forked from gitea/gitea
Add notification interface and refactor UI notifications (#5085)
* add notification interface and refactor UI notifications * add missing methods on notification interface and notifiy only issue status really changed * implement NotifyPullRequestReview for ui notification
This commit is contained in:
parent
dd62ca7ba9
commit
ea619b39b2
|
@ -112,6 +112,10 @@ func (issue *Issue) GetPullRequest() (pr *PullRequest, err error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
pr, err = getPullRequestByIssueID(x, issue.ID)
|
pr, err = getPullRequestByIssueID(x, issue.ID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
pr.Issue = issue
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -239,6 +239,8 @@ func getCurrentReview(e Engine, reviewer *User, issue *Issue) (*Review, error) {
|
||||||
if len(reviews) == 0 {
|
if len(reviews) == 0 {
|
||||||
return nil, ErrReviewNotExist{}
|
return nil, ErrReviewNotExist{}
|
||||||
}
|
}
|
||||||
|
reviews[0].Reviewer = reviewer
|
||||||
|
reviews[0].Issue = issue
|
||||||
return reviews[0], nil
|
return reviews[0], nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,43 @@
|
||||||
|
// Copyright 2018 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 base
|
||||||
|
|
||||||
|
import (
|
||||||
|
"code.gitea.io/git"
|
||||||
|
"code.gitea.io/gitea/models"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Notifier defines an interface to notify receiver
|
||||||
|
type Notifier interface {
|
||||||
|
Run()
|
||||||
|
|
||||||
|
NotifyCreateRepository(doer *models.User, u *models.User, repo *models.Repository)
|
||||||
|
NotifyMigrateRepository(doer *models.User, u *models.User, repo *models.Repository)
|
||||||
|
NotifyDeleteRepository(doer *models.User, repo *models.Repository)
|
||||||
|
NotifyForkRepository(doer *models.User, oldRepo, repo *models.Repository)
|
||||||
|
|
||||||
|
NotifyNewIssue(*models.Issue)
|
||||||
|
NotifyIssueChangeStatus(*models.User, *models.Issue, bool)
|
||||||
|
NotifyIssueChangeMilestone(doer *models.User, issue *models.Issue)
|
||||||
|
NotifyIssueChangeAssignee(doer *models.User, issue *models.Issue, removed bool)
|
||||||
|
NotifyIssueChangeContent(doer *models.User, issue *models.Issue, oldContent string)
|
||||||
|
NotifyIssueClearLabels(doer *models.User, issue *models.Issue)
|
||||||
|
NotifyIssueChangeTitle(doer *models.User, issue *models.Issue, oldTitle string)
|
||||||
|
NotifyIssueChangeLabels(doer *models.User, issue *models.Issue,
|
||||||
|
addedLabels []*models.Label, removedLabels []*models.Label)
|
||||||
|
|
||||||
|
NotifyNewPullRequest(*models.PullRequest)
|
||||||
|
NotifyMergePullRequest(*models.PullRequest, *models.User, *git.Repository)
|
||||||
|
NotifyPullRequestReview(*models.PullRequest, *models.Review, *models.Comment)
|
||||||
|
|
||||||
|
NotifyCreateIssueComment(*models.User, *models.Repository,
|
||||||
|
*models.Issue, *models.Comment)
|
||||||
|
NotifyUpdateComment(*models.User, *models.Comment, string)
|
||||||
|
NotifyDeleteComment(*models.User, *models.Comment)
|
||||||
|
|
||||||
|
NotifyNewRelease(rel *models.Release)
|
||||||
|
NotifyUpdateRelease(doer *models.User, rel *models.Release)
|
||||||
|
NotifyDeleteRelease(doer *models.User, rel *models.Release)
|
||||||
|
}
|
|
@ -1,50 +1,175 @@
|
||||||
// Copyright 2016 The Gitea Authors. All rights reserved.
|
// Copyright 2018 The Gitea Authors. All rights reserved.
|
||||||
// Use of this source code is governed by a MIT-style
|
// Use of this source code is governed by a MIT-style
|
||||||
// license that can be found in the LICENSE file.
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
package notification
|
package notification
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"code.gitea.io/git"
|
||||||
"code.gitea.io/gitea/models"
|
"code.gitea.io/gitea/models"
|
||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/notification/base"
|
||||||
)
|
"code.gitea.io/gitea/modules/notification/ui"
|
||||||
|
|
||||||
type (
|
|
||||||
notificationService struct {
|
|
||||||
issueQueue chan issueNotificationOpts
|
|
||||||
}
|
|
||||||
|
|
||||||
issueNotificationOpts struct {
|
|
||||||
issue *models.Issue
|
|
||||||
notificationAuthorID int64
|
|
||||||
}
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
// Service is the notification service
|
notifiers []base.Notifier
|
||||||
Service = ¬ificationService{
|
|
||||||
issueQueue: make(chan issueNotificationOpts, 100),
|
|
||||||
}
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// RegisterNotifier providers method to receive notify messages
|
||||||
|
func RegisterNotifier(notifier base.Notifier) {
|
||||||
|
go notifier.Run()
|
||||||
|
notifiers = append(notifiers, notifier)
|
||||||
|
}
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
go Service.Run()
|
RegisterNotifier(ui.NewNotifier())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ns *notificationService) Run() {
|
// NotifyCreateIssueComment notifies issue comment related message to notifiers
|
||||||
for {
|
func NotifyCreateIssueComment(doer *models.User, repo *models.Repository,
|
||||||
select {
|
issue *models.Issue, comment *models.Comment) {
|
||||||
case opts := <-ns.issueQueue:
|
for _, notifier := range notifiers {
|
||||||
if err := models.CreateOrUpdateIssueNotifications(opts.issue, opts.notificationAuthorID); err != nil {
|
notifier.NotifyCreateIssueComment(doer, repo, issue, comment)
|
||||||
log.Error(4, "Was unable to create issue notification: %v", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ns *notificationService) NotifyIssue(issue *models.Issue, notificationAuthorID int64) {
|
// NotifyNewIssue notifies new issue to notifiers
|
||||||
ns.issueQueue <- issueNotificationOpts{
|
func NotifyNewIssue(issue *models.Issue) {
|
||||||
issue,
|
for _, notifier := range notifiers {
|
||||||
notificationAuthorID,
|
notifier.NotifyNewIssue(issue)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyIssueChangeStatus notifies close or reopen issue to notifiers
|
||||||
|
func NotifyIssueChangeStatus(doer *models.User, issue *models.Issue, closeOrReopen bool) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyIssueChangeStatus(doer, issue, closeOrReopen)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyMergePullRequest notifies merge pull request to notifiers
|
||||||
|
func NotifyMergePullRequest(pr *models.PullRequest, doer *models.User, baseGitRepo *git.Repository) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyMergePullRequest(pr, doer, baseGitRepo)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyNewPullRequest notifies new pull request to notifiers
|
||||||
|
func NotifyNewPullRequest(pr *models.PullRequest) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyNewPullRequest(pr)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyPullRequestReview notifies new pull request review
|
||||||
|
func NotifyPullRequestReview(pr *models.PullRequest, review *models.Review, comment *models.Comment) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyPullRequestReview(pr, review, comment)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyUpdateComment notifies update comment to notifiers
|
||||||
|
func NotifyUpdateComment(doer *models.User, c *models.Comment, oldContent string) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyUpdateComment(doer, c, oldContent)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyDeleteComment notifies delete comment to notifiers
|
||||||
|
func NotifyDeleteComment(doer *models.User, c *models.Comment) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyDeleteComment(doer, c)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyDeleteRepository notifies delete repository to notifiers
|
||||||
|
func NotifyDeleteRepository(doer *models.User, repo *models.Repository) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyDeleteRepository(doer, repo)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyForkRepository notifies fork repository to notifiers
|
||||||
|
func NotifyForkRepository(doer *models.User, oldRepo, repo *models.Repository) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyForkRepository(doer, oldRepo, repo)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyNewRelease notifies new release to notifiers
|
||||||
|
func NotifyNewRelease(rel *models.Release) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyNewRelease(rel)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyUpdateRelease notifies update release to notifiers
|
||||||
|
func NotifyUpdateRelease(doer *models.User, rel *models.Release) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyUpdateRelease(doer, rel)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyDeleteRelease notifies delete release to notifiers
|
||||||
|
func NotifyDeleteRelease(doer *models.User, rel *models.Release) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyDeleteRelease(doer, rel)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyIssueChangeMilestone notifies change milestone to notifiers
|
||||||
|
func NotifyIssueChangeMilestone(doer *models.User, issue *models.Issue) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyIssueChangeMilestone(doer, issue)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyIssueChangeContent notifies change content to notifiers
|
||||||
|
func NotifyIssueChangeContent(doer *models.User, issue *models.Issue, oldContent string) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyIssueChangeContent(doer, issue, oldContent)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyIssueChangeAssignee notifies change content to notifiers
|
||||||
|
func NotifyIssueChangeAssignee(doer *models.User, issue *models.Issue, removed bool) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyIssueChangeAssignee(doer, issue, removed)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyIssueClearLabels notifies clear labels to notifiers
|
||||||
|
func NotifyIssueClearLabels(doer *models.User, issue *models.Issue) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyIssueClearLabels(doer, issue)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyIssueChangeTitle notifies change title to notifiers
|
||||||
|
func NotifyIssueChangeTitle(doer *models.User, issue *models.Issue, oldTitle string) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyIssueChangeTitle(doer, issue, oldTitle)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyIssueChangeLabels notifies change labels to notifiers
|
||||||
|
func NotifyIssueChangeLabels(doer *models.User, issue *models.Issue,
|
||||||
|
addedLabels []*models.Label, removedLabels []*models.Label) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyIssueChangeLabels(doer, issue, addedLabels, removedLabels)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyCreateRepository notifies create repository to notifiers
|
||||||
|
func NotifyCreateRepository(doer *models.User, u *models.User, repo *models.Repository) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyCreateRepository(doer, u, repo)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotifyMigrateRepository notifies create repository to notifiers
|
||||||
|
func NotifyMigrateRepository(doer *models.User, u *models.User, repo *models.Repository) {
|
||||||
|
for _, notifier := range notifiers {
|
||||||
|
notifier.NotifyMigrateRepository(doer, u, repo)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,134 @@
|
||||||
|
// Copyright 2018 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 ui
|
||||||
|
|
||||||
|
import (
|
||||||
|
"code.gitea.io/git"
|
||||||
|
"code.gitea.io/gitea/models"
|
||||||
|
"code.gitea.io/gitea/modules/log"
|
||||||
|
"code.gitea.io/gitea/modules/notification/base"
|
||||||
|
)
|
||||||
|
|
||||||
|
type (
|
||||||
|
notificationService struct {
|
||||||
|
issueQueue chan issueNotificationOpts
|
||||||
|
}
|
||||||
|
|
||||||
|
issueNotificationOpts struct {
|
||||||
|
issue *models.Issue
|
||||||
|
notificationAuthorID int64
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
_ base.Notifier = ¬ificationService{}
|
||||||
|
)
|
||||||
|
|
||||||
|
// NewNotifier create a new notificationService notifier
|
||||||
|
func NewNotifier() base.Notifier {
|
||||||
|
return ¬ificationService{
|
||||||
|
issueQueue: make(chan issueNotificationOpts, 100),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) Run() {
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case opts := <-ns.issueQueue:
|
||||||
|
if err := models.CreateOrUpdateIssueNotifications(opts.issue, opts.notificationAuthorID); err != nil {
|
||||||
|
log.Error(4, "Was unable to create issue notification: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyCreateIssueComment(doer *models.User, repo *models.Repository,
|
||||||
|
issue *models.Issue, comment *models.Comment) {
|
||||||
|
ns.issueQueue <- issueNotificationOpts{
|
||||||
|
issue,
|
||||||
|
doer.ID,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyNewIssue(issue *models.Issue) {
|
||||||
|
ns.issueQueue <- issueNotificationOpts{
|
||||||
|
issue,
|
||||||
|
issue.Poster.ID,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyIssueChangeStatus(doer *models.User, issue *models.Issue, isClosed bool) {
|
||||||
|
ns.issueQueue <- issueNotificationOpts{
|
||||||
|
issue,
|
||||||
|
doer.ID,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyMergePullRequest(pr *models.PullRequest, doer *models.User, gitRepo *git.Repository) {
|
||||||
|
ns.issueQueue <- issueNotificationOpts{
|
||||||
|
pr.Issue,
|
||||||
|
doer.ID,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyNewPullRequest(pr *models.PullRequest) {
|
||||||
|
ns.issueQueue <- issueNotificationOpts{
|
||||||
|
pr.Issue,
|
||||||
|
pr.Issue.PosterID,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyPullRequestReview(pr *models.PullRequest, r *models.Review, c *models.Comment) {
|
||||||
|
ns.issueQueue <- issueNotificationOpts{
|
||||||
|
pr.Issue,
|
||||||
|
r.Reviewer.ID,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyUpdateComment(doer *models.User, c *models.Comment, oldContent string) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyDeleteComment(doer *models.User, c *models.Comment) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyDeleteRepository(doer *models.User, repo *models.Repository) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyForkRepository(doer *models.User, oldRepo, repo *models.Repository) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyNewRelease(rel *models.Release) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyUpdateRelease(doer *models.User, rel *models.Release) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyDeleteRelease(doer *models.User, rel *models.Release) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyIssueChangeMilestone(doer *models.User, issue *models.Issue) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyIssueChangeContent(doer *models.User, issue *models.Issue, oldContent string) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyIssueChangeAssignee(doer *models.User, issue *models.Issue, removed bool) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyIssueClearLabels(doer *models.User, issue *models.Issue) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyIssueChangeTitle(doer *models.User, issue *models.Issue, oldTitle string) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyIssueChangeLabels(doer *models.User, issue *models.Issue,
|
||||||
|
addedLabels []*models.Label, removedLabels []*models.Label) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyCreateRepository(doer *models.User, u *models.User, repo *models.Repository) {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ns *notificationService) NotifyMigrateRepository(doer *models.User, u *models.User, repo *models.Repository) {
|
||||||
|
}
|
|
@ -13,6 +13,7 @@ import (
|
||||||
"code.gitea.io/gitea/models"
|
"code.gitea.io/gitea/models"
|
||||||
"code.gitea.io/gitea/modules/context"
|
"code.gitea.io/gitea/modules/context"
|
||||||
"code.gitea.io/gitea/modules/indexer"
|
"code.gitea.io/gitea/modules/indexer"
|
||||||
|
"code.gitea.io/gitea/modules/notification"
|
||||||
"code.gitea.io/gitea/modules/setting"
|
"code.gitea.io/gitea/modules/setting"
|
||||||
"code.gitea.io/gitea/modules/util"
|
"code.gitea.io/gitea/modules/util"
|
||||||
|
|
||||||
|
@ -207,6 +208,8 @@ func CreateIssue(ctx *context.APIContext, form api.CreateIssueOption) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
notification.NotifyNewIssue(issue)
|
||||||
|
|
||||||
if form.Closed {
|
if form.Closed {
|
||||||
if err := issue.ChangeStatus(ctx.User, ctx.Repo.Repository, true); err != nil {
|
if err := issue.ChangeStatus(ctx.User, ctx.Repo.Repository, true); err != nil {
|
||||||
if models.IsErrDependenciesLeft(err) {
|
if models.IsErrDependenciesLeft(err) {
|
||||||
|
@ -337,6 +340,8 @@ func EditIssue(ctx *context.APIContext, form api.EditIssueOption) {
|
||||||
ctx.Error(500, "ChangeStatus", err)
|
ctx.Error(500, "ChangeStatus", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
notification.NotifyIssueChangeStatus(ctx.User, issue, api.StateClosed == api.StateType(*form.State))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Refetch from database to assign some automatic values
|
// Refetch from database to assign some automatic values
|
||||||
|
|
|
@ -9,6 +9,7 @@ import (
|
||||||
|
|
||||||
"code.gitea.io/gitea/models"
|
"code.gitea.io/gitea/models"
|
||||||
"code.gitea.io/gitea/modules/context"
|
"code.gitea.io/gitea/modules/context"
|
||||||
|
"code.gitea.io/gitea/modules/notification"
|
||||||
|
|
||||||
api "code.gitea.io/sdk/gitea"
|
api "code.gitea.io/sdk/gitea"
|
||||||
)
|
)
|
||||||
|
@ -163,6 +164,8 @@ func CreateIssueComment(ctx *context.APIContext, form api.CreateIssueCommentOpti
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
notification.NotifyCreateIssueComment(ctx.User, ctx.Repo.Repository, issue, comment)
|
||||||
|
|
||||||
ctx.JSON(201, comment.APIFormat())
|
ctx.JSON(201, comment.APIFormat())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -14,6 +14,7 @@ import (
|
||||||
"code.gitea.io/gitea/modules/auth"
|
"code.gitea.io/gitea/modules/auth"
|
||||||
"code.gitea.io/gitea/modules/context"
|
"code.gitea.io/gitea/modules/context"
|
||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/log"
|
||||||
|
"code.gitea.io/gitea/modules/notification"
|
||||||
"code.gitea.io/gitea/modules/util"
|
"code.gitea.io/gitea/modules/util"
|
||||||
|
|
||||||
api "code.gitea.io/sdk/gitea"
|
api "code.gitea.io/sdk/gitea"
|
||||||
|
@ -270,6 +271,8 @@ func CreatePullRequest(ctx *context.APIContext, form api.CreatePullRequestOption
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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)
|
||||||
ctx.JSON(201, pr.APIFormat())
|
ctx.JSON(201, pr.APIFormat())
|
||||||
}
|
}
|
||||||
|
@ -386,6 +389,8 @@ func EditPullRequest(ctx *context.APIContext, form api.EditPullRequestOption) {
|
||||||
ctx.Error(500, "ChangeStatus", err)
|
ctx.Error(500, "ChangeStatus", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
notification.NotifyIssueChangeStatus(ctx.User, issue, api.StateClosed == api.StateType(*form.State))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Refetch from database
|
// Refetch from database
|
||||||
|
|
|
@ -490,7 +490,7 @@ func NewIssuePost(ctx *context.Context, form auth.CreateIssueForm) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
notification.Service.NotifyIssue(issue, ctx.User.ID)
|
notification.NotifyNewIssue(issue)
|
||||||
|
|
||||||
log.Trace("Issue created: %d/%d", repo.ID, issue.ID)
|
log.Trace("Issue created: %d/%d", repo.ID, issue.ID)
|
||||||
ctx.Redirect(ctx.Repo.RepoLink + "/issues/" + com.ToStr(issue.Index))
|
ctx.Redirect(ctx.Repo.RepoLink + "/issues/" + com.ToStr(issue.Index))
|
||||||
|
@ -1004,15 +1004,19 @@ func UpdateIssueStatus(ctx *context.Context) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
for _, issue := range issues {
|
for _, issue := range issues {
|
||||||
if err := issue.ChangeStatus(ctx.User, issue.Repo, isClosed); err != nil {
|
if issue.IsClosed != isClosed {
|
||||||
if models.IsErrDependenciesLeft(err) {
|
if err := issue.ChangeStatus(ctx.User, issue.Repo, isClosed); err != nil {
|
||||||
ctx.JSON(http.StatusPreconditionFailed, map[string]interface{}{
|
if models.IsErrDependenciesLeft(err) {
|
||||||
"error": "cannot close this issue because it still has open dependencies",
|
ctx.JSON(http.StatusPreconditionFailed, map[string]interface{}{
|
||||||
})
|
"error": "cannot close this issue because it still has open dependencies",
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.ServerError("ChangeStatus", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
ctx.ServerError("ChangeStatus", err)
|
|
||||||
return
|
notification.NotifyIssueChangeStatus(ctx.User, issue, isClosed)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ctx.JSON(200, map[string]interface{}{
|
ctx.JSON(200, map[string]interface{}{
|
||||||
|
@ -1072,7 +1076,8 @@ func NewComment(ctx *context.Context, form auth.CreateCommentForm) {
|
||||||
if pr != nil {
|
if pr != nil {
|
||||||
ctx.Flash.Info(ctx.Tr("repo.pulls.open_unmerged_pull_exists", pr.Index))
|
ctx.Flash.Info(ctx.Tr("repo.pulls.open_unmerged_pull_exists", pr.Index))
|
||||||
} else {
|
} else {
|
||||||
if err := issue.ChangeStatus(ctx.User, ctx.Repo.Repository, form.Status == "close"); err != nil {
|
isClosed := form.Status == "close"
|
||||||
|
if err := issue.ChangeStatus(ctx.User, ctx.Repo.Repository, isClosed); err != nil {
|
||||||
log.Error(4, "ChangeStatus: %v", err)
|
log.Error(4, "ChangeStatus: %v", err)
|
||||||
|
|
||||||
if models.IsErrDependenciesLeft(err) {
|
if models.IsErrDependenciesLeft(err) {
|
||||||
|
@ -1088,7 +1093,7 @@ func NewComment(ctx *context.Context, form auth.CreateCommentForm) {
|
||||||
} else {
|
} else {
|
||||||
log.Trace("Issue [%d] status changed to closed: %v", issue.ID, issue.IsClosed)
|
log.Trace("Issue [%d] status changed to closed: %v", issue.ID, issue.IsClosed)
|
||||||
|
|
||||||
notification.Service.NotifyIssue(issue, ctx.User.ID)
|
notification.NotifyIssueChangeStatus(ctx.User, issue, isClosed)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1116,7 +1121,7 @@ func NewComment(ctx *context.Context, form auth.CreateCommentForm) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
notification.Service.NotifyIssue(issue, ctx.User.ID)
|
notification.NotifyCreateIssueComment(ctx.User, ctx.Repo.Repository, issue, comment)
|
||||||
|
|
||||||
log.Trace("Comment created: %d/%d/%d", ctx.Repo.Repository.ID, issue.ID, comment.ID)
|
log.Trace("Comment created: %d/%d/%d", ctx.Repo.Repository.ID, issue.ID, comment.ID)
|
||||||
}
|
}
|
||||||
|
|
|
@ -580,7 +580,7 @@ func MergePullRequest(ctx *context.Context, form auth.MergePullRequestForm) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
notification.Service.NotifyIssue(pr.Issue, ctx.User.ID)
|
notification.NotifyMergePullRequest(pr, ctx.User, ctx.Repo.GitRepo)
|
||||||
|
|
||||||
log.Trace("Pull request merged: %d", pr.ID)
|
log.Trace("Pull request merged: %d", pr.ID)
|
||||||
ctx.Redirect(ctx.Repo.RepoLink + "/pulls/" + com.ToStr(pr.Index))
|
ctx.Redirect(ctx.Repo.RepoLink + "/pulls/" + com.ToStr(pr.Index))
|
||||||
|
@ -888,7 +888,7 @@ func CompareAndPullRequestPost(ctx *context.Context, form auth.CreateIssueForm)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
notification.Service.NotifyIssue(pullIssue, ctx.User.ID)
|
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)
|
||||||
ctx.Redirect(ctx.Repo.RepoLink + "/pulls/" + com.ToStr(pullIssue.Index))
|
ctx.Redirect(ctx.Repo.RepoLink + "/pulls/" + com.ToStr(pullIssue.Index))
|
||||||
|
|
|
@ -79,7 +79,7 @@ func CreateCodeComment(ctx *context.Context, form auth.CodeCommentForm) {
|
||||||
}
|
}
|
||||||
// Send no notification if comment is pending
|
// Send no notification if comment is pending
|
||||||
if !form.IsReview {
|
if !form.IsReview {
|
||||||
notification.Service.NotifyIssue(issue, ctx.User.ID)
|
notification.NotifyCreateIssueComment(ctx.User, issue.Repo, issue, comment)
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Trace("Comment created: %d/%d/%d", ctx.Repo.Repository.ID, issue.ID, comment.ID)
|
log.Trace("Comment created: %d/%d/%d", ctx.Repo.Repository.ID, issue.ID, comment.ID)
|
||||||
|
@ -184,5 +184,13 @@ func SubmitReview(ctx *context.Context, form auth.SubmitReviewForm) {
|
||||||
ctx.ServerError("Publish", err)
|
ctx.ServerError("Publish", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pr, err := issue.GetPullRequest()
|
||||||
|
if err != nil {
|
||||||
|
ctx.ServerError("GetPullRequest", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
notification.NotifyPullRequestReview(pr, review, comm)
|
||||||
|
|
||||||
ctx.Redirect(fmt.Sprintf("%s/pulls/%d#%s", ctx.Repo.RepoLink, issue.Index, comm.HashTag()))
|
ctx.Redirect(fmt.Sprintf("%s/pulls/%d#%s", ctx.Repo.RepoLink, issue.Index, comm.HashTag()))
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue