Move git diff codes from models to services/gitdiff (#7889)
* move git diff codes from models to services/gitdiff * fix template * fix test * fix template
This commit is contained in:
		
							parent
							
								
									b660a732ae
								
							
						
					
					
						commit
						c03d75fbd5
					
				|  | @ -7,7 +7,6 @@ | ||||||
| package models | package models | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"bytes" |  | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"strings" | 	"strings" | ||||||
| 
 | 
 | ||||||
|  | @ -15,7 +14,6 @@ import ( | ||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/markup" | 	"code.gitea.io/gitea/modules/markup" | ||||||
| 	"code.gitea.io/gitea/modules/markup/markdown" | 	"code.gitea.io/gitea/modules/markup/markdown" | ||||||
| 	"code.gitea.io/gitea/modules/setting" |  | ||||||
| 	api "code.gitea.io/gitea/modules/structs" | 	api "code.gitea.io/gitea/modules/structs" | ||||||
| 	"code.gitea.io/gitea/modules/timeutil" | 	"code.gitea.io/gitea/modules/timeutil" | ||||||
| 
 | 
 | ||||||
|  | @ -488,32 +486,6 @@ func (c *Comment) UnsignedLine() uint64 { | ||||||
| 	return uint64(c.Line) | 	return uint64(c.Line) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // AsDiff returns c.Patch as *Diff
 |  | ||||||
| func (c *Comment) AsDiff() (*Diff, error) { |  | ||||||
| 	diff, err := ParsePatch(setting.Git.MaxGitDiffLines, |  | ||||||
| 		setting.Git.MaxGitDiffLineCharacters, setting.Git.MaxGitDiffFiles, strings.NewReader(c.Patch)) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, err |  | ||||||
| 	} |  | ||||||
| 	if len(diff.Files) == 0 { |  | ||||||
| 		return nil, fmt.Errorf("no file found for comment ID: %d", c.ID) |  | ||||||
| 	} |  | ||||||
| 	secs := diff.Files[0].Sections |  | ||||||
| 	if len(secs) == 0 { |  | ||||||
| 		return nil, fmt.Errorf("no sections found for comment ID: %d", c.ID) |  | ||||||
| 	} |  | ||||||
| 	return diff, nil |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // MustAsDiff executes AsDiff and logs the error instead of returning
 |  | ||||||
| func (c *Comment) MustAsDiff() *Diff { |  | ||||||
| 	diff, err := c.AsDiff() |  | ||||||
| 	if err != nil { |  | ||||||
| 		log.Warn("MustAsDiff: %v", err) |  | ||||||
| 	} |  | ||||||
| 	return diff |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // CodeCommentURL returns the url to a comment in code
 | // CodeCommentURL returns the url to a comment in code
 | ||||||
| func (c *Comment) CodeCommentURL() string { | func (c *Comment) CodeCommentURL() string { | ||||||
| 	err := c.LoadIssue() | 	err := c.LoadIssue() | ||||||
|  | @ -873,59 +845,6 @@ func CreateIssueComment(doer *User, repo *Repository, issue *Issue, content stri | ||||||
| 	return comment, nil | 	return comment, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // CreateCodeComment creates a plain code comment at the specified line / path
 |  | ||||||
| func CreateCodeComment(doer *User, repo *Repository, issue *Issue, content, treePath string, line, reviewID int64) (*Comment, error) { |  | ||||||
| 	var commitID, patch string |  | ||||||
| 	pr, err := GetPullRequestByIssueID(issue.ID) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("GetPullRequestByIssueID: %v", err) |  | ||||||
| 	} |  | ||||||
| 	if err := pr.GetBaseRepo(); err != nil { |  | ||||||
| 		return nil, fmt.Errorf("GetHeadRepo: %v", err) |  | ||||||
| 	} |  | ||||||
| 	gitRepo, err := git.OpenRepository(pr.BaseRepo.RepoPath()) |  | ||||||
| 	if err != nil { |  | ||||||
| 		return nil, fmt.Errorf("OpenRepository: %v", err) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	// FIXME validate treePath
 |  | ||||||
| 	// Get latest commit referencing the commented line
 |  | ||||||
| 	// No need for get commit for base branch changes
 |  | ||||||
| 	if line > 0 { |  | ||||||
| 		commit, err := gitRepo.LineBlame(pr.GetGitRefName(), gitRepo.Path, treePath, uint(line)) |  | ||||||
| 		if err == nil { |  | ||||||
| 			commitID = commit.ID.String() |  | ||||||
| 		} else if !strings.Contains(err.Error(), "exit status 128 - fatal: no such path") { |  | ||||||
| 			return nil, fmt.Errorf("LineBlame[%s, %s, %s, %d]: %v", pr.GetGitRefName(), gitRepo.Path, treePath, line, err) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	// Only fetch diff if comment is review comment
 |  | ||||||
| 	if reviewID != 0 { |  | ||||||
| 		headCommitID, err := gitRepo.GetRefCommitID(pr.GetGitRefName()) |  | ||||||
| 		if err != nil { |  | ||||||
| 			return nil, fmt.Errorf("GetRefCommitID[%s]: %v", pr.GetGitRefName(), err) |  | ||||||
| 		} |  | ||||||
| 		patchBuf := new(bytes.Buffer) |  | ||||||
| 		if err := GetRawDiffForFile(gitRepo.Path, pr.MergeBase, headCommitID, RawDiffNormal, treePath, patchBuf); err != nil { |  | ||||||
| 			return nil, fmt.Errorf("GetRawDiffForLine[%s, %s, %s, %s]: %v", err, gitRepo.Path, pr.MergeBase, headCommitID, treePath) |  | ||||||
| 		} |  | ||||||
| 		patch = CutDiffAroundLine(patchBuf, int64((&Comment{Line: line}).UnsignedLine()), line < 0, setting.UI.CodeCommentLines) |  | ||||||
| 	} |  | ||||||
| 	return CreateComment(&CreateCommentOptions{ |  | ||||||
| 		Type:      CommentTypeCode, |  | ||||||
| 		Doer:      doer, |  | ||||||
| 		Repo:      repo, |  | ||||||
| 		Issue:     issue, |  | ||||||
| 		Content:   content, |  | ||||||
| 		LineNum:   line, |  | ||||||
| 		TreePath:  treePath, |  | ||||||
| 		CommitSHA: commitID, |  | ||||||
| 		ReviewID:  reviewID, |  | ||||||
| 		Patch:     patch, |  | ||||||
| 	}) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // CreateRefComment creates a commit reference comment to issue.
 | // CreateRefComment creates a commit reference comment to issue.
 | ||||||
| func CreateRefComment(doer *User, repo *Repository, issue *Issue, content, commitSHA string) error { | func CreateRefComment(doer *User, repo *Repository, issue *Issue, content, commitSHA string) error { | ||||||
| 	if len(commitSHA) == 0 { | 	if len(commitSHA) == 0 { | ||||||
|  |  | ||||||
|  | @ -250,3 +250,8 @@ func MaxBatchInsertSize(bean interface{}) int { | ||||||
| 	t := x.TableInfo(bean) | 	t := x.TableInfo(bean) | ||||||
| 	return 999 / len(t.ColumnsSeq()) | 	return 999 / len(t.ColumnsSeq()) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | // Count returns records number according struct's fields as database query conditions
 | ||||||
|  | func Count(bean interface{}) (int64, error) { | ||||||
|  | 	return x.Count(bean) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @ -8,10 +8,11 @@ import ( | ||||||
| 	"strings" | 	"strings" | ||||||
| 
 | 
 | ||||||
| 	"code.gitea.io/gitea/models" | 	"code.gitea.io/gitea/models" | ||||||
|  | 	"code.gitea.io/gitea/services/gitdiff" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // GetDiffPreview produces and returns diff result of a file which is not yet committed.
 | // GetDiffPreview produces and returns diff result of a file which is not yet committed.
 | ||||||
| func GetDiffPreview(repo *models.Repository, branch, treePath, content string) (*models.Diff, error) { | func GetDiffPreview(repo *models.Repository, branch, treePath, content string) (*gitdiff.Diff, error) { | ||||||
| 	if branch == "" { | 	if branch == "" { | ||||||
| 		branch = repo.DefaultBranch | 		branch = repo.DefaultBranch | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | @ -9,6 +9,7 @@ import ( | ||||||
| 
 | 
 | ||||||
| 	"code.gitea.io/gitea/models" | 	"code.gitea.io/gitea/models" | ||||||
| 	"code.gitea.io/gitea/modules/test" | 	"code.gitea.io/gitea/modules/test" | ||||||
|  | 	"code.gitea.io/gitea/services/gitdiff" | ||||||
| 
 | 
 | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
|  | @ -25,10 +26,10 @@ func TestGetDiffPreview(t *testing.T) { | ||||||
| 	treePath := "README.md" | 	treePath := "README.md" | ||||||
| 	content := "# repo1\n\nDescription for repo1\nthis is a new line" | 	content := "# repo1\n\nDescription for repo1\nthis is a new line" | ||||||
| 
 | 
 | ||||||
| 	expectedDiff := &models.Diff{ | 	expectedDiff := &gitdiff.Diff{ | ||||||
| 		TotalAddition: 2, | 		TotalAddition: 2, | ||||||
| 		TotalDeletion: 1, | 		TotalDeletion: 1, | ||||||
| 		Files: []*models.DiffFile{ | 		Files: []*gitdiff.DiffFile{ | ||||||
| 			{ | 			{ | ||||||
| 				Name:        "README.md", | 				Name:        "README.md", | ||||||
| 				OldName:     "README.md", | 				OldName:     "README.md", | ||||||
|  | @ -42,10 +43,10 @@ func TestGetDiffPreview(t *testing.T) { | ||||||
| 				IsLFSFile:   false, | 				IsLFSFile:   false, | ||||||
| 				IsRenamed:   false, | 				IsRenamed:   false, | ||||||
| 				IsSubmodule: false, | 				IsSubmodule: false, | ||||||
| 				Sections: []*models.DiffSection{ | 				Sections: []*gitdiff.DiffSection{ | ||||||
| 					{ | 					{ | ||||||
| 						Name: "", | 						Name: "", | ||||||
| 						Lines: []*models.DiffLine{ | 						Lines: []*gitdiff.DiffLine{ | ||||||
| 							{ | 							{ | ||||||
| 								LeftIdx:  0, | 								LeftIdx:  0, | ||||||
| 								RightIdx: 0, | 								RightIdx: 0, | ||||||
|  |  | ||||||
|  | @ -20,6 +20,7 @@ import ( | ||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/process" | 	"code.gitea.io/gitea/modules/process" | ||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
|  | 	"code.gitea.io/gitea/services/gitdiff" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // TemporaryUploadRepository is a type to wrap our upload repositories as a shallow clone
 | // TemporaryUploadRepository is a type to wrap our upload repositories as a shallow clone
 | ||||||
|  | @ -290,7 +291,7 @@ func (t *TemporaryUploadRepository) Push(doer *models.User, commitHash string, b | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // DiffIndex returns a Diff of the current index to the head
 | // DiffIndex returns a Diff of the current index to the head
 | ||||||
| func (t *TemporaryUploadRepository) DiffIndex() (diff *models.Diff, err error) { | func (t *TemporaryUploadRepository) DiffIndex() (diff *gitdiff.Diff, err error) { | ||||||
| 	timeout := 5 * time.Minute | 	timeout := 5 * time.Minute | ||||||
| 	ctx, cancel := context.WithTimeout(context.Background(), timeout) | 	ctx, cancel := context.WithTimeout(context.Background(), timeout) | ||||||
| 	defer cancel() | 	defer cancel() | ||||||
|  | @ -313,7 +314,7 @@ func (t *TemporaryUploadRepository) DiffIndex() (diff *models.Diff, err error) { | ||||||
| 	pid := process.GetManager().Add(fmt.Sprintf("diffIndex [repo_path: %s]", t.repo.RepoPath()), cmd) | 	pid := process.GetManager().Add(fmt.Sprintf("diffIndex [repo_path: %s]", t.repo.RepoPath()), cmd) | ||||||
| 	defer process.GetManager().Remove(pid) | 	defer process.GetManager().Remove(pid) | ||||||
| 
 | 
 | ||||||
| 	diff, err = models.ParsePatch(setting.Git.MaxGitDiffLines, setting.Git.MaxGitDiffLineCharacters, setting.Git.MaxGitDiffFiles, stdout) | 	diff, err = gitdiff.ParsePatch(setting.Git.MaxGitDiffLines, setting.Git.MaxGitDiffLineCharacters, setting.Git.MaxGitDiffFiles, stdout) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, fmt.Errorf("ParsePatch: %v", err) | 		return nil, fmt.Errorf("ParsePatch: %v", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | @ -27,6 +27,7 @@ import ( | ||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
| 	"code.gitea.io/gitea/modules/timeutil" | 	"code.gitea.io/gitea/modules/timeutil" | ||||||
| 	"code.gitea.io/gitea/modules/util" | 	"code.gitea.io/gitea/modules/util" | ||||||
|  | 	"code.gitea.io/gitea/services/gitdiff" | ||||||
| 
 | 
 | ||||||
| 	"gopkg.in/editorconfig/editorconfig-core-go.v1" | 	"gopkg.in/editorconfig/editorconfig-core-go.v1" | ||||||
| ) | ) | ||||||
|  | @ -230,6 +231,7 @@ func NewFuncMap() []template.FuncMap { | ||||||
| 			} | 			} | ||||||
| 			return float32(n) * 100 / float32(sum) | 			return float32(n) * 100 / float32(sum) | ||||||
| 		}, | 		}, | ||||||
|  | 		"CommentMustAsDiff": gitdiff.CommentMustAsDiff, | ||||||
| 	}} | 	}} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -16,6 +16,7 @@ import ( | ||||||
| 	"code.gitea.io/gitea/modules/git" | 	"code.gitea.io/gitea/modules/git" | ||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
|  | 	"code.gitea.io/gitea/services/gitdiff" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| const ( | const ( | ||||||
|  | @ -217,7 +218,7 @@ func Diff(ctx *context.Context) { | ||||||
| 
 | 
 | ||||||
| 	ctx.Data["CommitStatus"] = models.CalcCommitStatus(statuses) | 	ctx.Data["CommitStatus"] = models.CalcCommitStatus(statuses) | ||||||
| 
 | 
 | ||||||
| 	diff, err := models.GetDiffCommit(models.RepoPath(userName, repoName), | 	diff, err := gitdiff.GetDiffCommit(models.RepoPath(userName, repoName), | ||||||
| 		commitID, setting.Git.MaxGitDiffLines, | 		commitID, setting.Git.MaxGitDiffLines, | ||||||
| 		setting.Git.MaxGitDiffLineCharacters, setting.Git.MaxGitDiffFiles) | 		setting.Git.MaxGitDiffLineCharacters, setting.Git.MaxGitDiffFiles) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
|  | @ -269,10 +270,10 @@ func Diff(ctx *context.Context) { | ||||||
| 
 | 
 | ||||||
| // RawDiff dumps diff results of repository in given commit ID to io.Writer
 | // RawDiff dumps diff results of repository in given commit ID to io.Writer
 | ||||||
| func RawDiff(ctx *context.Context) { | func RawDiff(ctx *context.Context) { | ||||||
| 	if err := models.GetRawDiff( | 	if err := gitdiff.GetRawDiff( | ||||||
| 		models.RepoPath(ctx.Repo.Owner.Name, ctx.Repo.Repository.Name), | 		models.RepoPath(ctx.Repo.Owner.Name, ctx.Repo.Repository.Name), | ||||||
| 		ctx.Params(":sha"), | 		ctx.Params(":sha"), | ||||||
| 		models.RawDiffType(ctx.Params(":ext")), | 		gitdiff.RawDiffType(ctx.Params(":ext")), | ||||||
| 		ctx.Resp, | 		ctx.Resp, | ||||||
| 	); err != nil { | 	); err != nil { | ||||||
| 		ctx.ServerError("GetRawDiff", err) | 		ctx.ServerError("GetRawDiff", err) | ||||||
|  |  | ||||||
|  | @ -14,6 +14,7 @@ import ( | ||||||
| 	"code.gitea.io/gitea/modules/git" | 	"code.gitea.io/gitea/modules/git" | ||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
|  | 	"code.gitea.io/gitea/services/gitdiff" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| const ( | const ( | ||||||
|  | @ -230,7 +231,7 @@ func PrepareCompareDiff( | ||||||
| 		return true | 		return true | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	diff, err := models.GetDiffRange(models.RepoPath(headUser.Name, headRepo.Name), | 	diff, err := gitdiff.GetDiffRange(models.RepoPath(headUser.Name, headRepo.Name), | ||||||
| 		compareInfo.MergeBase, headCommitID, setting.Git.MaxGitDiffLines, | 		compareInfo.MergeBase, headCommitID, setting.Git.MaxGitDiffLines, | ||||||
| 		setting.Git.MaxGitDiffLineCharacters, setting.Git.MaxGitDiffFiles) | 		setting.Git.MaxGitDiffLineCharacters, setting.Git.MaxGitDiffFiles) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
|  |  | ||||||
|  | @ -24,6 +24,7 @@ import ( | ||||||
| 	"code.gitea.io/gitea/modules/pull" | 	"code.gitea.io/gitea/modules/pull" | ||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
| 	"code.gitea.io/gitea/modules/util" | 	"code.gitea.io/gitea/modules/util" | ||||||
|  | 	"code.gitea.io/gitea/services/gitdiff" | ||||||
| 
 | 
 | ||||||
| 	"github.com/unknwon/com" | 	"github.com/unknwon/com" | ||||||
| ) | ) | ||||||
|  | @ -517,7 +518,7 @@ func ViewPullFiles(ctx *context.Context) { | ||||||
| 		ctx.Data["Reponame"] = pull.HeadRepo.Name | 		ctx.Data["Reponame"] = pull.HeadRepo.Name | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	diff, err := models.GetDiffRangeWithWhitespaceBehavior(diffRepoPath, | 	diff, err := gitdiff.GetDiffRangeWithWhitespaceBehavior(diffRepoPath, | ||||||
| 		startCommitID, endCommitID, setting.Git.MaxGitDiffLines, | 		startCommitID, endCommitID, setting.Git.MaxGitDiffLines, | ||||||
| 		setting.Git.MaxGitDiffLineCharacters, setting.Git.MaxGitDiffFiles, | 		setting.Git.MaxGitDiffLineCharacters, setting.Git.MaxGitDiffFiles, | ||||||
| 		whitespaceFlags[ctx.Data["WhitespaceBehavior"].(string)]) | 		whitespaceFlags[ctx.Data["WhitespaceBehavior"].(string)]) | ||||||
|  |  | ||||||
|  | @ -13,6 +13,7 @@ import ( | ||||||
| 	"code.gitea.io/gitea/modules/log" | 	"code.gitea.io/gitea/modules/log" | ||||||
| 	"code.gitea.io/gitea/modules/notification" | 	"code.gitea.io/gitea/modules/notification" | ||||||
| 	pull_service "code.gitea.io/gitea/modules/pull" | 	pull_service "code.gitea.io/gitea/modules/pull" | ||||||
|  | 	comment_service "code.gitea.io/gitea/services/comments" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // CreateCodeComment will create a code comment including an pending review if required
 | // CreateCodeComment will create a code comment including an pending review if required
 | ||||||
|  | @ -69,7 +70,7 @@ func CreateCodeComment(ctx *context.Context, form auth.CodeCommentForm) { | ||||||
| 		review.ID = form.Reply | 		review.ID = form.Reply | ||||||
| 	} | 	} | ||||||
| 	//FIXME check if line, commit and treepath exist
 | 	//FIXME check if line, commit and treepath exist
 | ||||||
| 	comment, err := models.CreateCodeComment( | 	comment, err := comment_service.CreateCodeComment( | ||||||
| 		ctx.User, | 		ctx.User, | ||||||
| 		issue.Repo, | 		issue.Repo, | ||||||
| 		issue, | 		issue, | ||||||
|  |  | ||||||
|  | @ -0,0 +1,69 @@ | ||||||
|  | // Copyright 2019 The Gitea 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 comments | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"bytes" | ||||||
|  | 	"fmt" | ||||||
|  | 	"strings" | ||||||
|  | 
 | ||||||
|  | 	"code.gitea.io/gitea/models" | ||||||
|  | 	"code.gitea.io/gitea/modules/git" | ||||||
|  | 	"code.gitea.io/gitea/modules/setting" | ||||||
|  | 	"code.gitea.io/gitea/services/gitdiff" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // CreateCodeComment creates a plain code comment at the specified line / path
 | ||||||
|  | func CreateCodeComment(doer *models.User, repo *models.Repository, issue *models.Issue, content, treePath string, line, reviewID int64) (*models.Comment, error) { | ||||||
|  | 	var commitID, patch string | ||||||
|  | 	pr, err := models.GetPullRequestByIssueID(issue.ID) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("GetPullRequestByIssueID: %v", err) | ||||||
|  | 	} | ||||||
|  | 	if err := pr.GetBaseRepo(); err != nil { | ||||||
|  | 		return nil, fmt.Errorf("GetHeadRepo: %v", err) | ||||||
|  | 	} | ||||||
|  | 	gitRepo, err := git.OpenRepository(pr.BaseRepo.RepoPath()) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("OpenRepository: %v", err) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// FIXME validate treePath
 | ||||||
|  | 	// Get latest commit referencing the commented line
 | ||||||
|  | 	// No need for get commit for base branch changes
 | ||||||
|  | 	if line > 0 { | ||||||
|  | 		commit, err := gitRepo.LineBlame(pr.GetGitRefName(), gitRepo.Path, treePath, uint(line)) | ||||||
|  | 		if err == nil { | ||||||
|  | 			commitID = commit.ID.String() | ||||||
|  | 		} else if !strings.Contains(err.Error(), "exit status 128 - fatal: no such path") { | ||||||
|  | 			return nil, fmt.Errorf("LineBlame[%s, %s, %s, %d]: %v", pr.GetGitRefName(), gitRepo.Path, treePath, line, err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Only fetch diff if comment is review comment
 | ||||||
|  | 	if reviewID != 0 { | ||||||
|  | 		headCommitID, err := gitRepo.GetRefCommitID(pr.GetGitRefName()) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, fmt.Errorf("GetRefCommitID[%s]: %v", pr.GetGitRefName(), err) | ||||||
|  | 		} | ||||||
|  | 		patchBuf := new(bytes.Buffer) | ||||||
|  | 		if err := gitdiff.GetRawDiffForFile(gitRepo.Path, pr.MergeBase, headCommitID, gitdiff.RawDiffNormal, treePath, patchBuf); err != nil { | ||||||
|  | 			return nil, fmt.Errorf("GetRawDiffForLine[%s, %s, %s, %s]: %v", err, gitRepo.Path, pr.MergeBase, headCommitID, treePath) | ||||||
|  | 		} | ||||||
|  | 		patch = gitdiff.CutDiffAroundLine(patchBuf, int64((&models.Comment{Line: line}).UnsignedLine()), line < 0, setting.UI.CodeCommentLines) | ||||||
|  | 	} | ||||||
|  | 	return models.CreateComment(&models.CreateCommentOptions{ | ||||||
|  | 		Type:      models.CommentTypeCode, | ||||||
|  | 		Doer:      doer, | ||||||
|  | 		Repo:      repo, | ||||||
|  | 		Issue:     issue, | ||||||
|  | 		Content:   content, | ||||||
|  | 		LineNum:   line, | ||||||
|  | 		TreePath:  treePath, | ||||||
|  | 		CommitSHA: commitID, | ||||||
|  | 		ReviewID:  reviewID, | ||||||
|  | 		Patch:     patch, | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  | @ -1,8 +1,9 @@ | ||||||
| // Copyright 2014 The Gogs Authors. All rights reserved.
 | // Copyright 2014 The Gogs Authors. All rights reserved.
 | ||||||
|  | // Copyright 2019 The Gitea Authors. All rights reserved.
 | ||||||
| // Use of this source code is governed by a MIT-style
 | // Use of this source code is governed by a MIT-style
 | ||||||
| // license that can be found in the LICENSE file.
 | // license that can be found in the LICENSE file.
 | ||||||
| 
 | 
 | ||||||
| package models | package gitdiff | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"bufio" | 	"bufio" | ||||||
|  | @ -19,6 +20,7 @@ import ( | ||||||
| 	"strconv" | 	"strconv" | ||||||
| 	"strings" | 	"strings" | ||||||
| 
 | 
 | ||||||
|  | 	"code.gitea.io/gitea/models" | ||||||
| 	"code.gitea.io/gitea/modules/charset" | 	"code.gitea.io/gitea/modules/charset" | ||||||
| 	"code.gitea.io/gitea/modules/git" | 	"code.gitea.io/gitea/modules/git" | ||||||
| 	"code.gitea.io/gitea/modules/highlight" | 	"code.gitea.io/gitea/modules/highlight" | ||||||
|  | @ -60,7 +62,7 @@ type DiffLine struct { | ||||||
| 	RightIdx int | 	RightIdx int | ||||||
| 	Type     DiffLineType | 	Type     DiffLineType | ||||||
| 	Content  string | 	Content  string | ||||||
| 	Comments []*Comment | 	Comments []*models.Comment | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GetType returns the type of a DiffLine.
 | // GetType returns the type of a DiffLine.
 | ||||||
|  | @ -254,8 +256,8 @@ type Diff struct { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // LoadComments loads comments into each line
 | // LoadComments loads comments into each line
 | ||||||
| func (diff *Diff) LoadComments(issue *Issue, currentUser *User) error { | func (diff *Diff) LoadComments(issue *models.Issue, currentUser *models.User) error { | ||||||
| 	allComments, err := FetchCodeComments(issue, currentUser) | 	allComments, err := models.FetchCodeComments(issue, currentUser) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | @ -472,16 +474,16 @@ func ParsePatch(maxLines, maxLineCharacters, maxFiles int, reader io.Reader) (*D | ||||||
| 
 | 
 | ||||||
| 		trimLine := strings.Trim(line, "+- ") | 		trimLine := strings.Trim(line, "+- ") | ||||||
| 
 | 
 | ||||||
| 		if trimLine == LFSMetaFileIdentifier { | 		if trimLine == models.LFSMetaFileIdentifier { | ||||||
| 			curFileLFSPrefix = true | 			curFileLFSPrefix = true | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		if curFileLFSPrefix && strings.HasPrefix(trimLine, LFSMetaFileOidPrefix) { | 		if curFileLFSPrefix && strings.HasPrefix(trimLine, models.LFSMetaFileOidPrefix) { | ||||||
| 			oid := strings.TrimPrefix(trimLine, LFSMetaFileOidPrefix) | 			oid := strings.TrimPrefix(trimLine, models.LFSMetaFileOidPrefix) | ||||||
| 
 | 
 | ||||||
| 			if len(oid) == 64 { | 			if len(oid) == 64 { | ||||||
| 				m := &LFSMetaObject{Oid: oid} | 				m := &models.LFSMetaObject{Oid: oid} | ||||||
| 				count, err := x.Count(m) | 				count, err := models.Count(m) | ||||||
| 
 | 
 | ||||||
| 				if err == nil && count > 0 { | 				if err == nil && count > 0 { | ||||||
| 					curFile.IsBin = true | 					curFile.IsBin = true | ||||||
|  | @ -798,3 +800,29 @@ func GetRawDiffForFile(repoPath, startCommit, endCommit string, diffType RawDiff | ||||||
| func GetDiffCommit(repoPath, commitID string, maxLines, maxLineCharacters, maxFiles int) (*Diff, error) { | func GetDiffCommit(repoPath, commitID string, maxLines, maxLineCharacters, maxFiles int) (*Diff, error) { | ||||||
| 	return GetDiffRange(repoPath, "", commitID, maxLines, maxLineCharacters, maxFiles) | 	return GetDiffRange(repoPath, "", commitID, maxLines, maxLineCharacters, maxFiles) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | // CommentAsDiff returns c.Patch as *Diff
 | ||||||
|  | func CommentAsDiff(c *models.Comment) (*Diff, error) { | ||||||
|  | 	diff, err := ParsePatch(setting.Git.MaxGitDiffLines, | ||||||
|  | 		setting.Git.MaxGitDiffLineCharacters, setting.Git.MaxGitDiffFiles, strings.NewReader(c.Patch)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	if len(diff.Files) == 0 { | ||||||
|  | 		return nil, fmt.Errorf("no file found for comment ID: %d", c.ID) | ||||||
|  | 	} | ||||||
|  | 	secs := diff.Files[0].Sections | ||||||
|  | 	if len(secs) == 0 { | ||||||
|  | 		return nil, fmt.Errorf("no sections found for comment ID: %d", c.ID) | ||||||
|  | 	} | ||||||
|  | 	return diff, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // CommentMustAsDiff executes AsDiff and logs the error instead of returning
 | ||||||
|  | func CommentMustAsDiff(c *models.Comment) *Diff { | ||||||
|  | 	diff, err := CommentAsDiff(c) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Warn("CommentMustAsDiff: %v", err) | ||||||
|  | 	} | ||||||
|  | 	return diff | ||||||
|  | } | ||||||
|  | @ -1,10 +1,16 @@ | ||||||
| package models | // Copyright 2014 The Gogs Authors. All rights reserved.
 | ||||||
|  | // Copyright 2019 The Gitea 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 gitdiff | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"html/template" | 	"html/template" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"testing" | 	"testing" | ||||||
| 
 | 
 | ||||||
|  | 	"code.gitea.io/gitea/models" | ||||||
| 	"code.gitea.io/gitea/modules/setting" | 	"code.gitea.io/gitea/modules/setting" | ||||||
| 
 | 
 | ||||||
| 	dmp "github.com/sergi/go-diff/diffmatchpatch" | 	dmp "github.com/sergi/go-diff/diffmatchpatch" | ||||||
|  | @ -168,23 +174,24 @@ func setupDefaultDiff() *Diff { | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| func TestDiff_LoadComments(t *testing.T) { | func TestDiff_LoadComments(t *testing.T) { | ||||||
| 	issue := AssertExistsAndLoadBean(t, &Issue{ID: 2}).(*Issue) | 	assert.NoError(t, models.PrepareTestDatabase()) | ||||||
| 	user := AssertExistsAndLoadBean(t, &User{ID: 1}).(*User) | 
 | ||||||
|  | 	issue := models.AssertExistsAndLoadBean(t, &models.Issue{ID: 2}).(*models.Issue) | ||||||
|  | 	user := models.AssertExistsAndLoadBean(t, &models.User{ID: 1}).(*models.User) | ||||||
| 	diff := setupDefaultDiff() | 	diff := setupDefaultDiff() | ||||||
| 	assert.NoError(t, PrepareTestDatabase()) |  | ||||||
| 	assert.NoError(t, diff.LoadComments(issue, user)) | 	assert.NoError(t, diff.LoadComments(issue, user)) | ||||||
| 	assert.Len(t, diff.Files[0].Sections[0].Lines[0].Comments, 2) | 	assert.Len(t, diff.Files[0].Sections[0].Lines[0].Comments, 2) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestDiffLine_CanComment(t *testing.T) { | func TestDiffLine_CanComment(t *testing.T) { | ||||||
| 	assert.False(t, (&DiffLine{Type: DiffLineSection}).CanComment()) | 	assert.False(t, (&DiffLine{Type: DiffLineSection}).CanComment()) | ||||||
| 	assert.False(t, (&DiffLine{Type: DiffLineAdd, Comments: []*Comment{{Content: "bla"}}}).CanComment()) | 	assert.False(t, (&DiffLine{Type: DiffLineAdd, Comments: []*models.Comment{{Content: "bla"}}}).CanComment()) | ||||||
| 	assert.True(t, (&DiffLine{Type: DiffLineAdd}).CanComment()) | 	assert.True(t, (&DiffLine{Type: DiffLineAdd}).CanComment()) | ||||||
| 	assert.True(t, (&DiffLine{Type: DiffLineDel}).CanComment()) | 	assert.True(t, (&DiffLine{Type: DiffLineDel}).CanComment()) | ||||||
| 	assert.True(t, (&DiffLine{Type: DiffLinePlain}).CanComment()) | 	assert.True(t, (&DiffLine{Type: DiffLinePlain}).CanComment()) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestDiffLine_GetCommentSide(t *testing.T) { | func TestDiffLine_GetCommentSide(t *testing.T) { | ||||||
| 	assert.Equal(t, "previous", (&DiffLine{Comments: []*Comment{{Line: -3}}}).GetCommentSide()) | 	assert.Equal(t, "previous", (&DiffLine{Comments: []*models.Comment{{Line: -3}}}).GetCommentSide()) | ||||||
| 	assert.Equal(t, "proposed", (&DiffLine{Comments: []*Comment{{Line: 3}}}).GetCommentSide()) | 	assert.Equal(t, "proposed", (&DiffLine{Comments: []*models.Comment{{Line: 3}}}).GetCommentSide()) | ||||||
| } | } | ||||||
|  | @ -0,0 +1,16 @@ | ||||||
|  | // Copyright 2019 The Gitea 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 gitdiff | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"testing" | ||||||
|  | 
 | ||||||
|  | 	"code.gitea.io/gitea/models" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func TestMain(m *testing.M) { | ||||||
|  | 	models.MainTest(m, filepath.Join("..", "..")) | ||||||
|  | } | ||||||
|  | @ -319,7 +319,7 @@ | ||||||
| 							{{end}} | 							{{end}} | ||||||
| 								<a href="{{(index $comms 0).CodeCommentURL}}" class="file-comment">{{$filename}}</a> | 								<a href="{{(index $comms 0).CodeCommentURL}}" class="file-comment">{{$filename}}</a> | ||||||
| 							</div> | 							</div> | ||||||
| 							{{$diff := ((index $comms 0).MustAsDiff)}} | 							{{$diff := (CommentMustAsDiff (index $comms 0))}} | ||||||
| 							{{if $diff}} | 							{{if $diff}} | ||||||
| 								{{$file := (index $diff.Files 0)}} | 								{{$file := (index $diff.Files 0)}} | ||||||
| 								<div id="code-preview-{{(index $comms 0).ID}}" class="ui table segment{{if $invalid}} hide{{end}}"> | 								<div id="code-preview-{{(index $comms 0).ID}}" class="ui table segment{{if $invalid}} hide{{end}}"> | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue