* Send email to assigned user * Only send mail if enabled * Mail also when assigned through API * Need to refactor functions from models to issue service * Refer to issue index rather than ID * Disable email notifications completly at initalization if global disable * Check of user enbled mail shall be in mail notification function only * Initialize notifications from routers init function. * Use the assigned comment when sending assigned mail * Refactor so that assignees always added as separate step when new issue/pr. * Check error from AddAssignees * Check if user can be assiged to issue or pull request * Missing return * Refactor of CanBeAssigned check. CanBeAssigned shall have same check as UI. * Clarify function names (toggle rather than update/change), and clean up. * Fix review comments. * Flash error if assignees was not added when creating issue/pr * Generate error if assignee users doesn't existtags/v1.21.12.1
| @@ -896,7 +896,6 @@ type NewIssueOptions struct { | |||||
| Repo *Repository | Repo *Repository | ||||
| Issue *Issue | Issue *Issue | ||||
| LabelIDs []int64 | LabelIDs []int64 | ||||
| AssigneeIDs []int64 | |||||
| Attachments []string // In UUID format. | Attachments []string // In UUID format. | ||||
| IsPull bool | IsPull bool | ||||
| } | } | ||||
| @@ -918,40 +917,7 @@ func newIssue(e *xorm.Session, doer *User, opts NewIssueOptions) (err error) { | |||||
| } | } | ||||
| } | } | ||||
| // Keep the old assignee id thingy for compatibility reasons | |||||
| if opts.Issue.AssigneeID > 0 { | |||||
| isAdded := false | |||||
| // Check if the user has already been passed to issue.AssigneeIDs, if not, add it | |||||
| for _, aID := range opts.AssigneeIDs { | |||||
| if aID == opts.Issue.AssigneeID { | |||||
| isAdded = true | |||||
| break | |||||
| } | |||||
| } | |||||
| if !isAdded { | |||||
| opts.AssigneeIDs = append(opts.AssigneeIDs, opts.Issue.AssigneeID) | |||||
| } | |||||
| } | |||||
| // Check for and validate assignees | |||||
| if len(opts.AssigneeIDs) > 0 { | |||||
| for _, assigneeID := range opts.AssigneeIDs { | |||||
| user, err := getUserByID(e, assigneeID) | |||||
| if err != nil { | |||||
| return fmt.Errorf("getUserByID [user_id: %d, repo_id: %d]: %v", assigneeID, opts.Repo.ID, err) | |||||
| } | |||||
| valid, err := canBeAssigned(e, user, opts.Repo) | |||||
| if err != nil { | |||||
| return fmt.Errorf("canBeAssigned [user_id: %d, repo_id: %d]: %v", assigneeID, opts.Repo.ID, err) | |||||
| } | |||||
| if !valid { | |||||
| return ErrUserDoesNotHaveAccessToRepo{UserID: assigneeID, RepoName: opts.Repo.Name} | |||||
| } | |||||
| } | |||||
| } | |||||
| // Milestone and assignee validation should happen before insert actual object. | |||||
| // Milestone validation should happen before insert actual object. | |||||
| if _, err := e.SetExpr("`index`", "coalesce(MAX(`index`),0)+1"). | if _, err := e.SetExpr("`index`", "coalesce(MAX(`index`),0)+1"). | ||||
| Where("repo_id=?", opts.Issue.RepoID). | Where("repo_id=?", opts.Issue.RepoID). | ||||
| Insert(opts.Issue); err != nil { | Insert(opts.Issue); err != nil { | ||||
| @@ -976,14 +942,6 @@ func newIssue(e *xorm.Session, doer *User, opts NewIssueOptions) (err error) { | |||||
| } | } | ||||
| } | } | ||||
| // Insert the assignees | |||||
| for _, assigneeID := range opts.AssigneeIDs { | |||||
| err = opts.Issue.changeAssignee(e, doer, assigneeID, true) | |||||
| if err != nil { | |||||
| return err | |||||
| } | |||||
| } | |||||
| if opts.IsPull { | if opts.IsPull { | ||||
| _, err = e.Exec("UPDATE `repository` SET num_pulls = num_pulls + 1 WHERE id = ?", opts.Issue.RepoID) | _, err = e.Exec("UPDATE `repository` SET num_pulls = num_pulls + 1 WHERE id = ?", opts.Issue.RepoID) | ||||
| } else { | } else { | ||||
| @@ -1041,11 +999,11 @@ func newIssue(e *xorm.Session, doer *User, opts NewIssueOptions) (err error) { | |||||
| } | } | ||||
| // NewIssue creates new issue with labels for repository. | // NewIssue creates new issue with labels for repository. | ||||
| func NewIssue(repo *Repository, issue *Issue, labelIDs []int64, assigneeIDs []int64, uuids []string) (err error) { | |||||
| func NewIssue(repo *Repository, issue *Issue, labelIDs []int64, uuids []string) (err error) { | |||||
| // Retry several times in case INSERT fails due to duplicate key for (repo_id, index); see #7887 | // Retry several times in case INSERT fails due to duplicate key for (repo_id, index); see #7887 | ||||
| i := 0 | i := 0 | ||||
| for { | for { | ||||
| if err = newIssueAttempt(repo, issue, labelIDs, assigneeIDs, uuids); err == nil { | |||||
| if err = newIssueAttempt(repo, issue, labelIDs, uuids); err == nil { | |||||
| return nil | return nil | ||||
| } | } | ||||
| if !IsErrNewIssueInsert(err) { | if !IsErrNewIssueInsert(err) { | ||||
| @@ -1059,7 +1017,7 @@ func NewIssue(repo *Repository, issue *Issue, labelIDs []int64, assigneeIDs []in | |||||
| return fmt.Errorf("NewIssue: too many errors attempting to insert the new issue. Last error was: %v", err) | return fmt.Errorf("NewIssue: too many errors attempting to insert the new issue. Last error was: %v", err) | ||||
| } | } | ||||
| func newIssueAttempt(repo *Repository, issue *Issue, labelIDs []int64, assigneeIDs []int64, uuids []string) (err error) { | |||||
| func newIssueAttempt(repo *Repository, issue *Issue, labelIDs []int64, uuids []string) (err error) { | |||||
| sess := x.NewSession() | sess := x.NewSession() | ||||
| defer sess.Close() | defer sess.Close() | ||||
| if err = sess.Begin(); err != nil { | if err = sess.Begin(); err != nil { | ||||
| @@ -1071,7 +1029,6 @@ func newIssueAttempt(repo *Repository, issue *Issue, labelIDs []int64, assigneeI | |||||
| Issue: issue, | Issue: issue, | ||||
| LabelIDs: labelIDs, | LabelIDs: labelIDs, | ||||
| Attachments: uuids, | Attachments: uuids, | ||||
| AssigneeIDs: assigneeIDs, | |||||
| }); err != nil { | }); err != nil { | ||||
| if IsErrUserDoesNotHaveAccessToRepo(err) || IsErrNewIssueInsert(err) { | if IsErrUserDoesNotHaveAccessToRepo(err) || IsErrNewIssueInsert(err) { | ||||
| return err | return err | ||||
| @@ -58,8 +58,11 @@ func getAssigneesByIssue(e Engine, issue *Issue) (assignees []*User, err error) | |||||
| // IsUserAssignedToIssue returns true when the user is assigned to the issue | // IsUserAssignedToIssue returns true when the user is assigned to the issue | ||||
| func IsUserAssignedToIssue(issue *Issue, user *User) (isAssigned bool, err error) { | func IsUserAssignedToIssue(issue *Issue, user *User) (isAssigned bool, err error) { | ||||
| isAssigned, err = x.Exist(&IssueAssignees{IssueID: issue.ID, AssigneeID: user.ID}) | |||||
| return | |||||
| return isUserAssignedToIssue(x, issue, user) | |||||
| } | |||||
| func isUserAssignedToIssue(e Engine, issue *Issue, user *User) (isAssigned bool, err error) { | |||||
| return e.Get(&IssueAssignees{IssueID: issue.ID, AssigneeID: user.ID}) | |||||
| } | } | ||||
| // DeleteNotPassedAssignee deletes all assignees who aren't passed via the "assignees" array | // DeleteNotPassedAssignee deletes all assignees who aren't passed via the "assignees" array | ||||
| @@ -78,7 +81,7 @@ func DeleteNotPassedAssignee(issue *Issue, doer *User, assignees []*User) (err e | |||||
| if !found { | if !found { | ||||
| // This function also does comments and hooks, which is why we call it seperatly instead of directly removing the assignees here | // This function also does comments and hooks, which is why we call it seperatly instead of directly removing the assignees here | ||||
| if err := UpdateAssignee(issue, doer, assignee.ID); err != nil { | |||||
| if _, _, err := issue.ToggleAssignee(doer, assignee.ID); err != nil { | |||||
| return err | return err | ||||
| } | } | ||||
| } | } | ||||
| @@ -110,73 +113,56 @@ func clearAssigneeByUserID(sess *xorm.Session, userID int64) (err error) { | |||||
| return | return | ||||
| } | } | ||||
| // AddAssigneeIfNotAssigned adds an assignee only if he isn't aleady assigned to the issue | |||||
| func AddAssigneeIfNotAssigned(issue *Issue, doer *User, assigneeID int64) (err error) { | |||||
| // Check if the user is already assigned | |||||
| isAssigned, err := IsUserAssignedToIssue(issue, &User{ID: assigneeID}) | |||||
| if err != nil { | |||||
| return err | |||||
| } | |||||
| if !isAssigned { | |||||
| return issue.ChangeAssignee(doer, assigneeID) | |||||
| } | |||||
| return nil | |||||
| } | |||||
| // UpdateAssignee deletes or adds an assignee to an issue | |||||
| func UpdateAssignee(issue *Issue, doer *User, assigneeID int64) (err error) { | |||||
| return issue.ChangeAssignee(doer, assigneeID) | |||||
| } | |||||
| // ChangeAssignee changes the Assignee of this issue. | |||||
| func (issue *Issue) ChangeAssignee(doer *User, assigneeID int64) (err error) { | |||||
| // ToggleAssignee changes a user between assigned and not assigned for this issue, and make issue comment for it. | |||||
| func (issue *Issue) ToggleAssignee(doer *User, assigneeID int64) (removed bool, comment *Comment, err error) { | |||||
| sess := x.NewSession() | sess := x.NewSession() | ||||
| defer sess.Close() | defer sess.Close() | ||||
| if err := sess.Begin(); err != nil { | if err := sess.Begin(); err != nil { | ||||
| return err | |||||
| return false, nil, err | |||||
| } | } | ||||
| if err := issue.changeAssignee(sess, doer, assigneeID, false); err != nil { | |||||
| return err | |||||
| removed, comment, err = issue.toggleAssignee(sess, doer, assigneeID, false) | |||||
| if err != nil { | |||||
| return false, nil, err | |||||
| } | } | ||||
| if err := sess.Commit(); err != nil { | if err := sess.Commit(); err != nil { | ||||
| return err | |||||
| return false, nil, err | |||||
| } | } | ||||
| go HookQueue.Add(issue.RepoID) | go HookQueue.Add(issue.RepoID) | ||||
| return nil | |||||
| return removed, comment, nil | |||||
| } | } | ||||
| func (issue *Issue) changeAssignee(sess *xorm.Session, doer *User, assigneeID int64, isCreate bool) (err error) { | |||||
| // Update the assignee | |||||
| removed, err := updateIssueAssignee(sess, issue, assigneeID) | |||||
| func (issue *Issue) toggleAssignee(sess *xorm.Session, doer *User, assigneeID int64, isCreate bool) (removed bool, comment *Comment, err error) { | |||||
| removed, err = toggleUserAssignee(sess, issue, assigneeID) | |||||
| if err != nil { | if err != nil { | ||||
| return fmt.Errorf("UpdateIssueUserByAssignee: %v", err) | |||||
| return false, nil, fmt.Errorf("UpdateIssueUserByAssignee: %v", err) | |||||
| } | } | ||||
| // Repo infos | // Repo infos | ||||
| if err = issue.loadRepo(sess); err != nil { | if err = issue.loadRepo(sess); err != nil { | ||||
| return fmt.Errorf("loadRepo: %v", err) | |||||
| return false, nil, fmt.Errorf("loadRepo: %v", err) | |||||
| } | } | ||||
| // Comment | // Comment | ||||
| if _, err = createAssigneeComment(sess, doer, issue.Repo, issue, assigneeID, removed); err != nil { | |||||
| return fmt.Errorf("createAssigneeComment: %v", err) | |||||
| comment, err = createAssigneeComment(sess, doer, issue.Repo, issue, assigneeID, removed) | |||||
| if err != nil { | |||||
| return false, nil, fmt.Errorf("createAssigneeComment: %v", err) | |||||
| } | } | ||||
| // if pull request is in the middle of creation - don't call webhook | // if pull request is in the middle of creation - don't call webhook | ||||
| if isCreate { | if isCreate { | ||||
| return nil | |||||
| return removed, comment, err | |||||
| } | } | ||||
| if issue.IsPull { | if issue.IsPull { | ||||
| mode, _ := accessLevelUnit(sess, doer, issue.Repo, UnitTypePullRequests) | mode, _ := accessLevelUnit(sess, doer, issue.Repo, UnitTypePullRequests) | ||||
| if err = issue.loadPullRequest(sess); err != nil { | if err = issue.loadPullRequest(sess); err != nil { | ||||
| return fmt.Errorf("loadPullRequest: %v", err) | |||||
| return false, nil, fmt.Errorf("loadPullRequest: %v", err) | |||||
| } | } | ||||
| issue.PullRequest.Issue = issue | issue.PullRequest.Issue = issue | ||||
| apiPullRequest := &api.PullRequestPayload{ | apiPullRequest := &api.PullRequestPayload{ | ||||
| @@ -190,9 +176,10 @@ func (issue *Issue) changeAssignee(sess *xorm.Session, doer *User, assigneeID in | |||||
| } else { | } else { | ||||
| apiPullRequest.Action = api.HookIssueAssigned | apiPullRequest.Action = api.HookIssueAssigned | ||||
| } | } | ||||
| // Assignee comment triggers a webhook | |||||
| if err := prepareWebhooks(sess, issue.Repo, HookEventPullRequest, apiPullRequest); err != nil { | if err := prepareWebhooks(sess, issue.Repo, HookEventPullRequest, apiPullRequest); err != nil { | ||||
| log.Error("PrepareWebhooks [is_pull: %v, remove_assignee: %v]: %v", issue.IsPull, removed, err) | log.Error("PrepareWebhooks [is_pull: %v, remove_assignee: %v]: %v", issue.IsPull, removed, err) | ||||
| return nil | |||||
| return false, nil, err | |||||
| } | } | ||||
| } else { | } else { | ||||
| mode, _ := accessLevelUnit(sess, doer, issue.Repo, UnitTypeIssues) | mode, _ := accessLevelUnit(sess, doer, issue.Repo, UnitTypeIssues) | ||||
| @@ -208,67 +195,50 @@ func (issue *Issue) changeAssignee(sess *xorm.Session, doer *User, assigneeID in | |||||
| } else { | } else { | ||||
| apiIssue.Action = api.HookIssueAssigned | apiIssue.Action = api.HookIssueAssigned | ||||
| } | } | ||||
| // Assignee comment triggers a webhook | |||||
| if err := prepareWebhooks(sess, issue.Repo, HookEventIssues, apiIssue); err != nil { | if err := prepareWebhooks(sess, issue.Repo, HookEventIssues, apiIssue); err != nil { | ||||
| log.Error("PrepareWebhooks [is_pull: %v, remove_assignee: %v]: %v", issue.IsPull, removed, err) | log.Error("PrepareWebhooks [is_pull: %v, remove_assignee: %v]: %v", issue.IsPull, removed, err) | ||||
| return nil | |||||
| return false, nil, err | |||||
| } | } | ||||
| } | } | ||||
| return nil | |||||
| return removed, comment, nil | |||||
| } | } | ||||
| // UpdateAPIAssignee is a helper function to add or delete one or multiple issue assignee(s) | |||||
| // Deleting is done the GitHub way (quote from their api documentation): | |||||
| // https://developer.github.com/v3/issues/#edit-an-issue | |||||
| // "assignees" (array): Logins for Users to assign to this issue. | |||||
| // Pass one or more user logins to replace the set of assignees on this Issue. | |||||
| // Send an empty array ([]) to clear all assignees from the Issue. | |||||
| func UpdateAPIAssignee(issue *Issue, oneAssignee string, multipleAssignees []string, doer *User) (err error) { | |||||
| var allNewAssignees []*User | |||||
| // toggles user assignee state in database | |||||
| func toggleUserAssignee(e *xorm.Session, issue *Issue, assigneeID int64) (removed bool, err error) { | |||||
| // Keep the old assignee thingy for compatibility reasons | |||||
| if oneAssignee != "" { | |||||
| // Prevent double adding assignees | |||||
| var isDouble bool | |||||
| for _, assignee := range multipleAssignees { | |||||
| if assignee == oneAssignee { | |||||
| isDouble = true | |||||
| break | |||||
| } | |||||
| } | |||||
| if !isDouble { | |||||
| multipleAssignees = append(multipleAssignees, oneAssignee) | |||||
| } | |||||
| // Check if the user exists | |||||
| assignee, err := getUserByID(e, assigneeID) | |||||
| if err != nil { | |||||
| return false, err | |||||
| } | } | ||||
| // Loop through all assignees to add them | |||||
| for _, assigneeName := range multipleAssignees { | |||||
| assignee, err := GetUserByName(assigneeName) | |||||
| if err != nil { | |||||
| return err | |||||
| // Check if the submitted user is already assigned, if yes delete him otherwise add him | |||||
| var i int | |||||
| for i = 0; i < len(issue.Assignees); i++ { | |||||
| if issue.Assignees[i].ID == assigneeID { | |||||
| break | |||||
| } | } | ||||
| allNewAssignees = append(allNewAssignees, assignee) | |||||
| } | } | ||||
| // Delete all old assignees not passed | |||||
| if err = DeleteNotPassedAssignee(issue, doer, allNewAssignees); err != nil { | |||||
| return err | |||||
| } | |||||
| assigneeIn := IssueAssignees{AssigneeID: assigneeID, IssueID: issue.ID} | |||||
| // Add all new assignees | |||||
| // Update the assignee. The function will check if the user exists, is already | |||||
| // assigned (which he shouldn't as we deleted all assignees before) and | |||||
| // has access to the repo. | |||||
| for _, assignee := range allNewAssignees { | |||||
| // Extra method to prevent double adding (which would result in removing) | |||||
| err = AddAssigneeIfNotAssigned(issue, doer, assignee.ID) | |||||
| toBeDeleted := i < len(issue.Assignees) | |||||
| if toBeDeleted { | |||||
| issue.Assignees = append(issue.Assignees[:i], issue.Assignees[i:]...) | |||||
| _, err = e.Delete(assigneeIn) | |||||
| if err != nil { | if err != nil { | ||||
| return err | |||||
| return toBeDeleted, err | |||||
| } | |||||
| } else { | |||||
| issue.Assignees = append(issue.Assignees, assignee) | |||||
| _, err = e.Insert(assigneeIn) | |||||
| if err != nil { | |||||
| return toBeDeleted, err | |||||
| } | } | ||||
| } | } | ||||
| return | |||||
| return toBeDeleted, nil | |||||
| } | } | ||||
| // MakeIDsFromAPIAssigneesToAdd returns an array with all assignee IDs | // MakeIDsFromAPIAssigneesToAdd returns an array with all assignee IDs | ||||
| @@ -292,7 +262,7 @@ func MakeIDsFromAPIAssigneesToAdd(oneAssignee string, multipleAssignees []string | |||||
| } | } | ||||
| // Get the IDs of all assignees | // Get the IDs of all assignees | ||||
| assigneeIDs = GetUserIDsByNames(multipleAssignees) | |||||
| assigneeIDs, err = GetUserIDsByNames(multipleAssignees, false) | |||||
| return | return | ||||
| } | } | ||||
| @@ -20,17 +20,17 @@ func TestUpdateAssignee(t *testing.T) { | |||||
| // Assign multiple users | // Assign multiple users | ||||
| user2, err := GetUserByID(2) | user2, err := GetUserByID(2) | ||||
| assert.NoError(t, err) | assert.NoError(t, err) | ||||
| err = UpdateAssignee(issue, &User{ID: 1}, user2.ID) | |||||
| _, _, err = issue.ToggleAssignee(&User{ID: 1}, user2.ID) | |||||
| assert.NoError(t, err) | assert.NoError(t, err) | ||||
| user3, err := GetUserByID(3) | user3, err := GetUserByID(3) | ||||
| assert.NoError(t, err) | assert.NoError(t, err) | ||||
| err = UpdateAssignee(issue, &User{ID: 1}, user3.ID) | |||||
| _, _, err = issue.ToggleAssignee(&User{ID: 1}, user3.ID) | |||||
| assert.NoError(t, err) | assert.NoError(t, err) | ||||
| user1, err := GetUserByID(1) // This user is already assigned (see the definition in fixtures), so running UpdateAssignee should unassign him | user1, err := GetUserByID(1) // This user is already assigned (see the definition in fixtures), so running UpdateAssignee should unassign him | ||||
| assert.NoError(t, err) | assert.NoError(t, err) | ||||
| err = UpdateAssignee(issue, &User{ID: 1}, user1.ID) | |||||
| _, _, err = issue.ToggleAssignee(&User{ID: 1}, user1.ID) | |||||
| assert.NoError(t, err) | assert.NoError(t, err) | ||||
| // Check if he got removed | // Check if he got removed | ||||
| @@ -297,7 +297,7 @@ func testInsertIssue(t *testing.T, title, content string) { | |||||
| Title: title, | Title: title, | ||||
| Content: content, | Content: content, | ||||
| } | } | ||||
| err := NewIssue(repo, &issue, nil, nil, nil) | |||||
| err := NewIssue(repo, &issue, nil, nil) | |||||
| assert.NoError(t, err) | assert.NoError(t, err) | ||||
| var newIssue Issue | var newIssue Issue | ||||
| @@ -6,8 +6,6 @@ package models | |||||
| import ( | import ( | ||||
| "fmt" | "fmt" | ||||
| "xorm.io/xorm" | |||||
| ) | ) | ||||
| // IssueUser represents an issue-user relation. | // IssueUser represents an issue-user relation. | ||||
| @@ -51,42 +49,6 @@ func newIssueUsers(e Engine, repo *Repository, issue *Issue) error { | |||||
| return nil | return nil | ||||
| } | } | ||||
| func updateIssueAssignee(e *xorm.Session, issue *Issue, assigneeID int64) (removed bool, err error) { | |||||
| // Check if the user exists | |||||
| assignee, err := getUserByID(e, assigneeID) | |||||
| if err != nil { | |||||
| return false, err | |||||
| } | |||||
| // Check if the submitted user is already assigne, if yes delete him otherwise add him | |||||
| var i int | |||||
| for i = 0; i < len(issue.Assignees); i++ { | |||||
| if issue.Assignees[i].ID == assigneeID { | |||||
| break | |||||
| } | |||||
| } | |||||
| assigneeIn := IssueAssignees{AssigneeID: assigneeID, IssueID: issue.ID} | |||||
| toBeDeleted := i < len(issue.Assignees) | |||||
| if toBeDeleted { | |||||
| issue.Assignees = append(issue.Assignees[:i], issue.Assignees[i:]...) | |||||
| _, err = e.Delete(assigneeIn) | |||||
| if err != nil { | |||||
| return toBeDeleted, err | |||||
| } | |||||
| } else { | |||||
| issue.Assignees = append(issue.Assignees, assignee) | |||||
| _, err = e.Insert(assigneeIn) | |||||
| if err != nil { | |||||
| return toBeDeleted, err | |||||
| } | |||||
| } | |||||
| return toBeDeleted, nil | |||||
| } | |||||
| // UpdateIssueUserByRead updates issue-user relation for reading. | // UpdateIssueUserByRead updates issue-user relation for reading. | ||||
| func UpdateIssueUserByRead(uid, issueID int64) error { | func UpdateIssueUserByRead(uid, issueID int64) error { | ||||
| _, err := x.Exec("UPDATE `issue_user` SET is_read=? WHERE uid=? AND issue_id=?", true, uid, issueID) | _, err := x.Exec("UPDATE `issue_user` SET is_read=? WHERE uid=? AND issue_id=?", true, uid, issueID) | ||||
| @@ -686,11 +686,11 @@ func (pr *PullRequest) testPatch(e Engine) (err error) { | |||||
| } | } | ||||
| // NewPullRequest creates new pull request with labels for repository. | // NewPullRequest creates new pull request with labels for repository. | ||||
| func NewPullRequest(repo *Repository, pull *Issue, labelIDs []int64, uuids []string, pr *PullRequest, patch []byte, assigneeIDs []int64) (err error) { | |||||
| func NewPullRequest(repo *Repository, pull *Issue, labelIDs []int64, uuids []string, pr *PullRequest, patch []byte) (err error) { | |||||
| // Retry several times in case INSERT fails due to duplicate key for (repo_id, index); see #7887 | // Retry several times in case INSERT fails due to duplicate key for (repo_id, index); see #7887 | ||||
| i := 0 | i := 0 | ||||
| for { | for { | ||||
| if err = newPullRequestAttempt(repo, pull, labelIDs, uuids, pr, patch, assigneeIDs); err == nil { | |||||
| if err = newPullRequestAttempt(repo, pull, labelIDs, uuids, pr, patch); err == nil { | |||||
| return nil | return nil | ||||
| } | } | ||||
| if !IsErrNewIssueInsert(err) { | if !IsErrNewIssueInsert(err) { | ||||
| @@ -704,7 +704,7 @@ func NewPullRequest(repo *Repository, pull *Issue, labelIDs []int64, uuids []str | |||||
| return fmt.Errorf("NewPullRequest: too many errors attempting to insert the new issue. Last error was: %v", err) | return fmt.Errorf("NewPullRequest: too many errors attempting to insert the new issue. Last error was: %v", err) | ||||
| } | } | ||||
| func newPullRequestAttempt(repo *Repository, pull *Issue, labelIDs []int64, uuids []string, pr *PullRequest, patch []byte, assigneeIDs []int64) (err error) { | |||||
| func newPullRequestAttempt(repo *Repository, pull *Issue, labelIDs []int64, uuids []string, pr *PullRequest, patch []byte) (err error) { | |||||
| sess := x.NewSession() | sess := x.NewSession() | ||||
| defer sess.Close() | defer sess.Close() | ||||
| if err = sess.Begin(); err != nil { | if err = sess.Begin(); err != nil { | ||||
| @@ -717,7 +717,6 @@ func newPullRequestAttempt(repo *Repository, pull *Issue, labelIDs []int64, uuid | |||||
| LabelIDs: labelIDs, | LabelIDs: labelIDs, | ||||
| Attachments: uuids, | Attachments: uuids, | ||||
| IsPull: true, | IsPull: true, | ||||
| AssigneeIDs: assigneeIDs, | |||||
| }); err != nil { | }); err != nil { | ||||
| if IsErrUserDoesNotHaveAccessToRepo(err) || IsErrNewIssueInsert(err) { | if IsErrUserDoesNotHaveAccessToRepo(err) || IsErrNewIssueInsert(err) { | ||||
| return err | return err | ||||
| @@ -329,10 +329,18 @@ func HasAccessUnit(user *User, repo *Repository, unitType UnitType, testMode Acc | |||||
| return hasAccessUnit(x, user, repo, unitType, testMode) | return hasAccessUnit(x, user, repo, unitType, testMode) | ||||
| } | } | ||||
| // canBeAssigned return true if user could be assigned to a repo | |||||
| // CanBeAssigned return true if user can be assigned to issue or pull requests in repo | |||||
| // Currently any write access (code, issues or pr's) is assignable, to match assignee list in user interface. | |||||
| // FIXME: user could send PullRequest also could be assigned??? | // FIXME: user could send PullRequest also could be assigned??? | ||||
| func canBeAssigned(e Engine, user *User, repo *Repository) (bool, error) { | |||||
| return hasAccessUnit(e, user, repo, UnitTypeCode, AccessModeWrite) | |||||
| func CanBeAssigned(user *User, repo *Repository, isPull bool) (bool, error) { | |||||
| if user.IsOrganization() { | |||||
| return false, fmt.Errorf("Organization can't be added as assignee [user_id: %d, repo_id: %d]", user.ID, repo.ID) | |||||
| } | |||||
| perm, err := GetUserRepoPermission(repo, user) | |||||
| if err != nil { | |||||
| return false, err | |||||
| } | |||||
| return perm.CanAccessAny(AccessModeWrite, UnitTypeCode, UnitTypeIssues, UnitTypePullRequests), nil | |||||
| } | } | ||||
| func hasAccess(e Engine, userID int64, repo *Repository) (bool, error) { | func hasAccess(e Engine, userID int64, repo *Repository) (bool, error) { | ||||
| @@ -1320,16 +1320,20 @@ func GetUsersByIDs(ids []int64) ([]*User, error) { | |||||
| } | } | ||||
| // GetUserIDsByNames returns a slice of ids corresponds to names. | // GetUserIDsByNames returns a slice of ids corresponds to names. | ||||
| func GetUserIDsByNames(names []string) []int64 { | |||||
| func GetUserIDsByNames(names []string, ignoreNonExistent bool) ([]int64, error) { | |||||
| ids := make([]int64, 0, len(names)) | ids := make([]int64, 0, len(names)) | ||||
| for _, name := range names { | for _, name := range names { | ||||
| u, err := GetUserByName(name) | u, err := GetUserByName(name) | ||||
| if err != nil { | if err != nil { | ||||
| continue | |||||
| if ignoreNonExistent { | |||||
| continue | |||||
| } else { | |||||
| return nil, err | |||||
| } | |||||
| } | } | ||||
| ids = append(ids, u.ID) | ids = append(ids, u.ID) | ||||
| } | } | ||||
| return ids | |||||
| return ids, nil | |||||
| } | } | ||||
| // UserCommit represents a commit with validation of user. | // UserCommit represents a commit with validation of user. | ||||
| @@ -21,7 +21,7 @@ type Notifier interface { | |||||
| NotifyNewIssue(*models.Issue) | NotifyNewIssue(*models.Issue) | ||||
| NotifyIssueChangeStatus(*models.User, *models.Issue, bool) | NotifyIssueChangeStatus(*models.User, *models.Issue, bool) | ||||
| NotifyIssueChangeMilestone(doer *models.User, issue *models.Issue) | NotifyIssueChangeMilestone(doer *models.User, issue *models.Issue) | ||||
| NotifyIssueChangeAssignee(doer *models.User, issue *models.Issue, removed bool) | |||||
| NotifyIssueChangeAssignee(doer *models.User, issue *models.Issue, assignee *models.User, removed bool, comment *models.Comment) | |||||
| NotifyIssueChangeContent(doer *models.User, issue *models.Issue, oldContent string) | NotifyIssueChangeContent(doer *models.User, issue *models.Issue, oldContent string) | ||||
| NotifyIssueClearLabels(doer *models.User, issue *models.Issue) | NotifyIssueClearLabels(doer *models.User, issue *models.Issue) | ||||
| NotifyIssueChangeTitle(doer *models.User, issue *models.Issue, oldTitle string) | NotifyIssueChangeTitle(doer *models.User, issue *models.Issue, oldTitle string) | ||||
| @@ -83,7 +83,7 @@ func (*NullNotifier) NotifyIssueChangeContent(doer *models.User, issue *models.I | |||||
| } | } | ||||
| // NotifyIssueChangeAssignee places a place holder function | // NotifyIssueChangeAssignee places a place holder function | ||||
| func (*NullNotifier) NotifyIssueChangeAssignee(doer *models.User, issue *models.Issue, removed bool) { | |||||
| func (*NullNotifier) NotifyIssueChangeAssignee(doer *models.User, issue *models.Issue, assignee *models.User, removed bool, comment *models.Comment) { | |||||
| } | } | ||||
| // NotifyIssueClearLabels places a place holder function | // NotifyIssueClearLabels places a place holder function | ||||
| @@ -5,6 +5,8 @@ | |||||
| package mail | package mail | ||||
| import ( | import ( | ||||
| "fmt" | |||||
| "code.gitea.io/gitea/models" | "code.gitea.io/gitea/models" | ||||
| "code.gitea.io/gitea/modules/log" | "code.gitea.io/gitea/modules/log" | ||||
| "code.gitea.io/gitea/modules/notification/base" | "code.gitea.io/gitea/modules/notification/base" | ||||
| @@ -88,3 +90,11 @@ func (m *mailNotifier) NotifyPullRequestReview(pr *models.PullRequest, r *models | |||||
| log.Error("MailParticipants: %v", err) | log.Error("MailParticipants: %v", err) | ||||
| } | } | ||||
| } | } | ||||
| func (m *mailNotifier) NotifyIssueChangeAssignee(doer *models.User, issue *models.Issue, assignee *models.User, removed bool, comment *models.Comment) { | |||||
| // mail only sent to added assignees and not self-assignee | |||||
| if !removed && doer.ID != assignee.ID && assignee.EmailNotifications() == models.EmailNotificationsEnabled { | |||||
| ct := fmt.Sprintf("Assigned #%d.", issue.Index) | |||||
| mailer.SendIssueAssignedMail(issue, doer, ct, comment, []string{assignee.Email}) | |||||
| } | |||||
| } | |||||
| @@ -12,6 +12,7 @@ import ( | |||||
| "code.gitea.io/gitea/modules/notification/mail" | "code.gitea.io/gitea/modules/notification/mail" | ||||
| "code.gitea.io/gitea/modules/notification/ui" | "code.gitea.io/gitea/modules/notification/ui" | ||||
| "code.gitea.io/gitea/modules/notification/webhook" | "code.gitea.io/gitea/modules/notification/webhook" | ||||
| "code.gitea.io/gitea/modules/setting" | |||||
| ) | ) | ||||
| var ( | var ( | ||||
| @@ -24,9 +25,12 @@ func RegisterNotifier(notifier base.Notifier) { | |||||
| notifiers = append(notifiers, notifier) | notifiers = append(notifiers, notifier) | ||||
| } | } | ||||
| func init() { | |||||
| // NewContext registers notification handlers | |||||
| func NewContext() { | |||||
| RegisterNotifier(ui.NewNotifier()) | RegisterNotifier(ui.NewNotifier()) | ||||
| RegisterNotifier(mail.NewNotifier()) | |||||
| if setting.Service.EnableNotifyMail { | |||||
| RegisterNotifier(mail.NewNotifier()) | |||||
| } | |||||
| RegisterNotifier(indexer.NewNotifier()) | RegisterNotifier(indexer.NewNotifier()) | ||||
| RegisterNotifier(webhook.NewNotifier()) | RegisterNotifier(webhook.NewNotifier()) | ||||
| } | } | ||||
| @@ -138,9 +142,9 @@ func NotifyIssueChangeContent(doer *models.User, issue *models.Issue, oldContent | |||||
| } | } | ||||
| // NotifyIssueChangeAssignee notifies change content to notifiers | // NotifyIssueChangeAssignee notifies change content to notifiers | ||||
| func NotifyIssueChangeAssignee(doer *models.User, issue *models.Issue, removed bool) { | |||||
| func NotifyIssueChangeAssignee(doer *models.User, issue *models.Issue, assignee *models.User, removed bool, comment *models.Comment) { | |||||
| for _, notifier := range notifiers { | for _, notifier := range notifiers { | ||||
| notifier.NotifyIssueChangeAssignee(doer, issue, removed) | |||||
| notifier.NotifyIssueChangeAssignee(doer, issue, assignee, removed, comment) | |||||
| } | } | ||||
| } | } | ||||
| @@ -977,6 +977,7 @@ issues.review.review = Review | |||||
| issues.review.reviewers = Reviewers | issues.review.reviewers = Reviewers | ||||
| issues.review.show_outdated = Show outdated | issues.review.show_outdated = Show outdated | ||||
| issues.review.hide_outdated = Hide outdated | issues.review.hide_outdated = Hide outdated | ||||
| issues.assignee.error = Not all assignees was added due to an unexpected error. | |||||
| pulls.desc = Enable pull requests and code reviews. | pulls.desc = Enable pull requests and code reviews. | ||||
| pulls.new = New Pull Request | pulls.new = New Pull Request | ||||
| @@ -213,12 +213,31 @@ func CreateIssue(ctx *context.APIContext, form api.CreateIssueOption) { | |||||
| } | } | ||||
| return | return | ||||
| } | } | ||||
| // Check if the passed assignees is assignable | |||||
| for _, aID := range assigneeIDs { | |||||
| assignee, err := models.GetUserByID(aID) | |||||
| if err != nil { | |||||
| ctx.Error(500, "GetUserByID", err) | |||||
| return | |||||
| } | |||||
| valid, err := models.CanBeAssigned(assignee, ctx.Repo.Repository, false) | |||||
| if err != nil { | |||||
| ctx.Error(500, "canBeAssigned", err) | |||||
| return | |||||
| } | |||||
| if !valid { | |||||
| ctx.Error(422, "canBeAssigned", models.ErrUserDoesNotHaveAccessToRepo{UserID: aID, RepoName: ctx.Repo.Repository.Name}) | |||||
| return | |||||
| } | |||||
| } | |||||
| } else { | } else { | ||||
| // setting labels is not allowed if user is not a writer | // setting labels is not allowed if user is not a writer | ||||
| form.Labels = make([]int64, 0) | form.Labels = make([]int64, 0) | ||||
| } | } | ||||
| if err := issue_service.NewIssue(ctx.Repo.Repository, issue, form.Labels, assigneeIDs, nil); err != nil { | |||||
| if err := issue_service.NewIssue(ctx.Repo.Repository, issue, form.Labels, nil); err != nil { | |||||
| if models.IsErrUserDoesNotHaveAccessToRepo(err) { | if models.IsErrUserDoesNotHaveAccessToRepo(err) { | ||||
| ctx.Error(400, "UserDoesNotHaveAccessToRepo", err) | ctx.Error(400, "UserDoesNotHaveAccessToRepo", err) | ||||
| return | return | ||||
| @@ -227,6 +246,11 @@ func CreateIssue(ctx *context.APIContext, form api.CreateIssueOption) { | |||||
| return | return | ||||
| } | } | ||||
| if err := issue_service.AddAssignees(issue, ctx.User, assigneeIDs); err != nil { | |||||
| ctx.ServerError("AddAssignees", err) | |||||
| return | |||||
| } | |||||
| notification.NotifyNewIssue(issue) | notification.NotifyNewIssue(issue) | ||||
| if form.Closed { | if form.Closed { | ||||
| @@ -336,9 +360,9 @@ func EditIssue(ctx *context.APIContext, form api.EditIssueOption) { | |||||
| oneAssignee = *form.Assignee | oneAssignee = *form.Assignee | ||||
| } | } | ||||
| err = models.UpdateAPIAssignee(issue, oneAssignee, form.Assignees, ctx.User) | |||||
| err = issue_service.UpdateAssignees(issue, oneAssignee, form.Assignees, ctx.User) | |||||
| if err != nil { | if err != nil { | ||||
| ctx.Error(500, "UpdateAPIAssignee", err) | |||||
| ctx.Error(500, "UpdateAssignees", err) | |||||
| return | return | ||||
| } | } | ||||
| } | } | ||||
| @@ -17,6 +17,7 @@ import ( | |||||
| "code.gitea.io/gitea/modules/notification" | "code.gitea.io/gitea/modules/notification" | ||||
| api "code.gitea.io/gitea/modules/structs" | api "code.gitea.io/gitea/modules/structs" | ||||
| "code.gitea.io/gitea/modules/timeutil" | "code.gitea.io/gitea/modules/timeutil" | ||||
| issue_service "code.gitea.io/gitea/services/issue" | |||||
| milestone_service "code.gitea.io/gitea/services/milestone" | milestone_service "code.gitea.io/gitea/services/milestone" | ||||
| pull_service "code.gitea.io/gitea/services/pull" | pull_service "code.gitea.io/gitea/services/pull" | ||||
| ) | ) | ||||
| @@ -285,8 +286,26 @@ func CreatePullRequest(ctx *context.APIContext, form api.CreatePullRequestOption | |||||
| } | } | ||||
| return | return | ||||
| } | } | ||||
| // Check if the passed assignees is assignable | |||||
| for _, aID := range assigneeIDs { | |||||
| assignee, err := models.GetUserByID(aID) | |||||
| if err != nil { | |||||
| ctx.Error(500, "GetUserByID", err) | |||||
| return | |||||
| } | |||||
| if err := pull_service.NewPullRequest(repo, prIssue, labelIDs, []string{}, pr, patch, assigneeIDs); err != nil { | |||||
| valid, err := models.CanBeAssigned(assignee, repo, true) | |||||
| if err != nil { | |||||
| ctx.Error(500, "canBeAssigned", err) | |||||
| return | |||||
| } | |||||
| if !valid { | |||||
| ctx.Error(422, "canBeAssigned", models.ErrUserDoesNotHaveAccessToRepo{UserID: aID, RepoName: repo.Name}) | |||||
| return | |||||
| } | |||||
| } | |||||
| if err := pull_service.NewPullRequest(repo, prIssue, labelIDs, []string{}, pr, patch); err != nil { | |||||
| if models.IsErrUserDoesNotHaveAccessToRepo(err) { | if models.IsErrUserDoesNotHaveAccessToRepo(err) { | ||||
| ctx.Error(400, "UserDoesNotHaveAccessToRepo", err) | ctx.Error(400, "UserDoesNotHaveAccessToRepo", err) | ||||
| return | return | ||||
| @@ -298,6 +317,11 @@ func CreatePullRequest(ctx *context.APIContext, form api.CreatePullRequestOption | |||||
| return | return | ||||
| } | } | ||||
| if err := issue_service.AddAssignees(prIssue, ctx.User, assigneeIDs); err != nil { | |||||
| ctx.ServerError("AddAssignees", err) | |||||
| return | |||||
| } | |||||
| notification.NotifyNewPullRequest(pr) | notification.NotifyNewPullRequest(pr) | ||||
| log.Trace("Pull request created: %d/%d", repo.ID, prIssue.ID) | log.Trace("Pull request created: %d/%d", repo.ID, prIssue.ID) | ||||
| @@ -387,12 +411,12 @@ func EditPullRequest(ctx *context.APIContext, form api.EditPullRequestOption) { | |||||
| // Send an empty array ([]) to clear all assignees from the Issue. | // Send an empty array ([]) to clear all assignees from the Issue. | ||||
| if ctx.Repo.CanWrite(models.UnitTypePullRequests) && (form.Assignees != nil || len(form.Assignee) > 0) { | if ctx.Repo.CanWrite(models.UnitTypePullRequests) && (form.Assignees != nil || len(form.Assignee) > 0) { | ||||
| err = models.UpdateAPIAssignee(issue, form.Assignee, form.Assignees, ctx.User) | |||||
| err = issue_service.UpdateAssignees(issue, form.Assignee, form.Assignees, ctx.User) | |||||
| if err != nil { | if err != nil { | ||||
| if models.IsErrUserNotExist(err) { | if models.IsErrUserNotExist(err) { | ||||
| ctx.Error(422, "", fmt.Sprintf("Assignee does not exist: [name: %s]", err)) | ctx.Error(422, "", fmt.Sprintf("Assignee does not exist: [name: %s]", err)) | ||||
| } else { | } else { | ||||
| ctx.Error(500, "UpdateAPIAssignee", err) | |||||
| ctx.Error(500, "UpdateAssignees", err) | |||||
| } | } | ||||
| return | return | ||||
| } | } | ||||
| @@ -18,6 +18,7 @@ import ( | |||||
| "code.gitea.io/gitea/modules/log" | "code.gitea.io/gitea/modules/log" | ||||
| "code.gitea.io/gitea/modules/markup" | "code.gitea.io/gitea/modules/markup" | ||||
| "code.gitea.io/gitea/modules/markup/external" | "code.gitea.io/gitea/modules/markup/external" | ||||
| "code.gitea.io/gitea/modules/notification" | |||||
| "code.gitea.io/gitea/modules/setting" | "code.gitea.io/gitea/modules/setting" | ||||
| "code.gitea.io/gitea/modules/ssh" | "code.gitea.io/gitea/modules/ssh" | ||||
| "code.gitea.io/gitea/modules/task" | "code.gitea.io/gitea/modules/task" | ||||
| @@ -44,6 +45,7 @@ func NewServices() { | |||||
| setting.NewServices() | setting.NewServices() | ||||
| mailer.NewContext() | mailer.NewContext() | ||||
| _ = cache.NewContext() | _ = cache.NewContext() | ||||
| notification.NewContext() | |||||
| } | } | ||||
| // In case of problems connecting to DB, retry connection. Eg, PGSQL in Docker Container on Synology | // In case of problems connecting to DB, retry connection. Eg, PGSQL in Docker Container on Synology | ||||
| @@ -503,21 +503,21 @@ func ValidateRepoMetas(ctx *context.Context, form auth.CreateIssueForm, isPull b | |||||
| return nil, nil, 0 | return nil, nil, 0 | ||||
| } | } | ||||
| // Check if the passed assignees actually exists and has write access to the repo | |||||
| // Check if the passed assignees actually exists and is assignable | |||||
| for _, aID := range assigneeIDs { | for _, aID := range assigneeIDs { | ||||
| user, err := models.GetUserByID(aID) | |||||
| assignee, err := models.GetUserByID(aID) | |||||
| if err != nil { | if err != nil { | ||||
| ctx.ServerError("GetUserByID", err) | ctx.ServerError("GetUserByID", err) | ||||
| return nil, nil, 0 | return nil, nil, 0 | ||||
| } | } | ||||
| perm, err := models.GetUserRepoPermission(repo, user) | |||||
| valid, err := models.CanBeAssigned(assignee, repo, isPull) | |||||
| if err != nil { | if err != nil { | ||||
| ctx.ServerError("GetUserRepoPermission", err) | |||||
| ctx.ServerError("canBeAssigned", err) | |||||
| return nil, nil, 0 | return nil, nil, 0 | ||||
| } | } | ||||
| if !perm.CanWriteIssuesOrPulls(isPull) { | |||||
| ctx.ServerError("CanWriteIssuesOrPulls", fmt.Errorf("No permission for %s", user.Name)) | |||||
| if !valid { | |||||
| ctx.ServerError("canBeAssigned", models.ErrUserDoesNotHaveAccessToRepo{UserID: aID, RepoName: repo.Name}) | |||||
| return nil, nil, 0 | return nil, nil, 0 | ||||
| } | } | ||||
| } | } | ||||
| @@ -574,7 +574,7 @@ func NewIssuePost(ctx *context.Context, form auth.CreateIssueForm) { | |||||
| Content: form.Content, | Content: form.Content, | ||||
| Ref: form.Ref, | Ref: form.Ref, | ||||
| } | } | ||||
| if err := issue_service.NewIssue(repo, issue, labelIDs, assigneeIDs, attachments); err != nil { | |||||
| if err := issue_service.NewIssue(repo, issue, labelIDs, attachments); err != nil { | |||||
| if models.IsErrUserDoesNotHaveAccessToRepo(err) { | if models.IsErrUserDoesNotHaveAccessToRepo(err) { | ||||
| ctx.Error(400, "UserDoesNotHaveAccessToRepo", err.Error()) | ctx.Error(400, "UserDoesNotHaveAccessToRepo", err.Error()) | ||||
| return | return | ||||
| @@ -583,6 +583,11 @@ func NewIssuePost(ctx *context.Context, form auth.CreateIssueForm) { | |||||
| return | return | ||||
| } | } | ||||
| if err := issue_service.AddAssignees(issue, ctx.User, assigneeIDs); err != nil { | |||||
| log.Error("AddAssignees: %v", err) | |||||
| ctx.Flash.Error(ctx.Tr("issues.assignee.error")) | |||||
| } | |||||
| notification.NotifyNewIssue(issue) | notification.NotifyNewIssue(issue) | ||||
| log.Trace("Issue created: %d/%d", repo.ID, issue.ID) | log.Trace("Issue created: %d/%d", repo.ID, issue.ID) | ||||
| @@ -1112,7 +1117,7 @@ func UpdateIssueMilestone(ctx *context.Context) { | |||||
| }) | }) | ||||
| } | } | ||||
| // UpdateIssueAssignee change issue's assignee | |||||
| // UpdateIssueAssignee change issue's or pull's assignee | |||||
| func UpdateIssueAssignee(ctx *context.Context) { | func UpdateIssueAssignee(ctx *context.Context) { | ||||
| issues := getActionIssues(ctx) | issues := getActionIssues(ctx) | ||||
| if ctx.Written() { | if ctx.Written() { | ||||
| @@ -1130,10 +1135,29 @@ func UpdateIssueAssignee(ctx *context.Context) { | |||||
| return | return | ||||
| } | } | ||||
| default: | default: | ||||
| if err := issue.ChangeAssignee(ctx.User, assigneeID); err != nil { | |||||
| ctx.ServerError("ChangeAssignee", err) | |||||
| assignee, err := models.GetUserByID(assigneeID) | |||||
| if err != nil { | |||||
| ctx.ServerError("GetUserByID", err) | |||||
| return | |||||
| } | |||||
| valid, err := models.CanBeAssigned(assignee, issue.Repo, issue.IsPull) | |||||
| if err != nil { | |||||
| ctx.ServerError("canBeAssigned", err) | |||||
| return | return | ||||
| } | } | ||||
| if !valid { | |||||
| ctx.ServerError("canBeAssigned", models.ErrUserDoesNotHaveAccessToRepo{UserID: assigneeID, RepoName: issue.Repo.Name}) | |||||
| return | |||||
| } | |||||
| removed, comment, err := issue.ToggleAssignee(ctx.User, assigneeID) | |||||
| if err != nil { | |||||
| ctx.ServerError("ToggleAssignee", err) | |||||
| return | |||||
| } | |||||
| notification.NotifyIssueChangeAssignee(ctx.User, issue, assignee, removed, comment) | |||||
| } | } | ||||
| } | } | ||||
| ctx.JSON(200, map[string]interface{}{ | ctx.JSON(200, map[string]interface{}{ | ||||
| @@ -24,6 +24,7 @@ import ( | |||||
| "code.gitea.io/gitea/modules/setting" | "code.gitea.io/gitea/modules/setting" | ||||
| "code.gitea.io/gitea/modules/util" | "code.gitea.io/gitea/modules/util" | ||||
| "code.gitea.io/gitea/services/gitdiff" | "code.gitea.io/gitea/services/gitdiff" | ||||
| issue_service "code.gitea.io/gitea/services/issue" | |||||
| pull_service "code.gitea.io/gitea/services/pull" | pull_service "code.gitea.io/gitea/services/pull" | ||||
| "github.com/unknwon/com" | "github.com/unknwon/com" | ||||
| @@ -770,7 +771,7 @@ func CompareAndPullRequestPost(ctx *context.Context, form auth.CreateIssueForm) | |||||
| // FIXME: check error in the case two people send pull request at almost same time, give nice error prompt | // FIXME: check error in the case two people send pull request at almost same time, give nice error prompt | ||||
| // instead of 500. | // instead of 500. | ||||
| if err := pull_service.NewPullRequest(repo, pullIssue, labelIDs, attachments, pullRequest, patch, assigneeIDs); err != nil { | |||||
| if err := pull_service.NewPullRequest(repo, pullIssue, labelIDs, attachments, pullRequest, patch); err != nil { | |||||
| if models.IsErrUserDoesNotHaveAccessToRepo(err) { | if models.IsErrUserDoesNotHaveAccessToRepo(err) { | ||||
| ctx.Error(400, "UserDoesNotHaveAccessToRepo", err.Error()) | ctx.Error(400, "UserDoesNotHaveAccessToRepo", err.Error()) | ||||
| return | return | ||||
| @@ -782,6 +783,11 @@ func CompareAndPullRequestPost(ctx *context.Context, form auth.CreateIssueForm) | |||||
| return | return | ||||
| } | } | ||||
| if err := issue_service.AddAssignees(pullIssue, ctx.User, assigneeIDs); err != nil { | |||||
| log.Error("AddAssignees: %v", err) | |||||
| ctx.Flash.Error(ctx.Tr("issues.assignee.error")) | |||||
| } | |||||
| notification.NotifyNewPullRequest(pullRequest) | notification.NotifyNewPullRequest(pullRequest) | ||||
| log.Trace("Pull request created: %d/%d", repo.ID, pullIssue.ID) | log.Trace("Pull request created: %d/%d", repo.ID, pullIssue.ID) | ||||
| @@ -9,12 +9,13 @@ import ( | |||||
| "code.gitea.io/gitea/models" | "code.gitea.io/gitea/models" | ||||
| "code.gitea.io/gitea/modules/log" | "code.gitea.io/gitea/modules/log" | ||||
| "code.gitea.io/gitea/modules/notification" | |||||
| api "code.gitea.io/gitea/modules/structs" | api "code.gitea.io/gitea/modules/structs" | ||||
| ) | ) | ||||
| // NewIssue creates new issue with labels for repository. | // NewIssue creates new issue with labels for repository. | ||||
| func NewIssue(repo *models.Repository, issue *models.Issue, labelIDs []int64, assigneeIDs []int64, uuids []string) error { | |||||
| if err := models.NewIssue(repo, issue, labelIDs, assigneeIDs, uuids); err != nil { | |||||
| func NewIssue(repo *models.Repository, issue *models.Issue, labelIDs []int64, uuids []string) error { | |||||
| if err := models.NewIssue(repo, issue, labelIDs, uuids); err != nil { | |||||
| return err | return err | ||||
| } | } | ||||
| @@ -96,3 +97,104 @@ func ChangeTitle(issue *models.Issue, doer *models.User, title string) (err erro | |||||
| return nil | return nil | ||||
| } | } | ||||
| // UpdateAssignees is a helper function to add or delete one or multiple issue assignee(s) | |||||
| // Deleting is done the GitHub way (quote from their api documentation): | |||||
| // https://developer.github.com/v3/issues/#edit-an-issue | |||||
| // "assignees" (array): Logins for Users to assign to this issue. | |||||
| // Pass one or more user logins to replace the set of assignees on this Issue. | |||||
| // Send an empty array ([]) to clear all assignees from the Issue. | |||||
| func UpdateAssignees(issue *models.Issue, oneAssignee string, multipleAssignees []string, doer *models.User) (err error) { | |||||
| var allNewAssignees []*models.User | |||||
| // Keep the old assignee thingy for compatibility reasons | |||||
| if oneAssignee != "" { | |||||
| // Prevent double adding assignees | |||||
| var isDouble bool | |||||
| for _, assignee := range multipleAssignees { | |||||
| if assignee == oneAssignee { | |||||
| isDouble = true | |||||
| break | |||||
| } | |||||
| } | |||||
| if !isDouble { | |||||
| multipleAssignees = append(multipleAssignees, oneAssignee) | |||||
| } | |||||
| } | |||||
| // Loop through all assignees to add them | |||||
| for _, assigneeName := range multipleAssignees { | |||||
| assignee, err := models.GetUserByName(assigneeName) | |||||
| if err != nil { | |||||
| return err | |||||
| } | |||||
| allNewAssignees = append(allNewAssignees, assignee) | |||||
| } | |||||
| // Delete all old assignees not passed | |||||
| if err = models.DeleteNotPassedAssignee(issue, doer, allNewAssignees); err != nil { | |||||
| return err | |||||
| } | |||||
| // Add all new assignees | |||||
| // Update the assignee. The function will check if the user exists, is already | |||||
| // assigned (which he shouldn't as we deleted all assignees before) and | |||||
| // has access to the repo. | |||||
| for _, assignee := range allNewAssignees { | |||||
| // Extra method to prevent double adding (which would result in removing) | |||||
| err = AddAssigneeIfNotAssigned(issue, doer, assignee.ID) | |||||
| if err != nil { | |||||
| return err | |||||
| } | |||||
| } | |||||
| return | |||||
| } | |||||
| // AddAssigneeIfNotAssigned adds an assignee only if he isn't already assigned to the issue. | |||||
| // Also checks for access of assigned user | |||||
| func AddAssigneeIfNotAssigned(issue *models.Issue, doer *models.User, assigneeID int64) (err error) { | |||||
| assignee, err := models.GetUserByID(assigneeID) | |||||
| if err != nil { | |||||
| return err | |||||
| } | |||||
| // Check if the user is already assigned | |||||
| isAssigned, err := models.IsUserAssignedToIssue(issue, assignee) | |||||
| if err != nil { | |||||
| return err | |||||
| } | |||||
| if isAssigned { | |||||
| // nothing to to | |||||
| return nil | |||||
| } | |||||
| valid, err := models.CanBeAssigned(assignee, issue.Repo, issue.IsPull) | |||||
| if err != nil { | |||||
| return err | |||||
| } | |||||
| if !valid { | |||||
| return models.ErrUserDoesNotHaveAccessToRepo{UserID: assigneeID, RepoName: issue.Repo.Name} | |||||
| } | |||||
| removed, comment, err := issue.ToggleAssignee(doer, assigneeID) | |||||
| if err != nil { | |||||
| return err | |||||
| } | |||||
| notification.NotifyIssueChangeAssignee(doer, issue, assignee, removed, comment) | |||||
| return nil | |||||
| } | |||||
| // AddAssignees adds a list of assignes (from IDs) to an issue | |||||
| func AddAssignees(issue *models.Issue, doer *models.User, assigneeIDs []int64) (err error) { | |||||
| for _, assigneeID := range assigneeIDs { | |||||
| if err = AddAssigneeIfNotAssigned(issue, doer, assigneeID); err != nil { | |||||
| return err | |||||
| } | |||||
| } | |||||
| return nil | |||||
| } | |||||
| @@ -28,8 +28,9 @@ const ( | |||||
| mailAuthResetPassword base.TplName = "auth/reset_passwd" | mailAuthResetPassword base.TplName = "auth/reset_passwd" | ||||
| mailAuthRegisterNotify base.TplName = "auth/register_notify" | mailAuthRegisterNotify base.TplName = "auth/register_notify" | ||||
| mailIssueComment base.TplName = "issue/comment" | |||||
| mailIssueMention base.TplName = "issue/mention" | |||||
| mailIssueComment base.TplName = "issue/comment" | |||||
| mailIssueMention base.TplName = "issue/mention" | |||||
| mailIssueAssigned base.TplName = "issue/assigned" | |||||
| mailNotifyCollaborator base.TplName = "notify/collaborator" | mailNotifyCollaborator base.TplName = "notify/collaborator" | ||||
| ) | ) | ||||
| @@ -183,6 +184,7 @@ func composeIssueCommentMessage(issue *models.Issue, doer *models.User, content | |||||
| data = composeTplData(subject, body, issue.HTMLURL()) | data = composeTplData(subject, body, issue.HTMLURL()) | ||||
| } | } | ||||
| data["Doer"] = doer | data["Doer"] = doer | ||||
| data["Issue"] = issue | |||||
| var mailBody bytes.Buffer | var mailBody bytes.Buffer | ||||
| @@ -220,3 +222,8 @@ func SendIssueMentionMail(issue *models.Issue, doer *models.User, content string | |||||
| } | } | ||||
| SendAsync(composeIssueCommentMessage(issue, doer, content, comment, mailIssueMention, tos, "issue mention")) | SendAsync(composeIssueCommentMessage(issue, doer, content, comment, mailIssueMention, tos, "issue mention")) | ||||
| } | } | ||||
| // SendIssueAssignedMail composes and sends issue assigned email | |||||
| func SendIssueAssignedMail(issue *models.Issue, doer *models.User, content string, comment *models.Comment, tos []string) { | |||||
| SendAsync(composeIssueCommentMessage(issue, doer, content, comment, mailIssueAssigned, tos, "issue assigned")) | |||||
| } | |||||
| @@ -10,7 +10,6 @@ import ( | |||||
| "code.gitea.io/gitea/models" | "code.gitea.io/gitea/models" | ||||
| "code.gitea.io/gitea/modules/log" | "code.gitea.io/gitea/modules/log" | ||||
| "code.gitea.io/gitea/modules/references" | "code.gitea.io/gitea/modules/references" | ||||
| "code.gitea.io/gitea/modules/setting" | |||||
| "github.com/unknwon/com" | "github.com/unknwon/com" | ||||
| ) | ) | ||||
| @@ -24,9 +23,6 @@ func mailSubject(issue *models.Issue) string { | |||||
| // 1. Repository watchers and users who are participated in comments. | // 1. Repository watchers and users who are participated in comments. | ||||
| // 2. Users who are not in 1. but get mentioned in current issue/comment. | // 2. Users who are not in 1. but get mentioned in current issue/comment. | ||||
| func mailIssueCommentToParticipants(issue *models.Issue, doer *models.User, content string, comment *models.Comment, mentions []string) error { | func mailIssueCommentToParticipants(issue *models.Issue, doer *models.User, content string, comment *models.Comment, mentions []string) error { | ||||
| if !setting.Service.EnableNotifyMail { | |||||
| return nil | |||||
| } | |||||
| watchers, err := models.GetWatchers(issue.RepoID) | watchers, err := models.GetWatchers(issue.RepoID) | ||||
| if err != nil { | if err != nil { | ||||
| @@ -14,8 +14,8 @@ import ( | |||||
| ) | ) | ||||
| // NewPullRequest creates new pull request with labels for repository. | // NewPullRequest creates new pull request with labels for repository. | ||||
| func NewPullRequest(repo *models.Repository, pull *models.Issue, labelIDs []int64, uuids []string, pr *models.PullRequest, patch []byte, assigneeIDs []int64) error { | |||||
| if err := models.NewPullRequest(repo, pull, labelIDs, uuids, pr, patch, assigneeIDs); err != nil { | |||||
| func NewPullRequest(repo *models.Repository, pull *models.Issue, labelIDs []int64, uuids []string, pr *models.PullRequest, patch []byte) error { | |||||
| if err := models.NewPullRequest(repo, pull, labelIDs, uuids, pr, patch); err != nil { | |||||
| return err | return err | ||||
| } | } | ||||
| @@ -0,0 +1,17 @@ | |||||
| <!DOCTYPE html> | |||||
| <html> | |||||
| <head> | |||||
| <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> | |||||
| <title>{{.Subject}}</title> | |||||
| </head> | |||||
| <body> | |||||
| <p>@{{.Doer.Name}} assigned you to the {{if eq .Issue.IsPull true}}pull request{{else}}issue{{end}} <a href="{{.Link}}">#{{.Issue.Index}}</a> in repository {{.Issue.Repo.FullName}}.</p> | |||||
| <p> | |||||
| --- | |||||
| <br> | |||||
| <a href="{{.Link}}">View it on Gitea</a>. | |||||
| </p> | |||||
| </body> | |||||
| </html> | |||||