mirror of
				https://github.com/go-gitea/gitea
				synced 2025-10-26 08:58:24 +00:00 
			
		
		
		
	refactor: mv modules/notification/actions to services/actions
This commit is contained in:
		| @@ -1,527 +0,0 @@ | ||||
| // Copyright 2022 The Gitea Authors. All rights reserved. | ||||
| // SPDX-License-Identifier: MIT | ||||
|  | ||||
| package actions | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
|  | ||||
| 	"code.gitea.io/gitea/models/db" | ||||
| 	issues_model "code.gitea.io/gitea/models/issues" | ||||
| 	packages_model "code.gitea.io/gitea/models/packages" | ||||
| 	perm_model "code.gitea.io/gitea/models/perm" | ||||
| 	access_model "code.gitea.io/gitea/models/perm/access" | ||||
| 	repo_model "code.gitea.io/gitea/models/repo" | ||||
| 	user_model "code.gitea.io/gitea/models/user" | ||||
| 	"code.gitea.io/gitea/models/webhook" | ||||
| 	"code.gitea.io/gitea/modules/git" | ||||
| 	"code.gitea.io/gitea/modules/log" | ||||
| 	"code.gitea.io/gitea/modules/notification/base" | ||||
| 	"code.gitea.io/gitea/modules/repository" | ||||
| 	"code.gitea.io/gitea/modules/setting" | ||||
| 	api "code.gitea.io/gitea/modules/structs" | ||||
| 	"code.gitea.io/gitea/services/convert" | ||||
| ) | ||||
|  | ||||
| type actionsNotifier struct { | ||||
| 	base.NullNotifier | ||||
| } | ||||
|  | ||||
| var _ base.Notifier = &actionsNotifier{} | ||||
|  | ||||
| // NewNotifier create a new actionsNotifier notifier | ||||
| func NewNotifier() base.Notifier { | ||||
| 	return &actionsNotifier{} | ||||
| } | ||||
|  | ||||
| // NotifyNewIssue notifies issue created event | ||||
| func (n *actionsNotifier) NotifyNewIssue(ctx context.Context, issue *issues_model.Issue, _ []*user_model.User) { | ||||
| 	ctx = withMethod(ctx, "NotifyNewIssue") | ||||
| 	if err := issue.LoadRepo(ctx); err != nil { | ||||
| 		log.Error("issue.LoadRepo: %v", err) | ||||
| 		return | ||||
| 	} | ||||
| 	if err := issue.LoadPoster(ctx); err != nil { | ||||
| 		log.Error("issue.LoadPoster: %v", err) | ||||
| 		return | ||||
| 	} | ||||
| 	mode, _ := access_model.AccessLevel(ctx, issue.Poster, issue.Repo) | ||||
|  | ||||
| 	newNotifyInputFromIssue(issue, webhook.HookEventIssues).WithPayload(&api.IssuePayload{ | ||||
| 		Action:     api.HookIssueOpened, | ||||
| 		Index:      issue.Index, | ||||
| 		Issue:      convert.ToAPIIssue(ctx, issue), | ||||
| 		Repository: convert.ToRepo(ctx, issue.Repo, mode), | ||||
| 		Sender:     convert.ToUser(issue.Poster, nil), | ||||
| 	}).Notify(withMethod(ctx, "NotifyNewIssue")) | ||||
| } | ||||
|  | ||||
| // NotifyIssueChangeStatus notifies close or reopen issue to notifiers | ||||
| func (n *actionsNotifier) NotifyIssueChangeStatus(ctx context.Context, doer *user_model.User, issue *issues_model.Issue, _ *issues_model.Comment, isClosed bool) { | ||||
| 	ctx = withMethod(ctx, "NotifyIssueChangeStatus") | ||||
| 	mode, _ := access_model.AccessLevel(ctx, issue.Poster, issue.Repo) | ||||
| 	if issue.IsPull { | ||||
| 		if err := issue.LoadPullRequest(ctx); err != nil { | ||||
| 			log.Error("LoadPullRequest: %v", err) | ||||
| 			return | ||||
| 		} | ||||
| 		// Merge pull request calls issue.changeStatus so we need to handle separately. | ||||
| 		apiPullRequest := &api.PullRequestPayload{ | ||||
| 			Index:       issue.Index, | ||||
| 			PullRequest: convert.ToAPIPullRequest(db.DefaultContext, issue.PullRequest, nil), | ||||
| 			Repository:  convert.ToRepo(ctx, issue.Repo, mode), | ||||
| 			Sender:      convert.ToUser(doer, nil), | ||||
| 		} | ||||
| 		if isClosed { | ||||
| 			apiPullRequest.Action = api.HookIssueClosed | ||||
| 		} else { | ||||
| 			apiPullRequest.Action = api.HookIssueReOpened | ||||
| 		} | ||||
| 		newNotifyInputFromIssue(issue, webhook.HookEventPullRequest). | ||||
| 			WithDoer(doer). | ||||
| 			WithPayload(apiPullRequest). | ||||
| 			Notify(ctx) | ||||
| 		return | ||||
| 	} | ||||
| 	apiIssue := &api.IssuePayload{ | ||||
| 		Index:      issue.Index, | ||||
| 		Issue:      convert.ToAPIIssue(ctx, issue), | ||||
| 		Repository: convert.ToRepo(ctx, issue.Repo, mode), | ||||
| 		Sender:     convert.ToUser(doer, nil), | ||||
| 	} | ||||
| 	if isClosed { | ||||
| 		apiIssue.Action = api.HookIssueClosed | ||||
| 	} else { | ||||
| 		apiIssue.Action = api.HookIssueReOpened | ||||
| 	} | ||||
| 	newNotifyInputFromIssue(issue, webhook.HookEventIssues). | ||||
| 		WithDoer(doer). | ||||
| 		WithPayload(apiIssue). | ||||
| 		Notify(ctx) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyIssueChangeLabels(ctx context.Context, doer *user_model.User, issue *issues_model.Issue, | ||||
| 	_, _ []*issues_model.Label, | ||||
| ) { | ||||
| 	ctx = withMethod(ctx, "NotifyIssueChangeLabels") | ||||
|  | ||||
| 	var err error | ||||
| 	if err = issue.LoadRepo(ctx); err != nil { | ||||
| 		log.Error("LoadRepo: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	if err = issue.LoadPoster(ctx); err != nil { | ||||
| 		log.Error("LoadPoster: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	mode, _ := access_model.AccessLevel(ctx, issue.Poster, issue.Repo) | ||||
| 	if issue.IsPull { | ||||
| 		if err = issue.LoadPullRequest(ctx); err != nil { | ||||
| 			log.Error("loadPullRequest: %v", err) | ||||
| 			return | ||||
| 		} | ||||
| 		if err = issue.PullRequest.LoadIssue(ctx); err != nil { | ||||
| 			log.Error("LoadIssue: %v", err) | ||||
| 			return | ||||
| 		} | ||||
| 		newNotifyInputFromIssue(issue, webhook.HookEventPullRequestLabel). | ||||
| 			WithDoer(doer). | ||||
| 			WithPayload(&api.PullRequestPayload{ | ||||
| 				Action:      api.HookIssueLabelUpdated, | ||||
| 				Index:       issue.Index, | ||||
| 				PullRequest: convert.ToAPIPullRequest(ctx, issue.PullRequest, nil), | ||||
| 				Repository:  convert.ToRepo(ctx, issue.Repo, perm_model.AccessModeNone), | ||||
| 				Sender:      convert.ToUser(doer, nil), | ||||
| 			}). | ||||
| 			Notify(ctx) | ||||
| 		return | ||||
| 	} | ||||
| 	newNotifyInputFromIssue(issue, webhook.HookEventIssueLabel). | ||||
| 		WithDoer(doer). | ||||
| 		WithPayload(&api.IssuePayload{ | ||||
| 			Action:     api.HookIssueLabelUpdated, | ||||
| 			Index:      issue.Index, | ||||
| 			Issue:      convert.ToAPIIssue(ctx, issue), | ||||
| 			Repository: convert.ToRepo(ctx, issue.Repo, mode), | ||||
| 			Sender:     convert.ToUser(doer, nil), | ||||
| 		}). | ||||
| 		Notify(ctx) | ||||
| } | ||||
|  | ||||
| // NotifyCreateIssueComment notifies comment on an issue to notifiers | ||||
| func (n *actionsNotifier) NotifyCreateIssueComment(ctx context.Context, doer *user_model.User, repo *repo_model.Repository, | ||||
| 	issue *issues_model.Issue, comment *issues_model.Comment, _ []*user_model.User, | ||||
| ) { | ||||
| 	ctx = withMethod(ctx, "NotifyCreateIssueComment") | ||||
|  | ||||
| 	mode, _ := access_model.AccessLevel(ctx, doer, repo) | ||||
|  | ||||
| 	if issue.IsPull { | ||||
| 		newNotifyInputFromIssue(issue, webhook.HookEventPullRequestComment). | ||||
| 			WithDoer(doer). | ||||
| 			WithPayload(&api.IssueCommentPayload{ | ||||
| 				Action:     api.HookIssueCommentCreated, | ||||
| 				Issue:      convert.ToAPIIssue(ctx, issue), | ||||
| 				Comment:    convert.ToComment(comment), | ||||
| 				Repository: convert.ToRepo(ctx, repo, mode), | ||||
| 				Sender:     convert.ToUser(doer, nil), | ||||
| 				IsPull:     true, | ||||
| 			}). | ||||
| 			Notify(ctx) | ||||
| 		return | ||||
| 	} | ||||
| 	newNotifyInputFromIssue(issue, webhook.HookEventIssueComment). | ||||
| 		WithDoer(doer). | ||||
| 		WithPayload(&api.IssueCommentPayload{ | ||||
| 			Action:     api.HookIssueCommentCreated, | ||||
| 			Issue:      convert.ToAPIIssue(ctx, issue), | ||||
| 			Comment:    convert.ToComment(comment), | ||||
| 			Repository: convert.ToRepo(ctx, repo, mode), | ||||
| 			Sender:     convert.ToUser(doer, nil), | ||||
| 			IsPull:     false, | ||||
| 		}). | ||||
| 		Notify(ctx) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyNewPullRequest(ctx context.Context, pull *issues_model.PullRequest, _ []*user_model.User) { | ||||
| 	ctx = withMethod(ctx, "NotifyNewPullRequest") | ||||
|  | ||||
| 	if err := pull.LoadIssue(ctx); err != nil { | ||||
| 		log.Error("pull.LoadIssue: %v", err) | ||||
| 		return | ||||
| 	} | ||||
| 	if err := pull.Issue.LoadRepo(ctx); err != nil { | ||||
| 		log.Error("pull.Issue.LoadRepo: %v", err) | ||||
| 		return | ||||
| 	} | ||||
| 	if err := pull.Issue.LoadPoster(ctx); err != nil { | ||||
| 		log.Error("pull.Issue.LoadPoster: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	mode, _ := access_model.AccessLevel(ctx, pull.Issue.Poster, pull.Issue.Repo) | ||||
|  | ||||
| 	newNotifyInputFromIssue(pull.Issue, webhook.HookEventPullRequest). | ||||
| 		WithPayload(&api.PullRequestPayload{ | ||||
| 			Action:      api.HookIssueOpened, | ||||
| 			Index:       pull.Issue.Index, | ||||
| 			PullRequest: convert.ToAPIPullRequest(ctx, pull, nil), | ||||
| 			Repository:  convert.ToRepo(ctx, pull.Issue.Repo, mode), | ||||
| 			Sender:      convert.ToUser(pull.Issue.Poster, nil), | ||||
| 		}). | ||||
| 		WithPullRequest(pull). | ||||
| 		Notify(ctx) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyCreateRepository(ctx context.Context, doer, u *user_model.User, repo *repo_model.Repository) { | ||||
| 	ctx = withMethod(ctx, "NotifyCreateRepository") | ||||
|  | ||||
| 	newNotifyInput(repo, doer, webhook.HookEventRepository).WithPayload(&api.RepositoryPayload{ | ||||
| 		Action:       api.HookRepoCreated, | ||||
| 		Repository:   convert.ToRepo(ctx, repo, perm_model.AccessModeOwner), | ||||
| 		Organization: convert.ToUser(u, nil), | ||||
| 		Sender:       convert.ToUser(doer, nil), | ||||
| 	}).Notify(ctx) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyForkRepository(ctx context.Context, doer *user_model.User, oldRepo, repo *repo_model.Repository) { | ||||
| 	ctx = withMethod(ctx, "NotifyForkRepository") | ||||
|  | ||||
| 	oldMode, _ := access_model.AccessLevel(ctx, doer, oldRepo) | ||||
| 	mode, _ := access_model.AccessLevel(ctx, doer, repo) | ||||
|  | ||||
| 	// forked webhook | ||||
| 	newNotifyInput(oldRepo, doer, webhook.HookEventFork).WithPayload(&api.ForkPayload{ | ||||
| 		Forkee: convert.ToRepo(ctx, oldRepo, oldMode), | ||||
| 		Repo:   convert.ToRepo(ctx, repo, mode), | ||||
| 		Sender: convert.ToUser(doer, nil), | ||||
| 	}).Notify(ctx) | ||||
|  | ||||
| 	u := repo.MustOwner(ctx) | ||||
|  | ||||
| 	// Add to hook queue for created repo after session commit. | ||||
| 	if u.IsOrganization() { | ||||
| 		newNotifyInput(repo, doer, webhook.HookEventRepository). | ||||
| 			WithRef(oldRepo.DefaultBranch). | ||||
| 			WithPayload(&api.RepositoryPayload{ | ||||
| 				Action:       api.HookRepoCreated, | ||||
| 				Repository:   convert.ToRepo(ctx, repo, perm_model.AccessModeOwner), | ||||
| 				Organization: convert.ToUser(u, nil), | ||||
| 				Sender:       convert.ToUser(doer, nil), | ||||
| 			}).Notify(ctx) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyPullRequestReview(ctx context.Context, pr *issues_model.PullRequest, review *issues_model.Review, _ *issues_model.Comment, _ []*user_model.User) { | ||||
| 	ctx = withMethod(ctx, "NotifyPullRequestReview") | ||||
|  | ||||
| 	var reviewHookType webhook.HookEventType | ||||
|  | ||||
| 	switch review.Type { | ||||
| 	case issues_model.ReviewTypeApprove: | ||||
| 		reviewHookType = webhook.HookEventPullRequestReviewApproved | ||||
| 	case issues_model.ReviewTypeComment: | ||||
| 		reviewHookType = webhook.HookEventPullRequestComment | ||||
| 	case issues_model.ReviewTypeReject: | ||||
| 		reviewHookType = webhook.HookEventPullRequestReviewRejected | ||||
| 	default: | ||||
| 		// unsupported review webhook type here | ||||
| 		log.Error("Unsupported review webhook type") | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	if err := pr.LoadIssue(ctx); err != nil { | ||||
| 		log.Error("pr.LoadIssue: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	mode, err := access_model.AccessLevel(ctx, review.Issue.Poster, review.Issue.Repo) | ||||
| 	if err != nil { | ||||
| 		log.Error("models.AccessLevel: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	newNotifyInput(review.Issue.Repo, review.Reviewer, reviewHookType). | ||||
| 		WithRef(review.CommitID). | ||||
| 		WithPayload(&api.PullRequestPayload{ | ||||
| 			Action:      api.HookIssueReviewed, | ||||
| 			Index:       review.Issue.Index, | ||||
| 			PullRequest: convert.ToAPIPullRequest(db.DefaultContext, pr, nil), | ||||
| 			Repository:  convert.ToRepo(ctx, review.Issue.Repo, mode), | ||||
| 			Sender:      convert.ToUser(review.Reviewer, nil), | ||||
| 			Review: &api.ReviewPayload{ | ||||
| 				Type:    string(reviewHookType), | ||||
| 				Content: review.Content, | ||||
| 			}, | ||||
| 		}).Notify(ctx) | ||||
| } | ||||
|  | ||||
| func (*actionsNotifier) NotifyMergePullRequest(ctx context.Context, doer *user_model.User, pr *issues_model.PullRequest) { | ||||
| 	ctx = withMethod(ctx, "NotifyMergePullRequest") | ||||
|  | ||||
| 	// Reload pull request information. | ||||
| 	if err := pr.LoadAttributes(ctx); err != nil { | ||||
| 		log.Error("LoadAttributes: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	if err := pr.LoadIssue(ctx); err != nil { | ||||
| 		log.Error("LoadAttributes: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	if err := pr.Issue.LoadRepo(db.DefaultContext); err != nil { | ||||
| 		log.Error("pr.Issue.LoadRepo: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	mode, err := access_model.AccessLevel(ctx, doer, pr.Issue.Repo) | ||||
| 	if err != nil { | ||||
| 		log.Error("models.AccessLevel: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	// Merge pull request calls issue.changeStatus so we need to handle separately. | ||||
| 	apiPullRequest := &api.PullRequestPayload{ | ||||
| 		Index:       pr.Issue.Index, | ||||
| 		PullRequest: convert.ToAPIPullRequest(db.DefaultContext, pr, nil), | ||||
| 		Repository:  convert.ToRepo(ctx, pr.Issue.Repo, mode), | ||||
| 		Sender:      convert.ToUser(doer, nil), | ||||
| 		Action:      api.HookIssueClosed, | ||||
| 	} | ||||
|  | ||||
| 	newNotifyInput(pr.Issue.Repo, doer, webhook.HookEventPullRequest). | ||||
| 		WithRef(pr.MergedCommitID). | ||||
| 		WithPayload(apiPullRequest). | ||||
| 		WithPullRequest(pr). | ||||
| 		Notify(ctx) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyPushCommits(ctx context.Context, pusher *user_model.User, repo *repo_model.Repository, opts *repository.PushUpdateOptions, commits *repository.PushCommits) { | ||||
| 	ctx = withMethod(ctx, "NotifyPushCommits") | ||||
|  | ||||
| 	apiPusher := convert.ToUser(pusher, nil) | ||||
| 	apiCommits, apiHeadCommit, err := commits.ToAPIPayloadCommits(ctx, repo.RepoPath(), repo.HTMLURL()) | ||||
| 	if err != nil { | ||||
| 		log.Error("commits.ToAPIPayloadCommits failed: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	newNotifyInput(repo, pusher, webhook.HookEventPush). | ||||
| 		WithRef(opts.RefFullName). | ||||
| 		WithPayload(&api.PushPayload{ | ||||
| 			Ref:        opts.RefFullName, | ||||
| 			Before:     opts.OldCommitID, | ||||
| 			After:      opts.NewCommitID, | ||||
| 			CompareURL: setting.AppURL + commits.CompareURL, | ||||
| 			Commits:    apiCommits, | ||||
| 			HeadCommit: apiHeadCommit, | ||||
| 			Repo:       convert.ToRepo(ctx, repo, perm_model.AccessModeOwner), | ||||
| 			Pusher:     apiPusher, | ||||
| 			Sender:     apiPusher, | ||||
| 		}). | ||||
| 		Notify(ctx) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyCreateRef(ctx context.Context, pusher *user_model.User, repo *repo_model.Repository, refType, refFullName, refID string) { | ||||
| 	ctx = withMethod(ctx, "NotifyCreateRef") | ||||
|  | ||||
| 	apiPusher := convert.ToUser(pusher, nil) | ||||
| 	apiRepo := convert.ToRepo(ctx, repo, perm_model.AccessModeNone) | ||||
| 	refName := git.RefEndName(refFullName) | ||||
|  | ||||
| 	newNotifyInput(repo, pusher, webhook.HookEventCreate). | ||||
| 		WithRef(refName). | ||||
| 		WithPayload(&api.CreatePayload{ | ||||
| 			Ref:     refName, | ||||
| 			Sha:     refID, | ||||
| 			RefType: refType, | ||||
| 			Repo:    apiRepo, | ||||
| 			Sender:  apiPusher, | ||||
| 		}). | ||||
| 		Notify(ctx) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyDeleteRef(ctx context.Context, pusher *user_model.User, repo *repo_model.Repository, refType, refFullName string) { | ||||
| 	ctx = withMethod(ctx, "NotifyDeleteRef") | ||||
|  | ||||
| 	apiPusher := convert.ToUser(pusher, nil) | ||||
| 	apiRepo := convert.ToRepo(ctx, repo, perm_model.AccessModeNone) | ||||
| 	refName := git.RefEndName(refFullName) | ||||
|  | ||||
| 	newNotifyInput(repo, pusher, webhook.HookEventDelete). | ||||
| 		WithRef(refName). | ||||
| 		WithPayload(&api.DeletePayload{ | ||||
| 			Ref:        refName, | ||||
| 			RefType:    refType, | ||||
| 			PusherType: api.PusherTypeUser, | ||||
| 			Repo:       apiRepo, | ||||
| 			Sender:     apiPusher, | ||||
| 		}). | ||||
| 		Notify(ctx) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifySyncPushCommits(ctx context.Context, pusher *user_model.User, repo *repo_model.Repository, opts *repository.PushUpdateOptions, commits *repository.PushCommits) { | ||||
| 	ctx = withMethod(ctx, "NotifySyncPushCommits") | ||||
|  | ||||
| 	apiPusher := convert.ToUser(pusher, nil) | ||||
| 	apiCommits, apiHeadCommit, err := commits.ToAPIPayloadCommits(db.DefaultContext, repo.RepoPath(), repo.HTMLURL()) | ||||
| 	if err != nil { | ||||
| 		log.Error("commits.ToAPIPayloadCommits failed: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	newNotifyInput(repo, pusher, webhook.HookEventPush). | ||||
| 		WithRef(opts.RefFullName). | ||||
| 		WithPayload(&api.PushPayload{ | ||||
| 			Ref:          opts.RefFullName, | ||||
| 			Before:       opts.OldCommitID, | ||||
| 			After:        opts.NewCommitID, | ||||
| 			CompareURL:   setting.AppURL + commits.CompareURL, | ||||
| 			Commits:      apiCommits, | ||||
| 			TotalCommits: commits.Len, | ||||
| 			HeadCommit:   apiHeadCommit, | ||||
| 			Repo:         convert.ToRepo(ctx, repo, perm_model.AccessModeOwner), | ||||
| 			Pusher:       apiPusher, | ||||
| 			Sender:       apiPusher, | ||||
| 		}). | ||||
| 		Notify(ctx) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifySyncCreateRef(ctx context.Context, pusher *user_model.User, repo *repo_model.Repository, refType, refFullName, refID string) { | ||||
| 	ctx = withMethod(ctx, "NotifySyncCreateRef") | ||||
| 	n.NotifyCreateRef(ctx, pusher, repo, refType, refFullName, refID) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifySyncDeleteRef(ctx context.Context, pusher *user_model.User, repo *repo_model.Repository, refType, refFullName string) { | ||||
| 	ctx = withMethod(ctx, "NotifySyncDeleteRef") | ||||
| 	n.NotifyDeleteRef(ctx, pusher, repo, refType, refFullName) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyNewRelease(ctx context.Context, rel *repo_model.Release) { | ||||
| 	ctx = withMethod(ctx, "NotifyNewRelease") | ||||
| 	notifyRelease(ctx, rel.Publisher, rel, rel.Sha1, api.HookReleasePublished) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyUpdateRelease(ctx context.Context, doer *user_model.User, rel *repo_model.Release) { | ||||
| 	ctx = withMethod(ctx, "NotifyUpdateRelease") | ||||
| 	notifyRelease(ctx, doer, rel, rel.Sha1, api.HookReleaseUpdated) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyDeleteRelease(ctx context.Context, doer *user_model.User, rel *repo_model.Release) { | ||||
| 	ctx = withMethod(ctx, "NotifyDeleteRelease") | ||||
| 	notifyRelease(ctx, doer, rel, rel.Sha1, api.HookReleaseDeleted) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyPackageCreate(ctx context.Context, doer *user_model.User, pd *packages_model.PackageDescriptor) { | ||||
| 	ctx = withMethod(ctx, "NotifyPackageCreate") | ||||
| 	notifyPackage(ctx, doer, pd, api.HookPackageCreated) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyPackageDelete(ctx context.Context, doer *user_model.User, pd *packages_model.PackageDescriptor) { | ||||
| 	ctx = withMethod(ctx, "NotifyPackageDelete") | ||||
| 	notifyPackage(ctx, doer, pd, api.HookPackageDeleted) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyAutoMergePullRequest(ctx context.Context, doer *user_model.User, pr *issues_model.PullRequest) { | ||||
| 	ctx = withMethod(ctx, "NotifyAutoMergePullRequest") | ||||
| 	n.NotifyMergePullRequest(ctx, doer, pr) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyPullRequestSynchronized(ctx context.Context, doer *user_model.User, pr *issues_model.PullRequest) { | ||||
| 	ctx = withMethod(ctx, "NotifyPullRequestSynchronized") | ||||
|  | ||||
| 	if err := pr.LoadIssue(ctx); err != nil { | ||||
| 		log.Error("LoadAttributes: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	if err := pr.Issue.LoadRepo(db.DefaultContext); err != nil { | ||||
| 		log.Error("pr.Issue.LoadRepo: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	newNotifyInput(pr.Issue.Repo, doer, webhook.HookEventPullRequestSync). | ||||
| 		WithPayload(&api.PullRequestPayload{ | ||||
| 			Action:      api.HookIssueSynchronized, | ||||
| 			Index:       pr.Issue.Index, | ||||
| 			PullRequest: convert.ToAPIPullRequest(ctx, pr, nil), | ||||
| 			Repository:  convert.ToRepo(ctx, pr.Issue.Repo, perm_model.AccessModeNone), | ||||
| 			Sender:      convert.ToUser(doer, nil), | ||||
| 		}). | ||||
| 		WithPullRequest(pr). | ||||
| 		Notify(ctx) | ||||
| } | ||||
|  | ||||
| func (n *actionsNotifier) NotifyPullRequestChangeTargetBranch(ctx context.Context, doer *user_model.User, pr *issues_model.PullRequest, oldBranch string) { | ||||
| 	ctx = withMethod(ctx, "NotifyPullRequestChangeTargetBranch") | ||||
|  | ||||
| 	if err := pr.LoadIssue(ctx); err != nil { | ||||
| 		log.Error("LoadAttributes: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	if err := pr.Issue.LoadRepo(db.DefaultContext); err != nil { | ||||
| 		log.Error("pr.Issue.LoadRepo: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	mode, _ := access_model.AccessLevel(ctx, pr.Issue.Poster, pr.Issue.Repo) | ||||
| 	newNotifyInput(pr.Issue.Repo, doer, webhook.HookEventPullRequest). | ||||
| 		WithPayload(&api.PullRequestPayload{ | ||||
| 			Action: api.HookIssueEdited, | ||||
| 			Index:  pr.Issue.Index, | ||||
| 			Changes: &api.ChangesPayload{ | ||||
| 				Ref: &api.ChangesFromPayload{ | ||||
| 					From: oldBranch, | ||||
| 				}, | ||||
| 			}, | ||||
| 			PullRequest: convert.ToAPIPullRequest(ctx, pr, nil), | ||||
| 			Repository:  convert.ToRepo(ctx, pr.Issue.Repo, mode), | ||||
| 			Sender:      convert.ToUser(doer, nil), | ||||
| 		}). | ||||
| 		WithPullRequest(pr). | ||||
| 		Notify(ctx) | ||||
| } | ||||
| @@ -1,228 +0,0 @@ | ||||
| // Copyright 2022 The Gitea Authors. All rights reserved. | ||||
| // SPDX-License-Identifier: MIT | ||||
|  | ||||
| package actions | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"strings" | ||||
|  | ||||
| 	actions_model "code.gitea.io/gitea/models/actions" | ||||
| 	"code.gitea.io/gitea/models/db" | ||||
| 	issues_model "code.gitea.io/gitea/models/issues" | ||||
| 	packages_model "code.gitea.io/gitea/models/packages" | ||||
| 	access_model "code.gitea.io/gitea/models/perm/access" | ||||
| 	repo_model "code.gitea.io/gitea/models/repo" | ||||
| 	unit_model "code.gitea.io/gitea/models/unit" | ||||
| 	user_model "code.gitea.io/gitea/models/user" | ||||
| 	webhook_model "code.gitea.io/gitea/models/webhook" | ||||
| 	actions_module "code.gitea.io/gitea/modules/actions" | ||||
| 	"code.gitea.io/gitea/modules/git" | ||||
| 	"code.gitea.io/gitea/modules/json" | ||||
| 	"code.gitea.io/gitea/modules/log" | ||||
| 	api "code.gitea.io/gitea/modules/structs" | ||||
| 	"code.gitea.io/gitea/modules/util" | ||||
| 	actions_service "code.gitea.io/gitea/services/actions" | ||||
| 	"code.gitea.io/gitea/services/convert" | ||||
|  | ||||
| 	"github.com/nektos/act/pkg/jobparser" | ||||
| ) | ||||
|  | ||||
| var methodCtxKey struct{} | ||||
|  | ||||
| func withMethod(ctx context.Context, method string) context.Context { | ||||
| 	// don't overwrite | ||||
| 	if v := ctx.Value(methodCtxKey); v != nil { | ||||
| 		if _, ok := v.(string); ok { | ||||
| 			return ctx | ||||
| 		} | ||||
| 	} | ||||
| 	return context.WithValue(ctx, methodCtxKey, method) | ||||
| } | ||||
|  | ||||
| func getMethod(ctx context.Context) string { | ||||
| 	if v := ctx.Value(methodCtxKey); v != nil { | ||||
| 		if s, ok := v.(string); ok { | ||||
| 			return s | ||||
| 		} | ||||
| 	} | ||||
| 	return "notify" | ||||
| } | ||||
|  | ||||
| type notifyInput struct { | ||||
| 	// required | ||||
| 	Repo  *repo_model.Repository | ||||
| 	Doer  *user_model.User | ||||
| 	Event webhook_model.HookEventType | ||||
|  | ||||
| 	// optional | ||||
| 	Ref         string | ||||
| 	Payload     api.Payloader | ||||
| 	PullRequest *issues_model.PullRequest | ||||
| } | ||||
|  | ||||
| func newNotifyInput(repo *repo_model.Repository, doer *user_model.User, event webhook_model.HookEventType) *notifyInput { | ||||
| 	return ¬ifyInput{ | ||||
| 		Repo:  repo, | ||||
| 		Ref:   repo.DefaultBranch, | ||||
| 		Doer:  doer, | ||||
| 		Event: event, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (input *notifyInput) WithDoer(doer *user_model.User) *notifyInput { | ||||
| 	input.Doer = doer | ||||
| 	return input | ||||
| } | ||||
|  | ||||
| func (input *notifyInput) WithRef(ref string) *notifyInput { | ||||
| 	input.Ref = ref | ||||
| 	return input | ||||
| } | ||||
|  | ||||
| func (input *notifyInput) WithPayload(payload api.Payloader) *notifyInput { | ||||
| 	input.Payload = payload | ||||
| 	return input | ||||
| } | ||||
|  | ||||
| func (input *notifyInput) WithPullRequest(pr *issues_model.PullRequest) *notifyInput { | ||||
| 	input.PullRequest = pr | ||||
| 	return input | ||||
| } | ||||
|  | ||||
| func (input *notifyInput) Notify(ctx context.Context) { | ||||
| 	if err := notify(ctx, input); err != nil { | ||||
| 		log.Error("%s: %v", getMethod(ctx), err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func notify(ctx context.Context, input *notifyInput) error { | ||||
| 	if unit_model.TypeActions.UnitGlobalDisabled() { | ||||
| 		return nil | ||||
| 	} | ||||
| 	if err := input.Repo.LoadUnits(db.DefaultContext); err != nil { | ||||
| 		return fmt.Errorf("repo.LoadUnits: %w", err) | ||||
| 	} else if !input.Repo.UnitEnabled(ctx, unit_model.TypeActions) { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	gitRepo, err := git.OpenRepository(context.Background(), input.Repo.RepoPath()) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("git.OpenRepository: %w", err) | ||||
| 	} | ||||
| 	defer gitRepo.Close() | ||||
|  | ||||
| 	// Get the commit object for the ref | ||||
| 	commit, err := gitRepo.GetCommit(input.Ref) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("gitRepo.GetCommit: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	workflows, err := actions_module.DetectWorkflows(commit, input.Event) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("DetectWorkflows: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	if len(workflows) == 0 { | ||||
| 		log.Trace("repo %s with commit %s couldn't find workflows", input.Repo.RepoPath(), commit.ID) | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	p, err := json.Marshal(input.Payload) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("json.Marshal: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	for id, content := range workflows { | ||||
| 		run := actions_model.ActionRun{ | ||||
| 			Title:             truncateContent(strings.SplitN(commit.CommitMessage, "\n", 2)[0], 255), | ||||
| 			RepoID:            input.Repo.ID, | ||||
| 			OwnerID:           input.Repo.OwnerID, | ||||
| 			WorkflowID:        id, | ||||
| 			TriggerUserID:     input.Doer.ID, | ||||
| 			Ref:               input.Ref, | ||||
| 			CommitSHA:         commit.ID.String(), | ||||
| 			IsForkPullRequest: input.PullRequest != nil && input.PullRequest.IsFromFork(), | ||||
| 			Event:             input.Event, | ||||
| 			EventPayload:      string(p), | ||||
| 			Status:            actions_model.StatusWaiting, | ||||
| 		} | ||||
| 		jobs, err := jobparser.Parse(content) | ||||
| 		if err != nil { | ||||
| 			log.Error("jobparser.Parse: %v", err) | ||||
| 			continue | ||||
| 		} | ||||
| 		if err := actions_model.InsertRun(ctx, &run, jobs); err != nil { | ||||
| 			log.Error("InsertRun: %v", err) | ||||
| 			continue | ||||
| 		} | ||||
| 		if jobs, _, err := actions_model.FindRunJobs(ctx, actions_model.FindRunJobOptions{RunID: run.ID}); err != nil { | ||||
| 			log.Error("FindRunJobs: %v", err) | ||||
| 		} else { | ||||
| 			for _, job := range jobs { | ||||
| 				if err := actions_service.CreateCommitStatus(ctx, job); err != nil { | ||||
| 					log.Error("CreateCommitStatus: %v", err) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func newNotifyInputFromIssue(issue *issues_model.Issue, event webhook_model.HookEventType) *notifyInput { | ||||
| 	return newNotifyInput(issue.Repo, issue.Poster, event) | ||||
| } | ||||
|  | ||||
| func notifyRelease(ctx context.Context, doer *user_model.User, rel *repo_model.Release, ref string, action api.HookReleaseAction) { | ||||
| 	if err := rel.LoadAttributes(ctx); err != nil { | ||||
| 		log.Error("LoadAttributes: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	mode, _ := access_model.AccessLevel(ctx, doer, rel.Repo) | ||||
|  | ||||
| 	newNotifyInput(rel.Repo, doer, webhook_model.HookEventRelease). | ||||
| 		WithRef(ref). | ||||
| 		WithPayload(&api.ReleasePayload{ | ||||
| 			Action:     action, | ||||
| 			Release:    convert.ToRelease(rel), | ||||
| 			Repository: convert.ToRepo(ctx, rel.Repo, mode), | ||||
| 			Sender:     convert.ToUser(doer, nil), | ||||
| 		}). | ||||
| 		Notify(ctx) | ||||
| } | ||||
|  | ||||
| func notifyPackage(ctx context.Context, sender *user_model.User, pd *packages_model.PackageDescriptor, action api.HookPackageAction) { | ||||
| 	if pd.Repository == nil { | ||||
| 		// TODO https://github.com/go-gitea/gitea/pull/17940 | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	apiPackage, err := convert.ToPackage(ctx, pd, sender) | ||||
| 	if err != nil { | ||||
| 		log.Error("Error converting package: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	newNotifyInput(pd.Repository, sender, webhook_model.HookEventPackage). | ||||
| 		WithPayload(&api.PackagePayload{ | ||||
| 			Action:  action, | ||||
| 			Package: apiPackage, | ||||
| 			Sender:  convert.ToUser(sender, nil), | ||||
| 		}). | ||||
| 		Notify(ctx) | ||||
| } | ||||
|  | ||||
| func truncateContent(content string, n int) string { | ||||
| 	truncatedContent, truncatedRight := util.SplitStringAtByteN(content, n) | ||||
| 	if truncatedRight != "" { | ||||
| 		// in case the content is in a Latin family language, we remove the last broken word. | ||||
| 		lastSpaceIdx := strings.LastIndex(truncatedContent, " ") | ||||
| 		if lastSpaceIdx != -1 && (len(truncatedContent)-lastSpaceIdx < 15) { | ||||
| 			truncatedContent = truncatedContent[:lastSpaceIdx] + "…" | ||||
| 		} | ||||
| 	} | ||||
| 	return truncatedContent | ||||
| } | ||||
| @@ -12,7 +12,6 @@ import ( | ||||
| 	user_model "code.gitea.io/gitea/models/user" | ||||
| 	"code.gitea.io/gitea/modules/log" | ||||
| 	"code.gitea.io/gitea/modules/notification/action" | ||||
| 	"code.gitea.io/gitea/modules/notification/actions" | ||||
| 	"code.gitea.io/gitea/modules/notification/base" | ||||
| 	"code.gitea.io/gitea/modules/notification/indexer" | ||||
| 	"code.gitea.io/gitea/modules/notification/mail" | ||||
| @@ -41,9 +40,6 @@ func NewContext() { | ||||
| 	RegisterNotifier(webhook.NewNotifier()) | ||||
| 	RegisterNotifier(action.NewNotifier()) | ||||
| 	RegisterNotifier(mirror.NewNotifier()) | ||||
| 	if setting.Actions.Enabled { | ||||
| 		RegisterNotifier(actions.NewNotifier()) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // NotifyNewWikiPage notifies creating new wiki pages to notifiers | ||||
|   | ||||
		Reference in New Issue
	
	Block a user