mirror of https://github.com/go-gitea/gitea.git
This PR will avoid load pullrequest.Issue twice in pull request list page. It will reduce x times database queries for those WIP pull requests. Partially fix #29585 Backport #29900
This commit is contained in:
parent
c03b1e2854
commit
6ef986d474
|
@ -20,6 +20,7 @@ import (
|
||||||
"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/modules/timeutil"
|
"code.gitea.io/gitea/modules/timeutil"
|
||||||
|
"code.gitea.io/gitea/modules/util"
|
||||||
|
|
||||||
"xorm.io/builder"
|
"xorm.io/builder"
|
||||||
"xorm.io/xorm"
|
"xorm.io/xorm"
|
||||||
|
@ -821,3 +822,31 @@ func UpdateNotificationStatuses(ctx context.Context, user *user_model.User, curr
|
||||||
Update(n)
|
Update(n)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// LoadIssuePullRequests loads all issues' pull requests if possible
|
||||||
|
func (nl NotificationList) LoadIssuePullRequests(ctx context.Context) error {
|
||||||
|
issues := make(map[int64]*issues_model.Issue, len(nl))
|
||||||
|
for _, notification := range nl {
|
||||||
|
if notification.Issue != nil && notification.Issue.IsPull && notification.Issue.PullRequest == nil {
|
||||||
|
issues[notification.Issue.ID] = notification.Issue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(issues) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
pulls, err := issues_model.GetPullRequestByIssueIDs(ctx, util.KeysOfMap(issues))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, pull := range pulls {
|
||||||
|
if issue := issues[pull.IssueID]; issue != nil {
|
||||||
|
issue.PullRequest = pull
|
||||||
|
issue.PullRequest.Issue = issue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
|
@ -200,20 +200,6 @@ func (issue *Issue) IsTimetrackerEnabled(ctx context.Context) bool {
|
||||||
return issue.Repo.IsTimetrackerEnabled(ctx)
|
return issue.Repo.IsTimetrackerEnabled(ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetPullRequest returns the issue pull request
|
|
||||||
func (issue *Issue) GetPullRequest() (pr *PullRequest, err error) {
|
|
||||||
if !issue.IsPull {
|
|
||||||
return nil, fmt.Errorf("Issue is not a pull request")
|
|
||||||
}
|
|
||||||
|
|
||||||
pr, err = GetPullRequestByIssueID(db.DefaultContext, issue.ID)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
pr.Issue = issue
|
|
||||||
return pr, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// LoadPoster loads poster
|
// LoadPoster loads poster
|
||||||
func (issue *Issue) LoadPoster(ctx context.Context) (err error) {
|
func (issue *Issue) LoadPoster(ctx context.Context) (err error) {
|
||||||
if issue.Poster == nil && issue.PosterID != 0 {
|
if issue.Poster == nil && issue.PosterID != 0 {
|
||||||
|
|
|
@ -370,6 +370,9 @@ func (issues IssueList) LoadPullRequests(ctx context.Context) error {
|
||||||
|
|
||||||
for _, issue := range issues {
|
for _, issue := range issues {
|
||||||
issue.PullRequest = pullRequestMaps[issue.ID]
|
issue.PullRequest = pullRequestMaps[issue.ID]
|
||||||
|
if issue.PullRequest != nil {
|
||||||
|
issue.PullRequest.Issue = issue
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -212,3 +212,12 @@ func HasMergedPullRequestInRepo(ctx context.Context, repoID, posterID int64) (bo
|
||||||
Limit(1).
|
Limit(1).
|
||||||
Get(new(Issue))
|
Get(new(Issue))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GetPullRequestByIssueIDs returns all pull requests by issue ids
|
||||||
|
func GetPullRequestByIssueIDs(ctx context.Context, issueIDs []int64) (PullRequestList, error) {
|
||||||
|
prs := make([]*PullRequest, 0, len(issueIDs))
|
||||||
|
return prs, db.GetEngine(ctx).
|
||||||
|
Where("issue_id > 0").
|
||||||
|
In("issue_id", issueIDs).
|
||||||
|
Find(&prs)
|
||||||
|
}
|
||||||
|
|
|
@ -240,11 +240,11 @@ type CreateReviewOptions struct {
|
||||||
|
|
||||||
// IsOfficialReviewer check if at least one of the provided reviewers can make official reviews in issue (counts towards required approvals)
|
// IsOfficialReviewer check if at least one of the provided reviewers can make official reviews in issue (counts towards required approvals)
|
||||||
func IsOfficialReviewer(ctx context.Context, issue *Issue, reviewer *user_model.User) (bool, error) {
|
func IsOfficialReviewer(ctx context.Context, issue *Issue, reviewer *user_model.User) (bool, error) {
|
||||||
pr, err := GetPullRequestByIssueID(ctx, issue.ID)
|
if err := issue.LoadPullRequest(ctx); err != nil {
|
||||||
if err != nil {
|
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pr := issue.PullRequest
|
||||||
rule, err := git_model.GetFirstMatchProtectedBranchRule(ctx, pr.BaseRepoID, pr.BaseBranch)
|
rule, err := git_model.GetFirstMatchProtectedBranchRule(ctx, pr.BaseRepoID, pr.BaseBranch)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, err
|
return false, err
|
||||||
|
@ -272,11 +272,10 @@ func IsOfficialReviewer(ctx context.Context, issue *Issue, reviewer *user_model.
|
||||||
|
|
||||||
// IsOfficialReviewerTeam check if reviewer in this team can make official reviews in issue (counts towards required approvals)
|
// IsOfficialReviewerTeam check if reviewer in this team can make official reviews in issue (counts towards required approvals)
|
||||||
func IsOfficialReviewerTeam(ctx context.Context, issue *Issue, team *organization.Team) (bool, error) {
|
func IsOfficialReviewerTeam(ctx context.Context, issue *Issue, team *organization.Team) (bool, error) {
|
||||||
pr, err := GetPullRequestByIssueID(ctx, issue.ID)
|
if err := issue.LoadPullRequest(ctx); err != nil {
|
||||||
if err != nil {
|
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
pb, err := git_model.GetFirstMatchProtectedBranchRule(ctx, pr.BaseRepoID, pr.BaseBranch)
|
pb, err := git_model.GetFirstMatchProtectedBranchRule(ctx, issue.PullRequest.BaseRepoID, issue.PullRequest.BaseBranch)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
|
|
|
@ -45,3 +45,12 @@ func SliceSortedEqual[T comparable](s1, s2 []T) bool {
|
||||||
func SliceRemoveAll[T comparable](slice []T, target T) []T {
|
func SliceRemoveAll[T comparable](slice []T, target T) []T {
|
||||||
return slices.DeleteFunc(slice, func(t T) bool { return t == target })
|
return slices.DeleteFunc(slice, func(t T) bool { return t == target })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: Replace with "maps.Keys" once available, current it only in golang.org/x/exp/maps but not in standard library
|
||||||
|
func KeysOfMap[K comparable, V any](m map[K]V) []K {
|
||||||
|
keys := make([]K, 0, len(m))
|
||||||
|
for k := range m {
|
||||||
|
keys = append(keys, k)
|
||||||
|
}
|
||||||
|
return keys
|
||||||
|
}
|
||||||
|
|
|
@ -864,10 +864,11 @@ func EditIssue(ctx *context.APIContext) {
|
||||||
}
|
}
|
||||||
if form.State != nil {
|
if form.State != nil {
|
||||||
if issue.IsPull {
|
if issue.IsPull {
|
||||||
if pr, err := issue.GetPullRequest(); err != nil {
|
if err := issue.LoadPullRequest(ctx); err != nil {
|
||||||
ctx.Error(http.StatusInternalServerError, "GetPullRequest", err)
|
ctx.Error(http.StatusInternalServerError, "GetPullRequest", err)
|
||||||
return
|
return
|
||||||
} else if pr.HasMerged {
|
}
|
||||||
|
if issue.PullRequest.HasMerged {
|
||||||
ctx.Error(http.StatusPreconditionFailed, "MergedPRState", "cannot change state of this pull request, it was already merged")
|
ctx.Error(http.StatusPreconditionFailed, "MergedPRState", "cannot change state of this pull request, it was already merged")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
|
@ -240,18 +240,12 @@ func ListPinnedPullRequests(ctx *context.APIContext) {
|
||||||
}
|
}
|
||||||
|
|
||||||
apiPrs := make([]*api.PullRequest, len(issues))
|
apiPrs := make([]*api.PullRequest, len(issues))
|
||||||
|
if err := issues.LoadPullRequests(ctx); err != nil {
|
||||||
|
ctx.Error(http.StatusInternalServerError, "LoadPullRequests", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
for i, currentIssue := range issues {
|
for i, currentIssue := range issues {
|
||||||
pr, err := currentIssue.GetPullRequest()
|
pr := currentIssue.PullRequest
|
||||||
if err != nil {
|
|
||||||
ctx.Error(http.StatusInternalServerError, "GetPullRequest", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if err = pr.LoadIssue(ctx); err != nil {
|
|
||||||
ctx.Error(http.StatusInternalServerError, "LoadIssue", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if err = pr.LoadAttributes(ctx); err != nil {
|
if err = pr.LoadAttributes(ctx); err != nil {
|
||||||
ctx.Error(http.StatusInternalServerError, "LoadAttributes", err)
|
ctx.Error(http.StatusInternalServerError, "LoadAttributes", err)
|
||||||
return
|
return
|
||||||
|
|
|
@ -128,6 +128,12 @@ func getNotifications(ctx *context.Context) {
|
||||||
ctx.ServerError("LoadIssues", err)
|
ctx.ServerError("LoadIssues", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if err = notifications.LoadIssuePullRequests(ctx); err != nil {
|
||||||
|
ctx.ServerError("LoadIssuePullRequests", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
notifications = notifications.Without(failures)
|
notifications = notifications.Without(failures)
|
||||||
failCount += len(failures)
|
failCount += len(failures)
|
||||||
|
|
||||||
|
|
|
@ -61,8 +61,9 @@ func ToNotificationThread(ctx context.Context, n *activities_model.Notification)
|
||||||
result.Subject.LatestCommentHTMLURL = comment.HTMLURL(ctx)
|
result.Subject.LatestCommentHTMLURL = comment.HTMLURL(ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
pr, _ := n.Issue.GetPullRequest()
|
if err := n.Issue.LoadPullRequest(ctx); err == nil &&
|
||||||
if pr != nil && pr.HasMerged {
|
n.Issue.PullRequest != nil &&
|
||||||
|
n.Issue.PullRequest.HasMerged {
|
||||||
result.Subject.State = "merged"
|
result.Subject.State = "merged"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -264,11 +264,11 @@ func createCodeComment(ctx context.Context, doer *user_model.User, repo *repo_mo
|
||||||
|
|
||||||
// SubmitReview creates a review out of the existing pending review or creates a new one if no pending review exist
|
// SubmitReview creates a review out of the existing pending review or creates a new one if no pending review exist
|
||||||
func SubmitReview(ctx context.Context, doer *user_model.User, gitRepo *git.Repository, issue *issues_model.Issue, reviewType issues_model.ReviewType, content, commitID string, attachmentUUIDs []string) (*issues_model.Review, *issues_model.Comment, error) {
|
func SubmitReview(ctx context.Context, doer *user_model.User, gitRepo *git.Repository, issue *issues_model.Issue, reviewType issues_model.ReviewType, content, commitID string, attachmentUUIDs []string) (*issues_model.Review, *issues_model.Comment, error) {
|
||||||
pr, err := issue.GetPullRequest()
|
if err := issue.LoadPullRequest(ctx); err != nil {
|
||||||
if err != nil {
|
|
||||||
return nil, nil, err
|
return nil, nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pr := issue.PullRequest
|
||||||
var stale bool
|
var stale bool
|
||||||
if reviewType != issues_model.ReviewTypeApprove && reviewType != issues_model.ReviewTypeReject {
|
if reviewType != issues_model.ReviewTypeApprove && reviewType != issues_model.ReviewTypeReject {
|
||||||
stale = false
|
stale = false
|
||||||
|
|
|
@ -1,15 +1,15 @@
|
||||||
{{if .IsPull}}
|
{{if .IsPull}}
|
||||||
{{if and .PullRequest .PullRequest.HasMerged}}
|
{{if not .PullRequest}}
|
||||||
{{svg "octicon-git-merge" 16 "text purple"}}
|
No PullRequest
|
||||||
{{else if and .GetPullRequest .GetPullRequest.HasMerged}}
|
|
||||||
{{svg "octicon-git-merge" 16 "text purple"}}
|
|
||||||
{{else}}
|
{{else}}
|
||||||
{{if .IsClosed}}
|
{{if .IsClosed}}
|
||||||
{{svg "octicon-git-pull-request" 16 "text red"}}
|
{{if .PullRequest.HasMerged}}
|
||||||
|
{{svg "octicon-git-merge" 16 "text purple"}}
|
||||||
{{else}}
|
{{else}}
|
||||||
{{if and .PullRequest (.PullRequest.IsWorkInProgress ctx)}}
|
{{svg "octicon-git-pull-request" 16 "text red"}}
|
||||||
{{svg "octicon-git-pull-request-draft" 16 "text grey"}}
|
{{end}}
|
||||||
{{else if and .GetPullRequest (.GetPullRequest.IsWorkInProgress ctx)}}
|
{{else}}
|
||||||
|
{{if .PullRequest.IsWorkInProgress ctx}}
|
||||||
{{svg "octicon-git-pull-request-draft" 16 "text grey"}}
|
{{svg "octicon-git-pull-request-draft" 16 "text grey"}}
|
||||||
{{else}}
|
{{else}}
|
||||||
{{svg "octicon-git-pull-request" 16 "text green"}}
|
{{svg "octicon-git-pull-request" 16 "text green"}}
|
||||||
|
|
|
@ -424,8 +424,8 @@ func TestConflictChecking(t *testing.T) {
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
|
|
||||||
issue := unittest.AssertExistsAndLoadBean(t, &issues_model.Issue{Title: "PR with conflict!"})
|
issue := unittest.AssertExistsAndLoadBean(t, &issues_model.Issue{Title: "PR with conflict!"})
|
||||||
conflictingPR, err := issues_model.GetPullRequestByIssueID(db.DefaultContext, issue.ID)
|
assert.NoError(t, issue.LoadPullRequest(db.DefaultContext))
|
||||||
assert.NoError(t, err)
|
conflictingPR := issue.PullRequest
|
||||||
|
|
||||||
// Ensure conflictedFiles is populated.
|
// Ensure conflictedFiles is populated.
|
||||||
assert.Len(t, conflictingPR.ConflictedFiles, 1)
|
assert.Len(t, conflictingPR.ConflictedFiles, 1)
|
||||||
|
|
|
@ -175,8 +175,7 @@ func createOutdatedPR(t *testing.T, actor, forkOrg *user_model.User) *issues_mod
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
|
|
||||||
issue := unittest.AssertExistsAndLoadBean(t, &issues_model.Issue{Title: "Test Pull -to-update-"})
|
issue := unittest.AssertExistsAndLoadBean(t, &issues_model.Issue{Title: "Test Pull -to-update-"})
|
||||||
pr, err := issues_model.GetPullRequestByIssueID(db.DefaultContext, issue.ID)
|
assert.NoError(t, issue.LoadPullRequest(db.DefaultContext))
|
||||||
assert.NoError(t, err)
|
|
||||||
|
|
||||||
return pr
|
return issue.PullRequest
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue