gogs/models/update.go

223 lines
6.0 KiB
Go
Raw Permalink Normal View History

2014-04-27 11:05:13 +04:00
// 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.
2014-04-11 06:27:13 +04:00
package models
import (
"container/list"
2014-06-24 00:22:34 +04:00
"fmt"
2014-04-11 06:27:13 +04:00
"os/exec"
"strings"
2017-02-10 03:29:59 +03:00
git "github.com/gogits/git-module"
2014-04-11 06:27:13 +04:00
)
2016-08-12 12:29:29 +03:00
// 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.Committer.When,
2016-08-12 12:29:29 +03:00
}
}
func ListToPushCommits(l *list.List) *PushCommits {
if l == nil {
return &PushCommits{}
}
commits := make([]*PushCommit, 0)
var actEmail string
for e := l.Front(); e != nil; e = e.Next() {
commit := e.Value.(*git.Commit)
if actEmail == "" {
actEmail = commit.Committer.Email
}
2016-08-12 12:29:29 +03:00
commits = append(commits, CommitToPushCommit(commit))
}
return &PushCommits{l.Len(), commits, "", nil}
}
type PushUpdateOptions struct {
OldCommitID string
NewCommitID string
RefFullName string
PusherID int64
PusherName string
RepoUserName string
RepoName string
}
// PushUpdate must be called for any push actions in order to
// generates necessary push action history feeds.
func PushUpdate(opts PushUpdateOptions) (err error) {
2016-08-17 09:06:38 +03:00
isNewRef := opts.OldCommitID == git.EMPTY_SHA
isDelRef := opts.NewCommitID == git.EMPTY_SHA
if isNewRef && isDelRef {
2016-08-17 09:06:38 +03:00
return fmt.Errorf("Old and new revisions are both %s", git.EMPTY_SHA)
2014-04-11 06:27:13 +04:00
}
repoPath := RepoPath(opts.RepoUserName, opts.RepoName)
2014-04-11 06:27:13 +04:00
gitUpdate := exec.Command("git", "update-server-info")
gitUpdate.Dir = repoPath
if err = gitUpdate.Run(); err != nil {
return fmt.Errorf("Fail to call 'git update-server-info': %v", err)
}
2014-04-11 06:27:13 +04:00
gitRepo, err := git.OpenRepository(repoPath)
2014-04-11 06:27:13 +04:00
if err != nil {
return fmt.Errorf("OpenRepository: %v", err)
2014-04-11 06:27:13 +04:00
}
2016-08-17 09:06:38 +03:00
owner, err := GetUserByName(opts.RepoUserName)
2014-05-03 09:37:49 +04:00
if err != nil {
return fmt.Errorf("GetUserByName: %v", err)
2014-05-03 09:37:49 +04:00
}
2016-08-17 09:06:38 +03:00
repo, err := GetRepositoryByName(owner.ID, opts.RepoName)
2014-04-11 06:27:13 +04:00
if err != nil {
return fmt.Errorf("GetRepositoryByName: %v", err)
2014-04-11 06:27:13 +04:00
}
2017-03-12 09:59:48 +03:00
if err = repo.UpdateSize(); err != nil {
return fmt.Errorf("UpdateSize: %v", err)
}
// Push tags
2016-08-17 09:06:38 +03:00
if strings.HasPrefix(opts.RefFullName, git.TAG_PREFIX) {
if err := CommitRepoAction(CommitRepoActionOptions{
PusherName: opts.PusherName,
RepoOwnerID: owner.ID,
RepoName: repo.Name,
RefFullName: opts.RefFullName,
OldCommitID: opts.OldCommitID,
NewCommitID: opts.NewCommitID,
Commits: &PushCommits{},
}); err != nil {
return fmt.Errorf("CommitRepoAction.(tag): %v", err)
2014-06-28 10:55:33 +04:00
}
2016-08-17 09:06:38 +03:00
return nil
2014-06-28 10:55:33 +04:00
}
2014-06-28 19:56:41 +04:00
var l *list.List
// Skip read parent commits when delete branch
if !isDelRef {
var oldCommit *git.Commit
2017-03-12 09:59:48 +03:00
// Push new branch
newCommit, err := gitRepo.GetCommit(opts.NewCommitID)
2014-06-28 19:56:41 +04:00
if err != nil {
return fmt.Errorf("GetCommit [commit_id: %s]: %v", opts.NewCommitID, err)
2014-06-28 19:56:41 +04:00
}
if isNewRef {
2017-04-11 01:35:46 +03:00
l, err = newCommit.CommitsBefore()
if err != nil {
2017-04-11 01:35:46 +03:00
return fmt.Errorf("CommitsBefore [commit_id: %s]: %v", newCommit.ID, err)
}
} else {
oldCommit, err = gitRepo.GetCommit(opts.OldCommitID)
if err != nil {
return fmt.Errorf("GetCommit [commit_id: %s]: %v", opts.OldCommitID, err)
}
l, err = gitRepo.CommitsBefore(newCommit, oldCommit)
if err != nil {
return fmt.Errorf("CommitsBetween [before: %s, until: %s]: %v", opts.NewCommitID, opts.OldCommitID, err)
}
2014-06-28 19:56:41 +04:00
}
2017-04-11 01:35:46 +03:00
searchCommits := make([]*Commit, 0)
for e := l.Front(); e != nil; e = e.Next() {
commit := e.Value.(*git.Commit)
2017-04-27 02:05:33 +03:00
searchCommits = append(searchCommits, GitCommitToSearchCommit(repo.ID, commit))
2017-04-11 01:35:46 +03:00
}
x.Insert(searchCommits)
if !isNewRef {
// Remove deleted commits if any
var d *list.List
d, err = gitRepo.CommitsBefore(oldCommit, newCommit)
if err != nil {
return fmt.Errorf("CommitsBetween [before: %s, until: %s]: %v", opts.OldCommitID, opts.NewCommitID, err)
}
rebasedCommits := make([]string, 0)
for e := d.Front(); e != nil; e = e.Next() {
commit := e.Value.(*git.Commit)
rebasedCommits = append(rebasedCommits, commit.ID.String())
}
if len(rebasedCommits) > 0 {
d, err = gitRepo.CommitsUnreachable(rebasedCommits)
if err != nil {
return fmt.Errorf("CommitsUnreachable: %v", err)
}
rebasedCommits = make([]string, 0)
for e := d.Front(); e != nil; e = e.Next() {
commit := e.Value.(*git.Commit)
rebasedCommits = append(rebasedCommits, commit.ID.String())
}
if len(rebasedCommits) > 0 {
x.In("sha", rebasedCommits).Delete(new(Commit))
}
}
}
2014-06-28 19:56:41 +04:00
}
2016-08-17 09:06:38 +03:00
if err := CommitRepoAction(CommitRepoActionOptions{
PusherName: opts.PusherName,
RepoOwnerID: owner.ID,
RepoName: repo.Name,
RefFullName: opts.RefFullName,
OldCommitID: opts.OldCommitID,
NewCommitID: opts.NewCommitID,
Commits: ListToPushCommits(l),
}); err != nil {
return fmt.Errorf("CommitRepoAction.(branch): %v", err)
2014-04-11 06:27:13 +04:00
}
2014-06-24 00:22:34 +04:00
return nil
2014-04-11 06:27:13 +04:00
}
2017-04-27 02:05:33 +03:00
func ReindexCommits(RepoUserName string, RepoName string) (err error) {
repoPath := RepoPath(RepoUserName, RepoName)
gitRepo, err := git.OpenRepository(repoPath)
if err != nil {
return fmt.Errorf("OpenRepository: %v", err)
}
owner, err := GetUserByName(RepoUserName)
if err != nil {
return fmt.Errorf("GetUserByName: %v", err)
}
repo, err := GetRepositoryByName(owner.ID, RepoName)
if err != nil {
return fmt.Errorf("GetRepositoryByName: %v", err)
}
x.Delete(&Commit{RepoID: repo.ID})
for page := 0; ; page++ {
commits, err := gitRepo.CommitsByRangeSize("--all", page, 1000)
if err != nil {
return fmt.Errorf("CommitsByRange: %v", err)
}
searchCommits := make([]*Commit, 0)
for e := commits.Front(); e != nil; e = e.Next() {
commit := e.Value.(*git.Commit)
searchCommits = append(searchCommits, GitCommitToSearchCommit(repo.ID, commit))
}
x.Insert(searchCommits)
if len(searchCommits) < 1000 {
break
}
}
return nil
}