| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412 | // 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 modelsimport (	"bufio"	"bytes"	"fmt"	"io"	"io/ioutil"	"os"	"os/exec"	"strings"	"html/template"	"html"	"github.com/Unknwon/com"	"golang.org/x/net/html/charset"	"golang.org/x/text/transform"	"github.com/sergi/go-diff/diffmatchpatch"	"github.com/gogits/git-module"	"github.com/gogits/gogs/modules/base"	"github.com/gogits/gogs/modules/log"	"github.com/gogits/gogs/modules/process")type DiffLineType uint8const (	DIFF_LINE_PLAIN DiffLineType = iota + 1	DIFF_LINE_ADD	DIFF_LINE_DEL	DIFF_LINE_SECTION)type DiffFileType uint8const (	DIFF_FILE_ADD DiffFileType = iota + 1	DIFF_FILE_CHANGE	DIFF_FILE_DEL	DIFF_FILE_RENAME)type DiffLine struct {	LeftIdx  int	RightIdx int	Type     DiffLineType	Content  string	ParsedContent template.HTML}func (d *DiffLine) GetType() int {	return int(d.Type)}type DiffSection struct {	Name  string	Lines []*DiffLine}func diffToHtml(diffRecord []diffmatchpatch.Diff, lineType DiffLineType) template.HTML {	result := ""	for _, s := range diffRecord {		if s.Type == diffmatchpatch.DiffInsert && lineType == DIFF_LINE_ADD {			result = result + "<span class=\"added-code\">"+html.EscapeString(s.Text)+"</span>"		} else if s.Type == diffmatchpatch.DiffDelete && lineType == DIFF_LINE_DEL {			result = result + "<span class=\"removed-code\">"+html.EscapeString(s.Text)+"</span>"		} else if s.Type == diffmatchpatch.DiffEqual {			result = result + html.EscapeString(s.Text)		}	}	return template.HTML(result)}func (diffSection *DiffSection) GetLeftLine(idx int, sliceIdx int) *DiffLine {	for i, diffLine := range diffSection.Lines {		if diffLine.LeftIdx == idx && diffLine.RightIdx == 0 {			// ignore if the lines are too far from each other			if i > sliceIdx-5 && i < sliceIdx+5 {				return diffLine			} else {				return nil			}		}	}	return nil}func (diffSection *DiffSection) GetRightLine(idx int, sliceIdx int) *DiffLine {	for i, diffLine := range diffSection.Lines {		if diffLine.RightIdx == idx && diffLine.LeftIdx == 0 {			// ignore if the lines are too far from each other			if i > sliceIdx-5 && i < sliceIdx+5 {				return diffLine			} else {				return nil			}		}	}	return nil}// computes diff of each diff line and set the HTML on diffLine.ParsedContentfunc (diffSection *DiffSection) ComputeLinesDiff() {	for i, diffLine := range diffSection.Lines {		var compareDiffLine *DiffLine		var diff1, diff2 string		// default content: as is		diffLine.ParsedContent = template.HTML(html.EscapeString(diffLine.Content[1:]))		// just compute diff for adds and removes		if diffLine.Type != DIFF_LINE_ADD && diffLine.Type != DIFF_LINE_DEL {			continue		}		// try to find equivalent diff line. ignore, otherwise		if diffLine.Type == DIFF_LINE_ADD {			compareDiffLine = diffSection.GetLeftLine(diffLine.RightIdx, i)			if compareDiffLine == nil {				continue			}			diff1 = compareDiffLine.Content			diff2 = diffLine.Content		} else {			compareDiffLine = diffSection.GetRightLine(diffLine.LeftIdx, i)			if compareDiffLine == nil {				continue			}			diff1 = diffLine.Content			diff2 = compareDiffLine.Content		}		dmp := diffmatchpatch.New()		diffRecord := dmp.DiffMain(diff1[1:], diff2[1:], true)		diffRecord = dmp.DiffCleanupSemantic(diffRecord)		diffLine.ParsedContent = diffToHtml(diffRecord, diffLine.Type)	}}type DiffFile struct {	Name               string	OldName            string	Index              int	Addition, Deletion int	Type               DiffFileType	IsCreated          bool	IsDeleted          bool	IsBin              bool	IsRenamed          bool	Sections           []*DiffSection}func (diffFile *DiffFile) GetType() int {	return int(diffFile.Type)}type Diff struct {	TotalAddition, TotalDeletion int	Files                        []*DiffFile}func (diff *Diff) NumFiles() int {	return len(diff.Files)}const DIFF_HEAD = "diff --git "func ParsePatch(maxlines int, reader io.Reader) (*Diff, error) {	var (		diff = &Diff{Files: make([]*DiffFile, 0)}		curFile    *DiffFile		curSection = &DiffSection{			Lines: make([]*DiffLine, 0, 10),		}		leftLine, rightLine int		lineCount           int	)	input := bufio.NewReader(reader)	isEOF := false	for {		if isEOF {			break		}		line, err := input.ReadString('\n')		if err != nil {			if err == io.EOF {				isEOF = true			} else {				return nil, fmt.Errorf("ReadString: %v", err)			}		}		if len(line) > 0 && line[len(line)-1] == '\n' {			// Remove line break.			line = line[:len(line)-1]		}		if strings.HasPrefix(line, "+++ ") || strings.HasPrefix(line, "--- ") {			continue		} else if len(line) == 0 {			continue		}		lineCount++		// Diff data too large, we only show the first about maxlines lines		if lineCount >= maxlines {			log.Warn("Diff data too large")			io.Copy(ioutil.Discard, reader)			diff.Files = nil			return diff, nil		}		switch {		case line[0] == ' ':			diffLine := &DiffLine{Type: DIFF_LINE_PLAIN, Content: line, LeftIdx: leftLine, RightIdx: rightLine}			leftLine++			rightLine++			curSection.Lines = append(curSection.Lines, diffLine)			continue		case 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[1][1:], " ")			leftLine, _ = com.StrTo(strings.Split(ranges[0], ",")[0][1:]).Int()			if len(ranges) > 1 {				rightLine, _ = com.StrTo(strings.Split(ranges[1], ",")[0]).Int()			} else {				log.Warn("Parse line number failed: %v", line)				rightLine = leftLine			}			continue		case line[0] == '+':			curFile.Addition++			diff.TotalAddition++			diffLine := &DiffLine{Type: DIFF_LINE_ADD, Content: line, RightIdx: rightLine}			rightLine++			curSection.Lines = append(curSection.Lines, diffLine)			continue		case 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)		case strings.HasPrefix(line, "Binary"):			curFile.IsBin = true			continue		}		// Get new file.		if strings.HasPrefix(line, DIFF_HEAD) {			middle := -1			// Note: In case file name is surrounded by double quotes (it happens only in git-shell).			// e.g. diff --git "a/xxx" "b/xxx"			hasQuote := line[len(DIFF_HEAD)] == '"'			if hasQuote {				middle = strings.Index(line, ` "b/`)			} else {				middle = strings.Index(line, " b/")			}			beg := len(DIFF_HEAD)			a := line[beg+2 : middle]			b := line[middle+3:]			if hasQuote {				a = string(git.UnescapeChars([]byte(a[1 : len(a)-1])))				b = string(git.UnescapeChars([]byte(b[1 : len(b)-1])))			}			curFile = &DiffFile{				Name:     a,				Index:    len(diff.Files) + 1,				Type:     DIFF_FILE_CHANGE,				Sections: make([]*DiffSection, 0, 10),			}			diff.Files = append(diff.Files, curFile)			// Check file diff type.			for {				line, err := input.ReadString('\n')				if err != nil {					if err == io.EOF {						isEOF = true					} else {						return nil, fmt.Errorf("ReadString: %v", err)					}				}				switch {				case strings.HasPrefix(line, "new file"):					curFile.Type = DIFF_FILE_ADD					curFile.IsCreated = true				case strings.HasPrefix(line, "deleted"):					curFile.Type = DIFF_FILE_DEL					curFile.IsDeleted = true				case strings.HasPrefix(line, "index"):					curFile.Type = DIFF_FILE_CHANGE				case strings.HasPrefix(line, "similarity index 100%"):					curFile.Type = DIFF_FILE_RENAME					curFile.IsRenamed = true					curFile.OldName = curFile.Name					curFile.Name = b				}				if curFile.Type > 0 {					break				}			}		}	}	// FIXME: detect encoding while parsing.	var buf bytes.Buffer	for _, f := range diff.Files {		buf.Reset()		for _, sec := range f.Sections {			for _, l := range sec.Lines {				buf.WriteString(l.Content)				buf.WriteString("\n")			}		}		charsetLabel, err := base.DetectEncoding(buf.Bytes())		if charsetLabel != "UTF-8" && err == nil {			encoding, _ := charset.Lookup(charsetLabel)			if encoding != nil {				d := encoding.NewDecoder()				for _, sec := range f.Sections {					for _, l := range sec.Lines {						if c, _, err := transform.String(d, l.Content); err == nil {							l.Content = c						}					}				}			}		}	}	return diff, nil}func GetDiffRange(repoPath, beforeCommitID string, afterCommitID string, maxlines int) (*Diff, error) {	repo, err := git.OpenRepository(repoPath)	if err != nil {		return nil, err	}	commit, err := repo.GetCommit(afterCommitID)	if err != nil {		return nil, err	}	var cmd *exec.Cmd	// if "after" commit given	if len(beforeCommitID) == 0 {		// First commit of repository.		if commit.ParentCount() == 0 {			cmd = exec.Command("git", "show", afterCommitID)		} else {			c, _ := commit.Parent(0)			cmd = exec.Command("git", "diff", "-M", c.ID.String(), afterCommitID)		}	} else {		cmd = exec.Command("git", "diff", "-M", beforeCommitID, afterCommitID)	}	cmd.Dir = repoPath	cmd.Stderr = os.Stderr	stdout, err := cmd.StdoutPipe()	if err != nil {		return nil, fmt.Errorf("StdoutPipe: %v", err)	}	if err = cmd.Start(); err != nil {		return nil, fmt.Errorf("Start: %v", err)	}	pid := process.Add(fmt.Sprintf("GetDiffRange (%s)", repoPath), cmd)	defer process.Remove(pid)	diff, err := ParsePatch(maxlines, stdout)	if err != nil {		return nil, fmt.Errorf("ParsePatch: %v", err)	}	if err = cmd.Wait(); err != nil {		return nil, fmt.Errorf("Wait: %v", err)	}	return diff, nil}func GetDiffCommit(repoPath, commitId string, maxlines int) (*Diff, error) {	return GetDiffRange(repoPath, "", commitId, maxlines)}
 |