You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

dingtalk.go 6.9KB

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