gitea源码

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. // Copyright 2017 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package webhook
  4. import (
  5. "context"
  6. "fmt"
  7. "net/http"
  8. "net/url"
  9. "strings"
  10. webhook_model "code.gitea.io/gitea/models/webhook"
  11. "code.gitea.io/gitea/modules/git"
  12. api "code.gitea.io/gitea/modules/structs"
  13. "code.gitea.io/gitea/modules/util"
  14. webhook_module "code.gitea.io/gitea/modules/webhook"
  15. dingtalk "gitea.com/lunny/dingtalk_webhook"
  16. )
  17. type (
  18. DingtalkPayload dingtalk.Payload
  19. dingtalkConvertor struct{}
  20. )
  21. // Create implements PayloadConvertor Create method
  22. func (dc dingtalkConvertor) Create(p *api.CreatePayload) (DingtalkPayload, error) {
  23. // created tag/branch
  24. refName := git.RefName(p.Ref).ShortName()
  25. title := fmt.Sprintf("[%s] %s %s created", p.Repo.FullName, p.RefType, refName)
  26. return createDingtalkPayload(title, title, "view ref "+refName, p.Repo.HTMLURL+"/src/"+util.PathEscapeSegments(refName)), nil
  27. }
  28. // Delete implements PayloadConvertor Delete method
  29. func (dc dingtalkConvertor) Delete(p *api.DeletePayload) (DingtalkPayload, error) {
  30. // created tag/branch
  31. refName := git.RefName(p.Ref).ShortName()
  32. title := fmt.Sprintf("[%s] %s %s deleted", p.Repo.FullName, p.RefType, refName)
  33. return createDingtalkPayload(title, title, "view ref "+refName, p.Repo.HTMLURL+"/src/"+util.PathEscapeSegments(refName)), nil
  34. }
  35. // Fork implements PayloadConvertor Fork method
  36. func (dc dingtalkConvertor) Fork(p *api.ForkPayload) (DingtalkPayload, error) {
  37. title := fmt.Sprintf("%s is forked to %s", p.Forkee.FullName, p.Repo.FullName)
  38. return createDingtalkPayload(title, title, "view forked repo "+p.Repo.FullName, p.Repo.HTMLURL), nil
  39. }
  40. // Push implements PayloadConvertor Push method
  41. func (dc dingtalkConvertor) Push(p *api.PushPayload) (DingtalkPayload, error) {
  42. var (
  43. branchName = git.RefName(p.Ref).ShortName()
  44. commitDesc string
  45. )
  46. var titleLink, linkText string
  47. if p.TotalCommits == 1 {
  48. commitDesc = "1 new commit"
  49. titleLink = p.Commits[0].URL
  50. linkText = "view commit"
  51. } else {
  52. commitDesc = fmt.Sprintf("%d new commits", p.TotalCommits)
  53. titleLink = p.CompareURL
  54. linkText = "view commits"
  55. }
  56. if titleLink == "" {
  57. titleLink = p.Repo.HTMLURL + "/src/" + util.PathEscapeSegments(branchName)
  58. }
  59. title := fmt.Sprintf("[%s:%s] %s", p.Repo.FullName, branchName, commitDesc)
  60. var text string
  61. // for each commit, generate attachment text
  62. for i, commit := range p.Commits {
  63. var authorName string
  64. if commit.Author != nil {
  65. authorName = " - " + commit.Author.Name
  66. }
  67. text += fmt.Sprintf("[%s](%s) %s", commit.ID[:7], commit.URL,
  68. strings.TrimRight(commit.Message, "\r\n")) + authorName
  69. // add linebreak to each commit but the last
  70. if i < len(p.Commits)-1 {
  71. text += "\r\n"
  72. }
  73. }
  74. return createDingtalkPayload(title, text, linkText, titleLink), nil
  75. }
  76. // Issue implements PayloadConvertor Issue method
  77. func (dc dingtalkConvertor) Issue(p *api.IssuePayload) (DingtalkPayload, error) {
  78. text, issueTitle, extraMarkdown, _ := getIssuesPayloadInfo(p, noneLinkFormatter, true)
  79. return createDingtalkPayload(issueTitle, text+"\r\n\r\n"+extraMarkdown, "view issue", p.Issue.HTMLURL), nil
  80. }
  81. // Wiki implements PayloadConvertor Wiki method
  82. func (dc dingtalkConvertor) Wiki(p *api.WikiPayload) (DingtalkPayload, error) {
  83. text, _, _ := getWikiPayloadInfo(p, noneLinkFormatter, true)
  84. url := p.Repository.HTMLURL + "/wiki/" + url.PathEscape(p.Page)
  85. return createDingtalkPayload(text, text, "view wiki", url), nil
  86. }
  87. // IssueComment implements PayloadConvertor IssueComment method
  88. func (dc dingtalkConvertor) IssueComment(p *api.IssueCommentPayload) (DingtalkPayload, error) {
  89. text, issueTitle, _ := getIssueCommentPayloadInfo(p, noneLinkFormatter, true)
  90. return createDingtalkPayload(issueTitle, text+"\r\n\r\n"+p.Comment.Body, "view issue comment", p.Comment.HTMLURL), nil
  91. }
  92. // PullRequest implements PayloadConvertor PullRequest method
  93. func (dc dingtalkConvertor) PullRequest(p *api.PullRequestPayload) (DingtalkPayload, error) {
  94. text, issueTitle, extraMarkdown, _ := getPullRequestPayloadInfo(p, noneLinkFormatter, true)
  95. return createDingtalkPayload(issueTitle, text+"\r\n\r\n"+extraMarkdown, "view pull request", p.PullRequest.HTMLURL), nil
  96. }
  97. // Review implements PayloadConvertor Review method
  98. func (dc dingtalkConvertor) Review(p *api.PullRequestPayload, event webhook_module.HookEventType) (DingtalkPayload, error) {
  99. var text, title string
  100. switch p.Action {
  101. case api.HookIssueReviewed:
  102. action, err := parseHookPullRequestEventType(event)
  103. if err != nil {
  104. return DingtalkPayload{}, err
  105. }
  106. title = fmt.Sprintf("[%s] Pull request review %s : #%d %s", p.Repository.FullName, action, p.Index, p.PullRequest.Title)
  107. text = p.Review.Content
  108. }
  109. return createDingtalkPayload(title, title+"\r\n\r\n"+text, "view pull request", p.PullRequest.HTMLURL), nil
  110. }
  111. // Repository implements PayloadConvertor Repository method
  112. func (dc dingtalkConvertor) Repository(p *api.RepositoryPayload) (DingtalkPayload, error) {
  113. switch p.Action {
  114. case api.HookRepoCreated:
  115. title := fmt.Sprintf("[%s] Repository created", p.Repository.FullName)
  116. return createDingtalkPayload(title, title, "view repository", p.Repository.HTMLURL), nil
  117. case api.HookRepoDeleted:
  118. title := fmt.Sprintf("[%s] Repository deleted", p.Repository.FullName)
  119. return DingtalkPayload{
  120. MsgType: "text",
  121. Text: struct {
  122. Content string `json:"content"`
  123. }{
  124. Content: title,
  125. },
  126. }, nil
  127. }
  128. return DingtalkPayload{}, nil
  129. }
  130. // Release implements PayloadConvertor Release method
  131. func (dc dingtalkConvertor) Release(p *api.ReleasePayload) (DingtalkPayload, error) {
  132. text, _ := getReleasePayloadInfo(p, noneLinkFormatter, true)
  133. return createDingtalkPayload(text, text, "view release", p.Release.HTMLURL), nil
  134. }
  135. func (dc dingtalkConvertor) Package(p *api.PackagePayload) (DingtalkPayload, error) {
  136. text, _ := getPackagePayloadInfo(p, noneLinkFormatter, true)
  137. return createDingtalkPayload(text, text, "view package", p.Package.HTMLURL), nil
  138. }
  139. func (dc dingtalkConvertor) Status(p *api.CommitStatusPayload) (DingtalkPayload, error) {
  140. text, _ := getStatusPayloadInfo(p, noneLinkFormatter, true)
  141. return createDingtalkPayload(text, text, "Status Changed", p.TargetURL), nil
  142. }
  143. func (dingtalkConvertor) WorkflowRun(p *api.WorkflowRunPayload) (DingtalkPayload, error) {
  144. text, _ := getWorkflowRunPayloadInfo(p, noneLinkFormatter, true)
  145. return createDingtalkPayload(text, text, "Workflow Run", p.WorkflowRun.HTMLURL), nil
  146. }
  147. func (dingtalkConvertor) WorkflowJob(p *api.WorkflowJobPayload) (DingtalkPayload, error) {
  148. text, _ := getWorkflowJobPayloadInfo(p, noneLinkFormatter, true)
  149. return createDingtalkPayload(text, text, "Workflow Job", p.WorkflowJob.HTMLURL), nil
  150. }
  151. func createDingtalkPayload(title, text, singleTitle, singleURL string) DingtalkPayload {
  152. return DingtalkPayload{
  153. MsgType: "actionCard",
  154. ActionCard: dingtalk.ActionCard{
  155. Text: strings.TrimSpace(text),
  156. Title: strings.TrimSpace(title),
  157. HideAvatar: "0",
  158. SingleTitle: singleTitle,
  159. // https://developers.dingtalk.com/document/app/message-link-description
  160. // to open the link in browser, we should use this URL, otherwise the page is displayed inside DingTalk client, very difficult to visit non-public URLs.
  161. SingleURL: "dingtalk://dingtalkclient/page/link?pc_slide=false&url=" + url.QueryEscape(singleURL),
  162. },
  163. }
  164. }
  165. func newDingtalkRequest(_ context.Context, w *webhook_model.Webhook, t *webhook_model.HookTask) (*http.Request, []byte, error) {
  166. var pc payloadConvertor[DingtalkPayload] = dingtalkConvertor{}
  167. return newJSONRequest(pc, w, t, true)
  168. }
  169. func init() {
  170. RegisterWebhookRequester(webhook_module.DINGTALK, newDingtalkRequest)
  171. }