// Copyright 2014 The Gogs Authors. All rights reserved. // Use of this source code is governed by a MIT-style // license that can be found in the LICENSE file. package models import ( "container/list" "fmt" "os/exec" "strings" "time" "code.gitea.io/git" "code.gitea.io/gitea/modules/cache" "code.gitea.io/gitea/modules/log" "code.gitea.io/gitea/modules/util" ) // env keys for git hooks need const ( EnvRepoName = "GITEA_REPO_NAME" EnvRepoUsername = "GITEA_REPO_USER_NAME" EnvRepoIsWiki = "GITEA_REPO_IS_WIKI" EnvPusherName = "GITEA_PUSHER_NAME" EnvPusherID = "GITEA_PUSHER_ID" ) // CommitToPushCommit transforms a git.Commit to PushCommit type. func CommitToPushCommit(commit *git.Commit) *PushCommit { return &PushCommit{ Sha1: commit.ID.String(), Message: commit.Message(), AuthorEmail: commit.Author.Email, AuthorName: commit.Author.Name, CommitterEmail: commit.Committer.Email, CommitterName: commit.Committer.Name, Timestamp: commit.Author.When, } } // ListToPushCommits transforms a list.List to PushCommits type. func ListToPushCommits(l *list.List) *PushCommits { var commits []*PushCommit var actEmail string for e := l.Front(); e != nil; e = e.Next() { commit := e.Value.(*git.Commit) if actEmail == "" { actEmail = commit.Committer.Email } commits = append(commits, CommitToPushCommit(commit)) } return &PushCommits{l.Len(), commits, "", nil} } // PushUpdateOptions defines the push update options type PushUpdateOptions struct { PusherID int64 PusherName string RepoUserName string RepoName string RefFullName string OldCommitID string NewCommitID string } // PushUpdate must be called for any push actions in order to // generates necessary push action history feeds. func PushUpdate(branch string, opt PushUpdateOptions) error { repo, err := pushUpdate(branch, opt) if err != nil { return err } pusher, err := GetUserByID(opt.PusherID) if err != nil { return err } log.Trace("TriggerTask '%s/%s' by %s", repo.Name, branch, pusher.Name) go AddTestPullRequestTask(pusher, repo.ID, branch, true) return nil } func pushUpdateDeleteTag(repo *Repository, gitRepo *git.Repository, tagName string) error { rel, err := GetRelease(repo.ID, tagName) if err != nil { if IsErrReleaseNotExist(err) { return nil } return fmt.Errorf("GetRelease: %v", err) } if rel.IsTag { if _, err = x.ID(rel.ID).Delete(new(Release)); err != nil { return fmt.Errorf("Delete: %v", err) } } else { rel.IsDraft = true rel.NumCommits = 0 rel.Sha1 = "" if _, err = x.ID(rel.ID).AllCols().Update(rel); err != nil { return fmt.Errorf("Update: %v", err) } } return nil } func pushUpdateAddTag(repo *Repository, gitRepo *git.Repository, tagName string) error { rel, err := GetRelease(repo.ID, tagName) if err != nil && !IsErrReleaseNotExist(err) { return fmt.Errorf("GetRelease: %v", err) } tag, err := gitRepo.GetTag(tagName) if err != nil { return fmt.Errorf("GetTag: %v", err) } commit, err := tag.Commit() if err != nil { return fmt.Errorf("Commit: %v", err) } sig := tag.Tagger if sig == nil { sig = commit.Author } if sig == nil { sig = commit.Committer } var author *User var createdAt = time.Unix(1, 0) if sig != nil { author, err = GetUserByEmail(sig.Email) if err != nil && !IsErrUserNotExist(err) { return fmt.Errorf("GetUserByEmail: %v", err) } createdAt = sig.When } commitsCount, err := commit.CommitsCount() if err != nil { return fmt.Errorf("CommitsCount: %v", err) } if rel == nil { rel = &Release{ RepoID: repo.ID, Title: "", TagName: tagName, LowerTagName: strings.ToLower(tagName), Target: "", Sha1: commit.ID.String(), NumCommits: commitsCount, Note: "", IsDraft: false, IsPrerelease: false, IsTag: true, CreatedUnix: util.TimeStamp(createdAt.Unix()), } if author != nil { rel.PublisherID = author.ID } if _, err = x.InsertOne(rel); err != nil { return fmt.Errorf("InsertOne: %v", err) } } else { rel.Sha1 = commit.ID.String() rel.CreatedUnix = util.TimeStamp(createdAt.Unix()) rel.NumCommits = commitsCount rel.IsDraft = false if rel.IsTag && author != nil { rel.PublisherID = author.ID } if _, err = x.ID(rel.ID).AllCols().Update(rel); err != nil { return fmt.Errorf("Update: %v", err) } } return nil } func pushUpdate(branch string, opts PushUpdateOptions) (repo *Repository, err error) { isNewRef := opts.OldCommitID == git.EmptySHA isDelRef := opts.NewCommitID == git.EmptySHA if isNewRef && isDelRef { return nil, fmt.Errorf("Old and new revisions are both %s", git.EmptySHA) } repoPath := RepoPath(opts.RepoUserName, opts.RepoName) gitUpdate := exec.Command("git", "update-server-info") gitUpdate.Dir = repoPath if err = gitUpdate.Run(); err != nil { return nil, fmt.Errorf("Failed to call 'git update-server-info': %v", err) } owner, err := GetUserByName(opts.RepoUserName) if err != nil { return nil, fmt.Errorf("GetUserByName: %v", err) } repo, err = GetRepositoryByName(owner.ID, opts.RepoName) if err != nil { return nil, fmt.Errorf("GetRepositoryByName: %v", err) } gitRepo, err := git.OpenRepository(repoPath) if err != nil { return nil, fmt.Errorf("OpenRepository: %v", err) } if err = repo.UpdateSize(); err != nil { log.Error(4, "Failed to update size for repository: %v", err) } var commits = &PushCommits{} if strings.HasPrefix(opts.RefFullName, git.TagPrefix) { // If is tag reference tagName := opts.RefFullName[len(git.TagPrefix):] if isDelRef { err = pushUpdateDeleteTag(repo, gitRepo, tagName) if err != nil { return nil, fmt.Errorf("pushUpdateDeleteTag: %v", err) } } else { // Clear cache for tag commit count cache.Remove(repo.GetCommitsCountCacheKey(tagName, true)) err = pushUpdateAddTag(repo, gitRepo, tagName) if err != nil { return nil, fmt.Errorf("pushUpdateAddTag: %v", err) } } } else if !isDelRef { // If is branch reference // Clear cache for branch commit count cache.Remove(repo.GetCommitsCountCacheKey(opts.RefFullName[len(git.BranchPrefix):], true)) newCommit, err := gitRepo.GetCommit(opts.NewCommitID) if err != nil { return nil, fmt.Errorf("gitRepo.GetCommit: %v", err) } // Push new branch. var l *list.List if isNewRef { l, err = newCommit.CommitsBeforeLimit(10) if err != nil { return nil, fmt.Errorf("newCommit.CommitsBeforeLimit: %v", err) } } else { l, err = newCommit.CommitsBeforeUntil(opts.OldCommitID) if err != nil { return nil, fmt.Errorf("newCommit.CommitsBeforeUntil: %v", err) } } commits = ListToPushCommits(l) } if opts.RefFullName == git.BranchPrefix+repo.DefaultBranch { UpdateRepoIndexer(repo) } if err := CommitRepoAction(CommitRepoActionOptions{ PusherName: opts.PusherName, RepoOwnerID: owner.ID, RepoName: repo.Name, RefFullName: opts.RefFullName, OldCommitID: opts.OldCommitID, NewCommitID: opts.NewCommitID, Commits: commits, }); err != nil { return nil, fmt.Errorf("CommitRepoAction: %v", err) } // create actions that update pull requests tracking the branch that was pushed to prs, err := GetUnmergedPullRequestsByHeadInfo(repo.ID, branch) if err != nil { log.Error(4, "Find pull requests [head_repo_id: %d, head_branch: %s]: %v", repo.ID, branch, err) } else { pusher, err := GetUserByID(opts.PusherID) if err != nil { return nil, fmt.Errorf("GetUserByID: %v", err) } for _, pr := range prs { if err = pr.GetHeadRepo(); err != nil { log.Error(4, "GetHeadRepo: %v", err) continue } else if err = pr.GetBaseRepo(); err != nil { log.Error(4, "GetBaseRepo: %v", err) continue } var ( baseBranch *Branch headBranch *Branch baseCommit *git.Commit headCommit *git.Commit headGitRepo *git.Repository ) if baseBranch, err = pr.BaseRepo.GetBranch(pr.BaseBranch); err != nil { log.Error(4, "BaseRepo.GetBranch: %v", err) continue } if baseCommit, err = baseBranch.GetCommit(); err != nil { log.Error(4, "baseBranch.GetCommit: %v", err) continue } if headBranch, err = pr.HeadRepo.GetBranch(pr.HeadBranch); err != nil { log.Error(4, "HeadRepo.GetBranch: %v", err) continue } if headCommit, err = headBranch.GetCommit(); err != nil { log.Error(4, "headRepo.GetCommit: %v", err) continue } // NOTICE: this is using pr.HeadRepo rather than pr.BaseRepo to get commits since they are going to be pushed in a go routine if headGitRepo, err = git.OpenRepository(pr.HeadRepo.RepoPath()); err != nil { log.Error(4, "OpenRepository", err) continue } l, err := headGitRepo.CommitsBetweenIDs(headCommit.ID.String(), baseCommit.ID.String()) if err != nil { log.Error(4, "CommitsBetweenIDs: %v", err) continue } commits := ListToPushCommits(l) if err = CommitPullRequestAction(pusher, pr.BaseRepo, commits); err != nil { log.Error(4, "CommitPullRequestAction [%d]: %v", pr.ID, err) continue } } } return repo, nil }