2014-08-24 08: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.
package models
import (
"encoding/json"
"errors"
"fmt"
"strings"
2015-08-28 23:36:13 +08:00
2016-11-10 17:24:48 +01:00
"code.gitea.io/git"
2016-11-11 10:39:44 +01:00
api "code.gitea.io/sdk/gitea"
2016-08-14 03:32:24 -07:00
2016-11-10 17:24:48 +01:00
"code.gitea.io/gitea/modules/setting"
2014-08-24 08:59:47 -04:00
)
2017-01-04 19:50:34 -05:00
// SlackMeta contains the slack metadata
2015-08-28 23:36:13 +08:00
type SlackMeta struct {
2015-08-29 11:49:59 +08:00
Channel string ` json:"channel" `
Username string ` json:"username" `
IconURL string ` json:"icon_url" `
Color string ` json:"color" `
2014-08-24 08:59:47 -04:00
}
2016-11-24 22:35:47 +01:00
// SlackPayload contains the information about the slack channel
2014-08-24 08:59:47 -04:00
type SlackPayload struct {
Channel string ` json:"channel" `
Text string ` json:"text" `
Username string ` json:"username" `
2015-08-29 11:49:59 +08:00
IconURL string ` json:"icon_url" `
2014-08-24 08:59:47 -04:00
UnfurlLinks int ` json:"unfurl_links" `
LinkNames int ` json:"link_names" `
Attachments [ ] SlackAttachment ` json:"attachments" `
}
2016-11-24 22:35:47 +01:00
// SlackAttachment contains the slack message
2014-08-24 08:59:47 -04:00
type SlackAttachment struct {
2015-12-01 21:16:19 -05:00
Fallback string ` json:"fallback" `
Color string ` json:"color" `
2016-08-14 03:32:24 -07:00
Title string ` json:"title" `
2015-12-01 21:16:19 -05:00
Text string ` json:"text" `
2014-08-24 08:59:47 -04:00
}
2016-11-24 22:35:47 +01:00
// SetSecret sets the slack secret
2015-08-28 23:36:13 +08:00
func ( p * SlackPayload ) SetSecret ( _ string ) { }
2016-11-24 22:35:47 +01:00
// JSONPayload Marshals the SlackPayload to json
2015-08-28 23:36:13 +08:00
func ( p * SlackPayload ) JSONPayload ( ) ( [ ] byte , error ) {
2015-08-28 23:38:09 +08:00
data , err := json . MarshalIndent ( p , "" , " " )
2014-08-24 08:59:47 -04:00
if err != nil {
return [ ] byte { } , err
}
return data , nil
}
2016-11-24 22:35:47 +01:00
// SlackTextFormatter replaces &, <, > with HTML characters
2015-08-28 23:36:13 +08:00
// see: https://api.slack.com/docs/formatting
func SlackTextFormatter ( s string ) string {
// replace & < >
2016-08-14 03:32:24 -07:00
s = strings . Replace ( s , "&" , "&" , - 1 )
s = strings . Replace ( s , "<" , "<" , - 1 )
s = strings . Replace ( s , ">" , ">" , - 1 )
return s
}
2016-11-24 22:35:47 +01:00
// SlackShortTextFormatter replaces &, <, > with HTML characters
2016-08-14 03:32:24 -07: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 23:36:13 +08:00
}
2014-08-24 08:59:47 -04:00
2017-01-04 19:50:34 -05:00
// SlackLinkFormatter creates a link compatible with slack
2015-08-28 23:36:13 +08:00
func SlackLinkFormatter ( url string , text string ) string {
return fmt . Sprintf ( "<%s|%s>" , url , SlackTextFormatter ( text ) )
2014-08-24 08:59:47 -04:00
}
2015-08-28 23:36:13 +08:00
func getSlackCreatePayload ( p * api . CreatePayload , slack * SlackMeta ) ( * SlackPayload , error ) {
// created tag/branch
refName := git . RefEndName ( p . Ref )
2016-08-14 04:17:26 -07:00
repoLink := SlackLinkFormatter ( p . Repo . HTMLURL , p . Repo . Name )
refLink := SlackLinkFormatter ( p . Repo . HTMLURL + "/src/" + refName , refName )
2015-08-28 23:36:13 +08: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 11:49:59 +08:00
Username : slack . Username ,
IconURL : slack . IconURL ,
2015-08-28 23:36:13 +08:00
} , nil
}
func getSlackPushPayload ( p * api . PushPayload , slack * SlackMeta ) ( * SlackPayload , error ) {
2014-08-24 08:59:47 -04:00
// n new commits
2015-08-28 23:36:13 +08:00
var (
branchName = git . RefEndName ( p . Ref )
2015-12-04 15:41:56 -05:00
commitDesc string
2015-08-28 23:36:13 +08:00
commitString string
)
2014-08-24 08:59:47 -04:00
if len ( p . Commits ) == 1 {
2015-12-04 15:41:56 -05:00
commitDesc = "1 new commit"
2014-08-24 08:59:47 -04:00
} else {
2015-12-04 15:41:56 -05:00
commitDesc = fmt . Sprintf ( "%d new commits" , len ( p . Commits ) )
}
2016-08-14 04:17:26 -07:00
if len ( p . CompareURL ) > 0 {
commitString = SlackLinkFormatter ( p . CompareURL , commitDesc )
2015-12-04 15:41:56 -05:00
} else {
commitString = commitDesc
2014-08-24 08:59:47 -04:00
}
2016-08-14 04:17:26 -07:00
repoLink := SlackLinkFormatter ( p . Repo . HTMLURL , p . Repo . Name )
branchLink := SlackLinkFormatter ( p . Repo . HTMLURL + "/src/" + branchName , branchName )
text := fmt . Sprintf ( "[%s:%s] %s pushed by %s" , repoLink , branchLink , commitString , p . Pusher . UserName )
2014-08-24 08:59:47 -04:00
2015-08-28 23:36:13 +08:00
var attachmentText string
2014-08-24 08:59:47 -04:00
// for each commit, generate attachment text
for i , commit := range p . Commits {
2016-08-14 03:32:24 -07:00
attachmentText += fmt . Sprintf ( "%s: %s - %s" , SlackLinkFormatter ( commit . URL , commit . ID [ : 7 ] ) , SlackShortTextFormatter ( commit . Message ) , SlackTextFormatter ( commit . Author . Name ) )
2014-08-24 08:59:47 -04:00
// add linebreak to each commit but the last
if i < len ( p . Commits ) - 1 {
attachmentText += "\n"
}
}
2016-08-14 03:32:24 -07: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 ) {
2016-11-27 18:14:25 +08:00
senderLink := SlackLinkFormatter ( setting . AppURL + p . Sender . UserName , p . Sender . UserName )
2016-08-14 04:17:26 -07:00
titleLink := SlackLinkFormatter ( fmt . Sprintf ( "%s/pulls/%d" , p . Repository . HTMLURL , p . Index ) ,
2016-08-14 03:32:24 -07:00
fmt . Sprintf ( "#%d %s" , p . Index , p . PullRequest . Title ) )
var text , title , attachmentText string
switch p . Action {
2016-11-07 16:37:32 +01:00
case api . HookIssueOpened :
2016-08-14 03:32:24 -07:00
text = fmt . Sprintf ( "[%s] Pull request submitted by %s" , p . Repository . FullName , senderLink )
title = titleLink
attachmentText = SlackTextFormatter ( p . PullRequest . Body )
2016-11-07 16:37:32 +01:00
case api . HookIssueClosed :
2016-08-14 03:32:24 -07:00
if p . PullRequest . HasMerged {
text = fmt . Sprintf ( "[%s] Pull request merged: %s by %s" , p . Repository . FullName , titleLink , senderLink )
} else {
text = fmt . Sprintf ( "[%s] Pull request closed: %s by %s" , p . Repository . FullName , titleLink , senderLink )
}
2016-11-29 09:25:47 +01:00
case api . HookIssueReOpened :
2016-08-14 03:32:24 -07:00
text = fmt . Sprintf ( "[%s] Pull request re-opened: %s by %s" , p . Repository . FullName , titleLink , senderLink )
2016-11-07 16:37:32 +01:00
case api . HookIssueEdited :
2016-08-14 03:32:24 -07:00
text = fmt . Sprintf ( "[%s] Pull request edited: %s by %s" , p . Repository . FullName , titleLink , senderLink )
attachmentText = SlackTextFormatter ( p . PullRequest . Body )
2016-11-07 16:37:32 +01:00
case api . HookIssueAssigned :
2016-08-14 03:32:24 -07:00
text = fmt . Sprintf ( "[%s] Pull request assigned to %s: %s by %s" , p . Repository . FullName ,
2016-11-27 18:14:25 +08:00
SlackLinkFormatter ( setting . AppURL + p . PullRequest . Assignee . UserName , p . PullRequest . Assignee . UserName ) ,
2016-08-14 03:32:24 -07:00
titleLink , senderLink )
2016-11-07 16:37:32 +01:00
case api . HookIssueUnassigned :
2016-08-14 03:32:24 -07:00
text = fmt . Sprintf ( "[%s] Pull request unassigned: %s by %s" , p . Repository . FullName , titleLink , senderLink )
2016-11-07 16:37:32 +01:00
case api . HookIssueLabelUpdated :
2016-08-14 03:32:24 -07:00
text = fmt . Sprintf ( "[%s] Pull request labels updated: %s by %s" , p . Repository . FullName , titleLink , senderLink )
2016-11-07 16:37:32 +01:00
case api . HookIssueLabelCleared :
2016-08-14 03:32:24 -07:00
text = fmt . Sprintf ( "[%s] Pull request labels cleared: %s by %s" , p . Repository . FullName , titleLink , senderLink )
2016-11-07 16:37:32 +01:00
case api . HookIssueSynchronized :
2016-08-14 03:32:24 -07:00
text = fmt . Sprintf ( "[%s] Pull request synchronized: %s by %s" , p . Repository . FullName , titleLink , senderLink )
}
2014-08-24 08:59:47 -04:00
return & SlackPayload {
2016-08-14 03:32:24 -07:00
Channel : slack . Channel ,
Text : text ,
Username : slack . Username ,
IconURL : slack . IconURL ,
Attachments : [ ] SlackAttachment { {
Color : slack . Color ,
Title : title ,
Text : attachmentText ,
} } ,
2014-08-24 08:59:47 -04:00
} , nil
}
2016-11-24 22:35:47 +01:00
// GetSlackPayload converts a slack webhook into a SlackPayload
2015-08-28 23:36:13 +08:00
func GetSlackPayload ( p api . Payloader , event HookEventType , meta string ) ( * SlackPayload , error ) {
s := new ( SlackPayload )
2014-08-26 08:20:18 -04:00
2015-08-28 23:36:13 +08:00
slack := & SlackMeta { }
if err := json . Unmarshal ( [ ] byte ( meta ) , & slack ) ; err != nil {
return s , errors . New ( "GetSlackPayload meta json:" + err . Error ( ) )
}
switch event {
2016-11-07 16:37:32 +01:00
case HookEventCreate :
2015-08-28 23:36:13 +08:00
return getSlackCreatePayload ( p . ( * api . CreatePayload ) , slack )
2016-11-07 16:37:32 +01:00
case HookEventPush :
2015-08-28 23:36:13 +08:00
return getSlackPushPayload ( p . ( * api . PushPayload ) , slack )
2016-11-07 16:37:32 +01:00
case HookEventPullRequest :
2016-08-14 03:32:24 -07:00
return getSlackPullRequestPayload ( p . ( * api . PullRequestPayload ) , slack )
2015-08-28 23:36:13 +08:00
}
return s , nil
2014-08-26 08:20:18 -04:00
}