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.

wechatwork.go 6.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  1. // Copyright 2021 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package webhook
  4. import (
  5. "fmt"
  6. "strings"
  7. "code.gitea.io/gitea/modules/git"
  8. "code.gitea.io/gitea/modules/json"
  9. api "code.gitea.io/gitea/modules/structs"
  10. webhook_module "code.gitea.io/gitea/modules/webhook"
  11. )
  12. type (
  13. // WechatworkPayload represents
  14. WechatworkPayload struct {
  15. Msgtype string `json:"msgtype"`
  16. Text struct {
  17. Content string `json:"content"`
  18. MentionedList []string `json:"mentioned_list"`
  19. MentionedMobileList []string `json:"mentioned_mobile_list"`
  20. } `json:"text"`
  21. Markdown struct {
  22. Content string `json:"content"`
  23. } `json:"markdown"`
  24. }
  25. )
  26. // SetSecret sets the Wechatwork secret
  27. func (f *WechatworkPayload) SetSecret(_ string) {}
  28. // JSONPayload Marshals the WechatworkPayload to json
  29. func (f *WechatworkPayload) JSONPayload() ([]byte, error) {
  30. data, err := json.MarshalIndent(f, "", " ")
  31. if err != nil {
  32. return []byte{}, err
  33. }
  34. return data, nil
  35. }
  36. func newWechatworkMarkdownPayload(title string) *WechatworkPayload {
  37. return &WechatworkPayload{
  38. Msgtype: "markdown",
  39. Markdown: struct {
  40. Content string `json:"content"`
  41. }{
  42. Content: title,
  43. },
  44. }
  45. }
  46. var _ PayloadConvertor = &WechatworkPayload{}
  47. // Create implements PayloadConvertor Create method
  48. func (f *WechatworkPayload) Create(p *api.CreatePayload) (api.Payloader, error) {
  49. // created tag/branch
  50. refName := git.RefName(p.Ref).ShortName()
  51. title := fmt.Sprintf("[%s] %s %s created", p.Repo.FullName, p.RefType, refName)
  52. return newWechatworkMarkdownPayload(title), nil
  53. }
  54. // Delete implements PayloadConvertor Delete method
  55. func (f *WechatworkPayload) Delete(p *api.DeletePayload) (api.Payloader, error) {
  56. // created tag/branch
  57. refName := git.RefName(p.Ref).ShortName()
  58. title := fmt.Sprintf("[%s] %s %s deleted", p.Repo.FullName, p.RefType, refName)
  59. return newWechatworkMarkdownPayload(title), nil
  60. }
  61. // Fork implements PayloadConvertor Fork method
  62. func (f *WechatworkPayload) Fork(p *api.ForkPayload) (api.Payloader, error) {
  63. title := fmt.Sprintf("%s is forked to %s", p.Forkee.FullName, p.Repo.FullName)
  64. return newWechatworkMarkdownPayload(title), nil
  65. }
  66. // Push implements PayloadConvertor Push method
  67. func (f *WechatworkPayload) Push(p *api.PushPayload) (api.Payloader, error) {
  68. var (
  69. branchName = git.RefName(p.Ref).ShortName()
  70. commitDesc string
  71. )
  72. title := fmt.Sprintf("# %s:%s <font color=\"warning\"> %s </font>", p.Repo.FullName, branchName, commitDesc)
  73. var text string
  74. // for each commit, generate attachment text
  75. for i, commit := range p.Commits {
  76. var authorName string
  77. if commit.Author != nil {
  78. authorName = "Author: " + commit.Author.Name
  79. }
  80. message := strings.ReplaceAll(commit.Message, "\n\n", "\r\n")
  81. text += fmt.Sprintf(" > [%s](%s) \r\n ><font color=\"info\">%s</font> \n ><font color=\"warning\">%s</font>", commit.ID[:7], commit.URL,
  82. message, authorName)
  83. // add linebreak to each commit but the last
  84. if i < len(p.Commits)-1 {
  85. text += "\n"
  86. }
  87. }
  88. return newWechatworkMarkdownPayload(title + "\r\n\r\n" + text), nil
  89. }
  90. // Issue implements PayloadConvertor Issue method
  91. func (f *WechatworkPayload) Issue(p *api.IssuePayload) (api.Payloader, error) {
  92. text, issueTitle, attachmentText, _ := getIssuesPayloadInfo(p, noneLinkFormatter, true)
  93. var content string
  94. content += fmt.Sprintf(" ><font color=\"info\">%s</font>\n >%s \n ><font color=\"warning\"> %s</font> \n [%s](%s)", text, attachmentText, issueTitle, p.Issue.HTMLURL, p.Issue.HTMLURL)
  95. return newWechatworkMarkdownPayload(content), nil
  96. }
  97. // IssueComment implements PayloadConvertor IssueComment method
  98. func (f *WechatworkPayload) IssueComment(p *api.IssueCommentPayload) (api.Payloader, error) {
  99. text, issueTitle, _ := getIssueCommentPayloadInfo(p, noneLinkFormatter, true)
  100. var content string
  101. content += fmt.Sprintf(" ><font color=\"info\">%s</font>\n >%s \n ><font color=\"warning\">%s</font> \n [%s](%s)", text, p.Comment.Body, issueTitle, p.Comment.HTMLURL, p.Comment.HTMLURL)
  102. return newWechatworkMarkdownPayload(content), nil
  103. }
  104. // PullRequest implements PayloadConvertor PullRequest method
  105. func (f *WechatworkPayload) PullRequest(p *api.PullRequestPayload) (api.Payloader, error) {
  106. text, issueTitle, attachmentText, _ := getPullRequestPayloadInfo(p, noneLinkFormatter, true)
  107. pr := fmt.Sprintf("> <font color=\"info\"> %s </font> \r\n > <font color=\"comment\">%s </font> \r\n > <font color=\"comment\">%s </font> \r\n",
  108. text, issueTitle, attachmentText)
  109. return newWechatworkMarkdownPayload(pr), nil
  110. }
  111. // Review implements PayloadConvertor Review method
  112. func (f *WechatworkPayload) Review(p *api.PullRequestPayload, event webhook_module.HookEventType) (api.Payloader, error) {
  113. var text, title string
  114. switch p.Action {
  115. case api.HookIssueReviewed:
  116. action, err := parseHookPullRequestEventType(event)
  117. if err != nil {
  118. return nil, err
  119. }
  120. title = fmt.Sprintf("[%s] Pull request review %s : #%d %s", p.Repository.FullName, action, p.Index, p.PullRequest.Title)
  121. text = p.Review.Content
  122. }
  123. return newWechatworkMarkdownPayload("# " + title + "\r\n\r\n >" + text), nil
  124. }
  125. // Repository implements PayloadConvertor Repository method
  126. func (f *WechatworkPayload) Repository(p *api.RepositoryPayload) (api.Payloader, error) {
  127. var title string
  128. switch p.Action {
  129. case api.HookRepoCreated:
  130. title = fmt.Sprintf("[%s] Repository created", p.Repository.FullName)
  131. return newWechatworkMarkdownPayload(title), nil
  132. case api.HookRepoDeleted:
  133. title = fmt.Sprintf("[%s] Repository deleted", p.Repository.FullName)
  134. return newWechatworkMarkdownPayload(title), nil
  135. }
  136. return nil, nil
  137. }
  138. // Wiki implements PayloadConvertor Wiki method
  139. func (f *WechatworkPayload) Wiki(p *api.WikiPayload) (api.Payloader, error) {
  140. text, _, _ := getWikiPayloadInfo(p, noneLinkFormatter, true)
  141. return newWechatworkMarkdownPayload(text), nil
  142. }
  143. // Release implements PayloadConvertor Release method
  144. func (f *WechatworkPayload) Release(p *api.ReleasePayload) (api.Payloader, error) {
  145. text, _ := getReleasePayloadInfo(p, noneLinkFormatter, true)
  146. return newWechatworkMarkdownPayload(text), nil
  147. }
  148. func (f *WechatworkPayload) Package(p *api.PackagePayload) (api.Payloader, error) {
  149. text, _ := getPackagePayloadInfo(p, noneLinkFormatter, true)
  150. return newWechatworkMarkdownPayload(text), nil
  151. }
  152. // GetWechatworkPayload GetWechatworkPayload converts a ding talk webhook into a WechatworkPayload
  153. func GetWechatworkPayload(p api.Payloader, event webhook_module.HookEventType, _ string) (api.Payloader, error) {
  154. return convertPayloader(new(WechatworkPayload), p, event)
  155. }