mirror of https://github.com/go-gitea/gitea.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
566 lines
17 KiB
566 lines
17 KiB
// Copyright 2015 The Gogs Authors. All rights reserved. |
|
// Copyright 2019 The Gitea Authors. All rights reserved. |
|
// SPDX-License-Identifier: MIT |
|
|
|
package git |
|
|
|
import ( |
|
"bytes" |
|
"io" |
|
"os" |
|
"strconv" |
|
"strings" |
|
|
|
"code.gitea.io/gitea/modules/git/gitcmd" |
|
"code.gitea.io/gitea/modules/setting" |
|
) |
|
|
|
// GetBranchCommitID returns last commit ID string of given branch. |
|
func (repo *Repository) GetBranchCommitID(name string) (string, error) { |
|
return repo.GetRefCommitID(BranchPrefix + name) |
|
} |
|
|
|
// GetTagCommitID returns last commit ID string of given tag. |
|
func (repo *Repository) GetTagCommitID(name string) (string, error) { |
|
return repo.GetRefCommitID(TagPrefix + name) |
|
} |
|
|
|
// GetCommit returns commit object of by ID string. |
|
func (repo *Repository) GetCommit(commitID string) (*Commit, error) { |
|
id, err := repo.ConvertToGitID(commitID) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return repo.getCommit(id) |
|
} |
|
|
|
// GetBranchCommit returns the last commit of given branch. |
|
func (repo *Repository) GetBranchCommit(name string) (*Commit, error) { |
|
commitID, err := repo.GetBranchCommitID(name) |
|
if err != nil { |
|
return nil, err |
|
} |
|
return repo.GetCommit(commitID) |
|
} |
|
|
|
// GetTagCommit get the commit of the specific tag via name |
|
func (repo *Repository) GetTagCommit(name string) (*Commit, error) { |
|
commitID, err := repo.GetTagCommitID(name) |
|
if err != nil { |
|
return nil, err |
|
} |
|
return repo.GetCommit(commitID) |
|
} |
|
|
|
func (repo *Repository) getCommitByPathWithID(id ObjectID, relpath string) (*Commit, error) { |
|
// File name starts with ':' must be escaped. |
|
if relpath[0] == ':' { |
|
relpath = `\` + relpath |
|
} |
|
|
|
stdout, _, runErr := gitcmd.NewCommand("log", "-1", prettyLogFormat). |
|
AddDynamicArguments(id.String()). |
|
AddDashesAndList(relpath). |
|
WithDir(repo.Path). |
|
RunStdString(repo.Ctx) |
|
if runErr != nil { |
|
return nil, runErr |
|
} |
|
|
|
id, err := NewIDFromString(stdout) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return repo.getCommit(id) |
|
} |
|
|
|
// GetCommitByPath returns the last commit of relative path. |
|
func (repo *Repository) GetCommitByPath(relpath string) (*Commit, error) { |
|
stdout, _, runErr := gitcmd.NewCommand("log", "-1", prettyLogFormat). |
|
AddDashesAndList(relpath). |
|
WithDir(repo.Path). |
|
RunStdBytes(repo.Ctx) |
|
if runErr != nil { |
|
return nil, runErr |
|
} |
|
|
|
commits, err := repo.parsePrettyFormatLogToList(stdout) |
|
if err != nil { |
|
return nil, err |
|
} |
|
if len(commits) == 0 { |
|
return nil, ErrNotExist{ID: relpath} |
|
} |
|
return commits[0], nil |
|
} |
|
|
|
// commitsByRangeWithTime returns the specific page commits before current revision, with not, since, until support |
|
func (repo *Repository) commitsByRangeWithTime(id ObjectID, page, pageSize int, not, since, until string) ([]*Commit, error) { |
|
cmd := gitcmd.NewCommand("log"). |
|
AddOptionFormat("--skip=%d", (page-1)*pageSize). |
|
AddOptionFormat("--max-count=%d", pageSize). |
|
AddArguments(prettyLogFormat). |
|
AddDynamicArguments(id.String()) |
|
|
|
if not != "" { |
|
cmd.AddOptionValues("--not", not) |
|
} |
|
if since != "" { |
|
cmd.AddOptionFormat("--since=%s", since) |
|
} |
|
if until != "" { |
|
cmd.AddOptionFormat("--until=%s", until) |
|
} |
|
|
|
stdout, _, err := cmd.WithDir(repo.Path).RunStdBytes(repo.Ctx) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return repo.parsePrettyFormatLogToList(stdout) |
|
} |
|
|
|
func (repo *Repository) searchCommits(id ObjectID, opts SearchCommitsOptions) ([]*Commit, error) { |
|
// add common arguments to git command |
|
addCommonSearchArgs := func(c *gitcmd.Command) { |
|
// ignore case |
|
c.AddArguments("-i") |
|
|
|
// add authors if present in search query |
|
for _, v := range opts.Authors { |
|
c.AddOptionFormat("--author=%s", v) |
|
} |
|
|
|
// add committers if present in search query |
|
for _, v := range opts.Committers { |
|
c.AddOptionFormat("--committer=%s", v) |
|
} |
|
|
|
// add time constraints if present in search query |
|
if len(opts.After) > 0 { |
|
c.AddOptionFormat("--after=%s", opts.After) |
|
} |
|
if len(opts.Before) > 0 { |
|
c.AddOptionFormat("--before=%s", opts.Before) |
|
} |
|
} |
|
|
|
// create new git log command with limit of 100 commits |
|
cmd := gitcmd.NewCommand("log", "-100", prettyLogFormat).AddDynamicArguments(id.String()) |
|
|
|
// pretend that all refs along with HEAD were listed on command line as <commis> |
|
// https://git-scm.com/docs/git-log#Documentation/git-log.txt---all |
|
// note this is done only for command created above |
|
if opts.All { |
|
cmd.AddArguments("--all") |
|
} |
|
|
|
// interpret search string keywords as string instead of regex |
|
cmd.AddArguments("--fixed-strings") |
|
|
|
// add remaining keywords from search string |
|
// note this is done only for command created above |
|
for _, v := range opts.Keywords { |
|
cmd.AddOptionFormat("--grep=%s", v) |
|
} |
|
|
|
// search for commits matching given constraints and keywords in commit msg |
|
addCommonSearchArgs(cmd) |
|
stdout, _, err := cmd.WithDir(repo.Path).RunStdBytes(repo.Ctx) |
|
if err != nil { |
|
return nil, err |
|
} |
|
if len(stdout) != 0 { |
|
stdout = append(stdout, '\n') |
|
} |
|
|
|
// if there are any keywords (ie not committer:, author:, time:) |
|
// then let's iterate over them |
|
for _, v := range opts.Keywords { |
|
// ignore anything not matching a valid sha pattern |
|
if id.Type().IsValid(v) { |
|
// create new git log command with 1 commit limit |
|
hashCmd := gitcmd.NewCommand("log", "-1", prettyLogFormat) |
|
// add previous arguments except for --grep and --all |
|
addCommonSearchArgs(hashCmd) |
|
// add keyword as <commit> |
|
hashCmd.AddDynamicArguments(v) |
|
|
|
// search with given constraints for commit matching sha hash of v |
|
hashMatching, _, err := hashCmd.WithDir(repo.Path).RunStdBytes(repo.Ctx) |
|
if err != nil || bytes.Contains(stdout, hashMatching) { |
|
continue |
|
} |
|
stdout = append(stdout, hashMatching...) |
|
stdout = append(stdout, '\n') |
|
} |
|
} |
|
|
|
return repo.parsePrettyFormatLogToList(bytes.TrimSuffix(stdout, []byte{'\n'})) |
|
} |
|
|
|
// FileChangedBetweenCommits Returns true if the file changed between commit IDs id1 and id2 |
|
// You must ensure that id1 and id2 are valid commit ids. |
|
func (repo *Repository) FileChangedBetweenCommits(filename, id1, id2 string) (bool, error) { |
|
stdout, _, err := gitcmd.NewCommand("diff", "--name-only", "-z"). |
|
AddDynamicArguments(id1, id2). |
|
AddDashesAndList(filename). |
|
WithDir(repo.Path). |
|
RunStdBytes(repo.Ctx) |
|
if err != nil { |
|
return false, err |
|
} |
|
return len(strings.TrimSpace(string(stdout))) > 0, nil |
|
} |
|
|
|
type CommitsByFileAndRangeOptions struct { |
|
Revision string |
|
File string |
|
Not string |
|
Page int |
|
Since string |
|
Until string |
|
} |
|
|
|
// CommitsByFileAndRange return the commits according revision file and the page |
|
func (repo *Repository) CommitsByFileAndRange(opts CommitsByFileAndRangeOptions) ([]*Commit, error) { |
|
stdoutReader, stdoutWriter := io.Pipe() |
|
defer func() { |
|
_ = stdoutReader.Close() |
|
_ = stdoutWriter.Close() |
|
}() |
|
go func() { |
|
stderr := strings.Builder{} |
|
gitCmd := gitcmd.NewCommand("rev-list"). |
|
AddOptionFormat("--max-count=%d", setting.Git.CommitsRangeSize). |
|
AddOptionFormat("--skip=%d", (opts.Page-1)*setting.Git.CommitsRangeSize) |
|
gitCmd.AddDynamicArguments(opts.Revision) |
|
|
|
if opts.Not != "" { |
|
gitCmd.AddOptionValues("--not", opts.Not) |
|
} |
|
if opts.Since != "" { |
|
gitCmd.AddOptionFormat("--since=%s", opts.Since) |
|
} |
|
if opts.Until != "" { |
|
gitCmd.AddOptionFormat("--until=%s", opts.Until) |
|
} |
|
|
|
gitCmd.AddDashesAndList(opts.File) |
|
err := gitCmd.WithDir(repo.Path). |
|
WithStdout(stdoutWriter). |
|
WithStderr(&stderr). |
|
Run(repo.Ctx) |
|
if err != nil { |
|
_ = stdoutWriter.CloseWithError(gitcmd.ConcatenateError(err, (&stderr).String())) |
|
} else { |
|
_ = stdoutWriter.Close() |
|
} |
|
}() |
|
|
|
objectFormat, err := repo.GetObjectFormat() |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
length := objectFormat.FullLength() |
|
commits := []*Commit{} |
|
shaline := make([]byte, length+1) |
|
for { |
|
n, err := io.ReadFull(stdoutReader, shaline) |
|
if err != nil || n < length { |
|
if err == io.EOF { |
|
err = nil |
|
} |
|
return commits, err |
|
} |
|
objectID, err := NewIDFromString(string(shaline[0:length])) |
|
if err != nil { |
|
return nil, err |
|
} |
|
commit, err := repo.getCommit(objectID) |
|
if err != nil { |
|
return nil, err |
|
} |
|
commits = append(commits, commit) |
|
} |
|
} |
|
|
|
// FilesCountBetween return the number of files changed between two commits |
|
func (repo *Repository) FilesCountBetween(startCommitID, endCommitID string) (int, error) { |
|
stdout, _, err := gitcmd.NewCommand("diff", "--name-only"). |
|
AddDynamicArguments(startCommitID + "..." + endCommitID). |
|
WithDir(repo.Path). |
|
RunStdString(repo.Ctx) |
|
if err != nil && strings.Contains(err.Error(), "no merge base") { |
|
// git >= 2.28 now returns an error if startCommitID and endCommitID have become unrelated. |
|
// previously it would return the results of git diff --name-only startCommitID endCommitID so let's try that... |
|
stdout, _, err = gitcmd.NewCommand("diff", "--name-only"). |
|
AddDynamicArguments(startCommitID, endCommitID). |
|
WithDir(repo.Path). |
|
RunStdString(repo.Ctx) |
|
} |
|
if err != nil { |
|
return 0, err |
|
} |
|
return len(strings.Split(stdout, "\n")) - 1, nil |
|
} |
|
|
|
// CommitsBetween returns a list that contains commits between [before, last). |
|
// If before is detached (removed by reset + push) it is not included. |
|
func (repo *Repository) CommitsBetween(last, before *Commit) ([]*Commit, error) { |
|
var stdout []byte |
|
var err error |
|
if before == nil { |
|
stdout, _, err = gitcmd.NewCommand("rev-list"). |
|
AddDynamicArguments(last.ID.String()). |
|
WithDir(repo.Path). |
|
RunStdBytes(repo.Ctx) |
|
} else { |
|
stdout, _, err = gitcmd.NewCommand("rev-list"). |
|
AddDynamicArguments(before.ID.String() + ".." + last.ID.String()). |
|
WithDir(repo.Path). |
|
RunStdBytes(repo.Ctx) |
|
if err != nil && strings.Contains(err.Error(), "no merge base") { |
|
// future versions of git >= 2.28 are likely to return an error if before and last have become unrelated. |
|
// previously it would return the results of git rev-list before last so let's try that... |
|
stdout, _, err = gitcmd.NewCommand("rev-list"). |
|
AddDynamicArguments(before.ID.String(), last.ID.String()). |
|
WithDir(repo.Path). |
|
RunStdBytes(repo.Ctx) |
|
} |
|
} |
|
if err != nil { |
|
return nil, err |
|
} |
|
return repo.parsePrettyFormatLogToList(bytes.TrimSpace(stdout)) |
|
} |
|
|
|
// CommitsBetweenLimit returns a list that contains at most limit commits skipping the first skip commits between [before, last) |
|
func (repo *Repository) CommitsBetweenLimit(last, before *Commit, limit, skip int) ([]*Commit, error) { |
|
var stdout []byte |
|
var err error |
|
if before == nil { |
|
stdout, _, err = gitcmd.NewCommand("rev-list"). |
|
AddOptionValues("--max-count", strconv.Itoa(limit)). |
|
AddOptionValues("--skip", strconv.Itoa(skip)). |
|
AddDynamicArguments(last.ID.String()). |
|
WithDir(repo.Path). |
|
RunStdBytes(repo.Ctx) |
|
} else { |
|
stdout, _, err = gitcmd.NewCommand("rev-list"). |
|
AddOptionValues("--max-count", strconv.Itoa(limit)). |
|
AddOptionValues("--skip", strconv.Itoa(skip)). |
|
AddDynamicArguments(before.ID.String() + ".." + last.ID.String()). |
|
WithDir(repo.Path). |
|
RunStdBytes(repo.Ctx) |
|
if err != nil && strings.Contains(err.Error(), "no merge base") { |
|
// future versions of git >= 2.28 are likely to return an error if before and last have become unrelated. |
|
// previously it would return the results of git rev-list --max-count n before last so let's try that... |
|
stdout, _, err = gitcmd.NewCommand("rev-list"). |
|
AddOptionValues("--max-count", strconv.Itoa(limit)). |
|
AddOptionValues("--skip", strconv.Itoa(skip)). |
|
AddDynamicArguments(before.ID.String(), last.ID.String()). |
|
WithDir(repo.Path). |
|
RunStdBytes(repo.Ctx) |
|
} |
|
} |
|
if err != nil { |
|
return nil, err |
|
} |
|
return repo.parsePrettyFormatLogToList(bytes.TrimSpace(stdout)) |
|
} |
|
|
|
// CommitsBetweenNotBase returns a list that contains commits between [before, last), excluding commits in baseBranch. |
|
// If before is detached (removed by reset + push) it is not included. |
|
func (repo *Repository) CommitsBetweenNotBase(last, before *Commit, baseBranch string) ([]*Commit, error) { |
|
var stdout []byte |
|
var err error |
|
if before == nil { |
|
stdout, _, err = gitcmd.NewCommand("rev-list"). |
|
AddDynamicArguments(last.ID.String()). |
|
AddOptionValues("--not", baseBranch). |
|
WithDir(repo.Path). |
|
RunStdBytes(repo.Ctx) |
|
} else { |
|
stdout, _, err = gitcmd.NewCommand("rev-list"). |
|
AddDynamicArguments(before.ID.String()+".."+last.ID.String()). |
|
AddOptionValues("--not", baseBranch). |
|
WithDir(repo.Path). |
|
RunStdBytes(repo.Ctx) |
|
if err != nil && strings.Contains(err.Error(), "no merge base") { |
|
// future versions of git >= 2.28 are likely to return an error if before and last have become unrelated. |
|
// previously it would return the results of git rev-list before last so let's try that... |
|
stdout, _, err = gitcmd.NewCommand("rev-list"). |
|
AddDynamicArguments(before.ID.String(), last.ID.String()). |
|
AddOptionValues("--not", baseBranch). |
|
WithDir(repo.Path). |
|
RunStdBytes(repo.Ctx) |
|
} |
|
} |
|
if err != nil { |
|
return nil, err |
|
} |
|
return repo.parsePrettyFormatLogToList(bytes.TrimSpace(stdout)) |
|
} |
|
|
|
// CommitsBetweenIDs return commits between twoe commits |
|
func (repo *Repository) CommitsBetweenIDs(last, before string) ([]*Commit, error) { |
|
lastCommit, err := repo.GetCommit(last) |
|
if err != nil { |
|
return nil, err |
|
} |
|
if before == "" { |
|
return repo.CommitsBetween(lastCommit, nil) |
|
} |
|
beforeCommit, err := repo.GetCommit(before) |
|
if err != nil { |
|
return nil, err |
|
} |
|
return repo.CommitsBetween(lastCommit, beforeCommit) |
|
} |
|
|
|
// commitsBefore the limit is depth, not total number of returned commits. |
|
func (repo *Repository) commitsBefore(id ObjectID, limit int) ([]*Commit, error) { |
|
cmd := gitcmd.NewCommand("log", prettyLogFormat) |
|
if limit > 0 { |
|
cmd.AddOptionFormat("-%d", limit) |
|
} |
|
cmd.AddDynamicArguments(id.String()) |
|
|
|
stdout, _, runErr := cmd.WithDir(repo.Path).RunStdBytes(repo.Ctx) |
|
if runErr != nil { |
|
return nil, runErr |
|
} |
|
|
|
formattedLog, err := repo.parsePrettyFormatLogToList(bytes.TrimSpace(stdout)) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
commits := make([]*Commit, 0, len(formattedLog)) |
|
for _, commit := range formattedLog { |
|
branches, err := repo.getBranches(os.Environ(), commit.ID.String(), 2) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
if len(branches) > 1 { |
|
break |
|
} |
|
|
|
commits = append(commits, commit) |
|
} |
|
|
|
return commits, nil |
|
} |
|
|
|
func (repo *Repository) getCommitsBefore(id ObjectID) ([]*Commit, error) { |
|
return repo.commitsBefore(id, 0) |
|
} |
|
|
|
func (repo *Repository) getCommitsBeforeLimit(id ObjectID, num int) ([]*Commit, error) { |
|
return repo.commitsBefore(id, num) |
|
} |
|
|
|
func (repo *Repository) getBranches(env []string, commitID string, limit int) ([]string, error) { |
|
if DefaultFeatures().CheckVersionAtLeast("2.7.0") { |
|
stdout, _, err := gitcmd.NewCommand("for-each-ref", "--format=%(refname:strip=2)"). |
|
AddOptionFormat("--count=%d", limit). |
|
AddOptionValues("--contains", commitID, BranchPrefix). |
|
WithDir(repo.Path). |
|
WithEnv(env). |
|
RunStdString(repo.Ctx) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
branches := strings.Fields(stdout) |
|
return branches, nil |
|
} |
|
|
|
stdout, _, err := gitcmd.NewCommand("branch"). |
|
AddOptionValues("--contains", commitID). |
|
WithDir(repo.Path). |
|
WithEnv(env). |
|
RunStdString(repo.Ctx) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
refs := strings.Split(stdout, "\n") |
|
|
|
var maxNum int |
|
if len(refs) > limit { |
|
maxNum = limit |
|
} else { |
|
maxNum = len(refs) - 1 |
|
} |
|
|
|
branches := make([]string, maxNum) |
|
for i, ref := range refs[:maxNum] { |
|
parts := strings.Fields(ref) |
|
|
|
branches[i] = parts[len(parts)-1] |
|
} |
|
return branches, nil |
|
} |
|
|
|
// GetCommitsFromIDs get commits from commit IDs |
|
func (repo *Repository) GetCommitsFromIDs(commitIDs []string) []*Commit { |
|
commits := make([]*Commit, 0, len(commitIDs)) |
|
|
|
for _, commitID := range commitIDs { |
|
commit, err := repo.GetCommit(commitID) |
|
if err == nil && commit != nil { |
|
commits = append(commits, commit) |
|
} |
|
} |
|
|
|
return commits |
|
} |
|
|
|
// IsCommitInBranch check if the commit is on the branch |
|
func (repo *Repository) IsCommitInBranch(commitID, branch string) (r bool, err error) { |
|
stdout, _, err := gitcmd.NewCommand("branch", "--contains"). |
|
AddDynamicArguments(commitID, branch). |
|
WithDir(repo.Path). |
|
RunStdString(repo.Ctx) |
|
if err != nil { |
|
return false, err |
|
} |
|
return len(stdout) > 0, err |
|
} |
|
|
|
// GetCommitBranchStart returns the commit where the branch diverged |
|
func (repo *Repository) GetCommitBranchStart(env []string, branch, endCommitID string) (string, error) { |
|
cmd := gitcmd.NewCommand("log", prettyLogFormat) |
|
cmd.AddDynamicArguments(endCommitID) |
|
|
|
stdout, _, runErr := cmd.WithDir(repo.Path). |
|
WithEnv(env). |
|
RunStdBytes(repo.Ctx) |
|
if runErr != nil { |
|
return "", runErr |
|
} |
|
|
|
parts := bytes.SplitSeq(bytes.TrimSpace(stdout), []byte{'\n'}) |
|
|
|
// check the commits one by one until we find a commit contained by another branch |
|
// and we think this commit is the divergence point |
|
for commitID := range parts { |
|
branches, err := repo.getBranches(env, string(commitID), 2) |
|
if err != nil { |
|
return "", err |
|
} |
|
for _, b := range branches { |
|
if b != branch { |
|
return string(commitID), nil |
|
} |
|
} |
|
} |
|
|
|
return "", nil |
|
}
|
|
|