mirror of
https://codeberg.org/forgejo/forgejo.git
synced 2025-01-02 14:28:52 -05:00
Pull request conflict files detection (#5951)
* add conflict detection * test pull request conflict files * fix detection files number * fix comments
This commit is contained in:
parent
680a57ce92
commit
bf4badad1d
4 changed files with 45 additions and 5 deletions
|
@ -54,9 +54,10 @@ const (
|
||||||
|
|
||||||
// PullRequest represents relation between pull request and repositories.
|
// PullRequest represents relation between pull request and repositories.
|
||||||
type PullRequest struct {
|
type PullRequest struct {
|
||||||
ID int64 `xorm:"pk autoincr"`
|
ID int64 `xorm:"pk autoincr"`
|
||||||
Type PullRequestType
|
Type PullRequestType
|
||||||
Status PullRequestStatus
|
Status PullRequestStatus
|
||||||
|
ConflictedFiles []string `xorm:"TEXT JSON"`
|
||||||
|
|
||||||
IssueID int64 `xorm:"INDEX"`
|
IssueID int64 `xorm:"INDEX"`
|
||||||
Issue *Issue `xorm:"-"`
|
Issue *Issue `xorm:"-"`
|
||||||
|
@ -843,6 +844,7 @@ func (pr *PullRequest) testPatch(e Engine) (err error) {
|
||||||
args = append(args, "--ignore-whitespace")
|
args = append(args, "--ignore-whitespace")
|
||||||
}
|
}
|
||||||
args = append(args, patchPath)
|
args = append(args, patchPath)
|
||||||
|
pr.ConflictedFiles = []string{}
|
||||||
|
|
||||||
_, stderr, err = process.GetManager().ExecDirEnv(-1, "", fmt.Sprintf("testPatch (git apply --check): %d", pr.BaseRepo.ID),
|
_, stderr, err = process.GetManager().ExecDirEnv(-1, "", fmt.Sprintf("testPatch (git apply --check): %d", pr.BaseRepo.ID),
|
||||||
[]string{"GIT_INDEX_FILE=" + indexTmpPath, "GIT_DIR=" + pr.BaseRepo.RepoPath()},
|
[]string{"GIT_INDEX_FILE=" + indexTmpPath, "GIT_DIR=" + pr.BaseRepo.RepoPath()},
|
||||||
|
@ -851,8 +853,26 @@ func (pr *PullRequest) testPatch(e Engine) (err error) {
|
||||||
for i := range patchConflicts {
|
for i := range patchConflicts {
|
||||||
if strings.Contains(stderr, patchConflicts[i]) {
|
if strings.Contains(stderr, patchConflicts[i]) {
|
||||||
log.Trace("PullRequest[%d].testPatch (apply): has conflict", pr.ID)
|
log.Trace("PullRequest[%d].testPatch (apply): has conflict", pr.ID)
|
||||||
fmt.Println(stderr)
|
const prefix = "error: patch failed:"
|
||||||
pr.Status = PullRequestStatusConflict
|
pr.Status = PullRequestStatusConflict
|
||||||
|
pr.ConflictedFiles = make([]string, 0, 5)
|
||||||
|
scanner := bufio.NewScanner(strings.NewReader(stderr))
|
||||||
|
for scanner.Scan() {
|
||||||
|
line := scanner.Text()
|
||||||
|
|
||||||
|
if strings.HasPrefix(line, prefix) {
|
||||||
|
pr.ConflictedFiles = append(pr.ConflictedFiles, strings.TrimSpace(strings.Split(line[len(prefix):], ":")[0]))
|
||||||
|
}
|
||||||
|
// only list 10 conflicted files
|
||||||
|
if len(pr.ConflictedFiles) >= 10 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(pr.ConflictedFiles) > 0 {
|
||||||
|
log.Trace("Found %d files conflicted: %v", len(pr.ConflictedFiles), pr.ConflictedFiles)
|
||||||
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1375,7 +1395,7 @@ func (pr *PullRequest) checkAndUpdateStatus() {
|
||||||
|
|
||||||
// Make sure there is no waiting test to process before leaving the checking status.
|
// Make sure there is no waiting test to process before leaving the checking status.
|
||||||
if !pullRequestQueue.Exist(pr.ID) {
|
if !pullRequestQueue.Exist(pr.ID) {
|
||||||
if err := pr.UpdateCols("status"); err != nil {
|
if err := pr.UpdateCols("status, conflicted_files"); err != nil {
|
||||||
log.Error(4, "Update[%d]: %v", pr.ID, err)
|
log.Error(4, "Update[%d]: %v", pr.ID, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1396,6 +1416,11 @@ func (pr *PullRequest) IsWorkInProgress() bool {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// IsFilesConflicted determines if the Pull Request has changes conflicting with the target branch.
|
||||||
|
func (pr *PullRequest) IsFilesConflicted() bool {
|
||||||
|
return len(pr.ConflictedFiles) > 0
|
||||||
|
}
|
||||||
|
|
||||||
// GetWorkInProgressPrefix returns the prefix used to mark the pull request as a work in progress.
|
// GetWorkInProgressPrefix returns the prefix used to mark the pull request as a work in progress.
|
||||||
// It returns an empty string when none were found
|
// It returns an empty string when none were found
|
||||||
func (pr *PullRequest) GetWorkInProgressPrefix() string {
|
func (pr *PullRequest) GetWorkInProgressPrefix() string {
|
||||||
|
|
|
@ -872,6 +872,7 @@ pulls.has_merged = The pull request has been merged.
|
||||||
pulls.title_wip_desc = `<a href="#">Start the title with <strong>%s</strong></a> to prevent the pull request from being merged accidentally.`
|
pulls.title_wip_desc = `<a href="#">Start the title with <strong>%s</strong></a> to prevent the pull request from being merged accidentally.`
|
||||||
pulls.cannot_merge_work_in_progress = This pull request is marked as a work in progress. Remove the <strong>%s</strong> prefix from the title when it's ready
|
pulls.cannot_merge_work_in_progress = This pull request is marked as a work in progress. Remove the <strong>%s</strong> prefix from the title when it's ready
|
||||||
pulls.data_broken = This pull request is broken due to missing fork information.
|
pulls.data_broken = This pull request is broken due to missing fork information.
|
||||||
|
pulls.files_conflicted = This pull request has changes conflicting with the target branch.
|
||||||
pulls.is_checking = "Merge conflict checking is in progress. Try again in few moments."
|
pulls.is_checking = "Merge conflict checking is in progress. Try again in few moments."
|
||||||
pulls.blocked_by_approvals = "This Pull Request doesn't have enough approvals yet. %d of %d approvals granted."
|
pulls.blocked_by_approvals = "This Pull Request doesn't have enough approvals yet. %d of %d approvals granted."
|
||||||
pulls.can_auto_merge_desc = This pull request can be merged automatically.
|
pulls.can_auto_merge_desc = This pull request can be merged automatically.
|
||||||
|
|
|
@ -345,6 +345,11 @@ func PrepareViewPullInfo(ctx *context.Context, issue *models.Issue) *git.PullReq
|
||||||
ctx.Data["WorkInProgressPrefix"] = pull.GetWorkInProgressPrefix()
|
ctx.Data["WorkInProgressPrefix"] = pull.GetWorkInProgressPrefix()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if pull.IsFilesConflicted() {
|
||||||
|
ctx.Data["IsPullFilesConflicted"] = true
|
||||||
|
ctx.Data["ConflictedFiles"] = pull.ConflictedFiles
|
||||||
|
}
|
||||||
|
|
||||||
ctx.Data["NumCommits"] = prInfo.Commits.Len()
|
ctx.Data["NumCommits"] = prInfo.Commits.Len()
|
||||||
ctx.Data["NumFiles"] = prInfo.NumFiles
|
ctx.Data["NumFiles"] = prInfo.NumFiles
|
||||||
return prInfo
|
return prInfo
|
||||||
|
|
|
@ -38,6 +38,7 @@
|
||||||
{{if .Issue.PullRequest.HasMerged}}purple
|
{{if .Issue.PullRequest.HasMerged}}purple
|
||||||
{{else if .Issue.IsClosed}}grey
|
{{else if .Issue.IsClosed}}grey
|
||||||
{{else if .IsPullWorkInProgress}}grey
|
{{else if .IsPullWorkInProgress}}grey
|
||||||
|
{{else if .IsFilesConflicted}}grey
|
||||||
{{else if .IsPullRequestBroken}}red
|
{{else if .IsPullRequestBroken}}red
|
||||||
{{else if .IsBlockedByApprovals}}red
|
{{else if .IsBlockedByApprovals}}red
|
||||||
{{else if .Issue.PullRequest.IsChecking}}yellow
|
{{else if .Issue.PullRequest.IsChecking}}yellow
|
||||||
|
@ -59,6 +60,14 @@
|
||||||
<div class="item text grey">
|
<div class="item text grey">
|
||||||
{{$.i18n.Tr "repo.pulls.reopen_to_merge"}}
|
{{$.i18n.Tr "repo.pulls.reopen_to_merge"}}
|
||||||
</div>
|
</div>
|
||||||
|
{{else if .IsPullFilesConflicted}}
|
||||||
|
<div class="item text grey">
|
||||||
|
<span class="octicon octicon-x"></span>
|
||||||
|
{{$.i18n.Tr "repo.pulls.files_conflicted"}}
|
||||||
|
{{range .ConflictedFiles}}
|
||||||
|
<div>{{.}}</div>
|
||||||
|
{{end}}
|
||||||
|
</div>
|
||||||
{{else if .IsPullRequestBroken}}
|
{{else if .IsPullRequestBroken}}
|
||||||
<div class="item text red">
|
<div class="item text red">
|
||||||
<span class="octicon octicon-x"></span>
|
<span class="octicon octicon-x"></span>
|
||||||
|
|
Loading…
Reference in a new issue