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

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