webhook_discord.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420
  1. // Copyright 2015 - Present, The Gogs Authors. All rights reserved.
  2. // Copyright 2018 - Present, Gitote. All rights reserved.
  3. //
  4. // This source code is licensed under the MIT license found in the
  5. // LICENSE file in the root directory of this source tree.
  6. package models
  7. import (
  8. "fmt"
  9. "gitote/gitote/pkg/setting"
  10. "strconv"
  11. "strings"
  12. "github.com/json-iterator/go"
  13. "gitlab.com/gitote/git-module"
  14. api "gitlab.com/gitote/go-gitote-client"
  15. )
  16. // DiscordEmbedFooterObject for Embed Footer Structure.
  17. type DiscordEmbedFooterObject struct {
  18. Text string `json:"text"`
  19. }
  20. // DiscordEmbedAuthorObject for Embed Author Structure.
  21. type DiscordEmbedAuthorObject struct {
  22. Name string `json:"name"`
  23. URL string `json:"url"`
  24. IconURL string `json:"icon_url"`
  25. }
  26. // DiscordEmbedFieldObject for Embed Field Structure
  27. type DiscordEmbedFieldObject struct {
  28. Name string `json:"name"`
  29. Value string `json:"value"`
  30. }
  31. // DiscordEmbedObject is for Embed Structure
  32. type DiscordEmbedObject struct {
  33. Title string `json:"title"`
  34. Description string `json:"description"`
  35. URL string `json:"url"`
  36. Color int `json:"color"`
  37. Footer *DiscordEmbedFooterObject `json:"footer"`
  38. Author *DiscordEmbedAuthorObject `json:"author"`
  39. Fields []*DiscordEmbedFieldObject `json:"fields"`
  40. }
  41. // DiscordPayload represents
  42. type DiscordPayload struct {
  43. Content string `json:"content"`
  44. Username string `json:"username"`
  45. AvatarURL string `json:"avatar_url"`
  46. Embeds []*DiscordEmbedObject `json:"embeds"`
  47. }
  48. // JSONPayload Marshals the DiscordPayload to json
  49. func (p *DiscordPayload) JSONPayload() ([]byte, error) {
  50. data, err := jsoniter.MarshalIndent(p, "", " ")
  51. if err != nil {
  52. return []byte{}, err
  53. }
  54. return data, nil
  55. }
  56. // DiscordTextFormatter formats text
  57. func DiscordTextFormatter(s string) string {
  58. return strings.Split(s, "\n")[0]
  59. }
  60. // DiscordLinkFormatter formats link
  61. func DiscordLinkFormatter(url string, text string) string {
  62. return fmt.Sprintf("[%s](%s)", text, url)
  63. }
  64. // DiscordSHALinkFormatter formats SHA text
  65. func DiscordSHALinkFormatter(url string, text string) string {
  66. return fmt.Sprintf("[`%s`](%s)", text, url)
  67. }
  68. // getDiscordCreatePayload composes Discord payload for create new branch or tag.
  69. func getDiscordCreatePayload(p *api.CreatePayload) (*DiscordPayload, error) {
  70. refName := git.RefEndName(p.Ref)
  71. repoLink := DiscordLinkFormatter(p.Repo.HTMLURL, p.Repo.Name)
  72. refLink := DiscordLinkFormatter(p.Repo.HTMLURL+"/src/"+refName, refName)
  73. content := fmt.Sprintf("Created new %s: %s/%s", p.RefType, repoLink, refLink)
  74. return &DiscordPayload{
  75. Embeds: []*DiscordEmbedObject{{
  76. Description: content,
  77. URL: setting.AppURL + p.Sender.UserName,
  78. Author: &DiscordEmbedAuthorObject{
  79. Name: p.Sender.UserName,
  80. IconURL: p.Sender.AvatarUrl,
  81. },
  82. }},
  83. }, nil
  84. }
  85. // getDiscordDeletePayload composes Discord payload for delete a branch or tag.
  86. func getDiscordDeletePayload(p *api.DeletePayload) (*DiscordPayload, error) {
  87. refName := git.RefEndName(p.Ref)
  88. repoLink := DiscordLinkFormatter(p.Repo.HTMLURL, p.Repo.Name)
  89. content := fmt.Sprintf("Deleted %s: %s/%s", p.RefType, repoLink, refName)
  90. return &DiscordPayload{
  91. Embeds: []*DiscordEmbedObject{{
  92. Description: content,
  93. URL: setting.AppURL + p.Sender.UserName,
  94. Author: &DiscordEmbedAuthorObject{
  95. Name: p.Sender.UserName,
  96. IconURL: p.Sender.AvatarUrl,
  97. },
  98. }},
  99. }, nil
  100. }
  101. // getDiscordForkPayload composes Discord payload for forked by a repository.
  102. func getDiscordForkPayload(p *api.ForkPayload) (*DiscordPayload, error) {
  103. baseLink := DiscordLinkFormatter(p.Repo.HTMLURL, p.Repo.Name)
  104. forkLink := DiscordLinkFormatter(p.Forkee.HTMLURL, p.Forkee.FullName)
  105. content := fmt.Sprintf("%s is forked to %s", baseLink, forkLink)
  106. return &DiscordPayload{
  107. Embeds: []*DiscordEmbedObject{{
  108. Description: content,
  109. URL: setting.AppURL + p.Sender.UserName,
  110. Author: &DiscordEmbedAuthorObject{
  111. Name: p.Sender.UserName,
  112. IconURL: p.Sender.AvatarUrl,
  113. },
  114. }},
  115. }, nil
  116. }
  117. func getDiscordPushPayload(p *api.PushPayload, slack *SlackMeta) (*DiscordPayload, error) {
  118. // n new commits
  119. var (
  120. branchName = git.RefEndName(p.Ref)
  121. commitDesc string
  122. commitString string
  123. )
  124. if len(p.Commits) == 1 {
  125. commitDesc = "1 new commit"
  126. } else {
  127. commitDesc = fmt.Sprintf("%d new commits", len(p.Commits))
  128. }
  129. if len(p.CompareURL) > 0 {
  130. commitString = DiscordLinkFormatter(p.CompareURL, commitDesc)
  131. } else {
  132. commitString = commitDesc
  133. }
  134. repoLink := DiscordLinkFormatter(p.Repo.HTMLURL, p.Repo.Name)
  135. branchLink := DiscordLinkFormatter(p.Repo.HTMLURL+"/src/"+branchName, branchName)
  136. content := fmt.Sprintf("Pushed %s to %s/%s\n", commitString, repoLink, branchLink)
  137. // for each commit, generate attachment text
  138. for i, commit := range p.Commits {
  139. content += fmt.Sprintf("%s %s - %s", DiscordSHALinkFormatter(commit.URL, commit.ID[:7]), DiscordTextFormatter(commit.Message), commit.Author.Name)
  140. // add linebreak to each commit but the last
  141. if i < len(p.Commits)-1 {
  142. content += "\n"
  143. }
  144. }
  145. color, _ := strconv.ParseInt(strings.TrimLeft(slack.Color, "#"), 16, 32)
  146. return &DiscordPayload{
  147. Username: slack.Username,
  148. AvatarURL: slack.IconURL,
  149. Embeds: []*DiscordEmbedObject{{
  150. Description: content,
  151. URL: setting.AppURL + p.Sender.UserName,
  152. Color: int(color),
  153. Author: &DiscordEmbedAuthorObject{
  154. Name: p.Sender.UserName,
  155. IconURL: p.Sender.AvatarUrl,
  156. },
  157. }},
  158. }, nil
  159. }
  160. func getDiscordIssuesPayload(p *api.IssuesPayload, slack *SlackMeta) (*DiscordPayload, error) {
  161. title := fmt.Sprintf("#%d %s", p.Index, p.Issue.Title)
  162. url := fmt.Sprintf("%s/issues/%d", p.Repository.HTMLURL, p.Index)
  163. content := ""
  164. fields := make([]*DiscordEmbedFieldObject, 0, 1)
  165. switch p.Action {
  166. case api.HOOK_ISSUE_OPENED:
  167. title = "New issue: " + title
  168. content = p.Issue.Body
  169. case api.HOOK_ISSUE_CLOSED:
  170. title = "Issue closed: " + title
  171. case api.HOOK_ISSUE_REOPENED:
  172. title = "Issue re-opened: " + title
  173. case api.HOOK_ISSUE_EDITED:
  174. title = "Issue edited: " + title
  175. content = p.Issue.Body
  176. case api.HOOK_ISSUE_ASSIGNED:
  177. title = "Issue assigned: " + title
  178. fields = []*DiscordEmbedFieldObject{{
  179. Name: "New Assignee",
  180. Value: p.Issue.Assignee.UserName,
  181. }}
  182. case api.HOOK_ISSUE_UNASSIGNED:
  183. title = "Issue unassigned: " + title
  184. case api.HOOK_ISSUE_LABEL_UPDATED:
  185. title = "Issue labels updated: " + title
  186. labels := make([]string, len(p.Issue.Labels))
  187. for i := range p.Issue.Labels {
  188. labels[i] = p.Issue.Labels[i].Name
  189. }
  190. if len(labels) == 0 {
  191. labels = []string{"<empty>"}
  192. }
  193. fields = []*DiscordEmbedFieldObject{{
  194. Name: "Labels",
  195. Value: strings.Join(labels, ", "),
  196. }}
  197. case api.HOOK_ISSUE_LABEL_CLEARED:
  198. title = "Issue labels cleared: " + title
  199. case api.HOOK_ISSUE_SYNCHRONIZED:
  200. title = "Issue synchronized: " + title
  201. case api.HOOK_ISSUE_MILESTONED:
  202. title = "Issue milestoned: " + title
  203. fields = []*DiscordEmbedFieldObject{{
  204. Name: "New Milestone",
  205. Value: p.Issue.Milestone.Title,
  206. }}
  207. case api.HOOK_ISSUE_DEMILESTONED:
  208. title = "Issue demilestoned: " + title
  209. }
  210. color, _ := strconv.ParseInt(strings.TrimLeft(slack.Color, "#"), 16, 32)
  211. return &DiscordPayload{
  212. Username: slack.Username,
  213. AvatarURL: slack.IconURL,
  214. Embeds: []*DiscordEmbedObject{{
  215. Title: title,
  216. Description: content,
  217. URL: url,
  218. Color: int(color),
  219. Footer: &DiscordEmbedFooterObject{
  220. Text: p.Repository.FullName,
  221. },
  222. Author: &DiscordEmbedAuthorObject{
  223. Name: p.Sender.UserName,
  224. IconURL: p.Sender.AvatarUrl,
  225. },
  226. Fields: fields,
  227. }},
  228. }, nil
  229. }
  230. func getDiscordIssueCommentPayload(p *api.IssueCommentPayload, slack *SlackMeta) (*DiscordPayload, error) {
  231. title := fmt.Sprintf("#%d %s", p.Issue.Index, p.Issue.Title)
  232. url := fmt.Sprintf("%s/issues/%d#%s", p.Repository.HTMLURL, p.Issue.Index, CommentHashTag(p.Comment.ID))
  233. content := ""
  234. fields := make([]*DiscordEmbedFieldObject, 0, 1)
  235. switch p.Action {
  236. case api.HOOK_ISSUE_COMMENT_CREATED:
  237. title = "New comment: " + title
  238. content = p.Comment.Body
  239. case api.HOOK_ISSUE_COMMENT_EDITED:
  240. title = "Comment edited: " + title
  241. content = p.Comment.Body
  242. case api.HOOK_ISSUE_COMMENT_DELETED:
  243. title = "Comment deleted: " + title
  244. url = fmt.Sprintf("%s/issues/%d", p.Repository.HTMLURL, p.Issue.Index)
  245. content = p.Comment.Body
  246. }
  247. color, _ := strconv.ParseInt(strings.TrimLeft(slack.Color, "#"), 16, 32)
  248. return &DiscordPayload{
  249. Username: slack.Username,
  250. AvatarURL: slack.IconURL,
  251. Embeds: []*DiscordEmbedObject{{
  252. Title: title,
  253. Description: content,
  254. URL: url,
  255. Color: int(color),
  256. Footer: &DiscordEmbedFooterObject{
  257. Text: p.Repository.FullName,
  258. },
  259. Author: &DiscordEmbedAuthorObject{
  260. Name: p.Sender.UserName,
  261. IconURL: p.Sender.AvatarUrl,
  262. },
  263. Fields: fields,
  264. }},
  265. }, nil
  266. }
  267. func getDiscordPullRequestPayload(p *api.PullRequestPayload, slack *SlackMeta) (*DiscordPayload, error) {
  268. title := fmt.Sprintf("#%d %s", p.Index, p.PullRequest.Title)
  269. url := fmt.Sprintf("%s/pulls/%d", p.Repository.HTMLURL, p.Index)
  270. content := ""
  271. fields := make([]*DiscordEmbedFieldObject, 0, 1)
  272. switch p.Action {
  273. case api.HOOK_ISSUE_OPENED:
  274. title = "New pull request: " + title
  275. content = p.PullRequest.Body
  276. case api.HOOK_ISSUE_CLOSED:
  277. if p.PullRequest.HasMerged {
  278. title = "Pull request merged: " + title
  279. } else {
  280. title = "Pull request closed: " + title
  281. }
  282. case api.HOOK_ISSUE_REOPENED:
  283. title = "Pull request re-opened: " + title
  284. case api.HOOK_ISSUE_EDITED:
  285. title = "Pull request edited: " + title
  286. content = p.PullRequest.Body
  287. case api.HOOK_ISSUE_ASSIGNED:
  288. title = "Pull request assigned: " + title
  289. fields = []*DiscordEmbedFieldObject{{
  290. Name: "New Assignee",
  291. Value: p.PullRequest.Assignee.UserName,
  292. }}
  293. case api.HOOK_ISSUE_UNASSIGNED:
  294. title = "Pull request unassigned: " + title
  295. case api.HOOK_ISSUE_LABEL_UPDATED:
  296. title = "Pull request labels updated: " + title
  297. labels := make([]string, len(p.PullRequest.Labels))
  298. for i := range p.PullRequest.Labels {
  299. labels[i] = p.PullRequest.Labels[i].Name
  300. }
  301. fields = []*DiscordEmbedFieldObject{{
  302. Name: "Labels",
  303. Value: strings.Join(labels, ", "),
  304. }}
  305. case api.HOOK_ISSUE_LABEL_CLEARED:
  306. title = "Pull request labels cleared: " + title
  307. case api.HOOK_ISSUE_SYNCHRONIZED:
  308. title = "Pull request synchronized: " + title
  309. case api.HOOK_ISSUE_MILESTONED:
  310. title = "Pull request milestoned: " + title
  311. fields = []*DiscordEmbedFieldObject{{
  312. Name: "New Milestone",
  313. Value: p.PullRequest.Milestone.Title,
  314. }}
  315. case api.HOOK_ISSUE_DEMILESTONED:
  316. title = "Pull request demilestoned: " + title
  317. }
  318. color, _ := strconv.ParseInt(strings.TrimLeft(slack.Color, "#"), 16, 32)
  319. return &DiscordPayload{
  320. Username: slack.Username,
  321. AvatarURL: slack.IconURL,
  322. Embeds: []*DiscordEmbedObject{{
  323. Title: title,
  324. Description: content,
  325. URL: url,
  326. Color: int(color),
  327. Footer: &DiscordEmbedFooterObject{
  328. Text: p.Repository.FullName,
  329. },
  330. Author: &DiscordEmbedAuthorObject{
  331. Name: p.Sender.UserName,
  332. IconURL: p.Sender.AvatarUrl,
  333. },
  334. Fields: fields,
  335. }},
  336. }, nil
  337. }
  338. func getDiscordReleasePayload(p *api.ReleasePayload) (*DiscordPayload, error) {
  339. repoLink := DiscordLinkFormatter(p.Repository.HTMLURL, p.Repository.Name)
  340. refLink := DiscordLinkFormatter(p.Repository.HTMLURL+"/src/"+p.Release.TagName, p.Release.TagName)
  341. content := fmt.Sprintf("Published new release %s of %s", refLink, repoLink)
  342. return &DiscordPayload{
  343. Embeds: []*DiscordEmbedObject{{
  344. Description: content,
  345. URL: setting.AppURL + p.Sender.UserName,
  346. Author: &DiscordEmbedAuthorObject{
  347. Name: p.Sender.UserName,
  348. IconURL: p.Sender.AvatarUrl,
  349. },
  350. }},
  351. }, nil
  352. }
  353. // GetDiscordPayload converts a discord webhook into a DiscordPayload
  354. func GetDiscordPayload(p api.Payloader, event HookEventType, meta string) (payload *DiscordPayload, err error) {
  355. slack := &SlackMeta{}
  356. if err := jsoniter.Unmarshal([]byte(meta), &slack); err != nil {
  357. return nil, fmt.Errorf("jsoniter.Unmarshal: %v", err)
  358. }
  359. switch event {
  360. case HookEventCreate:
  361. payload, err = getDiscordCreatePayload(p.(*api.CreatePayload))
  362. case HookEventDelete:
  363. payload, err = getDiscordDeletePayload(p.(*api.DeletePayload))
  364. case HookEventFork:
  365. payload, err = getDiscordForkPayload(p.(*api.ForkPayload))
  366. case HookEventPush:
  367. payload, err = getDiscordPushPayload(p.(*api.PushPayload), slack)
  368. case HookEventIssues:
  369. payload, err = getDiscordIssuesPayload(p.(*api.IssuesPayload), slack)
  370. case HookEventIssueComment:
  371. payload, err = getDiscordIssueCommentPayload(p.(*api.IssueCommentPayload), slack)
  372. case HookEventPullRequest:
  373. payload, err = getDiscordPullRequestPayload(p.(*api.PullRequestPayload), slack)
  374. case HookEventRelease:
  375. payload, err = getDiscordReleasePayload(p.(*api.ReleasePayload))
  376. }
  377. if err != nil {
  378. return nil, fmt.Errorf("event '%s': %v", event, err)
  379. }
  380. payload.Username = slack.Username
  381. payload.AvatarURL = slack.IconURL
  382. if len(payload.Embeds) > 0 {
  383. color, _ := strconv.ParseInt(strings.TrimLeft(slack.Color, "#"), 16, 32)
  384. payload.Embeds[0].Color = int(color)
  385. }
  386. return payload, nil
  387. }