mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-31 21:28:11 +09:00 
			
		
		
		
	fix
This commit is contained in:
		
							
								
								
									
										411
									
								
								models/git.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										411
									
								
								models/git.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,411 @@ | ||||
| // 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 ( | ||||
| 	"bufio" | ||||
| 	"container/list" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"os" | ||||
| 	"os/exec" | ||||
| 	"path" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/gogits/git" | ||||
|  | ||||
| 	"github.com/gogits/gogs/modules/base" | ||||
| ) | ||||
|  | ||||
| // RepoFile represents a file object in git repository. | ||||
| type RepoFile struct { | ||||
| 	*git.TreeEntry | ||||
| 	Path   string | ||||
| 	Size   int64 | ||||
| 	Repo   *git.Repository | ||||
| 	Commit *git.Commit | ||||
| } | ||||
|  | ||||
| // LookupBlob returns the content of an object. | ||||
| func (file *RepoFile) LookupBlob() (*git.Blob, error) { | ||||
| 	if file.Repo == nil { | ||||
| 		return nil, ErrRepoFileNotLoaded | ||||
| 	} | ||||
|  | ||||
| 	return file.Repo.LookupBlob(file.Id) | ||||
| } | ||||
|  | ||||
| // GetBranches returns all branches of given repository. | ||||
| func GetBranches(userName, repoName string) ([]string, error) { | ||||
| 	repo, err := git.OpenRepository(RepoPath(userName, repoName)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	refs, err := repo.AllReferences() | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	brs := make([]string, len(refs)) | ||||
| 	for i, ref := range refs { | ||||
| 		brs[i] = ref.BranchName() | ||||
| 	} | ||||
| 	return brs, nil | ||||
| } | ||||
|  | ||||
| // GetTags returns all tags of given repository. | ||||
| func GetTags(userName, repoName string) ([]string, error) { | ||||
| 	repo, err := git.OpenRepository(RepoPath(userName, repoName)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	refs, err := repo.AllTags() | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	tags := make([]string, len(refs)) | ||||
| 	for i, ref := range refs { | ||||
| 		tags[i] = ref.Name | ||||
| 	} | ||||
| 	return tags, nil | ||||
| } | ||||
|  | ||||
| func IsBranchExist(userName, repoName, branchName string) bool { | ||||
| 	repo, err := git.OpenRepository(RepoPath(userName, repoName)) | ||||
| 	if err != nil { | ||||
| 		return false | ||||
| 	} | ||||
| 	return repo.IsBranchExist(branchName) | ||||
| } | ||||
|  | ||||
| func GetTargetFile(userName, repoName, branchName, commitId, rpath string) (*RepoFile, error) { | ||||
| 	repo, err := git.OpenRepository(RepoPath(userName, repoName)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	commit, err := repo.GetCommitOfBranch(branchName) | ||||
| 	if err != nil { | ||||
| 		commit, err = repo.GetCommit(commitId) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	parts := strings.Split(path.Clean(rpath), "/") | ||||
|  | ||||
| 	var entry *git.TreeEntry | ||||
| 	tree := commit.Tree | ||||
| 	for i, part := range parts { | ||||
| 		if i == len(parts)-1 { | ||||
| 			entry = tree.EntryByName(part) | ||||
| 			if entry == nil { | ||||
| 				return nil, ErrRepoFileNotExist | ||||
| 			} | ||||
| 		} else { | ||||
| 			tree, err = repo.SubTree(tree, part) | ||||
| 			if err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	size, err := repo.ObjectSize(entry.Id) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	repoFile := &RepoFile{ | ||||
| 		entry, | ||||
| 		rpath, | ||||
| 		size, | ||||
| 		repo, | ||||
| 		commit, | ||||
| 	} | ||||
|  | ||||
| 	return repoFile, nil | ||||
| } | ||||
|  | ||||
| // GetReposFiles returns a list of file object in given directory of repository. | ||||
| // func GetReposFilesOfBranch(userName, repoName, branchName, rpath string) ([]*RepoFile, error) { | ||||
| // 	return getReposFiles(userName, repoName, commitId, rpath) | ||||
| // } | ||||
|  | ||||
| // GetReposFiles returns a list of file object in given directory of repository. | ||||
| func GetReposFiles(userName, repoName, commitId, rpath string) ([]*RepoFile, error) { | ||||
| 	return getReposFiles(userName, repoName, commitId, rpath) | ||||
| } | ||||
|  | ||||
| func getReposFiles(userName, repoName, commitId string, rpath string) ([]*RepoFile, error) { | ||||
| 	repopath := RepoPath(userName, repoName) | ||||
| 	repo, err := git.OpenRepository(repopath) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	commit, err := repo.GetCommit(commitId) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	var repodirs []*RepoFile | ||||
| 	var repofiles []*RepoFile | ||||
| 	commit.Tree.Walk(func(dirname string, entry *git.TreeEntry) int { | ||||
| 		if dirname == rpath { | ||||
| 			// TODO: size get method shoule be improved | ||||
| 			size, err := repo.ObjectSize(entry.Id) | ||||
| 			if err != nil { | ||||
| 				return 0 | ||||
| 			} | ||||
|  | ||||
| 			cmd := exec.Command("git", "log", "-1", "--pretty=format:%H", commitId, "--", path.Join(dirname, entry.Name)) | ||||
| 			cmd.Dir = repopath | ||||
| 			out, err := cmd.Output() | ||||
| 			if err != nil { | ||||
| 				return 0 | ||||
| 			} | ||||
| 			filecm, err := repo.GetCommit(string(out)) | ||||
| 			if err != nil { | ||||
| 				return 0 | ||||
| 			} | ||||
|  | ||||
| 			rp := &RepoFile{ | ||||
| 				entry, | ||||
| 				path.Join(dirname, entry.Name), | ||||
| 				size, | ||||
| 				repo, | ||||
| 				filecm, | ||||
| 			} | ||||
|  | ||||
| 			if entry.IsFile() { | ||||
| 				repofiles = append(repofiles, rp) | ||||
| 			} else if entry.IsDir() { | ||||
| 				repodirs = append(repodirs, rp) | ||||
| 			} | ||||
| 		} | ||||
| 		return 0 | ||||
| 	}) | ||||
|  | ||||
| 	return append(repodirs, repofiles...), nil | ||||
| } | ||||
|  | ||||
| func GetCommit(userName, repoName, commitId string) (*git.Commit, error) { | ||||
| 	repo, err := git.OpenRepository(RepoPath(userName, repoName)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return repo.GetCommit(commitId) | ||||
| } | ||||
|  | ||||
| // GetCommitsByBranch returns all commits of given branch of repository. | ||||
| func GetCommitsByBranch(userName, repoName, branchName string) (*list.List, error) { | ||||
| 	repo, err := git.OpenRepository(RepoPath(userName, repoName)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	r, err := repo.LookupReference(fmt.Sprintf("refs/heads/%s", branchName)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return r.AllCommits() | ||||
| } | ||||
|  | ||||
| // GetCommitsByCommitId returns all commits of given commitId of repository. | ||||
| func GetCommitsByCommitId(userName, repoName, commitId string) (*list.List, error) { | ||||
| 	repo, err := git.OpenRepository(RepoPath(userName, repoName)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	oid, err := git.NewOidFromString(commitId) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return repo.CommitsBefore(oid) | ||||
| } | ||||
|  | ||||
| // Diff line types. | ||||
| const ( | ||||
| 	DIFF_LINE_PLAIN = iota + 1 | ||||
| 	DIFF_LINE_ADD | ||||
| 	DIFF_LINE_DEL | ||||
| 	DIFF_LINE_SECTION | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	DIFF_FILE_ADD = iota + 1 | ||||
| 	DIFF_FILE_CHANGE | ||||
| 	DIFF_FILE_DEL | ||||
| ) | ||||
|  | ||||
| type DiffLine struct { | ||||
| 	LeftIdx  int | ||||
| 	RightIdx int | ||||
| 	Type     int | ||||
| 	Content  string | ||||
| } | ||||
|  | ||||
| func (d DiffLine) GetType() int { | ||||
| 	return d.Type | ||||
| } | ||||
|  | ||||
| type DiffSection struct { | ||||
| 	Name  string | ||||
| 	Lines []*DiffLine | ||||
| } | ||||
|  | ||||
| type DiffFile struct { | ||||
| 	Name               string | ||||
| 	Addition, Deletion int | ||||
| 	Type               int | ||||
| 	Sections           []*DiffSection | ||||
| } | ||||
|  | ||||
| type Diff struct { | ||||
| 	TotalAddition, TotalDeletion int | ||||
| 	Files                        []*DiffFile | ||||
| } | ||||
|  | ||||
| func (diff *Diff) NumFiles() int { | ||||
| 	return len(diff.Files) | ||||
| } | ||||
|  | ||||
| const DIFF_HEAD = "diff --git " | ||||
|  | ||||
| func ParsePatch(reader io.Reader) (*Diff, error) { | ||||
| 	scanner := bufio.NewScanner(reader) | ||||
| 	var ( | ||||
| 		curFile    *DiffFile | ||||
| 		curSection = &DiffSection{ | ||||
| 			Lines: make([]*DiffLine, 0, 10), | ||||
| 		} | ||||
|  | ||||
| 		leftLine, rightLine int | ||||
| 	) | ||||
|  | ||||
| 	diff := &Diff{Files: make([]*DiffFile, 0)} | ||||
| 	var i int | ||||
| 	for scanner.Scan() { | ||||
| 		line := scanner.Text() | ||||
| 		// fmt.Println(i, line) | ||||
| 		if strings.HasPrefix(line, "+++ ") || strings.HasPrefix(line, "--- ") { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		i = i + 1 | ||||
| 		if line == "" { | ||||
| 			continue | ||||
| 		} | ||||
| 		if line[0] == ' ' { | ||||
| 			diffLine := &DiffLine{Type: DIFF_LINE_PLAIN, Content: line, LeftIdx: leftLine, RightIdx: rightLine} | ||||
| 			leftLine++ | ||||
| 			rightLine++ | ||||
| 			curSection.Lines = append(curSection.Lines, diffLine) | ||||
| 			continue | ||||
| 		} else if line[0] == '@' { | ||||
| 			curSection = &DiffSection{} | ||||
| 			curFile.Sections = append(curFile.Sections, curSection) | ||||
| 			ss := strings.Split(line, "@@") | ||||
| 			diffLine := &DiffLine{Type: DIFF_LINE_SECTION, Content: line} | ||||
| 			curSection.Lines = append(curSection.Lines, diffLine) | ||||
|  | ||||
| 			// Parse line number. | ||||
| 			ranges := strings.Split(ss[len(ss)-2][1:], " ") | ||||
| 			leftLine, _ = base.StrTo(strings.Split(ranges[0], ",")[0][1:]).Int() | ||||
| 			rightLine, _ = base.StrTo(strings.Split(ranges[1], ",")[0]).Int() | ||||
| 			continue | ||||
| 		} else if line[0] == '+' { | ||||
| 			curFile.Addition++ | ||||
| 			diff.TotalAddition++ | ||||
| 			diffLine := &DiffLine{Type: DIFF_LINE_ADD, Content: line, RightIdx: rightLine} | ||||
| 			rightLine++ | ||||
| 			curSection.Lines = append(curSection.Lines, diffLine) | ||||
| 			continue | ||||
| 		} else if line[0] == '-' { | ||||
| 			curFile.Deletion++ | ||||
| 			diff.TotalDeletion++ | ||||
| 			diffLine := &DiffLine{Type: DIFF_LINE_DEL, Content: line, LeftIdx: leftLine} | ||||
| 			if leftLine > 0 { | ||||
| 				leftLine++ | ||||
| 			} | ||||
| 			curSection.Lines = append(curSection.Lines, diffLine) | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		// Get new file. | ||||
| 		if strings.HasPrefix(line, DIFF_HEAD) { | ||||
| 			fs := strings.Split(line[len(DIFF_HEAD):], " ") | ||||
| 			a := fs[0] | ||||
|  | ||||
| 			curFile = &DiffFile{ | ||||
| 				Name:     a[strings.Index(a, "/")+1:], | ||||
| 				Type:     DIFF_FILE_CHANGE, | ||||
| 				Sections: make([]*DiffSection, 0, 10), | ||||
| 			} | ||||
| 			diff.Files = append(diff.Files, curFile) | ||||
|  | ||||
| 			// Check file diff type. | ||||
| 			for scanner.Scan() { | ||||
| 				switch { | ||||
| 				case strings.HasPrefix(scanner.Text(), "new file"): | ||||
| 					curFile.Type = DIFF_FILE_ADD | ||||
| 				case strings.HasPrefix(scanner.Text(), "deleted"): | ||||
| 					curFile.Type = DIFF_FILE_DEL | ||||
| 				case strings.HasPrefix(scanner.Text(), "index"): | ||||
| 					curFile.Type = DIFF_FILE_CHANGE | ||||
| 				} | ||||
| 				if curFile.Type > 0 { | ||||
| 					break | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return diff, nil | ||||
| } | ||||
|  | ||||
| func GetDiff(repoPath, commitid string) (*Diff, error) { | ||||
| 	repo, err := git.OpenRepository(repoPath) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	commit, err := repo.GetCommit(commitid) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// First commit of repository. | ||||
| 	if commit.ParentCount() == 0 { | ||||
| 		rd, wr := io.Pipe() | ||||
| 		go func() { | ||||
| 			cmd := exec.Command("git", "show", commitid) | ||||
| 			cmd.Dir = repoPath | ||||
| 			cmd.Stdout = wr | ||||
| 			cmd.Stdin = os.Stdin | ||||
| 			cmd.Stderr = os.Stderr | ||||
| 			cmd.Run() | ||||
| 			wr.Close() | ||||
| 		}() | ||||
| 		defer rd.Close() | ||||
| 		return ParsePatch(rd) | ||||
| 	} | ||||
|  | ||||
| 	rd, wr := io.Pipe() | ||||
| 	go func() { | ||||
| 		cmd := exec.Command("git", "diff", commit.Parent(0).Oid.String(), commitid) | ||||
| 		cmd.Dir = repoPath | ||||
| 		cmd.Stdout = wr | ||||
| 		cmd.Stdin = os.Stdin | ||||
| 		cmd.Stderr = os.Stderr | ||||
| 		cmd.Run() | ||||
| 		wr.Close() | ||||
| 	}() | ||||
| 	defer rd.Close() | ||||
| 	return ParsePatch(rd) | ||||
| } | ||||
		Reference in New Issue
	
	Block a user