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.

webhook.go 30KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100
  1. // Copyright 2015 The Gogs Authors. All rights reserved.
  2. // Copyright 2017 The Gitea Authors. All rights reserved.
  3. // Use of this source code is governed by a MIT-style
  4. // license that can be found in the LICENSE file.
  5. package repo
  6. import (
  7. "encoding/json"
  8. "errors"
  9. "fmt"
  10. "path"
  11. "strings"
  12. "code.gitea.io/gitea/models"
  13. "code.gitea.io/gitea/modules/auth"
  14. "code.gitea.io/gitea/modules/base"
  15. "code.gitea.io/gitea/modules/context"
  16. "code.gitea.io/gitea/modules/convert"
  17. "code.gitea.io/gitea/modules/git"
  18. "code.gitea.io/gitea/modules/setting"
  19. api "code.gitea.io/gitea/modules/structs"
  20. "code.gitea.io/gitea/modules/webhook"
  21. "github.com/unknwon/com"
  22. )
  23. const (
  24. tplHooks base.TplName = "repo/settings/webhook/base"
  25. tplHookNew base.TplName = "repo/settings/webhook/new"
  26. tplOrgHookNew base.TplName = "org/settings/hook_new"
  27. tplAdminHookNew base.TplName = "admin/hook_new"
  28. )
  29. // Webhooks render web hooks list page
  30. func Webhooks(ctx *context.Context) {
  31. ctx.Data["Title"] = ctx.Tr("repo.settings.hooks")
  32. ctx.Data["PageIsSettingsHooks"] = true
  33. ctx.Data["BaseLink"] = ctx.Repo.RepoLink + "/settings/hooks"
  34. ctx.Data["Description"] = ctx.Tr("repo.settings.hooks_desc", "https://docs.gitea.io/en-us/webhooks/")
  35. ws, err := models.GetWebhooksByRepoID(ctx.Repo.Repository.ID, models.ListOptions{})
  36. if err != nil {
  37. ctx.ServerError("GetWebhooksByRepoID", err)
  38. return
  39. }
  40. ctx.Data["Webhooks"] = ws
  41. ctx.HTML(200, tplHooks)
  42. }
  43. type orgRepoCtx struct {
  44. OrgID int64
  45. RepoID int64
  46. IsAdmin bool
  47. IsSystemWebhook bool
  48. Link string
  49. NewTemplate base.TplName
  50. }
  51. // getOrgRepoCtx determines whether this is a repo, organization, or admin (both default and system) context.
  52. func getOrgRepoCtx(ctx *context.Context) (*orgRepoCtx, error) {
  53. if len(ctx.Repo.RepoLink) > 0 {
  54. return &orgRepoCtx{
  55. RepoID: ctx.Repo.Repository.ID,
  56. Link: path.Join(ctx.Repo.RepoLink, "settings/hooks"),
  57. NewTemplate: tplHookNew,
  58. }, nil
  59. }
  60. if len(ctx.Org.OrgLink) > 0 {
  61. return &orgRepoCtx{
  62. OrgID: ctx.Org.Organization.ID,
  63. Link: path.Join(ctx.Org.OrgLink, "settings/hooks"),
  64. NewTemplate: tplOrgHookNew,
  65. }, nil
  66. }
  67. if ctx.User.IsAdmin {
  68. // Are we looking at default webhooks?
  69. if ctx.Params(":configType") == "hooks" {
  70. return &orgRepoCtx{
  71. IsAdmin: true,
  72. Link: path.Join(setting.AppSubURL, "/admin/hooks"),
  73. NewTemplate: tplAdminHookNew,
  74. }, nil
  75. }
  76. // Must be system webhooks instead
  77. return &orgRepoCtx{
  78. IsAdmin: true,
  79. IsSystemWebhook: true,
  80. Link: path.Join(setting.AppSubURL, "/admin/system-hooks"),
  81. NewTemplate: tplAdminHookNew,
  82. }, nil
  83. }
  84. return nil, errors.New("Unable to set OrgRepo context")
  85. }
  86. func checkHookType(ctx *context.Context) string {
  87. hookType := strings.ToLower(ctx.Params(":type"))
  88. if !com.IsSliceContainsStr(setting.Webhook.Types, hookType) {
  89. ctx.NotFound("checkHookType", nil)
  90. return ""
  91. }
  92. return hookType
  93. }
  94. // WebhooksNew render creating webhook page
  95. func WebhooksNew(ctx *context.Context) {
  96. ctx.Data["Title"] = ctx.Tr("repo.settings.add_webhook")
  97. ctx.Data["Webhook"] = models.Webhook{HookEvent: &models.HookEvent{}}
  98. orCtx, err := getOrgRepoCtx(ctx)
  99. if err != nil {
  100. ctx.ServerError("getOrgRepoCtx", err)
  101. return
  102. }
  103. if orCtx.IsAdmin && orCtx.IsSystemWebhook {
  104. ctx.Data["PageIsAdminSystemHooks"] = true
  105. ctx.Data["PageIsAdminSystemHooksNew"] = true
  106. } else if orCtx.IsAdmin {
  107. ctx.Data["PageIsAdminHooks"] = true
  108. ctx.Data["PageIsAdminHooksNew"] = true
  109. } else {
  110. ctx.Data["PageIsSettingsHooks"] = true
  111. ctx.Data["PageIsSettingsHooksNew"] = true
  112. }
  113. hookType := checkHookType(ctx)
  114. ctx.Data["HookType"] = hookType
  115. if ctx.Written() {
  116. return
  117. }
  118. if hookType == "discord" {
  119. ctx.Data["DiscordHook"] = map[string]interface{}{
  120. "Username": "Gitea",
  121. "IconURL": setting.AppURL + "img/favicon.png",
  122. }
  123. }
  124. ctx.Data["BaseLink"] = orCtx.Link
  125. ctx.HTML(200, orCtx.NewTemplate)
  126. }
  127. // ParseHookEvent convert web form content to models.HookEvent
  128. func ParseHookEvent(form auth.WebhookForm) *models.HookEvent {
  129. return &models.HookEvent{
  130. PushOnly: form.PushOnly(),
  131. SendEverything: form.SendEverything(),
  132. ChooseEvents: form.ChooseEvents(),
  133. HookEvents: models.HookEvents{
  134. Create: form.Create,
  135. Delete: form.Delete,
  136. Fork: form.Fork,
  137. Issues: form.Issues,
  138. IssueAssign: form.IssueAssign,
  139. IssueLabel: form.IssueLabel,
  140. IssueMilestone: form.IssueMilestone,
  141. IssueComment: form.IssueComment,
  142. Release: form.Release,
  143. Push: form.Push,
  144. PullRequest: form.PullRequest,
  145. PullRequestAssign: form.PullRequestAssign,
  146. PullRequestLabel: form.PullRequestLabel,
  147. PullRequestMilestone: form.PullRequestMilestone,
  148. PullRequestComment: form.PullRequestComment,
  149. PullRequestReview: form.PullRequestReview,
  150. PullRequestSync: form.PullRequestSync,
  151. Repository: form.Repository,
  152. },
  153. BranchFilter: form.BranchFilter,
  154. }
  155. }
  156. // GiteaHooksNewPost response for creating Gitea webhook
  157. func GiteaHooksNewPost(ctx *context.Context, form auth.NewWebhookForm) {
  158. ctx.Data["Title"] = ctx.Tr("repo.settings.add_webhook")
  159. ctx.Data["PageIsSettingsHooks"] = true
  160. ctx.Data["PageIsSettingsHooksNew"] = true
  161. ctx.Data["Webhook"] = models.Webhook{HookEvent: &models.HookEvent{}}
  162. ctx.Data["HookType"] = models.GITEA.Name()
  163. orCtx, err := getOrgRepoCtx(ctx)
  164. if err != nil {
  165. ctx.ServerError("getOrgRepoCtx", err)
  166. return
  167. }
  168. ctx.Data["BaseLink"] = orCtx.Link
  169. if ctx.HasError() {
  170. ctx.HTML(200, orCtx.NewTemplate)
  171. return
  172. }
  173. contentType := models.ContentTypeJSON
  174. if models.HookContentType(form.ContentType) == models.ContentTypeForm {
  175. contentType = models.ContentTypeForm
  176. }
  177. w := &models.Webhook{
  178. RepoID: orCtx.RepoID,
  179. URL: form.PayloadURL,
  180. HTTPMethod: form.HTTPMethod,
  181. ContentType: contentType,
  182. Secret: form.Secret,
  183. HookEvent: ParseHookEvent(form.WebhookForm),
  184. IsActive: form.Active,
  185. HookTaskType: models.GITEA,
  186. OrgID: orCtx.OrgID,
  187. IsSystemWebhook: orCtx.IsSystemWebhook,
  188. }
  189. if err := w.UpdateEvent(); err != nil {
  190. ctx.ServerError("UpdateEvent", err)
  191. return
  192. } else if err := models.CreateWebhook(w); err != nil {
  193. ctx.ServerError("CreateWebhook", err)
  194. return
  195. }
  196. ctx.Flash.Success(ctx.Tr("repo.settings.add_hook_success"))
  197. ctx.Redirect(orCtx.Link)
  198. }
  199. // GogsHooksNewPost response for creating webhook
  200. func GogsHooksNewPost(ctx *context.Context, form auth.NewGogshookForm) {
  201. newGogsWebhookPost(ctx, form, models.GOGS)
  202. }
  203. // newGogsWebhookPost response for creating gogs hook
  204. func newGogsWebhookPost(ctx *context.Context, form auth.NewGogshookForm, kind models.HookTaskType) {
  205. ctx.Data["Title"] = ctx.Tr("repo.settings.add_webhook")
  206. ctx.Data["PageIsSettingsHooks"] = true
  207. ctx.Data["PageIsSettingsHooksNew"] = true
  208. ctx.Data["Webhook"] = models.Webhook{HookEvent: &models.HookEvent{}}
  209. ctx.Data["HookType"] = models.GOGS.Name()
  210. orCtx, err := getOrgRepoCtx(ctx)
  211. if err != nil {
  212. ctx.ServerError("getOrgRepoCtx", err)
  213. return
  214. }
  215. ctx.Data["BaseLink"] = orCtx.Link
  216. if ctx.HasError() {
  217. ctx.HTML(200, orCtx.NewTemplate)
  218. return
  219. }
  220. contentType := models.ContentTypeJSON
  221. if models.HookContentType(form.ContentType) == models.ContentTypeForm {
  222. contentType = models.ContentTypeForm
  223. }
  224. w := &models.Webhook{
  225. RepoID: orCtx.RepoID,
  226. URL: form.PayloadURL,
  227. ContentType: contentType,
  228. Secret: form.Secret,
  229. HookEvent: ParseHookEvent(form.WebhookForm),
  230. IsActive: form.Active,
  231. HookTaskType: kind,
  232. OrgID: orCtx.OrgID,
  233. IsSystemWebhook: orCtx.IsSystemWebhook,
  234. }
  235. if err := w.UpdateEvent(); err != nil {
  236. ctx.ServerError("UpdateEvent", err)
  237. return
  238. } else if err := models.CreateWebhook(w); err != nil {
  239. ctx.ServerError("CreateWebhook", err)
  240. return
  241. }
  242. ctx.Flash.Success(ctx.Tr("repo.settings.add_hook_success"))
  243. ctx.Redirect(orCtx.Link)
  244. }
  245. // DiscordHooksNewPost response for creating discord hook
  246. func DiscordHooksNewPost(ctx *context.Context, form auth.NewDiscordHookForm) {
  247. ctx.Data["Title"] = ctx.Tr("repo.settings")
  248. ctx.Data["PageIsSettingsHooks"] = true
  249. ctx.Data["PageIsSettingsHooksNew"] = true
  250. ctx.Data["Webhook"] = models.Webhook{HookEvent: &models.HookEvent{}}
  251. ctx.Data["HookType"] = models.DISCORD.Name()
  252. orCtx, err := getOrgRepoCtx(ctx)
  253. if err != nil {
  254. ctx.ServerError("getOrgRepoCtx", err)
  255. return
  256. }
  257. if ctx.HasError() {
  258. ctx.HTML(200, orCtx.NewTemplate)
  259. return
  260. }
  261. meta, err := json.Marshal(&webhook.DiscordMeta{
  262. Username: form.Username,
  263. IconURL: form.IconURL,
  264. })
  265. if err != nil {
  266. ctx.ServerError("Marshal", err)
  267. return
  268. }
  269. w := &models.Webhook{
  270. RepoID: orCtx.RepoID,
  271. URL: form.PayloadURL,
  272. ContentType: models.ContentTypeJSON,
  273. HookEvent: ParseHookEvent(form.WebhookForm),
  274. IsActive: form.Active,
  275. HookTaskType: models.DISCORD,
  276. Meta: string(meta),
  277. OrgID: orCtx.OrgID,
  278. IsSystemWebhook: orCtx.IsSystemWebhook,
  279. }
  280. if err := w.UpdateEvent(); err != nil {
  281. ctx.ServerError("UpdateEvent", err)
  282. return
  283. } else if err := models.CreateWebhook(w); err != nil {
  284. ctx.ServerError("CreateWebhook", err)
  285. return
  286. }
  287. ctx.Flash.Success(ctx.Tr("repo.settings.add_hook_success"))
  288. ctx.Redirect(orCtx.Link)
  289. }
  290. // DingtalkHooksNewPost response for creating dingtalk hook
  291. func DingtalkHooksNewPost(ctx *context.Context, form auth.NewDingtalkHookForm) {
  292. ctx.Data["Title"] = ctx.Tr("repo.settings")
  293. ctx.Data["PageIsSettingsHooks"] = true
  294. ctx.Data["PageIsSettingsHooksNew"] = true
  295. ctx.Data["Webhook"] = models.Webhook{HookEvent: &models.HookEvent{}}
  296. ctx.Data["HookType"] = models.DINGTALK.Name()
  297. orCtx, err := getOrgRepoCtx(ctx)
  298. if err != nil {
  299. ctx.ServerError("getOrgRepoCtx", err)
  300. return
  301. }
  302. if ctx.HasError() {
  303. ctx.HTML(200, orCtx.NewTemplate)
  304. return
  305. }
  306. w := &models.Webhook{
  307. RepoID: orCtx.RepoID,
  308. URL: form.PayloadURL,
  309. ContentType: models.ContentTypeJSON,
  310. HookEvent: ParseHookEvent(form.WebhookForm),
  311. IsActive: form.Active,
  312. HookTaskType: models.DINGTALK,
  313. Meta: "",
  314. OrgID: orCtx.OrgID,
  315. IsSystemWebhook: orCtx.IsSystemWebhook,
  316. }
  317. if err := w.UpdateEvent(); err != nil {
  318. ctx.ServerError("UpdateEvent", err)
  319. return
  320. } else if err := models.CreateWebhook(w); err != nil {
  321. ctx.ServerError("CreateWebhook", err)
  322. return
  323. }
  324. ctx.Flash.Success(ctx.Tr("repo.settings.add_hook_success"))
  325. ctx.Redirect(orCtx.Link)
  326. }
  327. // TelegramHooksNewPost response for creating telegram hook
  328. func TelegramHooksNewPost(ctx *context.Context, form auth.NewTelegramHookForm) {
  329. ctx.Data["Title"] = ctx.Tr("repo.settings")
  330. ctx.Data["PageIsSettingsHooks"] = true
  331. ctx.Data["PageIsSettingsHooksNew"] = true
  332. ctx.Data["Webhook"] = models.Webhook{HookEvent: &models.HookEvent{}}
  333. ctx.Data["HookType"] = models.TELEGRAM.Name()
  334. orCtx, err := getOrgRepoCtx(ctx)
  335. if err != nil {
  336. ctx.ServerError("getOrgRepoCtx", err)
  337. return
  338. }
  339. if ctx.HasError() {
  340. ctx.HTML(200, orCtx.NewTemplate)
  341. return
  342. }
  343. meta, err := json.Marshal(&webhook.TelegramMeta{
  344. BotToken: form.BotToken,
  345. ChatID: form.ChatID,
  346. })
  347. if err != nil {
  348. ctx.ServerError("Marshal", err)
  349. return
  350. }
  351. w := &models.Webhook{
  352. RepoID: orCtx.RepoID,
  353. URL: fmt.Sprintf("https://api.telegram.org/bot%s/sendMessage?chat_id=%s", form.BotToken, form.ChatID),
  354. ContentType: models.ContentTypeJSON,
  355. HookEvent: ParseHookEvent(form.WebhookForm),
  356. IsActive: form.Active,
  357. HookTaskType: models.TELEGRAM,
  358. Meta: string(meta),
  359. OrgID: orCtx.OrgID,
  360. IsSystemWebhook: orCtx.IsSystemWebhook,
  361. }
  362. if err := w.UpdateEvent(); err != nil {
  363. ctx.ServerError("UpdateEvent", err)
  364. return
  365. } else if err := models.CreateWebhook(w); err != nil {
  366. ctx.ServerError("CreateWebhook", err)
  367. return
  368. }
  369. ctx.Flash.Success(ctx.Tr("repo.settings.add_hook_success"))
  370. ctx.Redirect(orCtx.Link)
  371. }
  372. // MatrixHooksNewPost response for creating a Matrix hook
  373. func MatrixHooksNewPost(ctx *context.Context, form auth.NewMatrixHookForm) {
  374. ctx.Data["Title"] = ctx.Tr("repo.settings")
  375. ctx.Data["PageIsSettingsHooks"] = true
  376. ctx.Data["PageIsSettingsHooksNew"] = true
  377. ctx.Data["Webhook"] = models.Webhook{HookEvent: &models.HookEvent{}}
  378. ctx.Data["HookType"] = models.MATRIX.Name()
  379. orCtx, err := getOrgRepoCtx(ctx)
  380. if err != nil {
  381. ctx.ServerError("getOrgRepoCtx", err)
  382. return
  383. }
  384. if ctx.HasError() {
  385. ctx.HTML(200, orCtx.NewTemplate)
  386. return
  387. }
  388. meta, err := json.Marshal(&webhook.MatrixMeta{
  389. HomeserverURL: form.HomeserverURL,
  390. Room: form.RoomID,
  391. AccessToken: form.AccessToken,
  392. MessageType: form.MessageType,
  393. })
  394. if err != nil {
  395. ctx.ServerError("Marshal", err)
  396. return
  397. }
  398. w := &models.Webhook{
  399. RepoID: orCtx.RepoID,
  400. URL: fmt.Sprintf("%s/_matrix/client/r0/rooms/%s/send/m.room.message", form.HomeserverURL, form.RoomID),
  401. ContentType: models.ContentTypeJSON,
  402. HTTPMethod: "PUT",
  403. HookEvent: ParseHookEvent(form.WebhookForm),
  404. IsActive: form.Active,
  405. HookTaskType: models.MATRIX,
  406. Meta: string(meta),
  407. OrgID: orCtx.OrgID,
  408. IsSystemWebhook: orCtx.IsSystemWebhook,
  409. }
  410. if err := w.UpdateEvent(); err != nil {
  411. ctx.ServerError("UpdateEvent", err)
  412. return
  413. } else if err := models.CreateWebhook(w); err != nil {
  414. ctx.ServerError("CreateWebhook", err)
  415. return
  416. }
  417. ctx.Flash.Success(ctx.Tr("repo.settings.add_hook_success"))
  418. ctx.Redirect(orCtx.Link)
  419. }
  420. // MSTeamsHooksNewPost response for creating MS Teams hook
  421. func MSTeamsHooksNewPost(ctx *context.Context, form auth.NewMSTeamsHookForm) {
  422. ctx.Data["Title"] = ctx.Tr("repo.settings")
  423. ctx.Data["PageIsSettingsHooks"] = true
  424. ctx.Data["PageIsSettingsHooksNew"] = true
  425. ctx.Data["Webhook"] = models.Webhook{HookEvent: &models.HookEvent{}}
  426. ctx.Data["HookType"] = models.MSTEAMS.Name()
  427. orCtx, err := getOrgRepoCtx(ctx)
  428. if err != nil {
  429. ctx.ServerError("getOrgRepoCtx", err)
  430. return
  431. }
  432. if ctx.HasError() {
  433. ctx.HTML(200, orCtx.NewTemplate)
  434. return
  435. }
  436. w := &models.Webhook{
  437. RepoID: orCtx.RepoID,
  438. URL: form.PayloadURL,
  439. ContentType: models.ContentTypeJSON,
  440. HookEvent: ParseHookEvent(form.WebhookForm),
  441. IsActive: form.Active,
  442. HookTaskType: models.MSTEAMS,
  443. Meta: "",
  444. OrgID: orCtx.OrgID,
  445. IsSystemWebhook: orCtx.IsSystemWebhook,
  446. }
  447. if err := w.UpdateEvent(); err != nil {
  448. ctx.ServerError("UpdateEvent", err)
  449. return
  450. } else if err := models.CreateWebhook(w); err != nil {
  451. ctx.ServerError("CreateWebhook", err)
  452. return
  453. }
  454. ctx.Flash.Success(ctx.Tr("repo.settings.add_hook_success"))
  455. ctx.Redirect(orCtx.Link)
  456. }
  457. // SlackHooksNewPost response for creating slack hook
  458. func SlackHooksNewPost(ctx *context.Context, form auth.NewSlackHookForm) {
  459. ctx.Data["Title"] = ctx.Tr("repo.settings")
  460. ctx.Data["PageIsSettingsHooks"] = true
  461. ctx.Data["PageIsSettingsHooksNew"] = true
  462. ctx.Data["Webhook"] = models.Webhook{HookEvent: &models.HookEvent{}}
  463. ctx.Data["HookType"] = models.SLACK.Name()
  464. orCtx, err := getOrgRepoCtx(ctx)
  465. if err != nil {
  466. ctx.ServerError("getOrgRepoCtx", err)
  467. return
  468. }
  469. if ctx.HasError() {
  470. ctx.HTML(200, orCtx.NewTemplate)
  471. return
  472. }
  473. if form.HasInvalidChannel() {
  474. ctx.Flash.Error(ctx.Tr("repo.settings.add_webhook.invalid_channel_name"))
  475. ctx.Redirect(orCtx.Link + "/slack/new")
  476. return
  477. }
  478. meta, err := json.Marshal(&webhook.SlackMeta{
  479. Channel: strings.TrimSpace(form.Channel),
  480. Username: form.Username,
  481. IconURL: form.IconURL,
  482. Color: form.Color,
  483. })
  484. if err != nil {
  485. ctx.ServerError("Marshal", err)
  486. return
  487. }
  488. w := &models.Webhook{
  489. RepoID: orCtx.RepoID,
  490. URL: form.PayloadURL,
  491. ContentType: models.ContentTypeJSON,
  492. HookEvent: ParseHookEvent(form.WebhookForm),
  493. IsActive: form.Active,
  494. HookTaskType: models.SLACK,
  495. Meta: string(meta),
  496. OrgID: orCtx.OrgID,
  497. IsSystemWebhook: orCtx.IsSystemWebhook,
  498. }
  499. if err := w.UpdateEvent(); err != nil {
  500. ctx.ServerError("UpdateEvent", err)
  501. return
  502. } else if err := models.CreateWebhook(w); err != nil {
  503. ctx.ServerError("CreateWebhook", err)
  504. return
  505. }
  506. ctx.Flash.Success(ctx.Tr("repo.settings.add_hook_success"))
  507. ctx.Redirect(orCtx.Link)
  508. }
  509. // FeishuHooksNewPost response for creating feishu hook
  510. func FeishuHooksNewPost(ctx *context.Context, form auth.NewFeishuHookForm) {
  511. ctx.Data["Title"] = ctx.Tr("repo.settings")
  512. ctx.Data["PageIsSettingsHooks"] = true
  513. ctx.Data["PageIsSettingsHooksNew"] = true
  514. ctx.Data["Webhook"] = models.Webhook{HookEvent: &models.HookEvent{}}
  515. ctx.Data["HookType"] = models.FEISHU.Name()
  516. orCtx, err := getOrgRepoCtx(ctx)
  517. if err != nil {
  518. ctx.ServerError("getOrgRepoCtx", err)
  519. return
  520. }
  521. if ctx.HasError() {
  522. ctx.HTML(200, orCtx.NewTemplate)
  523. return
  524. }
  525. w := &models.Webhook{
  526. RepoID: orCtx.RepoID,
  527. URL: form.PayloadURL,
  528. ContentType: models.ContentTypeJSON,
  529. HookEvent: ParseHookEvent(form.WebhookForm),
  530. IsActive: form.Active,
  531. HookTaskType: models.FEISHU,
  532. Meta: "",
  533. OrgID: orCtx.OrgID,
  534. IsSystemWebhook: orCtx.IsSystemWebhook,
  535. }
  536. if err := w.UpdateEvent(); err != nil {
  537. ctx.ServerError("UpdateEvent", err)
  538. return
  539. } else if err := models.CreateWebhook(w); err != nil {
  540. ctx.ServerError("CreateWebhook", err)
  541. return
  542. }
  543. ctx.Flash.Success(ctx.Tr("repo.settings.add_hook_success"))
  544. ctx.Redirect(orCtx.Link)
  545. }
  546. func checkWebhook(ctx *context.Context) (*orgRepoCtx, *models.Webhook) {
  547. ctx.Data["RequireHighlightJS"] = true
  548. orCtx, err := getOrgRepoCtx(ctx)
  549. if err != nil {
  550. ctx.ServerError("getOrgRepoCtx", err)
  551. return nil, nil
  552. }
  553. ctx.Data["BaseLink"] = orCtx.Link
  554. var w *models.Webhook
  555. if orCtx.RepoID > 0 {
  556. w, err = models.GetWebhookByRepoID(ctx.Repo.Repository.ID, ctx.ParamsInt64(":id"))
  557. } else if orCtx.OrgID > 0 {
  558. w, err = models.GetWebhookByOrgID(ctx.Org.Organization.ID, ctx.ParamsInt64(":id"))
  559. } else if orCtx.IsSystemWebhook {
  560. w, err = models.GetSystemWebhook(ctx.ParamsInt64(":id"))
  561. } else {
  562. w, err = models.GetDefaultWebhook(ctx.ParamsInt64(":id"))
  563. }
  564. if err != nil {
  565. if models.IsErrWebhookNotExist(err) {
  566. ctx.NotFound("GetWebhookByID", nil)
  567. } else {
  568. ctx.ServerError("GetWebhookByID", err)
  569. }
  570. return nil, nil
  571. }
  572. ctx.Data["HookType"] = w.HookTaskType.Name()
  573. switch w.HookTaskType {
  574. case models.SLACK:
  575. ctx.Data["SlackHook"] = webhook.GetSlackHook(w)
  576. case models.DISCORD:
  577. ctx.Data["DiscordHook"] = webhook.GetDiscordHook(w)
  578. case models.TELEGRAM:
  579. ctx.Data["TelegramHook"] = webhook.GetTelegramHook(w)
  580. case models.MATRIX:
  581. ctx.Data["MatrixHook"] = webhook.GetMatrixHook(w)
  582. }
  583. ctx.Data["History"], err = w.History(1)
  584. if err != nil {
  585. ctx.ServerError("History", err)
  586. }
  587. return orCtx, w
  588. }
  589. // WebHooksEdit render editing web hook page
  590. func WebHooksEdit(ctx *context.Context) {
  591. ctx.Data["Title"] = ctx.Tr("repo.settings.update_webhook")
  592. ctx.Data["PageIsSettingsHooks"] = true
  593. ctx.Data["PageIsSettingsHooksEdit"] = true
  594. orCtx, w := checkWebhook(ctx)
  595. if ctx.Written() {
  596. return
  597. }
  598. ctx.Data["Webhook"] = w
  599. ctx.HTML(200, orCtx.NewTemplate)
  600. }
  601. // WebHooksEditPost response for editing web hook
  602. func WebHooksEditPost(ctx *context.Context, form auth.NewWebhookForm) {
  603. ctx.Data["Title"] = ctx.Tr("repo.settings.update_webhook")
  604. ctx.Data["PageIsSettingsHooks"] = true
  605. ctx.Data["PageIsSettingsHooksEdit"] = true
  606. orCtx, w := checkWebhook(ctx)
  607. if ctx.Written() {
  608. return
  609. }
  610. ctx.Data["Webhook"] = w
  611. if ctx.HasError() {
  612. ctx.HTML(200, orCtx.NewTemplate)
  613. return
  614. }
  615. contentType := models.ContentTypeJSON
  616. if models.HookContentType(form.ContentType) == models.ContentTypeForm {
  617. contentType = models.ContentTypeForm
  618. }
  619. w.URL = form.PayloadURL
  620. w.ContentType = contentType
  621. w.Secret = form.Secret
  622. w.HookEvent = ParseHookEvent(form.WebhookForm)
  623. w.IsActive = form.Active
  624. w.HTTPMethod = form.HTTPMethod
  625. if err := w.UpdateEvent(); err != nil {
  626. ctx.ServerError("UpdateEvent", err)
  627. return
  628. } else if err := models.UpdateWebhook(w); err != nil {
  629. ctx.ServerError("WebHooksEditPost", err)
  630. return
  631. }
  632. ctx.Flash.Success(ctx.Tr("repo.settings.update_hook_success"))
  633. ctx.Redirect(fmt.Sprintf("%s/%d", orCtx.Link, w.ID))
  634. }
  635. // GogsHooksEditPost response for editing gogs hook
  636. func GogsHooksEditPost(ctx *context.Context, form auth.NewGogshookForm) {
  637. ctx.Data["Title"] = ctx.Tr("repo.settings.update_webhook")
  638. ctx.Data["PageIsSettingsHooks"] = true
  639. ctx.Data["PageIsSettingsHooksEdit"] = true
  640. orCtx, w := checkWebhook(ctx)
  641. if ctx.Written() {
  642. return
  643. }
  644. ctx.Data["Webhook"] = w
  645. if ctx.HasError() {
  646. ctx.HTML(200, orCtx.NewTemplate)
  647. return
  648. }
  649. contentType := models.ContentTypeJSON
  650. if models.HookContentType(form.ContentType) == models.ContentTypeForm {
  651. contentType = models.ContentTypeForm
  652. }
  653. w.URL = form.PayloadURL
  654. w.ContentType = contentType
  655. w.Secret = form.Secret
  656. w.HookEvent = ParseHookEvent(form.WebhookForm)
  657. w.IsActive = form.Active
  658. if err := w.UpdateEvent(); err != nil {
  659. ctx.ServerError("UpdateEvent", err)
  660. return
  661. } else if err := models.UpdateWebhook(w); err != nil {
  662. ctx.ServerError("GogsHooksEditPost", err)
  663. return
  664. }
  665. ctx.Flash.Success(ctx.Tr("repo.settings.update_hook_success"))
  666. ctx.Redirect(fmt.Sprintf("%s/%d", orCtx.Link, w.ID))
  667. }
  668. // SlackHooksEditPost response for editing slack hook
  669. func SlackHooksEditPost(ctx *context.Context, form auth.NewSlackHookForm) {
  670. ctx.Data["Title"] = ctx.Tr("repo.settings")
  671. ctx.Data["PageIsSettingsHooks"] = true
  672. ctx.Data["PageIsSettingsHooksEdit"] = true
  673. orCtx, w := checkWebhook(ctx)
  674. if ctx.Written() {
  675. return
  676. }
  677. ctx.Data["Webhook"] = w
  678. if ctx.HasError() {
  679. ctx.HTML(200, orCtx.NewTemplate)
  680. return
  681. }
  682. if form.HasInvalidChannel() {
  683. ctx.Flash.Error(ctx.Tr("repo.settings.add_webhook.invalid_channel_name"))
  684. ctx.Redirect(fmt.Sprintf("%s/%d", orCtx.Link, w.ID))
  685. return
  686. }
  687. meta, err := json.Marshal(&webhook.SlackMeta{
  688. Channel: strings.TrimSpace(form.Channel),
  689. Username: form.Username,
  690. IconURL: form.IconURL,
  691. Color: form.Color,
  692. })
  693. if err != nil {
  694. ctx.ServerError("Marshal", err)
  695. return
  696. }
  697. w.URL = form.PayloadURL
  698. w.Meta = string(meta)
  699. w.HookEvent = ParseHookEvent(form.WebhookForm)
  700. w.IsActive = form.Active
  701. if err := w.UpdateEvent(); err != nil {
  702. ctx.ServerError("UpdateEvent", err)
  703. return
  704. } else if err := models.UpdateWebhook(w); err != nil {
  705. ctx.ServerError("UpdateWebhook", err)
  706. return
  707. }
  708. ctx.Flash.Success(ctx.Tr("repo.settings.update_hook_success"))
  709. ctx.Redirect(fmt.Sprintf("%s/%d", orCtx.Link, w.ID))
  710. }
  711. // DiscordHooksEditPost response for editing discord hook
  712. func DiscordHooksEditPost(ctx *context.Context, form auth.NewDiscordHookForm) {
  713. ctx.Data["Title"] = ctx.Tr("repo.settings")
  714. ctx.Data["PageIsSettingsHooks"] = true
  715. ctx.Data["PageIsSettingsHooksEdit"] = true
  716. orCtx, w := checkWebhook(ctx)
  717. if ctx.Written() {
  718. return
  719. }
  720. ctx.Data["Webhook"] = w
  721. if ctx.HasError() {
  722. ctx.HTML(200, orCtx.NewTemplate)
  723. return
  724. }
  725. meta, err := json.Marshal(&webhook.DiscordMeta{
  726. Username: form.Username,
  727. IconURL: form.IconURL,
  728. })
  729. if err != nil {
  730. ctx.ServerError("Marshal", err)
  731. return
  732. }
  733. w.URL = form.PayloadURL
  734. w.Meta = string(meta)
  735. w.HookEvent = ParseHookEvent(form.WebhookForm)
  736. w.IsActive = form.Active
  737. if err := w.UpdateEvent(); err != nil {
  738. ctx.ServerError("UpdateEvent", err)
  739. return
  740. } else if err := models.UpdateWebhook(w); err != nil {
  741. ctx.ServerError("UpdateWebhook", err)
  742. return
  743. }
  744. ctx.Flash.Success(ctx.Tr("repo.settings.update_hook_success"))
  745. ctx.Redirect(fmt.Sprintf("%s/%d", orCtx.Link, w.ID))
  746. }
  747. // DingtalkHooksEditPost response for editing discord hook
  748. func DingtalkHooksEditPost(ctx *context.Context, form auth.NewDingtalkHookForm) {
  749. ctx.Data["Title"] = ctx.Tr("repo.settings")
  750. ctx.Data["PageIsSettingsHooks"] = true
  751. ctx.Data["PageIsSettingsHooksEdit"] = true
  752. orCtx, w := checkWebhook(ctx)
  753. if ctx.Written() {
  754. return
  755. }
  756. ctx.Data["Webhook"] = w
  757. if ctx.HasError() {
  758. ctx.HTML(200, orCtx.NewTemplate)
  759. return
  760. }
  761. w.URL = form.PayloadURL
  762. w.HookEvent = ParseHookEvent(form.WebhookForm)
  763. w.IsActive = form.Active
  764. if err := w.UpdateEvent(); err != nil {
  765. ctx.ServerError("UpdateEvent", err)
  766. return
  767. } else if err := models.UpdateWebhook(w); err != nil {
  768. ctx.ServerError("UpdateWebhook", err)
  769. return
  770. }
  771. ctx.Flash.Success(ctx.Tr("repo.settings.update_hook_success"))
  772. ctx.Redirect(fmt.Sprintf("%s/%d", orCtx.Link, w.ID))
  773. }
  774. // TelegramHooksEditPost response for editing discord hook
  775. func TelegramHooksEditPost(ctx *context.Context, form auth.NewTelegramHookForm) {
  776. ctx.Data["Title"] = ctx.Tr("repo.settings")
  777. ctx.Data["PageIsSettingsHooks"] = true
  778. ctx.Data["PageIsSettingsHooksEdit"] = true
  779. orCtx, w := checkWebhook(ctx)
  780. if ctx.Written() {
  781. return
  782. }
  783. ctx.Data["Webhook"] = w
  784. if ctx.HasError() {
  785. ctx.HTML(200, orCtx.NewTemplate)
  786. return
  787. }
  788. meta, err := json.Marshal(&webhook.TelegramMeta{
  789. BotToken: form.BotToken,
  790. ChatID: form.ChatID,
  791. })
  792. if err != nil {
  793. ctx.ServerError("Marshal", err)
  794. return
  795. }
  796. w.Meta = string(meta)
  797. w.URL = fmt.Sprintf("https://api.telegram.org/bot%s/sendMessage?chat_id=%s", form.BotToken, form.ChatID)
  798. w.HookEvent = ParseHookEvent(form.WebhookForm)
  799. w.IsActive = form.Active
  800. if err := w.UpdateEvent(); err != nil {
  801. ctx.ServerError("UpdateEvent", err)
  802. return
  803. } else if err := models.UpdateWebhook(w); err != nil {
  804. ctx.ServerError("UpdateWebhook", err)
  805. return
  806. }
  807. ctx.Flash.Success(ctx.Tr("repo.settings.update_hook_success"))
  808. ctx.Redirect(fmt.Sprintf("%s/%d", orCtx.Link, w.ID))
  809. }
  810. // MatrixHooksEditPost response for editing a Matrix hook
  811. func MatrixHooksEditPost(ctx *context.Context, form auth.NewMatrixHookForm) {
  812. ctx.Data["Title"] = ctx.Tr("repo.settings")
  813. ctx.Data["PageIsSettingsHooks"] = true
  814. ctx.Data["PageIsSettingsHooksEdit"] = true
  815. orCtx, w := checkWebhook(ctx)
  816. if ctx.Written() {
  817. return
  818. }
  819. ctx.Data["Webhook"] = w
  820. if ctx.HasError() {
  821. ctx.HTML(200, orCtx.NewTemplate)
  822. return
  823. }
  824. meta, err := json.Marshal(&webhook.MatrixMeta{
  825. HomeserverURL: form.HomeserverURL,
  826. Room: form.RoomID,
  827. AccessToken: form.AccessToken,
  828. MessageType: form.MessageType,
  829. })
  830. if err != nil {
  831. ctx.ServerError("Marshal", err)
  832. return
  833. }
  834. w.Meta = string(meta)
  835. w.URL = fmt.Sprintf("%s/_matrix/client/r0/rooms/%s/send/m.room.message", form.HomeserverURL, form.RoomID)
  836. w.HookEvent = ParseHookEvent(form.WebhookForm)
  837. w.IsActive = form.Active
  838. if err := w.UpdateEvent(); err != nil {
  839. ctx.ServerError("UpdateEvent", err)
  840. return
  841. } else if err := models.UpdateWebhook(w); err != nil {
  842. ctx.ServerError("UpdateWebhook", err)
  843. return
  844. }
  845. ctx.Flash.Success(ctx.Tr("repo.settings.update_hook_success"))
  846. ctx.Redirect(fmt.Sprintf("%s/%d", orCtx.Link, w.ID))
  847. }
  848. // MSTeamsHooksEditPost response for editing MS Teams hook
  849. func MSTeamsHooksEditPost(ctx *context.Context, form auth.NewMSTeamsHookForm) {
  850. ctx.Data["Title"] = ctx.Tr("repo.settings")
  851. ctx.Data["PageIsSettingsHooks"] = true
  852. ctx.Data["PageIsSettingsHooksEdit"] = true
  853. orCtx, w := checkWebhook(ctx)
  854. if ctx.Written() {
  855. return
  856. }
  857. ctx.Data["Webhook"] = w
  858. if ctx.HasError() {
  859. ctx.HTML(200, orCtx.NewTemplate)
  860. return
  861. }
  862. w.URL = form.PayloadURL
  863. w.HookEvent = ParseHookEvent(form.WebhookForm)
  864. w.IsActive = form.Active
  865. if err := w.UpdateEvent(); err != nil {
  866. ctx.ServerError("UpdateEvent", err)
  867. return
  868. } else if err := models.UpdateWebhook(w); err != nil {
  869. ctx.ServerError("UpdateWebhook", err)
  870. return
  871. }
  872. ctx.Flash.Success(ctx.Tr("repo.settings.update_hook_success"))
  873. ctx.Redirect(fmt.Sprintf("%s/%d", orCtx.Link, w.ID))
  874. }
  875. // FeishuHooksEditPost response for editing feishu hook
  876. func FeishuHooksEditPost(ctx *context.Context, form auth.NewFeishuHookForm) {
  877. ctx.Data["Title"] = ctx.Tr("repo.settings")
  878. ctx.Data["PageIsSettingsHooks"] = true
  879. ctx.Data["PageIsSettingsHooksEdit"] = true
  880. orCtx, w := checkWebhook(ctx)
  881. if ctx.Written() {
  882. return
  883. }
  884. ctx.Data["Webhook"] = w
  885. if ctx.HasError() {
  886. ctx.HTML(200, orCtx.NewTemplate)
  887. return
  888. }
  889. w.URL = form.PayloadURL
  890. w.HookEvent = ParseHookEvent(form.WebhookForm)
  891. w.IsActive = form.Active
  892. if err := w.UpdateEvent(); err != nil {
  893. ctx.ServerError("UpdateEvent", err)
  894. return
  895. } else if err := models.UpdateWebhook(w); err != nil {
  896. ctx.ServerError("UpdateWebhook", err)
  897. return
  898. }
  899. ctx.Flash.Success(ctx.Tr("repo.settings.update_hook_success"))
  900. ctx.Redirect(fmt.Sprintf("%s/%d", orCtx.Link, w.ID))
  901. }
  902. // TestWebhook test if web hook is work fine
  903. func TestWebhook(ctx *context.Context) {
  904. hookID := ctx.ParamsInt64(":id")
  905. w, err := models.GetWebhookByRepoID(ctx.Repo.Repository.ID, hookID)
  906. if err != nil {
  907. ctx.Flash.Error("GetWebhookByID: " + err.Error())
  908. ctx.Status(500)
  909. return
  910. }
  911. // Grab latest commit or fake one if it's empty repository.
  912. commit := ctx.Repo.Commit
  913. if commit == nil {
  914. ghost := models.NewGhostUser()
  915. commit = &git.Commit{
  916. ID: git.MustIDFromString(git.EmptySHA),
  917. Author: ghost.NewGitSig(),
  918. Committer: ghost.NewGitSig(),
  919. CommitMessage: "This is a fake commit",
  920. }
  921. }
  922. apiUser := convert.ToUser(ctx.User, true, true)
  923. p := &api.PushPayload{
  924. Ref: git.BranchPrefix + ctx.Repo.Repository.DefaultBranch,
  925. Before: commit.ID.String(),
  926. After: commit.ID.String(),
  927. Commits: []*api.PayloadCommit{
  928. {
  929. ID: commit.ID.String(),
  930. Message: commit.Message(),
  931. URL: ctx.Repo.Repository.HTMLURL() + "/commit/" + commit.ID.String(),
  932. Author: &api.PayloadUser{
  933. Name: commit.Author.Name,
  934. Email: commit.Author.Email,
  935. },
  936. Committer: &api.PayloadUser{
  937. Name: commit.Committer.Name,
  938. Email: commit.Committer.Email,
  939. },
  940. },
  941. },
  942. Repo: ctx.Repo.Repository.APIFormat(models.AccessModeNone),
  943. Pusher: apiUser,
  944. Sender: apiUser,
  945. }
  946. if err := webhook.PrepareWebhook(w, ctx.Repo.Repository, models.HookEventPush, p); err != nil {
  947. ctx.Flash.Error("PrepareWebhook: " + err.Error())
  948. ctx.Status(500)
  949. } else {
  950. ctx.Flash.Info(ctx.Tr("repo.settings.webhook.test_delivery_success"))
  951. ctx.Status(200)
  952. }
  953. }
  954. // DeleteWebhook delete a webhook
  955. func DeleteWebhook(ctx *context.Context) {
  956. if err := models.DeleteWebhookByRepoID(ctx.Repo.Repository.ID, ctx.QueryInt64("id")); err != nil {
  957. ctx.Flash.Error("DeleteWebhookByRepoID: " + err.Error())
  958. } else {
  959. ctx.Flash.Success(ctx.Tr("repo.settings.webhook_deletion_success"))
  960. }
  961. ctx.JSON(200, map[string]interface{}{
  962. "redirect": ctx.Repo.RepoLink + "/settings/hooks",
  963. })
  964. }