2014-08-24 16:59:47 +04:00
// Copyright 2014 The Gogs Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
2019-11-04 01:13:25 +03:00
package webhook
2014-08-24 16:59:47 +04:00
import (
"encoding/json"
"errors"
"fmt"
"strings"
2015-08-28 18:36:13 +03:00
2019-11-04 01:13:25 +03:00
"code.gitea.io/gitea/models"
2019-03-27 12:33:00 +03:00
"code.gitea.io/gitea/modules/git"
2019-11-04 01:13:25 +03:00
"code.gitea.io/gitea/modules/log"
2016-11-10 19:24:48 +03:00
"code.gitea.io/gitea/modules/setting"
2019-05-11 13:21:34 +03:00
api "code.gitea.io/gitea/modules/structs"
2014-08-24 16:59:47 +04:00
)
2017-01-05 03:50:34 +03:00
// SlackMeta contains the slack metadata
2015-08-28 18:36:13 +03:00
type SlackMeta struct {
2015-08-29 06:49:59 +03:00
Channel string ` json:"channel" `
Username string ` json:"username" `
IconURL string ` json:"icon_url" `
Color string ` json:"color" `
2014-08-24 16:59:47 +04:00
}
2019-11-04 01:13:25 +03:00
// GetSlackHook returns slack metadata
func GetSlackHook ( w * models . Webhook ) * SlackMeta {
s := & SlackMeta { }
if err := json . Unmarshal ( [ ] byte ( w . Meta ) , s ) ; err != nil {
log . Error ( "webhook.GetSlackHook(%d): %v" , w . ID , err )
}
return s
}
2016-11-25 00:35:47 +03:00
// SlackPayload contains the information about the slack channel
2014-08-24 16:59:47 +04:00
type SlackPayload struct {
Channel string ` json:"channel" `
Text string ` json:"text" `
Username string ` json:"username" `
2015-08-29 06:49:59 +03:00
IconURL string ` json:"icon_url" `
2014-08-24 16:59:47 +04:00
UnfurlLinks int ` json:"unfurl_links" `
LinkNames int ` json:"link_names" `
Attachments [ ] SlackAttachment ` json:"attachments" `
}
2016-11-25 00:35:47 +03:00
// SlackAttachment contains the slack message
2014-08-24 16:59:47 +04:00
type SlackAttachment struct {
2019-12-18 13:01:00 +03:00
Fallback string ` json:"fallback" `
Color string ` json:"color" `
Title string ` json:"title" `
TitleLink string ` json:"title_link" `
Text string ` json:"text" `
2014-08-24 16:59:47 +04:00
}
2016-11-25 00:35:47 +03:00
// SetSecret sets the slack secret
2015-08-28 18:36:13 +03:00
func ( p * SlackPayload ) SetSecret ( _ string ) { }
2016-11-25 00:35:47 +03:00
// JSONPayload Marshals the SlackPayload to json
2015-08-28 18:36:13 +03:00
func ( p * SlackPayload ) JSONPayload ( ) ( [ ] byte , error ) {
2015-08-28 18:38:09 +03:00
data , err := json . MarshalIndent ( p , "" , " " )
2014-08-24 16:59:47 +04:00
if err != nil {
return [ ] byte { } , err
}
return data , nil
}
2016-11-25 00:35:47 +03:00
// SlackTextFormatter replaces &, <, > with HTML characters
2015-08-28 18:36:13 +03:00
// see: https://api.slack.com/docs/formatting
func SlackTextFormatter ( s string ) string {
// replace & < >
2016-08-14 13:32:24 +03:00
s = strings . Replace ( s , "&" , "&" , - 1 )
s = strings . Replace ( s , "<" , "<" , - 1 )
s = strings . Replace ( s , ">" , ">" , - 1 )
return s
}
2016-11-25 00:35:47 +03:00
// SlackShortTextFormatter replaces &, <, > with HTML characters
2016-08-14 13:32:24 +03:00
func SlackShortTextFormatter ( s string ) string {
s = strings . Split ( s , "\n" ) [ 0 ]
// replace & < >
s = strings . Replace ( s , "&" , "&" , - 1 )
s = strings . Replace ( s , "<" , "<" , - 1 )
s = strings . Replace ( s , ">" , ">" , - 1 )
return s
2015-08-28 18:36:13 +03:00
}
2014-08-24 16:59:47 +04:00
2017-01-05 03:50:34 +03:00
// SlackLinkFormatter creates a link compatible with slack
2015-08-28 18:36:13 +03:00
func SlackLinkFormatter ( url string , text string ) string {
return fmt . Sprintf ( "<%s|%s>" , url , SlackTextFormatter ( text ) )
2014-08-24 16:59:47 +04:00
}
2017-10-30 05:04:25 +03:00
// SlackLinkToRef slack-formatter link to a repo ref
func SlackLinkToRef ( repoURL , ref string ) string {
refName := git . RefEndName ( ref )
switch {
case strings . HasPrefix ( ref , git . BranchPrefix ) :
return SlackLinkFormatter ( repoURL + "/src/branch/" + refName , refName )
case strings . HasPrefix ( ref , git . TagPrefix ) :
return SlackLinkFormatter ( repoURL + "/src/tag/" + refName , refName )
default :
return SlackLinkFormatter ( repoURL + "/src/commit/" + refName , refName )
}
}
2015-08-28 18:36:13 +03:00
2017-10-30 05:04:25 +03:00
func getSlackCreatePayload ( p * api . CreatePayload , slack * SlackMeta ) ( * SlackPayload , error ) {
2019-11-30 13:23:37 +03:00
repoLink := SlackLinkFormatter ( p . Repo . HTMLURL , p . Repo . FullName )
2017-10-30 05:04:25 +03:00
refLink := SlackLinkToRef ( p . Repo . HTMLURL , p . Ref )
2015-08-28 18:36:13 +03:00
text := fmt . Sprintf ( "[%s:%s] %s created by %s" , repoLink , refLink , p . RefType , p . Sender . UserName )
return & SlackPayload {
Channel : slack . Channel ,
Text : text ,
2015-08-29 06:49:59 +03:00
Username : slack . Username ,
IconURL : slack . IconURL ,
2015-08-28 18:36:13 +03:00
} , nil
}
2018-05-16 17:01:55 +03:00
// getSlackDeletePayload composes Slack payload for delete a branch or tag.
func getSlackDeletePayload ( p * api . DeletePayload , slack * SlackMeta ) ( * SlackPayload , error ) {
refName := git . RefEndName ( p . Ref )
2019-11-30 13:23:37 +03:00
repoLink := SlackLinkFormatter ( p . Repo . HTMLURL , p . Repo . FullName )
2018-05-16 17:01:55 +03:00
text := fmt . Sprintf ( "[%s:%s] %s deleted by %s" , repoLink , refName , p . RefType , p . Sender . UserName )
return & SlackPayload {
Channel : slack . Channel ,
Text : text ,
Username : slack . Username ,
IconURL : slack . IconURL ,
} , nil
}
// getSlackForkPayload composes Slack payload for forked by a repository.
func getSlackForkPayload ( p * api . ForkPayload , slack * SlackMeta ) ( * SlackPayload , error ) {
2019-08-07 02:27:10 +03:00
baseLink := SlackLinkFormatter ( p . Forkee . HTMLURL , p . Forkee . FullName )
forkLink := SlackLinkFormatter ( p . Repo . HTMLURL , p . Repo . FullName )
2018-05-16 17:01:55 +03:00
text := fmt . Sprintf ( "%s is forked to %s" , baseLink , forkLink )
return & SlackPayload {
Channel : slack . Channel ,
Text : text ,
Username : slack . Username ,
IconURL : slack . IconURL ,
} , nil
}
func getSlackIssuesPayload ( p * api . IssuePayload , slack * SlackMeta ) ( * SlackPayload , error ) {
2019-12-28 11:55:09 +03:00
text , issueTitle , attachmentText , color := getIssuesPayloadInfo ( p , SlackLinkFormatter )
2018-05-16 17:01:55 +03:00
2019-12-18 13:01:00 +03:00
pl := & SlackPayload {
2018-05-16 17:01:55 +03:00
Channel : slack . Channel ,
Text : text ,
Username : slack . Username ,
IconURL : slack . IconURL ,
2019-12-18 13:01:00 +03:00
}
if attachmentText != "" {
2019-12-28 11:55:09 +03:00
attachmentText = SlackTextFormatter ( attachmentText )
issueTitle = SlackTextFormatter ( issueTitle )
2019-12-18 13:01:00 +03:00
pl . Attachments = [ ] SlackAttachment { {
2019-12-28 11:55:09 +03:00
Color : fmt . Sprintf ( "%x" , color ) ,
Title : issueTitle ,
TitleLink : p . Issue . URL ,
2019-12-18 13:01:00 +03:00
Text : attachmentText ,
} }
}
return pl , nil
2018-05-16 17:01:55 +03:00
}
func getSlackIssueCommentPayload ( p * api . IssueCommentPayload , slack * SlackMeta ) ( * SlackPayload , error ) {
2019-12-28 11:55:09 +03:00
text , issueTitle , color := getIssueCommentPayloadInfo ( p , SlackLinkFormatter )
2018-05-16 17:01:55 +03:00
return & SlackPayload {
Channel : slack . Channel ,
Text : text ,
Username : slack . Username ,
IconURL : slack . IconURL ,
Attachments : [ ] SlackAttachment { {
2019-12-28 11:55:09 +03:00
Color : fmt . Sprintf ( "%x" , color ) ,
Title : issueTitle ,
TitleLink : p . Comment . HTMLURL ,
Text : SlackTextFormatter ( p . Comment . Body ) ,
2018-05-16 17:01:55 +03:00
} } ,
} , nil
}
func getSlackReleasePayload ( p * api . ReleasePayload , slack * SlackMeta ) ( * SlackPayload , error ) {
2019-12-28 11:55:09 +03:00
text , _ := getReleasePayloadInfo ( p , SlackLinkFormatter )
2018-05-21 05:28:29 +03:00
2018-05-16 17:01:55 +03:00
return & SlackPayload {
Channel : slack . Channel ,
Text : text ,
Username : slack . Username ,
IconURL : slack . IconURL ,
} , nil
}
2015-08-28 18:36:13 +03:00
func getSlackPushPayload ( p * api . PushPayload , slack * SlackMeta ) ( * SlackPayload , error ) {
2014-08-24 16:59:47 +04:00
// n new commits
2015-08-28 18:36:13 +03:00
var (
2015-12-04 23:41:56 +03:00
commitDesc string
2015-08-28 18:36:13 +03:00
commitString string
)
2014-08-24 16:59:47 +04:00
if len ( p . Commits ) == 1 {
2015-12-04 23:41:56 +03:00
commitDesc = "1 new commit"
2014-08-24 16:59:47 +04:00
} else {
2015-12-04 23:41:56 +03:00
commitDesc = fmt . Sprintf ( "%d new commits" , len ( p . Commits ) )
}
2016-08-14 14:17:26 +03:00
if len ( p . CompareURL ) > 0 {
commitString = SlackLinkFormatter ( p . CompareURL , commitDesc )
2015-12-04 23:41:56 +03:00
} else {
commitString = commitDesc
2014-08-24 16:59:47 +04:00
}
2019-11-30 13:23:37 +03:00
repoLink := SlackLinkFormatter ( p . Repo . HTMLURL , p . Repo . FullName )
2017-10-30 05:04:25 +03:00
branchLink := SlackLinkToRef ( p . Repo . HTMLURL , p . Ref )
2016-08-14 14:17:26 +03:00
text := fmt . Sprintf ( "[%s:%s] %s pushed by %s" , repoLink , branchLink , commitString , p . Pusher . UserName )
2014-08-24 16:59:47 +04:00
2015-08-28 18:36:13 +03:00
var attachmentText string
2014-08-24 16:59:47 +04:00
// for each commit, generate attachment text
for i , commit := range p . Commits {
2016-08-14 13:32:24 +03:00
attachmentText += fmt . Sprintf ( "%s: %s - %s" , SlackLinkFormatter ( commit . URL , commit . ID [ : 7 ] ) , SlackShortTextFormatter ( commit . Message ) , SlackTextFormatter ( commit . Author . Name ) )
2014-08-24 16:59:47 +04:00
// add linebreak to each commit but the last
if i < len ( p . Commits ) - 1 {
attachmentText += "\n"
}
}
2016-08-14 13:32:24 +03:00
return & SlackPayload {
Channel : slack . Channel ,
Text : text ,
Username : slack . Username ,
IconURL : slack . IconURL ,
Attachments : [ ] SlackAttachment { {
Color : slack . Color ,
Text : attachmentText ,
} } ,
} , nil
}
func getSlackPullRequestPayload ( p * api . PullRequestPayload , slack * SlackMeta ) ( * SlackPayload , error ) {
2019-12-28 11:55:09 +03:00
text , issueTitle , attachmentText , color := getPullRequestPayloadInfo ( p , SlackLinkFormatter )
2014-08-24 16:59:47 +04:00
2019-12-18 13:01:00 +03:00
pl := & SlackPayload {
2016-08-14 13:32:24 +03:00
Channel : slack . Channel ,
Text : text ,
Username : slack . Username ,
IconURL : slack . IconURL ,
2019-12-18 13:01:00 +03:00
}
if attachmentText != "" {
2019-12-28 11:55:09 +03:00
attachmentText = SlackTextFormatter ( p . PullRequest . Body )
issueTitle = SlackTextFormatter ( issueTitle )
2019-12-18 13:01:00 +03:00
pl . Attachments = [ ] SlackAttachment { {
2019-12-28 11:55:09 +03:00
Color : fmt . Sprintf ( "%x" , color ) ,
Title : issueTitle ,
TitleLink : p . PullRequest . URL ,
2019-12-18 13:01:00 +03:00
Text : attachmentText ,
} }
}
return pl , nil
2014-08-24 16:59:47 +04:00
}
2019-11-04 01:13:25 +03:00
func getSlackPullRequestApprovalPayload ( p * api . PullRequestPayload , slack * SlackMeta , event models . HookEventType ) ( * SlackPayload , error ) {
2018-12-27 21:04:30 +03:00
senderLink := SlackLinkFormatter ( setting . AppURL + p . Sender . UserName , p . Sender . UserName )
2019-12-18 13:01:00 +03:00
title := fmt . Sprintf ( "#%d %s" , p . Index , p . PullRequest . Title )
titleLink := fmt . Sprintf ( "%s/pulls/%d" , p . Repository . HTMLURL , p . Index )
2019-11-30 13:23:37 +03:00
repoLink := SlackLinkFormatter ( p . Repository . HTMLURL , p . Repository . FullName )
2019-12-18 13:01:00 +03:00
var text string
2018-12-27 21:04:30 +03:00
switch p . Action {
case api . HookIssueSynchronized :
action , err := parseHookPullRequestEventType ( event )
if err != nil {
return nil , err
}
2019-12-18 13:01:00 +03:00
text = fmt . Sprintf ( "[%s] Pull request review %s: [%s](%s) by %s" , repoLink , action , title , titleLink , senderLink )
2018-12-27 21:04:30 +03:00
}
return & SlackPayload {
Channel : slack . Channel ,
Text : text ,
Username : slack . Username ,
IconURL : slack . IconURL ,
} , nil
}
2017-09-03 11:20:24 +03:00
func getSlackRepositoryPayload ( p * api . RepositoryPayload , slack * SlackMeta ) ( * SlackPayload , error ) {
senderLink := SlackLinkFormatter ( setting . AppURL + p . Sender . UserName , p . Sender . UserName )
2019-11-30 13:23:37 +03:00
repoLink := SlackLinkFormatter ( p . Repository . HTMLURL , p . Repository . FullName )
2017-09-03 11:20:24 +03:00
var text , title , attachmentText string
2019-12-18 13:01:00 +03:00
2017-09-03 11:20:24 +03:00
switch p . Action {
case api . HookRepoCreated :
2019-11-30 13:23:37 +03:00
text = fmt . Sprintf ( "[%s] Repository created by %s" , repoLink , senderLink )
2017-09-03 11:20:24 +03:00
title = p . Repository . HTMLURL
case api . HookRepoDeleted :
2019-11-30 13:23:37 +03:00
text = fmt . Sprintf ( "[%s] Repository deleted by %s" , repoLink , senderLink )
2017-09-03 11:20:24 +03:00
}
return & SlackPayload {
Channel : slack . Channel ,
Text : text ,
Username : slack . Username ,
IconURL : slack . IconURL ,
Attachments : [ ] SlackAttachment { {
2019-12-18 13:01:00 +03:00
Color : slack . Color ,
Title : title ,
TitleLink : title ,
Text : attachmentText ,
2017-09-03 11:20:24 +03:00
} } ,
} , nil
}
2016-11-25 00:35:47 +03:00
// GetSlackPayload converts a slack webhook into a SlackPayload
2019-11-04 01:13:25 +03:00
func GetSlackPayload ( p api . Payloader , event models . HookEventType , meta string ) ( * SlackPayload , error ) {
2015-08-28 18:36:13 +03:00
s := new ( SlackPayload )
2014-08-26 16:20:18 +04:00
2015-08-28 18:36:13 +03:00
slack := & SlackMeta { }
if err := json . Unmarshal ( [ ] byte ( meta ) , & slack ) ; err != nil {
return s , errors . New ( "GetSlackPayload meta json:" + err . Error ( ) )
}
switch event {
2019-11-04 01:13:25 +03:00
case models . HookEventCreate :
2015-08-28 18:36:13 +03:00
return getSlackCreatePayload ( p . ( * api . CreatePayload ) , slack )
2019-11-04 01:13:25 +03:00
case models . HookEventDelete :
2018-05-16 17:01:55 +03:00
return getSlackDeletePayload ( p . ( * api . DeletePayload ) , slack )
2019-11-04 01:13:25 +03:00
case models . HookEventFork :
2018-05-16 17:01:55 +03:00
return getSlackForkPayload ( p . ( * api . ForkPayload ) , slack )
2019-11-04 01:13:25 +03:00
case models . HookEventIssues :
2018-05-16 17:01:55 +03:00
return getSlackIssuesPayload ( p . ( * api . IssuePayload ) , slack )
2019-11-04 01:13:25 +03:00
case models . HookEventIssueComment :
2018-05-16 17:01:55 +03:00
return getSlackIssueCommentPayload ( p . ( * api . IssueCommentPayload ) , slack )
2019-11-04 01:13:25 +03:00
case models . HookEventPush :
2015-08-28 18:36:13 +03:00
return getSlackPushPayload ( p . ( * api . PushPayload ) , slack )
2019-11-04 01:13:25 +03:00
case models . HookEventPullRequest :
2016-08-14 13:32:24 +03:00
return getSlackPullRequestPayload ( p . ( * api . PullRequestPayload ) , slack )
2019-11-04 01:13:25 +03:00
case models . HookEventPullRequestRejected , models . HookEventPullRequestApproved , models . HookEventPullRequestComment :
2018-12-27 21:04:30 +03:00
return getSlackPullRequestApprovalPayload ( p . ( * api . PullRequestPayload ) , slack , event )
2019-11-04 01:13:25 +03:00
case models . HookEventRepository :
2017-09-03 11:20:24 +03:00
return getSlackRepositoryPayload ( p . ( * api . RepositoryPayload ) , slack )
2019-11-04 01:13:25 +03:00
case models . HookEventRelease :
2018-05-16 17:01:55 +03:00
return getSlackReleasePayload ( p . ( * api . ReleasePayload ) , slack )
2015-08-28 18:36:13 +03:00
}
return s , nil
2014-08-26 16:20:18 +04:00
}