From f5be6276b0cf7c44bbcb3b882fa0691030fbc201 Mon Sep 17 00:00:00 2001 From: Lunny Xiao Date: Tue, 19 Aug 2025 11:32:00 -0700 Subject: [PATCH 1/8] Fix bug partially failure when commenting with status change in an issue --- options/locale/locale_en-US.ini | 3 + routers/api/v1/repo/issue.go | 6 +- routers/web/repo/issue_comment.go | 73 +++++++++++------------ routers/web/repo/issue_list.go | 4 +- services/issue/comments.go | 29 ++++++--- services/issue/commit.go | 4 +- services/issue/status.go | 68 +++++++++++++++++---- services/pull/merge.go | 4 +- services/pull/pull.go | 6 +- tests/integration/actions_trigger_test.go | 4 +- 10 files changed, 128 insertions(+), 73 deletions(-) diff --git a/options/locale/locale_en-US.ini b/options/locale/locale_en-US.ini index d7e73a0cfbb08..3869612c5c133 100644 --- a/options/locale/locale_en-US.ini +++ b/options/locale/locale_en-US.ini @@ -1629,6 +1629,9 @@ issues.reopen_issue = Reopen issues.reopen_comment_issue = Reopen with Comment issues.create_comment = Comment issues.comment.blocked_user = Cannot create or edit comment because you are blocked by the poster or repository owner. +issues.not_closed = The issue is not closed. +issues.comment.empty_content = The comment content cannot be empty. +issues.already_closed = The issue is already closed. issues.closed_at = `closed this issue %[2]s` issues.reopened_at = `reopened this issue %[2]s` issues.commit_ref_at = `referenced this issue from a commit %[2]s` diff --git a/routers/api/v1/repo/issue.go b/routers/api/v1/repo/issue.go index d4a5872fd1c27..3fffa5abb01e8 100644 --- a/routers/api/v1/repo/issue.go +++ b/routers/api/v1/repo/issue.go @@ -733,7 +733,7 @@ func CreateIssue(ctx *context.APIContext) { } if form.Closed { - if err := issue_service.CloseIssue(ctx, issue, ctx.Doer, ""); err != nil { + if _, err := issue_service.CloseIssue(ctx, issue, ctx.Doer, "", "", nil); err != nil { if issues_model.IsErrDependenciesLeft(err) { ctx.APIError(http.StatusPreconditionFailed, "cannot close this issue because it still has open dependencies") return @@ -1056,7 +1056,7 @@ func closeOrReopenIssue(ctx *context.APIContext, issue *issues_model.Issue, stat } if state == api.StateClosed && !issue.IsClosed { - if err := issue_service.CloseIssue(ctx, issue, ctx.Doer, ""); err != nil { + if _, err := issue_service.CloseIssue(ctx, issue, ctx.Doer, "", "", nil); err != nil { if issues_model.IsErrDependenciesLeft(err) { ctx.APIError(http.StatusPreconditionFailed, "cannot close this issue or pull request because it still has open dependencies") return @@ -1065,7 +1065,7 @@ func closeOrReopenIssue(ctx *context.APIContext, issue *issues_model.Issue, stat return } } else if state == api.StateOpen && issue.IsClosed { - if err := issue_service.ReopenIssue(ctx, issue, ctx.Doer, ""); err != nil { + if _, err := issue_service.ReopenIssue(ctx, issue, ctx.Doer, "", "", nil); err != nil { ctx.APIErrorInternal(err) return } diff --git a/routers/web/repo/issue_comment.go b/routers/web/repo/issue_comment.go index cb5b2d801952d..9b1458cd4003d 100644 --- a/routers/web/repo/issue_comment.go +++ b/routers/web/repo/issue_comment.go @@ -74,7 +74,6 @@ func NewComment(ctx *context.Context) { return } - var comment *issues_model.Comment defer func() { // Check if issue admin/poster changes the status of issue. if (ctx.Repo.CanWriteIssuesOrPulls(issue.IsPull) || (ctx.IsSigned && issue.IsPoster(ctx.Doer.ID))) && @@ -155,51 +154,37 @@ func NewComment(ctx *context.Context) { if pr != nil { ctx.Flash.Info(ctx.Tr("repo.pulls.open_unmerged_pull_exists", pr.Index)) - } else { - if form.Status == "close" && !issue.IsClosed { - if err := issue_service.CloseIssue(ctx, issue, ctx.Doer, ""); err != nil { - log.Error("CloseIssue: %v", err) - if issues_model.IsErrDependenciesLeft(err) { - if issue.IsPull { - ctx.JSONError(ctx.Tr("repo.issues.dependency.pr_close_blocked")) - } else { - ctx.JSONError(ctx.Tr("repo.issues.dependency.issue_close_blocked")) - } - return - } - } else { - if err := stopTimerIfAvailable(ctx, ctx.Doer, issue); err != nil { - ctx.ServerError("stopTimerIfAvailable", err) - return - } - log.Trace("Issue [%d] status changed to closed: %v", issue.ID, issue.IsClosed) - } - } else if form.Status == "reopen" && issue.IsClosed { - if err := issue_service.ReopenIssue(ctx, issue, ctx.Doer, ""); err != nil { - log.Error("ReopenIssue: %v", err) - } - } } } + }() - // Redirect to comment hashtag if there is any actual content. - typeName := "issues" - if issue.IsPull { - typeName = "pulls" + var createdComment *issues_model.Comment + var err error + + switch form.Status { + case "reopen": + if !issue.IsClosed { + ctx.JSONError(ctx.Tr("repo.issues.not_closed")) + return } - if comment != nil { - ctx.JSONRedirect(fmt.Sprintf("%s/%s/%d#%s", ctx.Repo.RepoLink, typeName, issue.Index, comment.HashTag())) - } else { - ctx.JSONRedirect(fmt.Sprintf("%s/%s/%d", ctx.Repo.RepoLink, typeName, issue.Index)) + + createdComment, err = issue_service.ReopenIssue(ctx, issue, ctx.Doer, "", form.Content, attachments) + case "close": + if issue.IsClosed { + ctx.JSONError(ctx.Tr("repo.issues.already_closed")) + return } - }() - // Fix #321: Allow empty comments, as long as we have attachments. - if len(form.Content) == 0 && len(attachments) == 0 { - return + createdComment, err = issue_service.CloseIssue(ctx, issue, ctx.Doer, "", form.Content, attachments) + default: + if len(form.Content) == 0 && len(attachments) == 0 { + ctx.JSONError(ctx.Tr("repo.issues.comment.empty_content")) + return + } + + createdComment, err = issue_service.CreateIssueComment(ctx, ctx.Doer, ctx.Repo.Repository, issue, form.Content, attachments) } - comment, err := issue_service.CreateIssueComment(ctx, ctx.Doer, ctx.Repo.Repository, issue, form.Content, attachments) if err != nil { if errors.Is(err, user_model.ErrBlockedUser) { ctx.JSONError(ctx.Tr("repo.issues.comment.blocked_user")) @@ -209,7 +194,17 @@ func NewComment(ctx *context.Context) { return } - log.Trace("Comment created: %d/%d/%d", ctx.Repo.Repository.ID, issue.ID, comment.ID) + // Redirect to comment hashtag if there is any actual content. + typeName := "issues" + if issue.IsPull { + typeName = "pulls" + } + if createdComment != nil { + log.Trace("Comment created: %d/%d/%d", ctx.Repo.Repository.ID, issue.ID, createdComment.ID) + ctx.JSONRedirect(fmt.Sprintf("%s/%s/%d#%s", ctx.Repo.RepoLink, typeName, issue.Index, createdComment.HashTag())) + } else { + ctx.JSONRedirect(fmt.Sprintf("%s/%s/%d", ctx.Repo.RepoLink, typeName, issue.Index)) + } } // UpdateCommentContent change comment of issue's content diff --git a/routers/web/repo/issue_list.go b/routers/web/repo/issue_list.go index fd34422cfcc65..9ec0bbb751079 100644 --- a/routers/web/repo/issue_list.go +++ b/routers/web/repo/issue_list.go @@ -434,7 +434,7 @@ func UpdateIssueStatus(ctx *context.Context) { continue } if action == "close" && !issue.IsClosed { - if err := issue_service.CloseIssue(ctx, issue, ctx.Doer, ""); err != nil { + if _, err := issue_service.CloseIssue(ctx, issue, ctx.Doer, "", "", nil); err != nil { if issues_model.IsErrDependenciesLeft(err) { ctx.JSON(http.StatusPreconditionFailed, map[string]any{ "error": ctx.Tr("repo.issues.dependency.issue_batch_close_blocked", issue.Index), @@ -445,7 +445,7 @@ func UpdateIssueStatus(ctx *context.Context) { return } } else if action == "open" && issue.IsClosed { - if err := issue_service.ReopenIssue(ctx, issue, ctx.Doer, ""); err != nil { + if _, err := issue_service.ReopenIssue(ctx, issue, ctx.Doer, "", "", nil); err != nil { ctx.ServerError("ReopenIssue", err) return } diff --git a/services/issue/comments.go b/services/issue/comments.go index 9442701029b57..beffb6a06fa35 100644 --- a/services/issue/comments.go +++ b/services/issue/comments.go @@ -55,6 +55,22 @@ func CreateRefComment(ctx context.Context, doer *user_model.User, repo *repo_mod return err } +func notifyCommentCreated(ctx context.Context, doer *user_model.User, repo *repo_model.Repository, issue *issues_model.Issue, comment *issues_model.Comment) error { + mentions, err := issues_model.FindAndUpdateIssueMentions(ctx, issue, doer, comment.Content) + if err != nil { + return err + } + + // reload issue to ensure it has the latest data, especially the number of comments + issue, err = issues_model.GetIssueByID(ctx, issue.ID) + if err != nil { + return err + } + + notify_service.CreateIssueComment(ctx, doer, repo, issue, comment, mentions) + return nil +} + // CreateIssueComment creates a plain issue comment. func CreateIssueComment(ctx context.Context, doer *user_model.User, repo *repo_model.Repository, issue *issues_model.Issue, content string, attachments []string) (*issues_model.Comment, error) { if user_model.IsUserBlockedBy(ctx, doer, issue.PosterID, repo.OwnerID) { @@ -75,19 +91,16 @@ func CreateIssueComment(ctx context.Context, doer *user_model.User, repo *repo_m return nil, err } - mentions, err := issues_model.FindAndUpdateIssueMentions(ctx, issue, doer, comment.Content) - if err != nil { - return nil, err - } + return comment, notifyCommentCreated(ctx, doer, repo, issue, comment) +} - // reload issue to ensure it has the latest data, especially the number of comments - issue, err = issues_model.GetIssueByID(ctx, issue.ID) +// CreateCommentAndChangeStatus creates a comment and changes the issue status. +func CreateCommentAndChangeStatus(ctx context.Context, doer *user_model.User, repo *repo_model.Repository, issue *issues_model.Issue, content string, attachments []string) (*issues_model.Comment, error) { + comment, err := CreateIssueComment(ctx, doer, repo, issue, content, attachments) if err != nil { return nil, err } - notify_service.CreateIssueComment(ctx, doer, repo, issue, comment, mentions) - return comment, nil } diff --git a/services/issue/commit.go b/services/issue/commit.go index 963d0359fd35d..c7776268d8541 100644 --- a/services/issue/commit.go +++ b/services/issue/commit.go @@ -196,11 +196,11 @@ func UpdateIssuesCommit(ctx context.Context, doer *user_model.User, repo *repo_m return err } } - if err := CloseIssue(ctx, refIssue, doer, c.Sha1); err != nil { + if _, err := CloseIssue(ctx, refIssue, doer, c.Sha1, "", nil); err != nil { return err } } else if ref.Action == references.XRefActionReopens && refIssue.IsClosed { - if err := ReopenIssue(ctx, refIssue, doer, c.Sha1); err != nil { + if _, err := ReopenIssue(ctx, refIssue, doer, c.Sha1, "", nil); err != nil { return err } } diff --git a/services/issue/status.go b/services/issue/status.go index fa59df93ba107..33c8c0d26112c 100644 --- a/services/issue/status.go +++ b/services/issue/status.go @@ -14,11 +14,25 @@ import ( ) // CloseIssue close an issue. -func CloseIssue(ctx context.Context, issue *issues_model.Issue, doer *user_model.User, commitID string) error { - var comment *issues_model.Comment +func CloseIssue(ctx context.Context, issue *issues_model.Issue, doer *user_model.User, commitID, commentContent string, attachments []string) (*issues_model.Comment, error) { + var refComment, createdComment *issues_model.Comment if err := db.WithTx(ctx, func(ctx context.Context) error { var err error - comment, err = issues_model.CloseIssue(ctx, issue, doer) + if commentContent != "" || len(attachments) > 0 { + createdComment, err = issues_model.CreateComment(ctx, &issues_model.CreateCommentOptions{ + Type: issues_model.CommentTypeComment, + Doer: doer, + Repo: issue.Repo, + Issue: issue, + Content: commentContent, + Attachments: attachments, + }) + if err != nil { + return err + } + } + + refComment, err = issues_model.CloseIssue(ctx, issue, doer) if err != nil { if issues_model.IsErrDependenciesLeft(err) { if _, err := issues_model.FinishIssueStopwatch(ctx, doer, issue); err != nil { @@ -31,23 +45,53 @@ func CloseIssue(ctx context.Context, issue *issues_model.Issue, doer *user_model _, err = issues_model.FinishIssueStopwatch(ctx, doer, issue) return err }); err != nil { - return err + return nil, err } - notify_service.IssueChangeStatus(ctx, doer, commitID, issue, comment, true) + if createdComment != nil { + if err := notifyCommentCreated(ctx, doer, issue.Repo, issue, createdComment); err != nil { + log.Error("Unable to notify comment created for issue[%d]#%d: %v", issue.ID, issue.Index, err) + } + } - return nil + notify_service.IssueChangeStatus(ctx, doer, commitID, issue, refComment, true) + + return createdComment, nil } -// ReopenIssue reopen an issue. +// ReopenIssue reopen an issue with or without a comment. // FIXME: If some issues dependent this one are closed, should we also reopen them? -func ReopenIssue(ctx context.Context, issue *issues_model.Issue, doer *user_model.User, commitID string) error { - comment, err := issues_model.ReopenIssue(ctx, issue, doer) +func ReopenIssue(ctx context.Context, issue *issues_model.Issue, doer *user_model.User, commitID, commentContent string, attachments []string) (*issues_model.Comment, error) { + var createdComment *issues_model.Comment + refComment, err := db.WithTx2(ctx, func(ctx context.Context) (*issues_model.Comment, error) { + var err error + if commentContent != "" || len(attachments) > 0 { + createdComment, err = issues_model.CreateComment(ctx, &issues_model.CreateCommentOptions{ + Type: issues_model.CommentTypeComment, + Doer: doer, + Repo: issue.Repo, + Issue: issue, + Content: commentContent, + Attachments: attachments, + }) + if err != nil { + return nil, err + } + } + + return issues_model.ReopenIssue(ctx, issue, doer) + }) if err != nil { - return err + return nil, err + } + + if createdComment != nil { + if err := notifyCommentCreated(ctx, doer, issue.Repo, issue, createdComment); err != nil { + log.Error("Unable to notify comment created for issue[%d]#%d: %v", issue.ID, issue.Index, err) + } } - notify_service.IssueChangeStatus(ctx, doer, commitID, issue, comment, false) + notify_service.IssueChangeStatus(ctx, doer, commitID, issue, refComment, false) - return nil + return createdComment, nil } diff --git a/services/pull/merge.go b/services/pull/merge.go index a941c204357ac..e7fb9525afde4 100644 --- a/services/pull/merge.go +++ b/services/pull/merge.go @@ -309,14 +309,14 @@ func handleCloseCrossReferences(ctx context.Context, pr *issues_model.PullReques return err } if ref.RefAction == references.XRefActionCloses && !ref.Issue.IsClosed { - if err = issue_service.CloseIssue(ctx, ref.Issue, doer, pr.MergedCommitID); err != nil { + if _, err = issue_service.CloseIssue(ctx, ref.Issue, doer, pr.MergedCommitID, "", nil); err != nil { // Allow ErrDependenciesLeft if !issues_model.IsErrDependenciesLeft(err) { return err } } } else if ref.RefAction == references.XRefActionReopens && ref.Issue.IsClosed { - if err = issue_service.ReopenIssue(ctx, ref.Issue, doer, pr.MergedCommitID); err != nil { + if _, err = issue_service.ReopenIssue(ctx, ref.Issue, doer, pr.MergedCommitID, "", nil); err != nil { return err } } diff --git a/services/pull/pull.go b/services/pull/pull.go index e55d4f5bb194f..9fc55bdbad4c2 100644 --- a/services/pull/pull.go +++ b/services/pull/pull.go @@ -700,7 +700,7 @@ func AdjustPullsCausedByBranchDeleted(ctx context.Context, doer *user_model.User var errs []error for _, pr := range prs { - if err = issue_service.CloseIssue(ctx, pr.Issue, doer, ""); err != nil && !issues_model.IsErrIssueIsClosed(err) && !issues_model.IsErrDependenciesLeft(err) { + if _, err = issue_service.CloseIssue(ctx, pr.Issue, doer, "", "", nil); err != nil && !issues_model.IsErrIssueIsClosed(err) && !issues_model.IsErrDependenciesLeft(err) { errs = append(errs, err) } if err == nil { @@ -740,7 +740,7 @@ func AdjustPullsCausedByBranchDeleted(ctx context.Context, doer *user_model.User errs = append(errs, err) } if err == nil { - if err = issue_service.CloseIssue(ctx, pr.Issue, doer, ""); err != nil && !issues_model.IsErrIssueIsClosed(err) && !issues_model.IsErrDependenciesLeft(err) { + if _, err = issue_service.CloseIssue(ctx, pr.Issue, doer, "", "", nil); err != nil && !issues_model.IsErrIssueIsClosed(err) && !issues_model.IsErrDependenciesLeft(err) { errs = append(errs, err) } } @@ -772,7 +772,7 @@ func CloseRepoBranchesPulls(ctx context.Context, doer *user_model.User, repo *re if pr.BaseRepoID == repo.ID { continue } - if err = issue_service.CloseIssue(ctx, pr.Issue, doer, ""); err != nil && !issues_model.IsErrIssueIsClosed(err) { + if _, err = issue_service.CloseIssue(ctx, pr.Issue, doer, "", "", nil); err != nil && !issues_model.IsErrIssueIsClosed(err) { errs = append(errs, err) } } diff --git a/tests/integration/actions_trigger_test.go b/tests/integration/actions_trigger_test.go index 088491d5705f4..8b6c146ac5074 100644 --- a/tests/integration/actions_trigger_test.go +++ b/tests/integration/actions_trigger_test.go @@ -565,12 +565,12 @@ jobs: checkCommitStatusAndInsertFakeStatus(t, repo, sha) // closed - err = issue_service.CloseIssue(db.DefaultContext, pullIssue, user2, "") + _, err = issue_service.CloseIssue(db.DefaultContext, pullIssue, user2, "", "", nil) assert.NoError(t, err) checkCommitStatusAndInsertFakeStatus(t, repo, sha) // reopened - err = issue_service.ReopenIssue(db.DefaultContext, pullIssue, user2, "") + _, err = issue_service.ReopenIssue(db.DefaultContext, pullIssue, user2, "", "", nil) assert.NoError(t, err) checkCommitStatusAndInsertFakeStatus(t, repo, sha) From 385fda5477d92e9f9c27d50abe99604d39e6f0ed Mon Sep 17 00:00:00 2001 From: Lunny Xiao Date: Tue, 19 Aug 2025 11:39:05 -0700 Subject: [PATCH 2/8] improvements --- services/issue/comments.go | 12 ++++-------- 1 file changed, 4 insertions(+), 8 deletions(-) diff --git a/services/issue/comments.go b/services/issue/comments.go index beffb6a06fa35..bbeb75f27244e 100644 --- a/services/issue/comments.go +++ b/services/issue/comments.go @@ -15,6 +15,7 @@ import ( user_model "code.gitea.io/gitea/models/user" "code.gitea.io/gitea/modules/gitrepo" "code.gitea.io/gitea/modules/json" + "code.gitea.io/gitea/modules/log" "code.gitea.io/gitea/modules/timeutil" git_service "code.gitea.io/gitea/services/git" notify_service "code.gitea.io/gitea/services/notify" @@ -91,14 +92,9 @@ func CreateIssueComment(ctx context.Context, doer *user_model.User, repo *repo_m return nil, err } - return comment, notifyCommentCreated(ctx, doer, repo, issue, comment) -} - -// CreateCommentAndChangeStatus creates a comment and changes the issue status. -func CreateCommentAndChangeStatus(ctx context.Context, doer *user_model.User, repo *repo_model.Repository, issue *issues_model.Issue, content string, attachments []string) (*issues_model.Comment, error) { - comment, err := CreateIssueComment(ctx, doer, repo, issue, content, attachments) - if err != nil { - return nil, err + if err := notifyCommentCreated(ctx, doer, repo, issue, comment); err != nil { + // If notification fails, we still return the comment but log the error. + log.Error("Failed to notify comment creation: %v", err) } return comment, nil From 0434311c7e6920107220d1f6e98957bbb9c283ca Mon Sep 17 00:00:00 2001 From: Lunny Xiao Date: Tue, 19 Aug 2025 14:52:10 -0700 Subject: [PATCH 3/8] Remove defer function in NewComment --- routers/web/repo/issue_comment.go | 182 ++++++++++++++++-------------- 1 file changed, 98 insertions(+), 84 deletions(-) diff --git a/routers/web/repo/issue_comment.go b/routers/web/repo/issue_comment.go index 9b1458cd4003d..e9c6891a350ce 100644 --- a/routers/web/repo/issue_comment.go +++ b/routers/web/repo/issue_comment.go @@ -74,90 +74,6 @@ func NewComment(ctx *context.Context) { return } - defer func() { - // Check if issue admin/poster changes the status of issue. - if (ctx.Repo.CanWriteIssuesOrPulls(issue.IsPull) || (ctx.IsSigned && issue.IsPoster(ctx.Doer.ID))) && - (form.Status == "reopen" || form.Status == "close") && - !(issue.IsPull && issue.PullRequest.HasMerged) { - // Duplication and conflict check should apply to reopen pull request. - var pr *issues_model.PullRequest - - if form.Status == "reopen" && issue.IsPull { - pull := issue.PullRequest - var err error - pr, err = issues_model.GetUnmergedPullRequest(ctx, pull.HeadRepoID, pull.BaseRepoID, pull.HeadBranch, pull.BaseBranch, pull.Flow) - if err != nil { - if !issues_model.IsErrPullRequestNotExist(err) { - ctx.JSONError(ctx.Tr("repo.issues.dependency.pr_close_blocked")) - return - } - } - - // Regenerate patch and test conflict. - if pr == nil { - issue.PullRequest.HeadCommitID = "" - pull_service.StartPullRequestCheckImmediately(ctx, issue.PullRequest) - } - - // check whether the ref of PR in base repo is consistent with the head commit of head branch in the head repo - // get head commit of PR - if pull.Flow == issues_model.PullRequestFlowGithub { - prHeadRef := pull.GetGitHeadRefName() - if err := pull.LoadBaseRepo(ctx); err != nil { - ctx.ServerError("Unable to load base repo", err) - return - } - prHeadCommitID, err := git.GetFullCommitID(ctx, pull.BaseRepo.RepoPath(), prHeadRef) - if err != nil { - ctx.ServerError("Get head commit Id of pr fail", err) - return - } - - // get head commit of branch in the head repo - if err := pull.LoadHeadRepo(ctx); err != nil { - ctx.ServerError("Unable to load head repo", err) - return - } - if ok := gitrepo.IsBranchExist(ctx, pull.HeadRepo, pull.BaseBranch); !ok { - // todo localize - ctx.JSONError("The origin branch is delete, cannot reopen.") - return - } - headBranchRef := pull.GetGitHeadBranchRefName() - headBranchCommitID, err := git.GetFullCommitID(ctx, pull.HeadRepo.RepoPath(), headBranchRef) - if err != nil { - ctx.ServerError("Get head commit Id of head branch fail", err) - return - } - - err = pull.LoadIssue(ctx) - if err != nil { - ctx.ServerError("load the issue of pull request error", err) - return - } - - if prHeadCommitID != headBranchCommitID { - // force push to base repo - err := git.Push(ctx, pull.HeadRepo.RepoPath(), git.PushOptions{ - Remote: pull.BaseRepo.RepoPath(), - Branch: pull.HeadBranch + ":" + prHeadRef, - Force: true, - Env: repo_module.InternalPushingEnvironment(pull.Issue.Poster, pull.BaseRepo), - }) - if err != nil { - ctx.ServerError("force push error", err) - return - } - } - } - } - - if pr != nil { - ctx.Flash.Info(ctx.Tr("repo.pulls.open_unmerged_pull_exists", pr.Index)) - } - } - }() - var createdComment *issues_model.Comment var err error @@ -168,13 +84,111 @@ func NewComment(ctx *context.Context) { return } + if !ctx.Repo.CanWriteIssuesOrPulls(issue.IsPull) && + !issue.IsPoster(ctx.Doer.ID) && + !ctx.Doer.IsAdmin { + ctx.JSONError(ctx.Tr("repo.issues.reopen_not_allowed")) + return + } + + if issue.IsPull && issue.PullRequest.HasMerged { + ctx.JSONError(ctx.Tr("repo.issues.reopen_not_allowed_merged")) + return + } + + // check if an opened pull request exists with the same head branch and base branch + pull := issue.PullRequest + var err error + pr, err := issues_model.GetUnmergedPullRequest(ctx, pull.HeadRepoID, pull.BaseRepoID, pull.HeadBranch, pull.BaseBranch, pull.Flow) + if err != nil { + if !issues_model.IsErrPullRequestNotExist(err) { + ctx.JSONError(ctx.Tr("repo.issues.dependency.pr_close_blocked")) + return + } + } + if pr != nil { + ctx.Flash.Info(ctx.Tr("repo.pulls.open_unmerged_pull_exists", pr.Index)) + return + } + createdComment, err = issue_service.ReopenIssue(ctx, issue, ctx.Doer, "", form.Content, attachments) + if err != nil { + if errors.Is(err, user_model.ErrBlockedUser) { + ctx.JSONError(ctx.Tr("repo.issues.comment.blocked_user")) + } else { + ctx.ServerError("ReopenIssue", err) + } + return + } + + // Regenerate patch and test conflict. + pull.HeadCommitID = "" + pull_service.StartPullRequestCheckImmediately(ctx, pull) + + // check whether the ref of PR in base repo is consistent with the head commit of head branch in the head repo + // get head commit of PR + if pull.Flow == issues_model.PullRequestFlowGithub { + prHeadRef := pull.GetGitHeadRefName() + if err := pull.LoadBaseRepo(ctx); err != nil { + ctx.ServerError("Unable to load base repo", err) + return + } + prHeadCommitID, err := git.GetFullCommitID(ctx, pull.BaseRepo.RepoPath(), prHeadRef) + if err != nil { + ctx.ServerError("Get head commit Id of pr fail", err) + return + } + + // get head commit of branch in the head repo + if err := pull.LoadHeadRepo(ctx); err != nil { + ctx.ServerError("Unable to load head repo", err) + return + } + if ok := gitrepo.IsBranchExist(ctx, pull.HeadRepo, pull.BaseBranch); !ok { + // todo localize + ctx.JSONError("The origin branch is delete, cannot reopen.") + return + } + headBranchRef := pull.GetGitHeadBranchRefName() + headBranchCommitID, err := git.GetFullCommitID(ctx, pull.HeadRepo.RepoPath(), headBranchRef) + if err != nil { + ctx.ServerError("Get head commit Id of head branch fail", err) + return + } + + err = pull.LoadIssue(ctx) + if err != nil { + ctx.ServerError("load the issue of pull request error", err) + return + } + + if prHeadCommitID != headBranchCommitID { + // force push to base repo + err := git.Push(ctx, pull.HeadRepo.RepoPath(), git.PushOptions{ + Remote: pull.BaseRepo.RepoPath(), + Branch: pull.HeadBranch + ":" + prHeadRef, + Force: true, + Env: repo_module.InternalPushingEnvironment(pull.Issue.Poster, pull.BaseRepo), + }) + if err != nil { + ctx.ServerError("force push error", err) + return + } + } + } case "close": if issue.IsClosed { ctx.JSONError(ctx.Tr("repo.issues.already_closed")) return } + if !ctx.Repo.CanWriteIssuesOrPulls(issue.IsPull) && + !issue.IsPoster(ctx.Doer.ID) && + !ctx.Doer.IsAdmin { + ctx.JSONError(ctx.Tr("repo.issues.close_not_allowed")) + return + } + createdComment, err = issue_service.CloseIssue(ctx, issue, ctx.Doer, "", form.Content, attachments) default: if len(form.Content) == 0 && len(attachments) == 0 { From 40b1ad32b1f7f519f459eadc5857b55765bb135f Mon Sep 17 00:00:00 2001 From: Lunny Xiao Date: Tue, 19 Aug 2025 15:19:03 -0700 Subject: [PATCH 4/8] improvements --- options/locale/locale_en-US.ini | 4 +++- routers/web/repo/issue_comment.go | 10 +++++----- 2 files changed, 8 insertions(+), 6 deletions(-) diff --git a/options/locale/locale_en-US.ini b/options/locale/locale_en-US.ini index 3869612c5c133..a36dcc9058e8d 100644 --- a/options/locale/locale_en-US.ini +++ b/options/locale/locale_en-US.ini @@ -1630,6 +1630,8 @@ issues.reopen_comment_issue = Reopen with Comment issues.create_comment = Comment issues.comment.blocked_user = Cannot create or edit comment because you are blocked by the poster or repository owner. issues.not_closed = The issue is not closed. +issues.reopen_not_allowed = No permission to reopen this issue. +issues.reopen_not_allowed_merged = A pull request cannot be reopened after it has been merged. issues.comment.empty_content = The comment content cannot be empty. issues.already_closed = The issue is already closed. issues.closed_at = `closed this issue %[2]s` @@ -1952,7 +1954,7 @@ pulls.has_merged = Failed: The pull request has been merged. You cannot merge ag pulls.push_rejected = Push Failed: The push was rejected. Review the Git Hooks for this repository. pulls.push_rejected_summary = Full Rejection Message pulls.push_rejected_no_message = Push Failed: The push was rejected but there was no remote message. Review the Git Hooks for this repository. -pulls.open_unmerged_pull_exists = `You cannot perform a reopen operation because there is a pending pull request (#%d) with identical properties.` +pulls.open_unmerged_pull_exists = You cannot perform a reopen operation because there is a pending pull request (#%d) with identical properties. pulls.status_checking = Some checks are pending pulls.status_checks_success = All checks were successful pulls.status_checks_warning = Some checks reported warnings diff --git a/routers/web/repo/issue_comment.go b/routers/web/repo/issue_comment.go index e9c6891a350ce..76dcaada1da82 100644 --- a/routers/web/repo/issue_comment.go +++ b/routers/web/repo/issue_comment.go @@ -102,7 +102,7 @@ func NewComment(ctx *context.Context) { pr, err := issues_model.GetUnmergedPullRequest(ctx, pull.HeadRepoID, pull.BaseRepoID, pull.HeadBranch, pull.BaseBranch, pull.Flow) if err != nil { if !issues_model.IsErrPullRequestNotExist(err) { - ctx.JSONError(ctx.Tr("repo.issues.dependency.pr_close_blocked")) + ctx.JSONError(err.Error()) return } } @@ -121,10 +121,6 @@ func NewComment(ctx *context.Context) { return } - // Regenerate patch and test conflict. - pull.HeadCommitID = "" - pull_service.StartPullRequestCheckImmediately(ctx, pull) - // check whether the ref of PR in base repo is consistent with the head commit of head branch in the head repo // get head commit of PR if pull.Flow == issues_model.PullRequestFlowGithub { @@ -176,6 +172,10 @@ func NewComment(ctx *context.Context) { } } } + + // Regenerate patch and test conflict. + pull.HeadCommitID = "" + pull_service.StartPullRequestCheckImmediately(ctx, pull) case "close": if issue.IsClosed { ctx.JSONError(ctx.Tr("repo.issues.already_closed")) From 1ce6f48e4cfd0d7e76ec46a196aa6d938f458904 Mon Sep 17 00:00:00 2001 From: Lunny Xiao Date: Wed, 10 Sep 2025 18:43:06 -0700 Subject: [PATCH 5/8] use a standalone function for close/reopen with comment --- options/locale/locale_en-US.ini | 1 + routers/api/v1/repo/issue.go | 6 +- routers/web/repo/issue_comment.go | 139 ++++++++++++---------- routers/web/repo/issue_list.go | 4 +- services/issue/commit.go | 4 +- services/issue/status.go | 47 +++++++- services/pull/merge.go | 4 +- services/pull/pull.go | 6 +- tests/integration/actions_trigger_test.go | 4 +- 9 files changed, 132 insertions(+), 83 deletions(-) diff --git a/options/locale/locale_en-US.ini b/options/locale/locale_en-US.ini index 52ab134a39869..6eb3da1761d4a 100644 --- a/options/locale/locale_en-US.ini +++ b/options/locale/locale_en-US.ini @@ -1956,6 +1956,7 @@ pulls.push_rejected = Push Failed: The push was rejected. Review the Git Hooks f pulls.push_rejected_summary = Full Rejection Message pulls.push_rejected_no_message = Push Failed: The push was rejected but there was no remote message. Review the Git Hooks for this repository. pulls.open_unmerged_pull_exists = You cannot perform a reopen operation because there is a pending pull request (#%d) with identical properties. +pulls.head_branch_not_exist = The head branch does not exist, cannot reopen the pull request. pulls.status_checking = Some checks are pending pulls.status_checks_success = All checks were successful pulls.status_checks_warning = Some checks reported warnings diff --git a/routers/api/v1/repo/issue.go b/routers/api/v1/repo/issue.go index 3fffa5abb01e8..d4a5872fd1c27 100644 --- a/routers/api/v1/repo/issue.go +++ b/routers/api/v1/repo/issue.go @@ -733,7 +733,7 @@ func CreateIssue(ctx *context.APIContext) { } if form.Closed { - if _, err := issue_service.CloseIssue(ctx, issue, ctx.Doer, "", "", nil); err != nil { + if err := issue_service.CloseIssue(ctx, issue, ctx.Doer, ""); err != nil { if issues_model.IsErrDependenciesLeft(err) { ctx.APIError(http.StatusPreconditionFailed, "cannot close this issue because it still has open dependencies") return @@ -1056,7 +1056,7 @@ func closeOrReopenIssue(ctx *context.APIContext, issue *issues_model.Issue, stat } if state == api.StateClosed && !issue.IsClosed { - if _, err := issue_service.CloseIssue(ctx, issue, ctx.Doer, "", "", nil); err != nil { + if err := issue_service.CloseIssue(ctx, issue, ctx.Doer, ""); err != nil { if issues_model.IsErrDependenciesLeft(err) { ctx.APIError(http.StatusPreconditionFailed, "cannot close this issue or pull request because it still has open dependencies") return @@ -1065,7 +1065,7 @@ func closeOrReopenIssue(ctx *context.APIContext, issue *issues_model.Issue, stat return } } else if state == api.StateOpen && issue.IsClosed { - if _, err := issue_service.ReopenIssue(ctx, issue, ctx.Doer, "", "", nil); err != nil { + if err := issue_service.ReopenIssue(ctx, issue, ctx.Doer, ""); err != nil { ctx.APIErrorInternal(err) return } diff --git a/routers/web/repo/issue_comment.go b/routers/web/repo/issue_comment.go index 76dcaada1da82..d0ea9fc061947 100644 --- a/routers/web/repo/issue_comment.go +++ b/routers/web/repo/issue_comment.go @@ -10,11 +10,11 @@ import ( "net/http" "strconv" + git_model "code.gitea.io/gitea/models/git" issues_model "code.gitea.io/gitea/models/issues" "code.gitea.io/gitea/models/renderhelper" user_model "code.gitea.io/gitea/models/user" "code.gitea.io/gitea/modules/git" - "code.gitea.io/gitea/modules/gitrepo" "code.gitea.io/gitea/modules/log" "code.gitea.io/gitea/modules/markup/markdown" repo_module "code.gitea.io/gitea/modules/repository" @@ -91,47 +91,10 @@ func NewComment(ctx *context.Context) { return } - if issue.IsPull && issue.PullRequest.HasMerged { - ctx.JSONError(ctx.Tr("repo.issues.reopen_not_allowed_merged")) - return - } - - // check if an opened pull request exists with the same head branch and base branch - pull := issue.PullRequest - var err error - pr, err := issues_model.GetUnmergedPullRequest(ctx, pull.HeadRepoID, pull.BaseRepoID, pull.HeadBranch, pull.BaseBranch, pull.Flow) - if err != nil { - if !issues_model.IsErrPullRequestNotExist(err) { - ctx.JSONError(err.Error()) - return - } - } - if pr != nil { - ctx.Flash.Info(ctx.Tr("repo.pulls.open_unmerged_pull_exists", pr.Index)) - return - } - - createdComment, err = issue_service.ReopenIssue(ctx, issue, ctx.Doer, "", form.Content, attachments) - if err != nil { - if errors.Is(err, user_model.ErrBlockedUser) { - ctx.JSONError(ctx.Tr("repo.issues.comment.blocked_user")) - } else { - ctx.ServerError("ReopenIssue", err) - } - return - } - - // check whether the ref of PR in base repo is consistent with the head commit of head branch in the head repo - // get head commit of PR - if pull.Flow == issues_model.PullRequestFlowGithub { - prHeadRef := pull.GetGitHeadRefName() - if err := pull.LoadBaseRepo(ctx); err != nil { - ctx.ServerError("Unable to load base repo", err) - return - } - prHeadCommitID, err := git.GetFullCommitID(ctx, pull.BaseRepo.RepoPath(), prHeadRef) - if err != nil { - ctx.ServerError("Get head commit Id of pr fail", err) + if issue.IsPull { + pull := issue.PullRequest + if pull.HasMerged { + ctx.JSONError(ctx.Tr("repo.issues.reopen_not_allowed_merged")) return } @@ -140,42 +103,88 @@ func NewComment(ctx *context.Context) { ctx.ServerError("Unable to load head repo", err) return } - if ok := gitrepo.IsBranchExist(ctx, pull.HeadRepo, pull.BaseBranch); !ok { - // todo localize - ctx.JSONError("The origin branch is delete, cannot reopen.") + branchExist, err := git_model.IsBranchExist(ctx, pull.HeadRepo.ID, pull.HeadBranch) + if err != nil { + ctx.ServerError("IsBranchExist", err) return } - headBranchRef := pull.GetGitHeadBranchRefName() - headBranchCommitID, err := git.GetFullCommitID(ctx, pull.HeadRepo.RepoPath(), headBranchRef) - if err != nil { - ctx.ServerError("Get head commit Id of head branch fail", err) + if !branchExist { + ctx.JSONError(ctx.Tr("repo.pulls.head_branch_not_exist")) return } - err = pull.LoadIssue(ctx) + // check if an opened pull request exists with the same head branch and base branch + pr, err := issues_model.GetUnmergedPullRequest(ctx, pull.HeadRepoID, pull.BaseRepoID, pull.HeadBranch, pull.BaseBranch, pull.Flow) if err != nil { - ctx.ServerError("load the issue of pull request error", err) + if !issues_model.IsErrPullRequestNotExist(err) { + ctx.JSONError(err.Error()) + return + } + } + if pr != nil { + ctx.Flash.Info(ctx.Tr("repo.pulls.open_unmerged_pull_exists", pr.Index)) return } + } - if prHeadCommitID != headBranchCommitID { - // force push to base repo - err := git.Push(ctx, pull.HeadRepo.RepoPath(), git.PushOptions{ - Remote: pull.BaseRepo.RepoPath(), - Branch: pull.HeadBranch + ":" + prHeadRef, - Force: true, - Env: repo_module.InternalPushingEnvironment(pull.Issue.Poster, pull.BaseRepo), - }) + createdComment, err = issue_service.ReopenIssueWithComment(ctx, issue, ctx.Doer, "", form.Content, attachments) + if err != nil { + if errors.Is(err, user_model.ErrBlockedUser) { + ctx.JSONError(ctx.Tr("repo.issues.comment.blocked_user")) + } else { + ctx.ServerError("ReopenIssue", err) + } + return + } + + if issue.IsPull { + pull := issue.PullRequest + // check whether the ref of PR in base repo is consistent with the head commit of head branch in the head repo + // get head commit of PR + if pull.Flow == issues_model.PullRequestFlowGithub { + prHeadRef := pull.GetGitHeadRefName() + if err := pull.LoadBaseRepo(ctx); err != nil { + ctx.ServerError("Unable to load base repo", err) + return + } + prHeadCommitID, err := git.GetFullCommitID(ctx, pull.BaseRepo.RepoPath(), prHeadRef) + if err != nil { + ctx.ServerError("Get head commit Id of pr fail", err) + return + } + + headBranchRef := pull.GetGitHeadBranchRefName() + headBranchCommitID, err := git.GetFullCommitID(ctx, pull.HeadRepo.RepoPath(), headBranchRef) if err != nil { - ctx.ServerError("force push error", err) + ctx.ServerError("Get head commit Id of head branch fail", err) + return + } + + if err = pull.LoadIssue(ctx); err != nil { + ctx.ServerError("load the issue of pull request error", err) return } + + // if the head commit ID of the PR is different from the head branch + if prHeadCommitID != headBranchCommitID { + // force push to base repo + err := git.Push(ctx, pull.HeadRepo.RepoPath(), git.PushOptions{ + Remote: pull.BaseRepo.RepoPath(), + Branch: pull.HeadBranch + ":" + prHeadRef, + Force: true, + Env: repo_module.InternalPushingEnvironment(pull.Issue.Poster, pull.BaseRepo), + }) + if err != nil { + ctx.ServerError("force push error", err) + return + } + } } - } - // Regenerate patch and test conflict. - pull.HeadCommitID = "" - pull_service.StartPullRequestCheckImmediately(ctx, pull) + // Regenerate patch and test conflict. + pull.HeadCommitID = "" + pull_service.StartPullRequestCheckImmediately(ctx, pull) + } case "close": if issue.IsClosed { ctx.JSONError(ctx.Tr("repo.issues.already_closed")) @@ -189,7 +198,7 @@ func NewComment(ctx *context.Context) { return } - createdComment, err = issue_service.CloseIssue(ctx, issue, ctx.Doer, "", form.Content, attachments) + createdComment, err = issue_service.CloseIssueWithComment(ctx, issue, ctx.Doer, "", form.Content, attachments) default: if len(form.Content) == 0 && len(attachments) == 0 { ctx.JSONError(ctx.Tr("repo.issues.comment.empty_content")) diff --git a/routers/web/repo/issue_list.go b/routers/web/repo/issue_list.go index 9ec0bbb751079..fd34422cfcc65 100644 --- a/routers/web/repo/issue_list.go +++ b/routers/web/repo/issue_list.go @@ -434,7 +434,7 @@ func UpdateIssueStatus(ctx *context.Context) { continue } if action == "close" && !issue.IsClosed { - if _, err := issue_service.CloseIssue(ctx, issue, ctx.Doer, "", "", nil); err != nil { + if err := issue_service.CloseIssue(ctx, issue, ctx.Doer, ""); err != nil { if issues_model.IsErrDependenciesLeft(err) { ctx.JSON(http.StatusPreconditionFailed, map[string]any{ "error": ctx.Tr("repo.issues.dependency.issue_batch_close_blocked", issue.Index), @@ -445,7 +445,7 @@ func UpdateIssueStatus(ctx *context.Context) { return } } else if action == "open" && issue.IsClosed { - if _, err := issue_service.ReopenIssue(ctx, issue, ctx.Doer, "", "", nil); err != nil { + if err := issue_service.ReopenIssue(ctx, issue, ctx.Doer, ""); err != nil { ctx.ServerError("ReopenIssue", err) return } diff --git a/services/issue/commit.go b/services/issue/commit.go index c7776268d8541..963d0359fd35d 100644 --- a/services/issue/commit.go +++ b/services/issue/commit.go @@ -196,11 +196,11 @@ func UpdateIssuesCommit(ctx context.Context, doer *user_model.User, repo *repo_m return err } } - if _, err := CloseIssue(ctx, refIssue, doer, c.Sha1, "", nil); err != nil { + if err := CloseIssue(ctx, refIssue, doer, c.Sha1); err != nil { return err } } else if ref.Action == references.XRefActionReopens && refIssue.IsClosed { - if _, err := ReopenIssue(ctx, refIssue, doer, c.Sha1, "", nil); err != nil { + if err := ReopenIssue(ctx, refIssue, doer, c.Sha1); err != nil { return err } } diff --git a/services/issue/status.go b/services/issue/status.go index 33c8c0d26112c..06d9ad5f6426b 100644 --- a/services/issue/status.go +++ b/services/issue/status.go @@ -13,8 +13,34 @@ import ( notify_service "code.gitea.io/gitea/services/notify" ) -// CloseIssue close an issue. -func CloseIssue(ctx context.Context, issue *issues_model.Issue, doer *user_model.User, commitID, commentContent string, attachments []string) (*issues_model.Comment, error) { +// CloseIssue closes an issue +func CloseIssue(ctx context.Context, issue *issues_model.Issue, doer *user_model.User, commitID string) error { + var comment *issues_model.Comment + if err := db.WithTx(ctx, func(ctx context.Context) error { + var err error + comment, err = issues_model.CloseIssue(ctx, issue, doer) + if err != nil { + if issues_model.IsErrDependenciesLeft(err) { + if _, err := issues_model.FinishIssueStopwatch(ctx, doer, issue); err != nil { + log.Error("Unable to stop stopwatch for issue[%d]#%d: %v", issue.ID, issue.Index, err) + } + } + return err + } + + _, err = issues_model.FinishIssueStopwatch(ctx, doer, issue) + return err + }); err != nil { + return err + } + + notify_service.IssueChangeStatus(ctx, doer, commitID, issue, comment, true) + + return nil +} + +// CloseIssueWithComment close an issue with comment +func CloseIssueWithComment(ctx context.Context, issue *issues_model.Issue, doer *user_model.User, commitID, commentContent string, attachments []string) (*issues_model.Comment, error) { var refComment, createdComment *issues_model.Comment if err := db.WithTx(ctx, func(ctx context.Context) error { var err error @@ -59,9 +85,22 @@ func CloseIssue(ctx context.Context, issue *issues_model.Issue, doer *user_model return createdComment, nil } -// ReopenIssue reopen an issue with or without a comment. +// ReopenIssue reopen an issue +// FIXME: If some issues dependent this one are closed, should we also reopen them? +func ReopenIssue(ctx context.Context, issue *issues_model.Issue, doer *user_model.User, commitID string) error { + comment, err := issues_model.ReopenIssue(ctx, issue, doer) + if err != nil { + return err + } + + notify_service.IssueChangeStatus(ctx, doer, commitID, issue, comment, false) + + return nil +} + +// ReopenIssue reopen an issue with a comment. // FIXME: If some issues dependent this one are closed, should we also reopen them? -func ReopenIssue(ctx context.Context, issue *issues_model.Issue, doer *user_model.User, commitID, commentContent string, attachments []string) (*issues_model.Comment, error) { +func ReopenIssueWithComment(ctx context.Context, issue *issues_model.Issue, doer *user_model.User, commitID, commentContent string, attachments []string) (*issues_model.Comment, error) { var createdComment *issues_model.Comment refComment, err := db.WithTx2(ctx, func(ctx context.Context) (*issues_model.Comment, error) { var err error diff --git a/services/pull/merge.go b/services/pull/merge.go index e7fb9525afde4..a941c204357ac 100644 --- a/services/pull/merge.go +++ b/services/pull/merge.go @@ -309,14 +309,14 @@ func handleCloseCrossReferences(ctx context.Context, pr *issues_model.PullReques return err } if ref.RefAction == references.XRefActionCloses && !ref.Issue.IsClosed { - if _, err = issue_service.CloseIssue(ctx, ref.Issue, doer, pr.MergedCommitID, "", nil); err != nil { + if err = issue_service.CloseIssue(ctx, ref.Issue, doer, pr.MergedCommitID); err != nil { // Allow ErrDependenciesLeft if !issues_model.IsErrDependenciesLeft(err) { return err } } } else if ref.RefAction == references.XRefActionReopens && ref.Issue.IsClosed { - if _, err = issue_service.ReopenIssue(ctx, ref.Issue, doer, pr.MergedCommitID, "", nil); err != nil { + if err = issue_service.ReopenIssue(ctx, ref.Issue, doer, pr.MergedCommitID); err != nil { return err } } diff --git a/services/pull/pull.go b/services/pull/pull.go index 9fc55bdbad4c2..e55d4f5bb194f 100644 --- a/services/pull/pull.go +++ b/services/pull/pull.go @@ -700,7 +700,7 @@ func AdjustPullsCausedByBranchDeleted(ctx context.Context, doer *user_model.User var errs []error for _, pr := range prs { - if _, err = issue_service.CloseIssue(ctx, pr.Issue, doer, "", "", nil); err != nil && !issues_model.IsErrIssueIsClosed(err) && !issues_model.IsErrDependenciesLeft(err) { + if err = issue_service.CloseIssue(ctx, pr.Issue, doer, ""); err != nil && !issues_model.IsErrIssueIsClosed(err) && !issues_model.IsErrDependenciesLeft(err) { errs = append(errs, err) } if err == nil { @@ -740,7 +740,7 @@ func AdjustPullsCausedByBranchDeleted(ctx context.Context, doer *user_model.User errs = append(errs, err) } if err == nil { - if _, err = issue_service.CloseIssue(ctx, pr.Issue, doer, "", "", nil); err != nil && !issues_model.IsErrIssueIsClosed(err) && !issues_model.IsErrDependenciesLeft(err) { + if err = issue_service.CloseIssue(ctx, pr.Issue, doer, ""); err != nil && !issues_model.IsErrIssueIsClosed(err) && !issues_model.IsErrDependenciesLeft(err) { errs = append(errs, err) } } @@ -772,7 +772,7 @@ func CloseRepoBranchesPulls(ctx context.Context, doer *user_model.User, repo *re if pr.BaseRepoID == repo.ID { continue } - if _, err = issue_service.CloseIssue(ctx, pr.Issue, doer, "", "", nil); err != nil && !issues_model.IsErrIssueIsClosed(err) { + if err = issue_service.CloseIssue(ctx, pr.Issue, doer, ""); err != nil && !issues_model.IsErrIssueIsClosed(err) { errs = append(errs, err) } } diff --git a/tests/integration/actions_trigger_test.go b/tests/integration/actions_trigger_test.go index c7901705f0ea5..3edb6017b4047 100644 --- a/tests/integration/actions_trigger_test.go +++ b/tests/integration/actions_trigger_test.go @@ -565,12 +565,12 @@ jobs: checkCommitStatusAndInsertFakeStatus(t, repo, sha) // closed - _, err = issue_service.CloseIssue(t.Context(), pullIssue, user2, "", "", nil) + err = issue_service.CloseIssue(t.Context(), pullIssue, user2, "") assert.NoError(t, err) checkCommitStatusAndInsertFakeStatus(t, repo, sha) // reopened - _, err = issue_service.ReopenIssue(t.Context(), pullIssue, user2, "", "", nil) + err = issue_service.ReopenIssue(t.Context(), pullIssue, user2, "") assert.NoError(t, err) checkCommitStatusAndInsertFakeStatus(t, repo, sha) From 602b905e6a6c62ffd98ca1249538786c042dbbf6 Mon Sep 17 00:00:00 2001 From: Lunny Xiao Date: Mon, 29 Sep 2025 16:55:50 -0700 Subject: [PATCH 6/8] refactor --- routers/web/repo/issue_comment.go | 311 +++++++++++++++--------------- 1 file changed, 159 insertions(+), 152 deletions(-) diff --git a/routers/web/repo/issue_comment.go b/routers/web/repo/issue_comment.go index 0d09005c83b7c..9128ce51992c3 100644 --- a/routers/web/repo/issue_comment.go +++ b/routers/web/repo/issue_comment.go @@ -31,6 +31,163 @@ import ( pull_service "code.gitea.io/gitea/services/pull" ) +func newCommentWithReopen(ctx *context.Context, issue *issues_model.Issue, content string, attachments []string) *issues_model.Comment { + if !issue.IsClosed { + ctx.JSONError(ctx.Tr("repo.issues.not_closed")) + return nil + } + + if !ctx.Repo.CanWriteIssuesOrPulls(issue.IsPull) && + !issue.IsPoster(ctx.Doer.ID) && + !ctx.Doer.IsAdmin { + ctx.JSONError(ctx.Tr("repo.issues.reopen_not_allowed")) + return nil + } + + if issue.IsPull { + pull := issue.PullRequest + if pull.HasMerged { + ctx.JSONError(ctx.Tr("repo.issues.reopen_not_allowed_merged")) + return nil + } + + // get head commit of branch in the head repo + if err := pull.LoadHeadRepo(ctx); err != nil { + ctx.ServerError("Unable to load head repo", err) + return nil + } + + // check whether the ref of PR in base repo is consistent with the head commit of head branch in the head repo + // get head commit of PR + if pull.Flow == issues_model.PullRequestFlowGithub { + prHeadRef := pull.GetGitHeadRefName() + if err := pull.LoadBaseRepo(ctx); err != nil { + ctx.ServerError("Unable to load base repo", err) + return nil + } + prHeadCommitID, err := git.GetFullCommitID(ctx, pull.BaseRepo.RepoPath(), prHeadRef) + if err != nil { + ctx.ServerError("Get head commit Id of pr fail", err) + return nil + } + + if ok := gitrepo.IsBranchExist(ctx, pull.HeadRepo, pull.BaseBranch); !ok { + // todo localize + ctx.JSONError("The origin branch is delete, cannot reopen.") + return nil + } + headBranchRef := git.RefNameFromBranch(pull.HeadBranch) + headBranchCommitID, err := git.GetFullCommitID(ctx, pull.HeadRepo.RepoPath(), headBranchRef.String()) + if err != nil { + ctx.ServerError("Get head commit Id of head branch fail", err) + return nil + } + + err = pull.LoadIssue(ctx) + if err != nil { + ctx.ServerError("load the issue of pull request error", err) + return nil + } + + if prHeadCommitID != headBranchCommitID { + // force push to base repo + err := git.Push(ctx, pull.HeadRepo.RepoPath(), git.PushOptions{ + Remote: pull.BaseRepo.RepoPath(), + Branch: pull.HeadBranch + ":" + prHeadRef, + Force: true, + Env: repo_module.InternalPushingEnvironment(pull.Issue.Poster, pull.BaseRepo), + }) + if err != nil { + ctx.ServerError("force push error", err) + return nil + } + } + } + + branchExist, err := git_model.IsBranchExist(ctx, pull.HeadRepo.ID, pull.HeadBranch) + if err != nil { + ctx.ServerError("IsBranchExist", err) + return nil + } + if !branchExist { + ctx.JSONError(ctx.Tr("repo.pulls.head_branch_not_exist")) + return nil + } + + // check if an opened pull request exists with the same head branch and base branch + pr, err := issues_model.GetUnmergedPullRequest(ctx, pull.HeadRepoID, pull.BaseRepoID, pull.HeadBranch, pull.BaseBranch, pull.Flow) + if err != nil { + if !issues_model.IsErrPullRequestNotExist(err) { + ctx.JSONError(err.Error()) + return nil + } + } + if pr != nil { + ctx.Flash.Info(ctx.Tr("repo.pulls.open_unmerged_pull_exists", pr.Index)) + return nil + } + } + + createdComment, err := issue_service.ReopenIssueWithComment(ctx, issue, ctx.Doer, "", content, attachments) + if err != nil { + if errors.Is(err, user_model.ErrBlockedUser) { + ctx.JSONError(ctx.Tr("repo.issues.comment.blocked_user")) + } else { + ctx.ServerError("ReopenIssue", err) + } + return nil + } + + if issue.IsPull { + pull := issue.PullRequest + // check whether the ref of PR in base repo is consistent with the head commit of head branch in the head repo + // get head commit of PR + if pull.Flow == issues_model.PullRequestFlowGithub { + prHeadRef := pull.GetGitHeadRefName() + if err := pull.LoadBaseRepo(ctx); err != nil { + ctx.ServerError("Unable to load base repo", err) + return nil + } + prHeadCommitID, err := ctx.Repo.GitRepo.GetRefCommitID(prHeadRef) + if err != nil { + ctx.ServerError("Get head commit Id of pr fail", err) + return nil + } + + headBranchCommitID, err := gitrepo.GetBranchCommitID(ctx, pull.HeadRepo, pull.HeadBranch) + if err != nil { + ctx.ServerError("Get head commit Id of head branch fail", err) + return nil + } + + if err = pull.LoadIssue(ctx); err != nil { + ctx.ServerError("load the issue of pull request error", err) + return nil + } + + // if the head commit ID of the PR is different from the head branch + if prHeadCommitID != headBranchCommitID { + // force push to base repo + err := git.Push(ctx, pull.HeadRepo.RepoPath(), git.PushOptions{ + Remote: pull.BaseRepo.RepoPath(), + Branch: pull.HeadBranch + ":" + prHeadRef, + Force: true, + Env: repo_module.InternalPushingEnvironment(pull.Issue.Poster, pull.BaseRepo), + }) + if err != nil { + ctx.ServerError("force push error", err) + return nil + } + } + } + + // Regenerate patch and test conflict. + pull.HeadCommitID = "" + pull_service.StartPullRequestCheckImmediately(ctx, pull) + } + return createdComment +} + // NewComment create a comment for issue func NewComment(ctx *context.Context) { form := web.GetForm(ctx).(*forms.CreateCommentForm) @@ -82,160 +239,10 @@ func NewComment(ctx *context.Context) { switch form.Status { case "reopen": - if !issue.IsClosed { - ctx.JSONError(ctx.Tr("repo.issues.not_closed")) + createdComment = newCommentWithReopen(ctx, issue, form.Content, attachments) + if ctx.Written() { return } - - if !ctx.Repo.CanWriteIssuesOrPulls(issue.IsPull) && - !issue.IsPoster(ctx.Doer.ID) && - !ctx.Doer.IsAdmin { - ctx.JSONError(ctx.Tr("repo.issues.reopen_not_allowed")) - return - } - - if issue.IsPull { - pull := issue.PullRequest - if pull.HasMerged { - ctx.JSONError(ctx.Tr("repo.issues.reopen_not_allowed_merged")) - return - } - - // get head commit of branch in the head repo - if err := pull.LoadHeadRepo(ctx); err != nil { - ctx.ServerError("Unable to load head repo", err) - return - } - - // check whether the ref of PR in base repo is consistent with the head commit of head branch in the head repo - // get head commit of PR - if pull.Flow == issues_model.PullRequestFlowGithub { - prHeadRef := pull.GetGitHeadRefName() - if err := pull.LoadBaseRepo(ctx); err != nil { - ctx.ServerError("Unable to load base repo", err) - return - } - prHeadCommitID, err := git.GetFullCommitID(ctx, pull.BaseRepo.RepoPath(), prHeadRef) - if err != nil { - ctx.ServerError("Get head commit Id of pr fail", err) - return - } - - if ok := gitrepo.IsBranchExist(ctx, pull.HeadRepo, pull.BaseBranch); !ok { - // todo localize - ctx.JSONError("The origin branch is delete, cannot reopen.") - return - } - headBranchRef := git.RefNameFromBranch(pull.HeadBranch) - headBranchCommitID, err := git.GetFullCommitID(ctx, pull.HeadRepo.RepoPath(), headBranchRef.String()) - if err != nil { - ctx.ServerError("Get head commit Id of head branch fail", err) - return - } - - err = pull.LoadIssue(ctx) - if err != nil { - ctx.ServerError("load the issue of pull request error", err) - return - } - - if prHeadCommitID != headBranchCommitID { - // force push to base repo - err := git.Push(ctx, pull.HeadRepo.RepoPath(), git.PushOptions{ - Remote: pull.BaseRepo.RepoPath(), - Branch: pull.HeadBranch + ":" + prHeadRef, - Force: true, - Env: repo_module.InternalPushingEnvironment(pull.Issue.Poster, pull.BaseRepo), - }) - if err != nil { - ctx.ServerError("force push error", err) - return - } - } - } - - branchExist, err := git_model.IsBranchExist(ctx, pull.HeadRepo.ID, pull.HeadBranch) - if err != nil { - ctx.ServerError("IsBranchExist", err) - return - } - if !branchExist { - ctx.JSONError(ctx.Tr("repo.pulls.head_branch_not_exist")) - return - } - - // check if an opened pull request exists with the same head branch and base branch - pr, err := issues_model.GetUnmergedPullRequest(ctx, pull.HeadRepoID, pull.BaseRepoID, pull.HeadBranch, pull.BaseBranch, pull.Flow) - if err != nil { - if !issues_model.IsErrPullRequestNotExist(err) { - ctx.JSONError(err.Error()) - return - } - } - if pr != nil { - ctx.Flash.Info(ctx.Tr("repo.pulls.open_unmerged_pull_exists", pr.Index)) - return - } - } - - createdComment, err = issue_service.ReopenIssueWithComment(ctx, issue, ctx.Doer, "", form.Content, attachments) - if err != nil { - if errors.Is(err, user_model.ErrBlockedUser) { - ctx.JSONError(ctx.Tr("repo.issues.comment.blocked_user")) - } else { - ctx.ServerError("ReopenIssue", err) - } - return - } - - if issue.IsPull { - pull := issue.PullRequest - // check whether the ref of PR in base repo is consistent with the head commit of head branch in the head repo - // get head commit of PR - if pull.Flow == issues_model.PullRequestFlowGithub { - prHeadRef := pull.GetGitHeadRefName() - if err := pull.LoadBaseRepo(ctx); err != nil { - ctx.ServerError("Unable to load base repo", err) - return - } - prHeadCommitID, err := git.GetFullCommitID(ctx, pull.BaseRepo.RepoPath(), prHeadRef) - if err != nil { - ctx.ServerError("Get head commit Id of pr fail", err) - return - } - - headBranchRef := pull.GetGitHeadBranchRefName() - headBranchCommitID, err := git.GetFullCommitID(ctx, pull.HeadRepo.RepoPath(), headBranchRef) - if err != nil { - ctx.ServerError("Get head commit Id of head branch fail", err) - return - } - - if err = pull.LoadIssue(ctx); err != nil { - ctx.ServerError("load the issue of pull request error", err) - return - } - - // if the head commit ID of the PR is different from the head branch - if prHeadCommitID != headBranchCommitID { - // force push to base repo - err := git.Push(ctx, pull.HeadRepo.RepoPath(), git.PushOptions{ - Remote: pull.BaseRepo.RepoPath(), - Branch: pull.HeadBranch + ":" + prHeadRef, - Force: true, - Env: repo_module.InternalPushingEnvironment(pull.Issue.Poster, pull.BaseRepo), - }) - if err != nil { - ctx.ServerError("force push error", err) - return - } - } - } - - // Regenerate patch and test conflict. - pull.HeadCommitID = "" - pull_service.StartPullRequestCheckImmediately(ctx, pull) - } case "close": if issue.IsClosed { ctx.JSONError(ctx.Tr("repo.issues.already_closed")) From d9d2b03d97e824a28a4fb0a64e524315ededbd24 Mon Sep 17 00:00:00 2001 From: Lunny Xiao Date: Mon, 6 Oct 2025 21:19:17 -0700 Subject: [PATCH 7/8] Add tests --- routers/web/repo/issue_comment.go | 238 +++++++++++++------------- tests/integration/issue_test.go | 20 +++ tests/integration/pull_create_test.go | 46 +++++ 3 files changed, 188 insertions(+), 116 deletions(-) diff --git a/routers/web/repo/issue_comment.go b/routers/web/repo/issue_comment.go index 9128ce51992c3..cf88b6582a343 100644 --- a/routers/web/repo/issue_comment.go +++ b/routers/web/repo/issue_comment.go @@ -23,6 +23,7 @@ import ( repo_module "code.gitea.io/gitea/modules/repository" "code.gitea.io/gitea/modules/setting" api "code.gitea.io/gitea/modules/structs" + "code.gitea.io/gitea/modules/util" "code.gitea.io/gitea/modules/web" "code.gitea.io/gitea/services/context" "code.gitea.io/gitea/services/convert" @@ -31,102 +32,84 @@ import ( pull_service "code.gitea.io/gitea/services/pull" ) -func newCommentWithReopen(ctx *context.Context, issue *issues_model.Issue, content string, attachments []string) *issues_model.Comment { - if !issue.IsClosed { - ctx.JSONError(ctx.Tr("repo.issues.not_closed")) - return nil - } +func reopenPullWithComment(ctx *context.Context, issue *issues_model.Issue, content string, attachments []string) *issues_model.Comment { + pull := issue.PullRequest - if !ctx.Repo.CanWriteIssuesOrPulls(issue.IsPull) && - !issue.IsPoster(ctx.Doer.ID) && - !ctx.Doer.IsAdmin { - ctx.JSONError(ctx.Tr("repo.issues.reopen_not_allowed")) + // get head commit of branch in the head repo + if err := pull.LoadHeadRepo(ctx); err != nil { + ctx.ServerError("Unable to load head repo", err) return nil } - if issue.IsPull { - pull := issue.PullRequest - if pull.HasMerged { - ctx.JSONError(ctx.Tr("repo.issues.reopen_not_allowed_merged")) + // check whether the ref of PR in base repo is consistent with the head commit of head branch in the head repo + // get head commit of PR + if pull.Flow == issues_model.PullRequestFlowGithub { + prHeadRef := pull.GetGitHeadRefName() + if err := pull.LoadBaseRepo(ctx); err != nil { + ctx.ServerError("Unable to load base repo", err) return nil } - - // get head commit of branch in the head repo - if err := pull.LoadHeadRepo(ctx); err != nil { - ctx.ServerError("Unable to load head repo", err) + prHeadCommitID, err := git.GetFullCommitID(ctx, pull.BaseRepo.RepoPath(), prHeadRef) + if err != nil { + ctx.ServerError("Get head commit Id of pr fail", err) return nil } - // check whether the ref of PR in base repo is consistent with the head commit of head branch in the head repo - // get head commit of PR - if pull.Flow == issues_model.PullRequestFlowGithub { - prHeadRef := pull.GetGitHeadRefName() - if err := pull.LoadBaseRepo(ctx); err != nil { - ctx.ServerError("Unable to load base repo", err) - return nil - } - prHeadCommitID, err := git.GetFullCommitID(ctx, pull.BaseRepo.RepoPath(), prHeadRef) - if err != nil { - ctx.ServerError("Get head commit Id of pr fail", err) - return nil - } - - if ok := gitrepo.IsBranchExist(ctx, pull.HeadRepo, pull.BaseBranch); !ok { - // todo localize - ctx.JSONError("The origin branch is delete, cannot reopen.") - return nil - } - headBranchRef := git.RefNameFromBranch(pull.HeadBranch) - headBranchCommitID, err := git.GetFullCommitID(ctx, pull.HeadRepo.RepoPath(), headBranchRef.String()) - if err != nil { - ctx.ServerError("Get head commit Id of head branch fail", err) - return nil - } - - err = pull.LoadIssue(ctx) - if err != nil { - ctx.ServerError("load the issue of pull request error", err) - return nil - } - - if prHeadCommitID != headBranchCommitID { - // force push to base repo - err := git.Push(ctx, pull.HeadRepo.RepoPath(), git.PushOptions{ - Remote: pull.BaseRepo.RepoPath(), - Branch: pull.HeadBranch + ":" + prHeadRef, - Force: true, - Env: repo_module.InternalPushingEnvironment(pull.Issue.Poster, pull.BaseRepo), - }) - if err != nil { - ctx.ServerError("force push error", err) - return nil - } - } + if ok := gitrepo.IsBranchExist(ctx, pull.HeadRepo, pull.BaseBranch); !ok { + // todo localize + ctx.JSONError("The origin branch is delete, cannot reopen.") + return nil } - - branchExist, err := git_model.IsBranchExist(ctx, pull.HeadRepo.ID, pull.HeadBranch) + headBranchRef := git.RefNameFromBranch(pull.HeadBranch) + headBranchCommitID, err := git.GetFullCommitID(ctx, pull.HeadRepo.RepoPath(), headBranchRef.String()) if err != nil { - ctx.ServerError("IsBranchExist", err) + ctx.ServerError("Get head commit Id of head branch fail", err) return nil } - if !branchExist { - ctx.JSONError(ctx.Tr("repo.pulls.head_branch_not_exist")) + + err = pull.LoadIssue(ctx) + if err != nil { + ctx.ServerError("load the issue of pull request error", err) return nil } - // check if an opened pull request exists with the same head branch and base branch - pr, err := issues_model.GetUnmergedPullRequest(ctx, pull.HeadRepoID, pull.BaseRepoID, pull.HeadBranch, pull.BaseBranch, pull.Flow) - if err != nil { - if !issues_model.IsErrPullRequestNotExist(err) { - ctx.JSONError(err.Error()) + if prHeadCommitID != headBranchCommitID { + // force push to base repo + err := git.Push(ctx, pull.HeadRepo.RepoPath(), git.PushOptions{ + Remote: pull.BaseRepo.RepoPath(), + Branch: pull.HeadBranch + ":" + prHeadRef, + Force: true, + Env: repo_module.InternalPushingEnvironment(pull.Issue.Poster, pull.BaseRepo), + }) + if err != nil { + ctx.ServerError("force push error", err) return nil } } - if pr != nil { - ctx.Flash.Info(ctx.Tr("repo.pulls.open_unmerged_pull_exists", pr.Index)) + } + + branchExist, err := git_model.IsBranchExist(ctx, pull.HeadRepo.ID, pull.HeadBranch) + if err != nil { + ctx.ServerError("IsBranchExist", err) + return nil + } + if !branchExist { + ctx.JSONError(ctx.Tr("repo.pulls.head_branch_not_exist")) + return nil + } + + // check if an opened pull request exists with the same head branch and base branch + pr, err := issues_model.GetUnmergedPullRequest(ctx, pull.HeadRepoID, pull.BaseRepoID, pull.HeadBranch, pull.BaseBranch, pull.Flow) + if err != nil { + if !issues_model.IsErrPullRequestNotExist(err) { + ctx.JSONError(err.Error()) return nil } } + if pr != nil { + ctx.Flash.Info(ctx.Tr("repo.pulls.open_unmerged_pull_exists", pr.Index)) + return nil + } createdComment, err := issue_service.ReopenIssueWithComment(ctx, issue, ctx.Doer, "", content, attachments) if err != nil { @@ -138,46 +121,43 @@ func newCommentWithReopen(ctx *context.Context, issue *issues_model.Issue, conte return nil } - if issue.IsPull { - pull := issue.PullRequest - // check whether the ref of PR in base repo is consistent with the head commit of head branch in the head repo - // get head commit of PR - if pull.Flow == issues_model.PullRequestFlowGithub { - prHeadRef := pull.GetGitHeadRefName() - if err := pull.LoadBaseRepo(ctx); err != nil { - ctx.ServerError("Unable to load base repo", err) - return nil - } - prHeadCommitID, err := ctx.Repo.GitRepo.GetRefCommitID(prHeadRef) - if err != nil { - ctx.ServerError("Get head commit Id of pr fail", err) - return nil - } + // check whether the ref of PR in base repo is consistent with the head commit of head branch in the head repo + // get head commit of PR + if pull.Flow == issues_model.PullRequestFlowGithub { + prHeadRef := pull.GetGitHeadRefName() + if err := pull.LoadBaseRepo(ctx); err != nil { + ctx.ServerError("Unable to load base repo", err) + return nil + } + prHeadCommitID, err := ctx.Repo.GitRepo.GetRefCommitID(prHeadRef) + if err != nil { + ctx.ServerError("Get head commit Id of pr fail", err) + return nil + } - headBranchCommitID, err := gitrepo.GetBranchCommitID(ctx, pull.HeadRepo, pull.HeadBranch) - if err != nil { - ctx.ServerError("Get head commit Id of head branch fail", err) - return nil - } + headBranchCommitID, err := gitrepo.GetBranchCommitID(ctx, pull.HeadRepo, pull.HeadBranch) + if err != nil { + ctx.ServerError("Get head commit Id of head branch fail", err) + return nil + } - if err = pull.LoadIssue(ctx); err != nil { - ctx.ServerError("load the issue of pull request error", err) - return nil - } + if err = pull.LoadIssue(ctx); err != nil { + ctx.ServerError("load the issue of pull request error", err) + return nil + } - // if the head commit ID of the PR is different from the head branch - if prHeadCommitID != headBranchCommitID { - // force push to base repo - err := git.Push(ctx, pull.HeadRepo.RepoPath(), git.PushOptions{ - Remote: pull.BaseRepo.RepoPath(), - Branch: pull.HeadBranch + ":" + prHeadRef, - Force: true, - Env: repo_module.InternalPushingEnvironment(pull.Issue.Poster, pull.BaseRepo), - }) - if err != nil { - ctx.ServerError("force push error", err) - return nil - } + // if the head commit ID of the PR is different from the head branch + if prHeadCommitID != headBranchCommitID { + // force push to base repo + err := git.Push(ctx, pull.HeadRepo.RepoPath(), git.PushOptions{ + Remote: pull.BaseRepo.RepoPath(), + Branch: pull.HeadBranch + ":" + prHeadRef, + Force: true, + Env: repo_module.InternalPushingEnvironment(pull.Issue.Poster, pull.BaseRepo), + }) + if err != nil { + ctx.ServerError("force push error", err) + return nil } } @@ -224,6 +204,11 @@ func NewComment(ctx *context.Context) { return } + if form.Content == "" { + ctx.JSONError(ctx.Tr("repo.issues.comment.empty_content")) + return + } + var attachments []string if setting.Attachment.Enabled { attachments = form.Files @@ -239,7 +224,30 @@ func NewComment(ctx *context.Context) { switch form.Status { case "reopen": - createdComment = newCommentWithReopen(ctx, issue, form.Content, attachments) + if !issue.IsClosed { + ctx.JSONError(ctx.Tr("repo.issues.not_closed")) + return + } + if !ctx.Repo.CanWriteIssuesOrPulls(issue.IsPull) && + !issue.IsPoster(ctx.Doer.ID) && + !ctx.Doer.IsAdmin { + ctx.JSONError(ctx.Tr("repo.issues.reopen_not_allowed")) + return + } + + if issue.IsPull { + createdComment = reopenPullWithComment(ctx, issue, form.Content, attachments) + } else { + createdComment, err = issue_service.ReopenIssueWithComment(ctx, issue, ctx.Doer, "", form.Content, attachments) + if err != nil { + if errors.Is(err, user_model.ErrBlockedUser) { + ctx.JSONError(ctx.Tr("repo.issues.comment.blocked_user")) + } else { + ctx.ServerError("ReopenIssue", err) + } + return + } + } if ctx.Written() { return } @@ -276,10 +284,8 @@ func NewComment(ctx *context.Context) { } // Redirect to comment hashtag if there is any actual content. - typeName := "issues" - if issue.IsPull { - typeName = "pulls" - } + typeName := util.Iif(issue.IsPull, "pulls", "issues") + if createdComment != nil { log.Trace("Comment created: %d/%d/%d", ctx.Repo.Repository.ID, issue.ID, createdComment.ID) ctx.JSONRedirect(fmt.Sprintf("%s/%s/%d#%s", ctx.Repo.RepoLink, typeName, issue.Index, createdComment.HashTag())) diff --git a/tests/integration/issue_test.go b/tests/integration/issue_test.go index 613353e55cbb9..09292b52b66e9 100644 --- a/tests/integration/issue_test.go +++ b/tests/integration/issue_test.go @@ -255,6 +255,26 @@ func TestIssueCommentClose(t *testing.T) { htmlDoc := NewHTMLParser(t, resp.Body) val := htmlDoc.doc.Find(".comment-list .comment .render-content p").First().Text() assert.Equal(t, "Description", val) + val = strings.TrimSpace(htmlDoc.doc.Find(".issue-title-header .issue-state-label").Text()) + assert.Equal(t, "Closed", val) +} + +func TestIssueCommentReopen(t *testing.T) { + defer tests.PrepareTestEnv(t)() + session := loginUser(t, "user2") + issueURL := testNewIssue(t, session, "user2", "repo1", "Title", "Description") + testIssueAddComment(t, session, issueURL, "Test comment 1", "") + testIssueAddComment(t, session, issueURL, "Test comment 2", "close") + testIssueAddComment(t, session, issueURL, "Test comment 2", "reopen") + + // Validate that issue content has not been updated + req := NewRequest(t, "GET", issueURL) + resp := session.MakeRequest(t, req, http.StatusOK) + htmlDoc := NewHTMLParser(t, resp.Body) + val := htmlDoc.doc.Find(".comment-list .comment .render-content p").First().Text() + assert.Equal(t, "Description", val) + val = strings.TrimSpace(htmlDoc.doc.Find(".issue-title-header .issue-state-label").Text()) + assert.Equal(t, "Open", val) } func TestIssueCommentDelete(t *testing.T) { diff --git a/tests/integration/pull_create_test.go b/tests/integration/pull_create_test.go index 18f7e80e8276f..727bb0c50f322 100644 --- a/tests/integration/pull_create_test.go +++ b/tests/integration/pull_create_test.go @@ -313,3 +313,49 @@ func TestCreatePullWhenBlocked(t *testing.T) { MakeRequest(t, req, http.StatusNoContent) }) } + +func TestPullRequestCommentClose(t *testing.T) { + onGiteaRun(t, func(t *testing.T, u *url.URL) { + session := loginUser(t, "user1") + testRepoFork(t, session, "user2", "repo1", "user1", "repo1", "") + testEditFile(t, session, "user1", "repo1", "master", "README.md", "Hello, World (Edited)\n") + resp := testPullCreate(t, session, "user1", "repo1", false, "master", "master", "This is a pull title") + pullURL := test.RedirectURL(resp) + + testIssueAddComment(t, session, pullURL, "Test comment 1", "") + testIssueAddComment(t, session, pullURL, "Test comment 2", "") + testIssueAddComment(t, session, pullURL, "Test comment 3", "close") + + // Validate that issue content has not been updated + req := NewRequest(t, "GET", pullURL) + resp = session.MakeRequest(t, req, http.StatusOK) + htmlDoc := NewHTMLParser(t, resp.Body) + val := strings.Split(strings.TrimSpace(htmlDoc.doc.Find("#issue-title-display > h1").First().Text()), "\n")[0] + assert.Equal(t, "This is a pull title", val) + val = strings.TrimSpace(htmlDoc.doc.Find(".issue-title-header .issue-state-label").Text()) + assert.Equal(t, "Closed", val) + }) +} + +func TestPullRequestCommentReopen(t *testing.T) { + onGiteaRun(t, func(t *testing.T, u *url.URL) { + session := loginUser(t, "user1") + testRepoFork(t, session, "user2", "repo1", "user1", "repo1", "") + testEditFile(t, session, "user1", "repo1", "master", "README.md", "Hello, World (Edited)\n") + resp := testPullCreate(t, session, "user1", "repo1", false, "master", "master", "This is a pull title") + pullURL := test.RedirectURL(resp) + + testIssueAddComment(t, session, pullURL, "Test comment 1", "") + testIssueAddComment(t, session, pullURL, "Test comment 2", "close") + testIssueAddComment(t, session, pullURL, "Test comment 2", "reopen") + + // Validate that issue content has not been updated + req := NewRequest(t, "GET", pullURL) + resp = session.MakeRequest(t, req, http.StatusOK) + htmlDoc := NewHTMLParser(t, resp.Body) + val := strings.Split(strings.TrimSpace(htmlDoc.doc.Find("#issue-title-display > h1").First().Text()), "\n")[0] + assert.Equal(t, "This is a pull title", val) + val = strings.TrimSpace(htmlDoc.doc.Find(".issue-title-header .issue-state-label").Text()) + assert.Equal(t, "Open", val) + }) +} From f30e54785f779cd553a6b728a2e9b315d6cee459 Mon Sep 17 00:00:00 2001 From: Lunny Xiao Date: Mon, 6 Oct 2025 21:21:25 -0700 Subject: [PATCH 8/8] fix comment --- services/issue/status.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/services/issue/status.go b/services/issue/status.go index 06d9ad5f6426b..50a26547357ac 100644 --- a/services/issue/status.go +++ b/services/issue/status.go @@ -98,7 +98,7 @@ func ReopenIssue(ctx context.Context, issue *issues_model.Issue, doer *user_mode return nil } -// ReopenIssue reopen an issue with a comment. +// ReopenIssueWithComment reopen an issue with a comment. // FIXME: If some issues dependent this one are closed, should we also reopen them? func ReopenIssueWithComment(ctx context.Context, issue *issues_model.Issue, doer *user_model.User, commitID, commentContent string, attachments []string) (*issues_model.Comment, error) { var createdComment *issues_model.Comment