mirror of
				https://github.com/go-gitea/gitea
				synced 2025-09-28 03:28:13 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			465 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			465 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // 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) {
 | |
| 	repo, err := git.OpenRepository(RepoPath(userName, repoName))
 | |
| 	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
 | |
| 			}
 | |
| 
 | |
| 			var cm = commit
 | |
| 			var i int
 | |
| 			for {
 | |
| 				i = i + 1
 | |
| 				//fmt.Println(".....", i, cm.Id(), cm.ParentCount())
 | |
| 				if cm.ParentCount() == 0 {
 | |
| 					break
 | |
| 				} else if cm.ParentCount() == 1 {
 | |
| 					pt, _ := repo.SubTree(cm.Parent(0).Tree, dirname)
 | |
| 					if pt == nil {
 | |
| 						break
 | |
| 					}
 | |
| 					pEntry := pt.EntryByName(entry.Name)
 | |
| 					if pEntry == nil || !pEntry.Id.Equal(entry.Id) {
 | |
| 						break
 | |
| 					} else {
 | |
| 						cm = cm.Parent(0)
 | |
| 					}
 | |
| 				} else {
 | |
| 					var emptyCnt = 0
 | |
| 					var sameIdcnt = 0
 | |
| 					var lastSameCm *git.Commit
 | |
| 					//fmt.Println(".....", cm.ParentCount())
 | |
| 					for i := 0; i < cm.ParentCount(); i++ {
 | |
| 						//fmt.Println("parent", i, cm.Parent(i).Id())
 | |
| 						p := cm.Parent(i)
 | |
| 						pt, _ := repo.SubTree(p.Tree, dirname)
 | |
| 						var pEntry *git.TreeEntry
 | |
| 						if pt != nil {
 | |
| 							pEntry = pt.EntryByName(entry.Name)
 | |
| 						}
 | |
| 
 | |
| 						//fmt.Println("pEntry", pEntry)
 | |
| 
 | |
| 						if pEntry == nil {
 | |
| 							emptyCnt = emptyCnt + 1
 | |
| 							if emptyCnt+sameIdcnt == cm.ParentCount() {
 | |
| 								if lastSameCm == nil {
 | |
| 									goto loop
 | |
| 								} else {
 | |
| 									cm = lastSameCm
 | |
| 									break
 | |
| 								}
 | |
| 							}
 | |
| 						} else {
 | |
| 							//fmt.Println(i, "pEntry", pEntry.Id, "entry", entry.Id)
 | |
| 							if !pEntry.Id.Equal(entry.Id) {
 | |
| 								goto loop
 | |
| 							} else {
 | |
| 								lastSameCm = cm.Parent(i)
 | |
| 								sameIdcnt = sameIdcnt + 1
 | |
| 								if emptyCnt+sameIdcnt == cm.ParentCount() {
 | |
| 									// TODO: now follow the first parent commit?
 | |
| 									cm = lastSameCm
 | |
| 									//fmt.Println("sameId...")
 | |
| 									break
 | |
| 								}
 | |
| 							}
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 		loop:
 | |
| 
 | |
| 			rp := &RepoFile{
 | |
| 				entry,
 | |
| 				path.Join(dirname, entry.Name),
 | |
| 				size,
 | |
| 				repo,
 | |
| 				cm,
 | |
| 			}
 | |
| 
 | |
| 			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)
 | |
| }
 |