From cfa6db75ae694756dd553d15e1ae7404335186bb Mon Sep 17 00:00:00 2001
From: a1012112796 <1012112796@qq.com>
Date: Sat, 20 Jun 2020 18:16:26 +0800
Subject: [PATCH 01/24] Add team support for review request
Block #11355
Signed-off-by: a1012112796 <1012112796@qq.com>
---
models/issue_comment.go | 20 ++
models/repo.go | 18 +-
models/review.go | 195 +++++++++++++++++-
routers/repo/issue.go | 81 +++++++-
services/issue/assignee.go | 33 ++-
.../repo/issue/view_content/comments.tmpl | 18 +-
templates/repo/issue/view_content/pull.tmpl | 11 +-
.../repo/issue/view_content/sidebar.tmpl | 41 +++-
8 files changed, 400 insertions(+), 17 deletions(-)
diff --git a/models/issue_comment.go b/models/issue_comment.go
index 452afc79f00df..37f09ca3de211 100644
--- a/models/issue_comment.go
+++ b/models/issue_comment.go
@@ -129,6 +129,7 @@ type Comment struct {
AssigneeID int64
RemovedAssignee bool
Assignee *User `xorm:"-"`
+ TeamAssignee *Team `xorm:"-"`
ResolveDoerID int64
ResolveDoer *User `xorm:"-"`
OldTitle string
@@ -465,6 +466,25 @@ func (c *Comment) LoadAssigneeUser() error {
}
c.Assignee = NewGhostUser()
}
+ } else {
+ if err = c.LoadIssue(); err != nil {
+ return err
+ }
+
+ if err = c.Issue.LoadRepo(); err != nil {
+ return err
+ }
+
+ if err = c.Issue.Repo.GetOwner(); err != nil {
+ return err
+ }
+
+ if c.Issue.Repo.Owner.IsOrganization() {
+ c.TeamAssignee, err = GetTeamByID(-c.AssigneeID)
+ if err != nil && !IsErrTeamNotExist(err) {
+ return err
+ }
+ }
}
return nil
}
diff --git a/models/repo.go b/models/repo.go
index 3b874f3359af6..fc38b7240f683 100644
--- a/models/repo.go
+++ b/models/repo.go
@@ -685,9 +685,9 @@ func (repo *Repository) getReviewersPublic(e Engine, doerID, posterID int64) (_
return users, nil
}
-func (repo *Repository) getReviewers(e Engine, doerID, posterID int64) (users []*User, err error) {
+func (repo *Repository) getReviewers(e Engine, doerID, posterID int64) (users []*User, teams []*Team, err error) {
if err = repo.getOwner(e); err != nil {
- return nil, err
+ return nil, nil, err
}
if repo.IsPrivate ||
@@ -696,6 +696,18 @@ func (repo *Repository) getReviewers(e Engine, doerID, posterID int64) (users []
} else {
users, err = repo.getReviewersPublic(x, doerID, posterID)
}
+
+ if repo.Owner.IsOrganization() {
+ teams, err = GetTeamsWithAccessToRepo(repo.OwnerID, repo.ID, AccessModeRead)
+ if err != nil {
+ return nil, nil, err
+ }
+
+ for _, team := range teams {
+ team.ID = -team.ID
+ }
+ }
+
return
}
@@ -704,7 +716,7 @@ func (repo *Repository) getReviewers(e Engine, doerID, posterID int64) (users []
// but for public rpo, that return all users that have write access or higher to the repository,
// and all repo watchers.
// TODO: may be we should hava a busy choice for users to block review request to them.
-func (repo *Repository) GetReviewers(doerID, posterID int64) (_ []*User, err error) {
+func (repo *Repository) GetReviewers(doerID, posterID int64) (_ []*User, _ []*Team, err error) {
return repo.getReviewers(x, doerID, posterID)
}
diff --git a/models/review.go b/models/review.go
index 522fe5886c372..0dafa58d6af56 100644
--- a/models/review.go
+++ b/models/review.go
@@ -8,6 +8,7 @@ import (
"fmt"
"strings"
+ "code.gitea.io/gitea/modules/base"
"code.gitea.io/gitea/modules/timeutil"
"xorm.io/builder"
@@ -53,6 +54,7 @@ type Review struct {
ID int64 `xorm:"pk autoincr"`
Type ReviewType
Reviewer *User `xorm:"-"`
+ ReviewerTeam *Team `xorm:"-"`
ReviewerID int64 `xorm:"index"`
OriginalAuthor string
OriginalAuthorID int64
@@ -101,6 +103,11 @@ func (r *Review) loadReviewer(e Engine) (err error) {
if r.Reviewer != nil || r.ReviewerID == 0 {
return nil
}
+
+ if r.ReviewerID < 0 {
+ r.ReviewerTeam, err = getTeamByID(e, -r.ReviewerID)
+ return
+ }
r.Reviewer, err = getUserByID(e, r.ReviewerID)
return
}
@@ -218,6 +225,30 @@ func isOfficialReviewer(e Engine, issue *Issue, reviewer *User) (bool, error) {
return pr.ProtectedBranch.isUserOfficialReviewer(e, reviewer)
}
+// IsOfficialReviewerTeam check if reviewer in this team can make official reviews in issue (counts towards required approvals)
+func IsOfficialReviewerTeam(issue *Issue, team *Team) (bool, error) {
+ return isOfficialReviewerTeam(x, issue, team)
+}
+
+func isOfficialReviewerTeam(e Engine, issue *Issue, team *Team) (bool, error) {
+ pr, err := getPullRequestByIssueID(e, issue.ID)
+ if err != nil {
+ return false, err
+ }
+ if err = pr.loadProtectedBranch(e); err != nil {
+ return false, err
+ }
+ if pr.ProtectedBranch == nil {
+ return false, nil
+ }
+
+ if !pr.ProtectedBranch.EnableApprovalsWhitelist {
+ return team.Authorize >= AccessModeWrite, nil
+ }
+
+ return base.Int64sContains(pr.ProtectedBranch.ApprovalsWhitelistTeamIDs, team.ID), nil
+}
+
func createReview(e Engine, opts CreateReviewOptions) (*Review, error) {
review := &Review{
Type: opts.Type,
@@ -373,6 +404,29 @@ func SubmitReview(doer *User, issue *Issue, reviewType ReviewType, content, comm
return nil, nil, err
}
+ // try to remove team review request if need
+ if issue.Repo.Owner.IsOrganization() && (reviewType == ReviewTypeApprove || reviewType == ReviewTypeReject) {
+ teamReviewRequests := make([]*Review, 0, 10)
+ if err = sess.SQL("SELECT * FROM review WHERE reviewer_id < 0 AND type = ?", ReviewTypeRequest).Find(&teamReviewRequests); err != nil {
+ return nil, nil, err
+ }
+
+ if len(teamReviewRequests) > 0 {
+ for _, teamReviewRequest := range teamReviewRequests {
+ ok := false
+ if ok, err = isTeamMember(sess, issue.Repo.OwnerID, -teamReviewRequest.ReviewerID, doer.ID); err != nil {
+ return nil, nil, err
+ }
+
+ if ok {
+ if _, err := sess.Delete(teamReviewRequest); err != nil {
+ return nil, nil, err
+ }
+ }
+ }
+ }
+ }
+
comm.Review = review
return review, comm, sess.Commit()
}
@@ -397,7 +451,7 @@ func GetReviewersByIssueID(issueID int64) (reviews []*Review, err error) {
// Load reviewer and skip if user is deleted
for _, review := range reviewsUnfiltered {
if err = review.loadReviewer(sess); err != nil {
- if !IsErrUserNotExist(err) {
+ if !IsErrUserNotExist(err) && !IsErrTeamNotExist(err) {
return nil, err
}
} else {
@@ -482,7 +536,7 @@ func InsertReviews(reviews []*Review) error {
}
// AddReviewRequest add a review request from one reviewer
-func AddReviewRequest(issue *Issue, reviewer *User, doer *User) (comment *Comment, err error) {
+func AddReviewRequest(issue *Issue, reviewer, doer *User) (comment *Comment, err error) {
review, err := GetReviewerByIssueIDAndUserID(issue.ID, reviewer.ID)
if err != nil {
return
@@ -549,7 +603,7 @@ func AddReviewRequest(issue *Issue, reviewer *User, doer *User) (comment *Commen
}
//RemoveReviewRequest remove a review request from one reviewer
-func RemoveReviewRequest(issue *Issue, reviewer *User, doer *User) (comment *Comment, err error) {
+func RemoveReviewRequest(issue *Issue, reviewer, doer *User) (comment *Comment, err error) {
review, err := GetReviewerByIssueIDAndUserID(issue.ID, reviewer.ID)
if err != nil {
return
@@ -612,6 +666,141 @@ func RemoveReviewRequest(issue *Issue, reviewer *User, doer *User) (comment *Com
return comment, sess.Commit()
}
+// AddTeamReviewRequest add a review request from one team
+func AddTeamReviewRequest(issue *Issue, reviewer *Team, doer *User) (comment *Comment, err error) {
+ review, err := GetReviewerByIssueIDAndUserID(issue.ID, -reviewer.ID)
+ if err != nil {
+ return
+ }
+
+ // skip it when reviewer hase been request to review
+ if review != nil && review.Type == ReviewTypeRequest {
+ return nil, nil
+ }
+
+ sess := x.NewSession()
+ defer sess.Close()
+ if err := sess.Begin(); err != nil {
+ return nil, err
+ }
+
+ var official bool
+ official, err = isOfficialReviewerTeam(sess, issue, reviewer)
+
+ if err != nil {
+ return nil, err
+ }
+
+ if !official {
+ official, err = isOfficialReviewer(sess, issue, doer)
+
+ if err != nil {
+ return nil, err
+ }
+ }
+
+ if official {
+ if _, err := sess.Exec("UPDATE `review` SET official=? WHERE issue_id=? AND reviewer_id=?", false, issue.ID, -reviewer.ID); err != nil {
+ return nil, err
+ }
+ }
+
+ _, err = createReview(sess, CreateReviewOptions{
+ Type: ReviewTypeRequest,
+ Issue: issue,
+ Reviewer: &User{ID: -reviewer.ID},
+ Official: official,
+ Stale: false,
+ })
+
+ if err != nil {
+ return
+ }
+
+ comment, err = createComment(sess, &CreateCommentOptions{
+ Type: CommentTypeReviewRequest,
+ Doer: doer,
+ Repo: issue.Repo,
+ Issue: issue,
+ RemovedAssignee: false, // Use RemovedAssignee as !isRequest
+ AssigneeID: -reviewer.ID, // Use AssigneeID as reviewer ID
+ })
+
+ if err != nil {
+ return nil, err
+ }
+
+ return comment, sess.Commit()
+}
+
+//RemoveTeamReviewRequest remove a review request from one team
+func RemoveTeamReviewRequest(issue *Issue, reviewer *Team, doer *User) (comment *Comment, err error) {
+ review, err := GetReviewerByIssueIDAndUserID(issue.ID, -reviewer.ID)
+ if err != nil {
+ return
+ }
+
+ if review.Type != ReviewTypeRequest {
+ return nil, nil
+ }
+
+ sess := x.NewSession()
+ defer sess.Close()
+ if err := sess.Begin(); err != nil {
+ return nil, err
+ }
+
+ _, err = sess.Delete(review)
+ if err != nil {
+ return nil, err
+ }
+
+ var official bool
+ official, err = isOfficialReviewerTeam(sess, issue, reviewer)
+ if err != nil {
+ return
+ }
+
+ if official {
+ // recalculate which is the latest official review from that team
+ var review *Review
+
+ review, err = getReviewerByIssueIDAndUserID(sess, issue.ID, -reviewer.ID)
+ if err != nil {
+ return nil, err
+ }
+
+ if review != nil {
+ if _, err := sess.Exec("UPDATE `review` SET official=? WHERE id=?", true, review.ID); err != nil {
+ return nil, err
+ }
+ }
+ }
+
+ if err != nil {
+ return nil, err
+ }
+
+ if doer == nil {
+ return nil, sess.Commit()
+ }
+
+ comment, err = createComment(sess, &CreateCommentOptions{
+ Type: CommentTypeReviewRequest,
+ Doer: doer,
+ Repo: issue.Repo,
+ Issue: issue,
+ RemovedAssignee: true, // Use RemovedAssignee as !isRequest
+ AssigneeID: -reviewer.ID, // Use AssigneeID as reviewer ID
+ })
+
+ if err != nil {
+ return nil, err
+ }
+
+ return comment, sess.Commit()
+}
+
// MarkConversation Add or remove Conversation mark for a code comment
func MarkConversation(comment *Comment, doer *User, isResolve bool) (err error) {
if comment.Type != CommentTypeCode {
diff --git a/routers/repo/issue.go b/routers/repo/issue.go
index 181ed59a8c1a8..1e3fdcbde8bc0 100644
--- a/routers/repo/issue.go
+++ b/routers/repo/issue.go
@@ -390,7 +390,7 @@ func RetrieveRepoMilestonesAndAssignees(ctx *context.Context, repo *models.Repos
// RetrieveRepoReviewers find all reviewers of a repository
func RetrieveRepoReviewers(ctx *context.Context, repo *models.Repository, issuePosterID int64) {
var err error
- ctx.Data["Reviewers"], err = repo.GetReviewers(ctx.User.ID, issuePosterID)
+ ctx.Data["Reviewers"], ctx.Data["TeamReviewers"], err = repo.GetReviewers(ctx.User.ID, issuePosterID)
if err != nil {
ctx.ServerError("GetReviewers", err)
return
@@ -1395,6 +1395,46 @@ func isLegalReviewRequest(reviewer, doer *models.User, isAdd bool, issue *models
return nil
}
+func isLegalTeamReviewRequest(reviewer *models.Team, doer *models.User, isAdd bool, issue *models.Issue) error {
+ if doer.IsOrganization() {
+ return fmt.Errorf("Organization can't be doer to add reviewer [user_id: %d, repo_id: %d]", doer.ID, issue.PullRequest.BaseRepo.ID)
+ }
+
+ permDoer, err := models.GetUserRepoPermission(issue.Repo, doer)
+ if err != nil {
+ return err
+ }
+
+ var pemResult bool
+ if isAdd {
+ if issue.Repo.IsPrivate {
+ pemResult = models.HasTeamRepo(reviewer.OrgID, reviewer.ID, issue.RepoID)
+
+ if !pemResult {
+ return fmt.Errorf("Reviewer can't read [team_id: %d, repo_name: %s]", reviewer.ID, issue.Repo.Name)
+ }
+ }
+
+ pemResult = permDoer.CanAccessAny(models.AccessModeWrite, models.UnitTypePullRequests)
+ if !pemResult {
+ pemResult, err = models.IsOfficialReviewer(issue, doer)
+ if err != nil {
+ return err
+ }
+ if !pemResult {
+ return fmt.Errorf("Doer can't choose reviewer [user_id: %d, repo_name: %s, issue_id: %d]", doer.ID, issue.Repo.Name, issue.ID)
+ }
+ }
+ } else {
+ pemResult = permDoer.IsAdmin()
+ if !pemResult {
+ return fmt.Errorf("Doer is not admin [user_id: %d, repo_name: %s]", doer.ID, issue.Repo.Name)
+ }
+ }
+
+ return nil
+}
+
// updatePullReviewRequest change pull's request reviewers
func updatePullReviewRequest(ctx *context.Context) {
issues := getActionIssues(ctx)
@@ -1413,6 +1453,45 @@ func updatePullReviewRequest(ctx *context.Context) {
for _, issue := range issues {
if issue.IsPull {
+ if reviewID < 0 {
+ if err := issue.LoadRepo(); err != nil {
+ ctx.ServerError("issue.LoadRepo", err)
+ return
+ }
+
+ if err := issue.Repo.GetOwner(); err != nil {
+ ctx.ServerError("issue.Repo.GetOwner", err)
+ return
+ }
+
+ if issue.Repo.Owner.IsOrganization() {
+ team, err := models.GetTeamByID(-reviewID)
+ if err != nil {
+ ctx.ServerError("models.GetTeamByID", err)
+ return
+ }
+
+ if team.OrgID != issue.Repo.OwnerID {
+ ctx.Status(403)
+ return
+ }
+
+ err = isLegalTeamReviewRequest(team, ctx.User, event == "add", issue)
+ if err != nil {
+ ctx.ServerError("isLegalTeamReviewRequest", err)
+ return
+ }
+
+ err = issue_service.TeamReviewRequest(issue, ctx.User, team, event == "add")
+ if err != nil {
+ ctx.ServerError("ReviewRequest", err)
+ return
+ }
+
+ continue
+ }
+ }
+
reviewer, err := models.GetUserByID(reviewID)
if err != nil {
ctx.ServerError("GetUserByID", err)
diff --git a/services/issue/assignee.go b/services/issue/assignee.go
index d63c7bf032e65..ec17f9e7febea 100644
--- a/services/issue/assignee.go
+++ b/services/issue/assignee.go
@@ -52,7 +52,7 @@ func ToggleAssignee(issue *models.Issue, doer *models.User, assigneeID int64) (r
return
}
-// ReviewRequest add or remove a review for this PR, and make comment for it.
+// ReviewRequest add or remove a review request from a user for this PR, and make comment for it.
func ReviewRequest(issue *models.Issue, doer *models.User, reviewer *models.User, isAdd bool) (err error) {
var comment *models.Comment
if isAdd {
@@ -71,3 +71,34 @@ func ReviewRequest(issue *models.Issue, doer *models.User, reviewer *models.User
return nil
}
+
+// TeamReviewRequest add or remove a review request from a team for this PR, and make comment for it.
+func TeamReviewRequest(issue *models.Issue, doer *models.User, reviewer *models.Team, isAdd bool) (err error) {
+ var comment *models.Comment
+ if isAdd {
+ comment, err = models.AddTeamReviewRequest(issue, reviewer, doer)
+ } else {
+ comment, err = models.RemoveTeamReviewRequest(issue, reviewer, doer)
+ }
+
+ if err != nil {
+ return
+ }
+
+ if comment != nil && isAdd {
+ // notify all user in this team
+ if err = comment.LoadIssue(); err != nil {
+ return
+ }
+ err = reviewer.GetMembers(&models.SearchMembersOptions{})
+ for _, member := range reviewer.Members {
+ if member.ID == comment.Issue.PosterID {
+ continue
+ }
+ comment.AssigneeID = member.ID
+ notification.NotifyPullReviewRequest(doer, issue, member, isAdd, comment)
+ }
+ }
+
+ return nil
+}
diff --git a/templates/repo/issue/view_content/comments.tmpl b/templates/repo/issue/view_content/comments.tmpl
index b3ef258a6c863..7be149d936cf7 100644
--- a/templates/repo/issue/view_content/comments.tmpl
+++ b/templates/repo/issue/view_content/comments.tmpl
@@ -590,14 +590,22 @@
{{.Poster.GetDisplayName}}
- {{if .RemovedAssignee}}
- {{if eq .PosterID .AssigneeID}}
- {{$.i18n.Tr "repo.issues.review.remove_review_request_self" $createdStr | Safe}}
+ {{if (gt .AssigneeID 0)}}
+ {{if .RemovedAssignee}}
+ {{if eq .PosterID .AssigneeID}}
+ {{$.i18n.Tr "repo.issues.review.remove_review_request_self" $createdStr | Safe}}
+ {{else}}
+ {{$.i18n.Tr "repo.issues.review.remove_review_request" (.Assignee.GetDisplayName|Escape) $createdStr | Safe}}
+ {{end}}
{{else}}
- {{$.i18n.Tr "repo.issues.review.remove_review_request" (.Assignee.GetDisplayName|Escape) $createdStr | Safe}}
+ {{$.i18n.Tr "repo.issues.review.add_review_request" (.Assignee.GetDisplayName|Escape) $createdStr | Safe}}
{{end}}
{{else}}
- {{$.i18n.Tr "repo.issues.review.add_review_request" (.Assignee.GetDisplayName|Escape) $createdStr | Safe}}
+ {{if .RemovedAssignee}}
+ {{$.i18n.Tr "repo.issues.review.remove_review_request" (.TeamAssignee.Name|Escape) $createdStr | Safe}}
+ {{else}}
+ {{$.i18n.Tr "repo.issues.review.add_review_request" (.TeamAssignee.Name|Escape) $createdStr | Safe}}
+ {{end}}
{{end}}
diff --git a/templates/repo/issue/view_content/pull.tmpl b/templates/repo/issue/view_content/pull.tmpl
index dc897bd7b9371..37aa2c9a86a0c 100644
--- a/templates/repo/issue/view_content/pull.tmpl
+++ b/templates/repo/issue/view_content/pull.tmpl
@@ -8,10 +8,17 @@
+ {{if (gt .ReviewerID 0)}}
-
{{.Reviewer.Name}}
+ {{end}}
+
+ {{if (gt .ReviewerID 0)}}
+ {{.Reviewer.Name}}
+ {{else}}
+ {{$.Issue.Repo.OwnerName}}/{{.ReviewerTeam.Name}}
+ {{end}}
{{if eq .Type 1}}
{{$.i18n.Tr "repo.issues.review.approve" $createdStr | Safe}}
{{else if eq .Type 2}}
@@ -39,7 +46,7 @@
{{$canChoose := false}}
{{if eq .Type 4}}
- {{if or (eq .ReviewerID $.SignedUserID) $.Permission.IsAdmin}}
+ {{if or (and (eq .ReviewerID $.SignedUserID) (gt .ReviewerID 0)) $.Permission.IsAdmin}}
{{$canChoose = true}}
{{end}}
{{else}}
diff --git a/templates/repo/issue/view_content/sidebar.tmpl b/templates/repo/issue/view_content/sidebar.tmpl
index f0c7f69b5857e..beeeb149da353 100644
--- a/templates/repo/issue/view_content/sidebar.tmpl
+++ b/templates/repo/issue/view_content/sidebar.tmpl
@@ -51,6 +51,39 @@
{{end}}
+ {{if .TeamReviewers}}
+ {{range .TeamReviewers}}
+ {{$ReviewerID := .ID}}
+ {{$checked := false}}
+ {{$canChoose := false}}
+ {{$notReviewed := true}}
+
+ {{range $.PullReviewers}}
+ {{if eq .ReviewerID $ReviewerID }}
+ {{$notReviewed = false }}
+ {{if eq .Type 4 }}
+ {{$checked = true}}
+ {{if $.Permission.IsAdmin}}
+ {{$canChoose = true}}
+ {{end}}
+ {{else}}
+ {{$canChoose = true}}
+ {{end}}
+ {{end}}
+ {{end}}
+
+ {{ if $notReviewed}}
+ {{$canChoose = true}}
+ {{end}}
+
+
+ {{svg "octicon-check" 16}}
+
+ {{$.Issue.Repo.OwnerName}}/{{.Name}}
+
+
+ {{end}}
+ {{end}}
@@ -59,7 +92,11 @@
{{range .PullReviewers}}
-
{{.Reviewer.GetDisplayName}}
+ {{if (gt .ReviewerID 0)}}
+
{{.Reviewer.GetDisplayName}}
+ {{else}}
+
{{$.Issue.Repo.OwnerName}}/{{.ReviewerTeam.Name}}
+ {{end}}
Date: Thu, 17 Sep 2020 17:21:52 +0800
Subject: [PATCH 04/24] Apply suggestion from review @lafriks
---
models/issue_comment.go | 15 +-
models/migrations/migrations.go | 2 +
models/migrations/v152.go | 29 ++++
models/review.go | 154 ++++++++++++------
routers/repo/issue.go | 4 +-
.../repo/issue/view_content/comments.tmpl | 4 +-
.../repo/issue/view_content/sidebar.tmpl | 2 +-
7 files changed, 151 insertions(+), 59 deletions(-)
create mode 100644 models/migrations/v152.go
diff --git a/models/issue_comment.go b/models/issue_comment.go
index 77a759913351c..270a10e240141 100644
--- a/models/issue_comment.go
+++ b/models/issue_comment.go
@@ -137,7 +137,8 @@ type Comment struct {
AssigneeID int64
RemovedAssignee bool
Assignee *User `xorm:"-"`
- TeamAssignee *Team `xorm:"-"`
+ AssigneeTeamID int64 `xorm:"NOT NULL DEFAULT 0"`
+ AssigneeTeam *Team `xorm:"-"`
ResolveDoerID int64
ResolveDoer *User `xorm:"-"`
OldTitle string
@@ -488,11 +489,11 @@ func (c *Comment) UpdateAttachments(uuids []string) error {
return sess.Commit()
}
-// LoadAssigneeUser if comment.Type is CommentTypeAssignees, then load assignees
-func (c *Comment) LoadAssigneeUser() error {
+// LoadAssigneeUserAndTeam if comment.Type is CommentTypeAssignees, then load assignees
+func (c *Comment) LoadAssigneeUserAndTeam() error {
var err error
- if c.AssigneeID > 0 {
+ if c.AssigneeID > 0 && c.Assignee == nil {
c.Assignee, err = getUserByID(x, c.AssigneeID)
if err != nil {
if !IsErrUserNotExist(err) {
@@ -500,7 +501,7 @@ func (c *Comment) LoadAssigneeUser() error {
}
c.Assignee = NewGhostUser()
}
- } else {
+ } else if c.AssigneeTeamID > 0 && c.AssigneeTeam == nil {
if err = c.LoadIssue(); err != nil {
return err
}
@@ -514,7 +515,7 @@ func (c *Comment) LoadAssigneeUser() error {
}
if c.Issue.Repo.Owner.IsOrganization() {
- c.TeamAssignee, err = GetTeamByID(-c.AssigneeID)
+ c.AssigneeTeam, err = GetTeamByID(c.AssigneeTeamID)
if err != nil && !IsErrTeamNotExist(err) {
return err
}
@@ -705,6 +706,7 @@ func createComment(e *xorm.Session, opts *CreateCommentOptions) (_ *Comment, err
ProjectID: opts.ProjectID,
RemovedAssignee: opts.RemovedAssignee,
AssigneeID: opts.AssigneeID,
+ AssigneeTeamID: opts.AssigneeTeamID,
CommitID: opts.CommitID,
CommitSHA: opts.CommitSHA,
Line: opts.LineNum,
@@ -869,6 +871,7 @@ type CreateCommentOptions struct {
OldProjectID int64
ProjectID int64
AssigneeID int64
+ AssigneeTeamID int64
RemovedAssignee bool
OldTitle string
NewTitle string
diff --git a/models/migrations/migrations.go b/models/migrations/migrations.go
index 5317cc5743d5d..c197902138aec 100644
--- a/models/migrations/migrations.go
+++ b/models/migrations/migrations.go
@@ -237,6 +237,8 @@ var migrations = []Migration{
NewMigration("add primary key to repo_topic", addPrimaryKeyToRepoTopic),
// v151 -> v152
NewMigration("set default password algorithm to Argon2", setDefaultPasswordToArgon2),
+ // v152 -> v153
+ NewMigration("add Team review request support", addTeamReviewRequestSupport),
}
// GetCurrentDBVersion returns the current db version
diff --git a/models/migrations/v152.go b/models/migrations/v152.go
new file mode 100644
index 0000000000000..4ef8898b1ce86
--- /dev/null
+++ b/models/migrations/v152.go
@@ -0,0 +1,29 @@
+// Copyright 2020 The Gitea Authors. All rights reserved.
+// Use of this source code is governed by a MIT-style
+// license that can be found in the LICENSE file.
+
+package migrations
+
+import (
+ "xorm.io/xorm"
+)
+
+func addTeamReviewRequestSupport(x *xorm.Engine) error {
+ type Review struct {
+ ReviewerTeamID int64 `xorm:"NOT NULL DEFAULT 0"`
+ }
+
+ type Comment struct {
+ AssigneeTeamID int64 `xorm:"NOT NULL DEFAULT 0"`
+ }
+
+ if err := x.Sync2(new(Review)); err != nil {
+ return err
+ }
+
+ if err := x.Sync2(new(Comment)); err != nil {
+ return err
+ }
+
+ return nil
+}
diff --git a/models/review.go b/models/review.go
index b51180e5abf86..9c9a56131bee3 100644
--- a/models/review.go
+++ b/models/review.go
@@ -54,8 +54,9 @@ type Review struct {
ID int64 `xorm:"pk autoincr"`
Type ReviewType
Reviewer *User `xorm:"-"`
- ReviewerTeam *Team `xorm:"-"`
ReviewerID int64 `xorm:"index"`
+ ReviewerTeamID int64 `xorm:"NOT NULL DEFAULT 0"`
+ ReviewerTeam *Team `xorm:"-"`
OriginalAuthor string
OriginalAuthorID int64
Issue *Issue `xorm:"-"`
@@ -100,15 +101,19 @@ func (r *Review) loadIssue(e Engine) (err error) {
}
func (r *Review) loadReviewer(e Engine) (err error) {
- if r.Reviewer != nil || r.ReviewerID == 0 {
- return nil
+ if r.ReviewerID == 0 || r.Reviewer != nil {
+ return
}
+ r.Reviewer, err = getUserByID(e, r.ReviewerID)
+ return
+}
- if r.ReviewerID < 0 {
- r.ReviewerTeam, err = getTeamByID(e, -r.ReviewerID)
+func (r *Review) loadReviewerTeam(e Engine) (err error) {
+ if r.ReviewerTeamID == 0 || r.ReviewerTeam != nil {
return
}
- r.Reviewer, err = getUserByID(e, r.ReviewerID)
+
+ r.ReviewerTeam, err = getTeamByID(e, r.ReviewerTeamID)
return
}
@@ -127,6 +132,9 @@ func (r *Review) loadAttributes(e Engine) (err error) {
if err = r.loadReviewer(e); err != nil {
return
}
+ if err = r.loadReviewerTeam(e); err != nil {
+ return
+ }
return
}
@@ -196,13 +204,14 @@ func FindReviews(opts FindReviewOptions) ([]*Review, error) {
// CreateReviewOptions represent the options to create a review. Type, Issue and Reviewer are required.
type CreateReviewOptions struct {
- Content string
- Type ReviewType
- Issue *Issue
- Reviewer *User
- Official bool
- CommitID string
- Stale bool
+ Content string
+ Type ReviewType
+ Issue *Issue
+ Reviewer *User
+ ReviewerTeam *Team
+ Official bool
+ CommitID string
+ Stale bool
}
// IsOfficialReviewer check if reviewer can make official reviews in issue (counts towards required approvals)
@@ -251,15 +260,20 @@ func isOfficialReviewerTeam(e Engine, issue *Issue, team *Team) (bool, error) {
func createReview(e Engine, opts CreateReviewOptions) (*Review, error) {
review := &Review{
- Type: opts.Type,
- Issue: opts.Issue,
- IssueID: opts.Issue.ID,
- Reviewer: opts.Reviewer,
- ReviewerID: opts.Reviewer.ID,
- Content: opts.Content,
- Official: opts.Official,
- CommitID: opts.CommitID,
- Stale: opts.Stale,
+ Type: opts.Type,
+ Issue: opts.Issue,
+ IssueID: opts.Issue.ID,
+ Reviewer: opts.Reviewer,
+ ReviewerTeam: opts.ReviewerTeam,
+ Content: opts.Content,
+ Official: opts.Official,
+ CommitID: opts.CommitID,
+ Stale: opts.Stale,
+ }
+ if opts.Reviewer != nil {
+ review.ReviewerID = opts.Reviewer.ID
+ } else {
+ review.ReviewerTeamID = opts.ReviewerTeam.ID
}
if _, err := e.Insert(review); err != nil {
return nil, err
@@ -407,14 +421,14 @@ func SubmitReview(doer *User, issue *Issue, reviewType ReviewType, content, comm
// try to remove team review request if need
if issue.Repo.Owner.IsOrganization() && (reviewType == ReviewTypeApprove || reviewType == ReviewTypeReject) {
teamReviewRequests := make([]*Review, 0, 10)
- if err = sess.SQL("SELECT * FROM review WHERE reviewer_id < 0 AND type = ?", ReviewTypeRequest).Find(&teamReviewRequests); err != nil {
+ if err = sess.SQL("SELECT * FROM review WHERE reviewer_team_id > 0 AND type = ?", ReviewTypeRequest).Find(&teamReviewRequests); err != nil {
return nil, nil, err
}
if len(teamReviewRequests) > 0 {
for _, teamReviewRequest := range teamReviewRequests {
ok := false
- if ok, err = isTeamMember(sess, issue.Repo.OwnerID, -teamReviewRequest.ReviewerID, doer.ID); err != nil {
+ if ok, err = isTeamMember(sess, issue.Repo.OwnerID, teamReviewRequest.ReviewerTeamID, doer.ID); err != nil {
return nil, nil, err
}
@@ -442,7 +456,7 @@ func GetReviewersByIssueID(issueID int64) (reviews []*Review, err error) {
}
// Get latest review of each reviwer, sorted in order they were made
- if err := sess.SQL("SELECT * FROM review WHERE id IN (SELECT max(id) as id FROM review WHERE issue_id = ? AND type in (?, ?, ?) GROUP BY issue_id, reviewer_id) ORDER BY review.updated_unix ASC",
+ if err := sess.SQL("SELECT * FROM review WHERE id IN (SELECT max(id) as id FROM review WHERE issue_id = ? AND reviewer_team_id = 0 AND type in (?, ?, ?) GROUP BY issue_id, reviewer_id) ORDER BY review.updated_unix ASC",
issueID, ReviewTypeApprove, ReviewTypeReject, ReviewTypeRequest).
Find(&reviewsUnfiltered); err != nil {
return nil, err
@@ -451,10 +465,32 @@ func GetReviewersByIssueID(issueID int64) (reviews []*Review, err error) {
// Load reviewer and skip if user is deleted
for _, review := range reviewsUnfiltered {
if err = review.loadReviewer(sess); err != nil {
- if !IsErrUserNotExist(err) && !IsErrTeamNotExist(err) {
+ if !IsErrUserNotExist(err) {
+ return nil, err
+ }
+ } else {
+ reviews = append(reviews, review)
+ }
+ }
+
+ teamReviewRequests := make([]*Review, 0, 5)
+ if err := sess.SQL("SELECT * FROM review WHERE id IN (SELECT max(id) as id FROM review WHERE issue_id = ? AND reviewer_team_id <> 0 GROUP BY issue_id, reviewer_team_id) ORDER BY review.updated_unix ASC",
+ issueID).
+ Find(&teamReviewRequests); err != nil {
+ return nil, err
+ }
+
+ if len(teamReviewRequests) == 0 {
+ return reviews, nil
+ }
+
+ for _, review := range teamReviewRequests {
+ if err = review.loadReviewerTeam(sess); err != nil {
+ if !IsErrTeamNotExist(err) {
return nil, err
}
} else {
+ review.ReviewerID = -review.ReviewerTeamID
reviews = append(reviews, review)
}
}
@@ -479,6 +515,28 @@ func getReviewerByIssueIDAndUserID(e Engine, issueID, userID int64) (review *Rev
return
}
+// GetTeamReviewerByIssueIDAndTeamID get the latest review requst of reviewer team for a pull request
+func GetTeamReviewerByIssueIDAndTeamID(issueID, teamID int64) (review *Review, err error) {
+ return getTeamReviewerByIssueIDAndTeamID(x, issueID, teamID)
+}
+
+func getTeamReviewerByIssueIDAndTeamID(e Engine, issueID, teamID int64) (review *Review, err error) {
+ review = new(Review)
+
+ has := false
+ if has, err = e.SQL("SELECT * FROM review WHERE id IN (SELECT max(id) as id FROM review WHERE issue_id = ? AND reviewer_team_id = ?)",
+ issueID, teamID).
+ Get(review); err != nil {
+ return nil, err
+ }
+
+ if !has {
+ return nil, ErrReviewNotExist{0}
+ }
+
+ return
+}
+
// MarkReviewsAsStale marks existing reviews as stale
func MarkReviewsAsStale(issueID int64) (err error) {
_, err = x.Exec("UPDATE `review` SET stale=? WHERE issue_id=?", true, issueID)
@@ -668,13 +726,13 @@ func RemoveReviewRequest(issue *Issue, reviewer, doer *User) (comment *Comment,
// AddTeamReviewRequest add a review request from one team
func AddTeamReviewRequest(issue *Issue, reviewer *Team, doer *User) (comment *Comment, err error) {
- review, err := GetReviewerByIssueIDAndUserID(issue.ID, -reviewer.ID)
- if err != nil {
+ review, err := GetTeamReviewerByIssueIDAndTeamID(issue.ID, reviewer.ID)
+ if err != nil && !IsErrReviewNotExist(err) {
return
}
- // skip it when reviewer hase been request to review
- if review != nil && review.Type == ReviewTypeRequest {
+ // skip it when reviewer team hase been request to review
+ if review != nil {
return nil, nil
}
@@ -699,31 +757,31 @@ func AddTeamReviewRequest(issue *Issue, reviewer *Team, doer *User) (comment *Co
}
}
- if official {
- if _, err := sess.Exec("UPDATE `review` SET official=? WHERE issue_id=? AND reviewer_id=?", false, issue.ID, -reviewer.ID); err != nil {
- return nil, err
- }
- }
-
_, err = createReview(sess, CreateReviewOptions{
- Type: ReviewTypeRequest,
- Issue: issue,
- Reviewer: &User{ID: -reviewer.ID},
- Official: official,
- Stale: false,
+ Type: ReviewTypeRequest,
+ Issue: issue,
+ ReviewerTeam: reviewer,
+ Official: official,
+ Stale: false,
})
if err != nil {
return
}
+ if official {
+ if _, err := sess.Exec("UPDATE `review` SET official=? WHERE issue_id=? AND reviewer_team_id = ?", false, issue.ID, reviewer.ID); err != nil {
+ return nil, err
+ }
+ }
+
comment, err = createComment(sess, &CreateCommentOptions{
Type: CommentTypeReviewRequest,
Doer: doer,
Repo: issue.Repo,
Issue: issue,
- RemovedAssignee: false, // Use RemovedAssignee as !isRequest
- AssigneeID: -reviewer.ID, // Use AssigneeID as reviewer ID
+ RemovedAssignee: false, // Use RemovedAssignee as !isRequest
+ AssigneeTeamID: reviewer.ID, // Use AssigneeTeamID as reviewer team ID
})
if err != nil {
@@ -735,12 +793,12 @@ func AddTeamReviewRequest(issue *Issue, reviewer *Team, doer *User) (comment *Co
//RemoveTeamReviewRequest remove a review request from one team
func RemoveTeamReviewRequest(issue *Issue, reviewer *Team, doer *User) (comment *Comment, err error) {
- review, err := GetReviewerByIssueIDAndUserID(issue.ID, -reviewer.ID)
- if err != nil {
+ review, err := GetTeamReviewerByIssueIDAndTeamID(issue.ID, reviewer.ID)
+ if err != nil && !IsErrReviewNotExist(err) {
return
}
- if review.Type != ReviewTypeRequest {
+ if review == nil {
return nil, nil
}
@@ -790,8 +848,8 @@ func RemoveTeamReviewRequest(issue *Issue, reviewer *Team, doer *User) (comment
Doer: doer,
Repo: issue.Repo,
Issue: issue,
- RemovedAssignee: true, // Use RemovedAssignee as !isRequest
- AssigneeID: -reviewer.ID, // Use AssigneeID as reviewer ID
+ RemovedAssignee: true, // Use RemovedAssignee as !isRequest
+ AssigneeTeamID: reviewer.ID, // Use AssigneeTeamID as reviewer team ID
})
if err != nil {
diff --git a/routers/repo/issue.go b/routers/repo/issue.go
index c04b0c3714a53..11f60985104e5 100644
--- a/routers/repo/issue.go
+++ b/routers/repo/issue.go
@@ -1138,8 +1138,8 @@ func ViewIssue(ctx *context.Context) {
}
} else if comment.Type == models.CommentTypeAssignees || comment.Type == models.CommentTypeReviewRequest {
- if err = comment.LoadAssigneeUser(); err != nil {
- ctx.ServerError("LoadAssigneeUser", err)
+ if err = comment.LoadAssigneeUserAndTeam(); err != nil {
+ ctx.ServerError("LoadAssigneeUserAndTeam", err)
return
}
} else if comment.Type == models.CommentTypeRemoveDependency || comment.Type == models.CommentTypeAddDependency {
diff --git a/templates/repo/issue/view_content/comments.tmpl b/templates/repo/issue/view_content/comments.tmpl
index 41b1599e5eb4e..b3403592e74b9 100644
--- a/templates/repo/issue/view_content/comments.tmpl
+++ b/templates/repo/issue/view_content/comments.tmpl
@@ -605,9 +605,9 @@
{{end}}
{{else}}
{{if .RemovedAssignee}}
- {{$.i18n.Tr "repo.issues.review.remove_review_request" (.TeamAssignee.Name|Escape) $createdStr | Safe}}
+ {{$.i18n.Tr "repo.issues.review.remove_review_request" (.AssigneeTeam.Name|Escape) $createdStr | Safe}}
{{else}}
- {{$.i18n.Tr "repo.issues.review.add_review_request" (.TeamAssignee.Name|Escape) $createdStr | Safe}}
+ {{$.i18n.Tr "repo.issues.review.add_review_request" (.AssigneeTeam.Name|Escape) $createdStr | Safe}}
{{end}}
{{end}}
diff --git a/templates/repo/issue/view_content/sidebar.tmpl b/templates/repo/issue/view_content/sidebar.tmpl
index 2dd68fe662ca0..ce27a96b5de1b 100644
--- a/templates/repo/issue/view_content/sidebar.tmpl
+++ b/templates/repo/issue/view_content/sidebar.tmpl
@@ -76,7 +76,7 @@
{{$canChoose = true}}
{{end}}
-
+
{{svg "octicon-check" 16}}
{{$.Issue.Repo.OwnerName}}/{{.Name}}
From 66a0f478e71decd6e0ef01c6ead1503f2ffdd661 Mon Sep 17 00:00:00 2001
From: a1012112796 <1012112796@qq.com>
Date: Thu, 17 Sep 2020 17:29:31 +0800
Subject: [PATCH 05/24] fix lint
---
models/migrations/v152.go | 7 ++-----
1 file changed, 2 insertions(+), 5 deletions(-)
diff --git a/models/migrations/v152.go b/models/migrations/v152.go
index 4ef8898b1ce86..a73a591256770 100644
--- a/models/migrations/v152.go
+++ b/models/migrations/v152.go
@@ -21,9 +21,6 @@ func addTeamReviewRequestSupport(x *xorm.Engine) error {
return err
}
- if err := x.Sync2(new(Comment)); err != nil {
- return err
- }
-
- return nil
+ err := x.Sync2(new(Comment))
+ return err
}
From f36a5ec99aad5158684c53271c8a6cbfd96968b5 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E8=B5=B5=E6=99=BA=E8=B6=85?= <1012112796@qq.com>
Date: Mon, 28 Sep 2020 17:42:54 +0800
Subject: [PATCH 06/24] Update models/migrations/v153.go
Co-authored-by: Lauris BH
---
models/migrations/v153.go | 3 +--
1 file changed, 1 insertion(+), 2 deletions(-)
diff --git a/models/migrations/v153.go b/models/migrations/v153.go
index a73a591256770..1e5ae9f7da47b 100644
--- a/models/migrations/v153.go
+++ b/models/migrations/v153.go
@@ -21,6 +21,5 @@ func addTeamReviewRequestSupport(x *xorm.Engine) error {
return err
}
- err := x.Sync2(new(Comment))
- return err
+ return x.Sync2(new(Comment))
}
From 26eb8c41e1b65d285cbbcc19e5fc065e6cf3c8a5 Mon Sep 17 00:00:00 2001
From: a1012112796 <1012112796@qq.com>
Date: Tue, 29 Sep 2020 13:08:46 +0800
Subject: [PATCH 07/24] Apply suggestion from code review @lafriks
* Fix getReviewers() return
* Simplify ui logic
* Fix some bugs about Original author
---
models/repo.go | 18 +-
models/review.go | 49 ++---
routers/repo/issue.go | 208 ++++++++++++++++--
templates/repo/issue/view_content/pull.tmpl | 55 ++---
.../repo/issue/view_content/sidebar.tmpl | 108 +++------
web_src/js/index.js | 2 +-
6 files changed, 263 insertions(+), 177 deletions(-)
diff --git a/models/repo.go b/models/repo.go
index ee51f7907a70e..db55c7bbf9242 100644
--- a/models/repo.go
+++ b/models/repo.go
@@ -729,26 +729,36 @@ func (repo *Repository) getReviewersPublic(e Engine, doerID, posterID int64) (_
return users, nil
}
-func (repo *Repository) getReviewers(e Engine, doerID, posterID int64) (users []*User, teams []*Team, err error) {
- if err = repo.getOwner(e); err != nil {
+func (repo *Repository) getReviewers(e Engine, doerID, posterID int64) ([]*User, []*Team, error) {
+ if err := repo.getOwner(e); err != nil {
return nil, nil, err
}
+ var (
+ users []*User
+ err error
+ )
+
if repo.IsPrivate ||
(repo.Owner.IsOrganization() && repo.Owner.Visibility == api.VisibleTypePrivate) {
users, err = repo.getReviewersPrivate(x, doerID, posterID)
} else {
users, err = repo.getReviewersPublic(x, doerID, posterID)
}
+ if err != nil {
+ return nil, nil, err
+ }
if repo.Owner.IsOrganization() {
- teams, err = GetTeamsWithAccessToRepo(repo.OwnerID, repo.ID, AccessModeRead)
+ teams, err := GetTeamsWithAccessToRepo(repo.OwnerID, repo.ID, AccessModeRead)
if err != nil {
return nil, nil, err
}
+
+ return users, teams, nil
}
- return
+ return users, nil, nil
}
// GetReviewers get all users can be requested to review
diff --git a/models/review.go b/models/review.go
index 9c9a56131bee3..93f49ceb6604b 100644
--- a/models/review.go
+++ b/models/review.go
@@ -122,6 +122,11 @@ func (r *Review) LoadReviewer() error {
return r.loadReviewer(x)
}
+// LoadReviewerTeam loads reviewer team
+func (r *Review) LoadReviewerTeam() error {
+ return r.loadReviewerTeam(x)
+}
+
func (r *Review) loadAttributes(e Engine) (err error) {
if err = r.loadIssue(e); err != nil {
return
@@ -446,53 +451,25 @@ func SubmitReview(doer *User, issue *Issue, reviewType ReviewType, content, comm
}
// GetReviewersByIssueID gets the latest review of each reviewer for a pull request
-func GetReviewersByIssueID(issueID int64) (reviews []*Review, err error) {
- reviewsUnfiltered := []*Review{}
-
- sess := x.NewSession()
- defer sess.Close()
- if err := sess.Begin(); err != nil {
- return nil, err
- }
+func GetReviewersByIssueID(issueID int64) ([]*Review, error) {
+ reviews := make([]*Review, 0, 10)
// Get latest review of each reviwer, sorted in order they were made
- if err := sess.SQL("SELECT * FROM review WHERE id IN (SELECT max(id) as id FROM review WHERE issue_id = ? AND reviewer_team_id = 0 AND type in (?, ?, ?) GROUP BY issue_id, reviewer_id) ORDER BY review.updated_unix ASC",
+ if err := x.SQL("SELECT * FROM review WHERE id IN (SELECT max(id) as id FROM review WHERE issue_id = ? AND reviewer_team_id = 0 AND type in (?, ?, ?) AND original_author_id = 0 GROUP BY issue_id, reviewer_id) ORDER BY review.updated_unix ASC",
issueID, ReviewTypeApprove, ReviewTypeReject, ReviewTypeRequest).
- Find(&reviewsUnfiltered); err != nil {
+ Find(&reviews); err != nil {
return nil, err
}
- // Load reviewer and skip if user is deleted
- for _, review := range reviewsUnfiltered {
- if err = review.loadReviewer(sess); err != nil {
- if !IsErrUserNotExist(err) {
- return nil, err
- }
- } else {
- reviews = append(reviews, review)
- }
- }
-
teamReviewRequests := make([]*Review, 0, 5)
- if err := sess.SQL("SELECT * FROM review WHERE id IN (SELECT max(id) as id FROM review WHERE issue_id = ? AND reviewer_team_id <> 0 GROUP BY issue_id, reviewer_team_id) ORDER BY review.updated_unix ASC",
+ if err := x.SQL("SELECT * FROM review WHERE id IN (SELECT max(id) as id FROM review WHERE issue_id = ? AND reviewer_team_id <> 0 AND original_author_id = 0 GROUP BY issue_id, reviewer_team_id) ORDER BY review.updated_unix ASC",
issueID).
Find(&teamReviewRequests); err != nil {
return nil, err
}
- if len(teamReviewRequests) == 0 {
- return reviews, nil
- }
-
- for _, review := range teamReviewRequests {
- if err = review.loadReviewerTeam(sess); err != nil {
- if !IsErrTeamNotExist(err) {
- return nil, err
- }
- } else {
- review.ReviewerID = -review.ReviewerTeamID
- reviews = append(reviews, review)
- }
+ if len(teamReviewRequests) > 0 {
+ reviews = append(reviews, teamReviewRequests...)
}
return reviews, nil
@@ -506,7 +483,7 @@ func GetReviewerByIssueIDAndUserID(issueID, userID int64) (review *Review, err e
func getReviewerByIssueIDAndUserID(e Engine, issueID, userID int64) (review *Review, err error) {
review = new(Review)
- if _, err := e.SQL("SELECT * FROM review WHERE id IN (SELECT max(id) as id FROM review WHERE issue_id = ? AND reviewer_id = ? AND type in (?, ?, ?))",
+ if _, err := e.SQL("SELECT * FROM review WHERE id IN (SELECT max(id) as id FROM review WHERE issue_id = ? AND reviewer_id = ? AND original_author_id = 0 AND type in (?, ?, ?))",
issueID, userID, ReviewTypeApprove, ReviewTypeReject, ReviewTypeRequest).
Get(review); err != nil {
return nil, err
diff --git a/routers/repo/issue.go b/routers/repo/issue.go
index e8aaad2a448f9..45db5b617d196 100644
--- a/routers/repo/issue.go
+++ b/routers/repo/issue.go
@@ -434,23 +434,195 @@ func retrieveProjects(ctx *context.Context, repo *models.Repository) {
}
}
+// ReviewerChoseItem items to bee shown
+type ReviewerChoseItem struct {
+ IsTeam bool
+ Team *models.Team
+ User *models.User
+ Review *models.Review
+ CanChange bool
+ Checked bool
+ ItemID int64
+}
+
// RetrieveRepoReviewers find all reviewers of a repository
-func RetrieveRepoReviewers(ctx *context.Context, repo *models.Repository, issuePosterID int64) {
+func RetrieveRepoReviewers(ctx *context.Context, repo *models.Repository, issue *models.Issue, canChooseReviewer bool) {
+ ctx.Data["CanChooseReviewer"] = canChooseReviewer
+
+ reviews, err := models.GetReviewersByIssueID(issue.ID)
+ if err != nil {
+ ctx.ServerError("GetReviewersByIssueID", err)
+ return
+ }
+
var (
- err error
- teamReviewers []*models.Team
+ pullReviews []*ReviewerChoseItem
+ reviewersResult []*ReviewerChoseItem
+ teamReviewersResult []*ReviewerChoseItem
+ teamReviewers []*models.Team
+ reviewers []*models.User
)
- ctx.Data["Reviewers"], teamReviewers, err = repo.GetReviewers(ctx.User.ID, issuePosterID)
- if err != nil {
- ctx.ServerError("GetReviewers", err)
+
+ if canChooseReviewer {
+ posterID := issue.PosterID
+ if issue.OriginalAuthorID > 0 {
+ posterID = 0
+ }
+ reviewers, teamReviewers, err = repo.GetReviewers(ctx.User.ID, posterID)
+ if err != nil {
+ ctx.ServerError("GetReviewers", err)
+ return
+ }
+ if len(reviewers) > 0 {
+ reviewersResult = make([]*ReviewerChoseItem, 0, len(reviewers))
+ }
+ if len(teamReviewers) > 0 {
+ teamReviewersResult = make([]*ReviewerChoseItem, 0, len(teamReviewers))
+ }
+ }
+
+ if len(reviews) != 0 {
+ pullReviews = make([]*ReviewerChoseItem, 0, len(reviews))
+
+ for _, review := range reviews {
+ tmp := &ReviewerChoseItem{
+ Checked: review.Type == models.ReviewTypeRequest,
+ Review: review,
+ ItemID: review.ReviewerID,
+ }
+ if review.ReviewerTeamID > 0 {
+ tmp.IsTeam = true
+ tmp.ItemID = -review.ReviewerTeamID
+ }
+ if ctx.User != nil && ctx.User.ID == review.ReviewerID && review.Type == models.ReviewTypeRequest {
+ // all user can refuse the review request to them
+ tmp.CanChange = true
+ } else if (canChooseReviewer || (ctx.User != nil && ctx.User.ID == issue.PosterID)) && review.Type != models.ReviewTypeRequest &&
+ ctx.User.ID != review.ReviewerID {
+ // manager and official reviewers call re-requst review from other reviewer
+ tmp.CanChange = true
+ } else if ctx.User != nil && ctx.Repo.IsAdmin() && ctx.User.ID != review.ReviewerID {
+ // only admin can dissims review request to other reviewers
+ tmp.CanChange = true
+ }
+
+ pullReviews = append(pullReviews, tmp)
+
+ if canChooseReviewer {
+ if tmp.IsTeam {
+ teamReviewersResult = append(teamReviewersResult, tmp)
+ } else {
+ reviewersResult = append(reviewersResult, tmp)
+ }
+ }
+ }
+ } else {
+ if !canChooseReviewer {
+ return
+ }
+ }
+
+ if !canChooseReviewer {
+ exitsReviers := make([]*ReviewerChoseItem, 0, len(pullReviews))
+ for _, item := range pullReviews {
+ if item.Review.ReviewerID > 0 {
+ if err = item.Review.LoadReviewer(); err != nil {
+ if models.IsErrUserNotExist(err) {
+ continue
+ } else {
+ ctx.ServerError("LoadReviewer", err)
+ return
+ }
+ }
+ item.User = item.Review.Reviewer
+ } else if item.Review.ReviewerTeamID > 0 {
+ if err = item.Review.LoadReviewerTeam(); err != nil {
+ if models.IsErrTeamNotExist(err) {
+ continue
+ } else {
+ ctx.ServerError("LoadReviewerTeam", err)
+ return
+ }
+ }
+ item.Team = item.Review.ReviewerTeam
+ } else {
+ continue
+ }
+
+ exitsReviers = append(exitsReviers, item)
+ }
+ ctx.Data["PullReviewers"] = exitsReviers
return
}
- for _, team := range teamReviewers {
- team.ID = -team.ID
+ if reviewersResult != nil {
+ exitLen := len(reviewersResult)
+ for _, reviewer := range reviewers {
+ exist := false
+ for index, tmp := range reviewersResult {
+ if index >= exitLen {
+ break
+ }
+ if tmp.ItemID == reviewer.ID {
+ tmp.User = reviewer
+ exist = true
+ }
+ }
+ if !exist {
+ reviewersResult = append(reviewersResult, &ReviewerChoseItem{
+ IsTeam: false,
+ CanChange: true,
+ User: reviewer,
+ ItemID: reviewer.ID,
+ })
+ }
+ }
+
+ ctx.Data["Reviewers"] = reviewersResult
}
- ctx.Data["TeamReviewers"] = teamReviewers
+ if teamReviewersResult != nil {
+ exitLen := len(teamReviewersResult)
+ for _, team := range teamReviewers {
+ exist := false
+ for index, tmp := range teamReviewersResult {
+ if index >= exitLen {
+ break
+ }
+ if tmp.ItemID == -team.ID {
+ tmp.Team = team
+ exist = true
+ }
+ }
+ if !exist {
+ teamReviewersResult = append(teamReviewersResult, &ReviewerChoseItem{
+ IsTeam: true,
+ CanChange: true,
+ Team: team,
+ ItemID: -team.ID,
+ })
+ }
+ }
+
+ ctx.Data["TeamReviewers"] = teamReviewersResult
+ }
+
+ if len(pullReviews) > 0 {
+ exitsReviers := make([]*ReviewerChoseItem, 0, len(pullReviews))
+ for _, item := range pullReviews {
+ if ctx.User != nil && item.ItemID == ctx.User.ID {
+ if err = item.Review.LoadReviewer(); err != nil {
+ ctx.ServerError("LoadReviewer", err)
+ return
+ }
+ item.User = item.Review.Reviewer
+ }
+ if item.Team != nil || item.User != nil {
+ exitsReviers = append(exitsReviers, item)
+ }
+ }
+ ctx.Data["PullReviewers"] = exitsReviers
+ }
}
// RetrieveRepoMetas find all the meta information of a repository
@@ -988,13 +1160,7 @@ func ViewIssue(ctx *context.Context) {
}
}
- if canChooseReviewer {
- RetrieveRepoReviewers(ctx, repo, issue.PosterID)
- ctx.Data["CanChooseReviewer"] = true
- } else {
- ctx.Data["CanChooseReviewer"] = false
- }
-
+ RetrieveRepoReviewers(ctx, repo, issue, canChooseReviewer)
if ctx.Written() {
return
}
@@ -1286,12 +1452,6 @@ func ViewIssue(ctx *context.Context) {
pull.HeadRepo != nil &&
git.IsBranchExist(pull.HeadRepo.RepoPath(), pull.HeadBranch) &&
(!pull.HasMerged || ctx.Data["HeadBranchCommitID"] == ctx.Data["PullHeadCommitID"])
-
- ctx.Data["PullReviewers"], err = models.GetReviewersByIssueID(issue.ID)
- if err != nil {
- ctx.ServerError("GetReviewersByIssueID", err)
- return
- }
}
// Get Dependencies
@@ -1563,7 +1723,7 @@ func isLegalReviewRequest(reviewer, doer *models.User, isAdd bool, issue *models
return fmt.Errorf("Reviewer can't read [user_id: %d, repo_name: %s]", reviewer.ID, issue.Repo.Name)
}
- if doer.ID == issue.PosterID && lastreview != nil && lastreview.Type != models.ReviewTypeRequest {
+ if doer.ID == issue.PosterID && issue.OriginalAuthorID == 0 && lastreview != nil && lastreview.Type != models.ReviewTypeRequest {
return nil
}
@@ -1582,7 +1742,7 @@ func isLegalReviewRequest(reviewer, doer *models.User, isAdd bool, issue *models
return fmt.Errorf("doer can't be reviewer [user_id: %d, repo_name: %s]", doer.ID, issue.Repo.Name)
}
- if reviewer.ID == issue.PosterID {
+ if reviewer.ID == issue.PosterID && issue.OriginalAuthorID == 0 {
return fmt.Errorf("poster of pr can't be reviewer [user_id: %d, repo_name: %s]", reviewer.ID, issue.Repo.Name)
}
} else {
diff --git a/templates/repo/issue/view_content/pull.tmpl b/templates/repo/issue/view_content/pull.tmpl
index be1462037efe8..b9be5327870fb 100644
--- a/templates/repo/issue/view_content/pull.tmpl
+++ b/templates/repo/issue/view_content/pull.tmpl
@@ -1,31 +1,31 @@
-{{if gt (len .PullReviewers) 0}}
+{{if .PullReviewers }}
{{$.i18n.Tr "repo.issues.review.reviewers"}}
{{range .PullReviewers}} - {{ $createdStr:= TimeSinceUnix .UpdatedUnix $.Lang }} + {{ $createdStr:= TimeSinceUnix .Review.UpdatedUnix $.Lang }}