2019-04-20 00:18:06 +10:00
// Copyright 2019 The Gitea Authors. All rights reserved.
2022-11-27 13:20:29 -05:00
// SPDX-License-Identifier: MIT
2019-04-20 00:18:06 +10:00
2019-11-04 06:13:25 +08:00
package webhook
2019-04-20 00:18:06 +10:00
import (
"fmt"
2022-09-04 21:54:23 +02:00
"net/url"
2019-04-20 00:18:06 +10:00
"strings"
"code.gitea.io/gitea/modules/git"
2021-07-25 00:03:58 +08:00
"code.gitea.io/gitea/modules/json"
2019-05-11 18:21:34 +08:00
api "code.gitea.io/gitea/modules/structs"
2021-11-16 18:18:25 +00:00
"code.gitea.io/gitea/modules/util"
2023-01-01 16:23:15 +01:00
webhook_module "code.gitea.io/gitea/modules/webhook"
2019-04-20 00:18:06 +10:00
)
type (
// MSTeamsFact for Fact Structure
MSTeamsFact struct {
Name string ` json:"name" `
Value string ` json:"value" `
}
// MSTeamsSection is a MessageCard section
MSTeamsSection struct {
ActivityTitle string ` json:"activityTitle" `
ActivitySubtitle string ` json:"activitySubtitle" `
ActivityImage string ` json:"activityImage" `
Facts [ ] MSTeamsFact ` json:"facts" `
Text string ` json:"text" `
}
// MSTeamsAction is an action (creates buttons, links etc)
MSTeamsAction struct {
Type string ` json:"@type" `
Name string ` json:"name" `
Targets [ ] MSTeamsActionTarget ` json:"targets,omitempty" `
}
// MSTeamsActionTarget is the actual link to follow, etc
MSTeamsActionTarget struct {
Os string ` json:"os" `
URI string ` json:"uri" `
}
// MSTeamsPayload is the parent object
MSTeamsPayload struct {
Type string ` json:"@type" `
Context string ` json:"@context" `
ThemeColor string ` json:"themeColor" `
Title string ` json:"title" `
Summary string ` json:"summary" `
Sections [ ] MSTeamsSection ` json:"sections" `
PotentialAction [ ] MSTeamsAction ` json:"potentialAction" `
}
)
// JSONPayload Marshals the MSTeamsPayload to json
2020-09-05 10:57:13 +08:00
func ( m * MSTeamsPayload ) JSONPayload ( ) ( [ ] byte , error ) {
data , err := json . MarshalIndent ( m , "" , " " )
2019-04-20 00:18:06 +10:00
if err != nil {
return [ ] byte { } , err
}
return data , nil
}
2022-01-20 18:46:10 +01:00
var _ PayloadConvertor = & MSTeamsPayload { }
2020-09-05 10:57:13 +08:00
// Create implements PayloadConvertor Create method
func ( m * MSTeamsPayload ) Create ( p * api . CreatePayload ) ( api . Payloader , error ) {
2019-04-20 00:18:06 +10:00
// created tag/branch
2023-05-26 09:04:48 +08:00
refName := git . RefName ( p . Ref ) . ShortName ( )
2019-04-20 00:18:06 +10:00
title := fmt . Sprintf ( "[%s] %s %s created" , p . Repo . FullName , p . RefType , refName )
2021-06-21 04:12:19 +02:00
return createMSTeamsPayload (
p . Repo ,
p . Sender ,
title ,
"" ,
2021-11-16 18:18:25 +00:00
p . Repo . HTMLURL + "/src/" + util . PathEscapeSegments ( refName ) ,
2021-06-21 04:12:19 +02:00
greenColor ,
& MSTeamsFact { fmt . Sprintf ( "%s:" , p . RefType ) , refName } ,
) , nil
2019-04-20 00:18:06 +10:00
}
2020-09-05 10:57:13 +08:00
// Delete implements PayloadConvertor Delete method
func ( m * MSTeamsPayload ) Delete ( p * api . DeletePayload ) ( api . Payloader , error ) {
2019-04-20 00:18:06 +10:00
// deleted tag/branch
2023-05-26 09:04:48 +08:00
refName := git . RefName ( p . Ref ) . ShortName ( )
2019-04-20 00:18:06 +10:00
title := fmt . Sprintf ( "[%s] %s %s deleted" , p . Repo . FullName , p . RefType , refName )
2021-06-21 04:12:19 +02:00
return createMSTeamsPayload (
p . Repo ,
p . Sender ,
title ,
"" ,
2021-11-16 18:18:25 +00:00
p . Repo . HTMLURL + "/src/" + util . PathEscapeSegments ( refName ) ,
2021-06-21 04:12:19 +02:00
yellowColor ,
& MSTeamsFact { fmt . Sprintf ( "%s:" , p . RefType ) , refName } ,
) , nil
2019-04-20 00:18:06 +10:00
}
2020-09-05 10:57:13 +08:00
// Fork implements PayloadConvertor Fork method
func ( m * MSTeamsPayload ) Fork ( p * api . ForkPayload ) ( api . Payloader , error ) {
2019-04-20 00:18:06 +10:00
title := fmt . Sprintf ( "%s is forked to %s" , p . Forkee . FullName , p . Repo . FullName )
2021-06-21 04:12:19 +02:00
return createMSTeamsPayload (
p . Repo ,
p . Sender ,
title ,
"" ,
p . Repo . HTMLURL ,
greenColor ,
& MSTeamsFact { "Forkee:" , p . Forkee . FullName } ,
) , nil
2019-04-20 00:18:06 +10:00
}
2020-09-05 10:57:13 +08:00
// Push implements PayloadConvertor Push method
func ( m * MSTeamsPayload ) Push ( p * api . PushPayload ) ( api . Payloader , error ) {
2019-04-20 00:18:06 +10:00
var (
2023-05-26 09:04:48 +08:00
branchName = git . RefName ( p . Ref ) . ShortName ( )
2019-04-20 00:18:06 +10:00
commitDesc string
)
var titleLink string
2022-10-16 18:22:34 +02:00
if p . TotalCommits == 1 {
2019-04-20 00:18:06 +10:00
commitDesc = "1 new commit"
titleLink = p . Commits [ 0 ] . URL
} else {
2022-10-16 18:22:34 +02:00
commitDesc = fmt . Sprintf ( "%d new commits" , p . TotalCommits )
2019-04-20 00:18:06 +10:00
titleLink = p . CompareURL
}
if titleLink == "" {
2021-11-16 18:18:25 +00:00
titleLink = p . Repo . HTMLURL + "/src/" + util . PathEscapeSegments ( branchName )
2019-04-20 00:18:06 +10:00
}
title := fmt . Sprintf ( "[%s:%s] %s" , p . Repo . FullName , branchName , commitDesc )
var text string
// for each commit, generate attachment text
for i , commit := range p . Commits {
text += fmt . Sprintf ( "[%s](%s) %s - %s" , commit . ID [ : 7 ] , commit . URL ,
strings . TrimRight ( commit . Message , "\r\n" ) , commit . Author . Name )
// add linebreak to each commit but the last
if i < len ( p . Commits ) - 1 {
2020-10-08 22:30:55 -07:00
text += "\n\n"
2019-04-20 00:18:06 +10:00
}
}
2021-06-21 04:12:19 +02:00
return createMSTeamsPayload (
p . Repo ,
p . Sender ,
title ,
text ,
titleLink ,
greenColor ,
2022-10-16 18:22:34 +02:00
& MSTeamsFact { "Commit count:" , fmt . Sprintf ( "%d" , p . TotalCommits ) } ,
2021-06-21 04:12:19 +02:00
) , nil
2019-04-20 00:18:06 +10:00
}
2020-09-05 10:57:13 +08:00
// Issue implements PayloadConvertor Issue method
func ( m * MSTeamsPayload ) Issue ( p * api . IssuePayload ) ( api . Payloader , error ) {
2021-06-21 04:12:19 +02:00
title , _ , attachmentText , color := getIssuesPayloadInfo ( p , noneLinkFormatter , false )
return createMSTeamsPayload (
p . Repository ,
p . Sender ,
title ,
attachmentText ,
p . Issue . HTMLURL ,
color ,
& MSTeamsFact { "Issue #:" , fmt . Sprintf ( "%d" , p . Issue . ID ) } ,
) , nil
2019-04-20 00:18:06 +10:00
}
2020-09-05 10:57:13 +08:00
// IssueComment implements PayloadConvertor IssueComment method
func ( m * MSTeamsPayload ) IssueComment ( p * api . IssueCommentPayload ) ( api . Payloader , error ) {
2021-06-21 04:12:19 +02:00
title , _ , color := getIssueCommentPayloadInfo ( p , noneLinkFormatter , false )
return createMSTeamsPayload (
p . Repository ,
p . Sender ,
title ,
p . Comment . Body ,
p . Comment . HTMLURL ,
color ,
& MSTeamsFact { "Issue #:" , fmt . Sprintf ( "%d" , p . Issue . ID ) } ,
) , nil
2019-04-20 00:18:06 +10:00
}
2020-09-05 10:57:13 +08:00
// PullRequest implements PayloadConvertor PullRequest method
func ( m * MSTeamsPayload ) PullRequest ( p * api . PullRequestPayload ) ( api . Payloader , error ) {
2021-06-21 04:12:19 +02:00
title , _ , attachmentText , color := getPullRequestPayloadInfo ( p , noneLinkFormatter , false )
return createMSTeamsPayload (
p . Repository ,
p . Sender ,
title ,
attachmentText ,
p . PullRequest . HTMLURL ,
color ,
& MSTeamsFact { "Pull request #:" , fmt . Sprintf ( "%d" , p . PullRequest . ID ) } ,
) , nil
2019-04-20 00:18:06 +10:00
}
2020-09-05 10:57:13 +08:00
// Review implements PayloadConvertor Review method
2023-01-01 16:23:15 +01:00
func ( m * MSTeamsPayload ) Review ( p * api . PullRequestPayload , event webhook_module . HookEventType ) ( api . Payloader , error ) {
2019-04-20 00:18:06 +10:00
var text , title string
var color int
switch p . Action {
2020-03-05 23:10:48 -06:00
case api . HookIssueReviewed :
2019-04-20 00:18:06 +10:00
action , err := parseHookPullRequestEventType ( event )
if err != nil {
return nil , err
}
title = fmt . Sprintf ( "[%s] Pull request review %s: #%d %s" , p . Repository . FullName , action , p . Index , p . PullRequest . Title )
2019-10-18 17:42:04 -05:00
text = p . Review . Content
switch event {
2023-01-01 16:23:15 +01:00
case webhook_module . HookEventPullRequestReviewApproved :
2019-10-18 17:42:04 -05:00
color = greenColor
2023-01-01 16:23:15 +01:00
case webhook_module . HookEventPullRequestReviewRejected :
2019-10-18 17:42:04 -05:00
color = redColor
2023-03-24 13:13:04 +08:00
case webhook_module . HookEventPullRequestReviewComment :
2019-10-18 17:42:04 -05:00
color = greyColor
default :
color = yellowColor
}
2019-04-20 00:18:06 +10:00
}
2021-06-21 04:12:19 +02:00
return createMSTeamsPayload (
p . Repository ,
p . Sender ,
title ,
text ,
p . PullRequest . HTMLURL ,
color ,
& MSTeamsFact { "Pull request #:" , fmt . Sprintf ( "%d" , p . PullRequest . ID ) } ,
) , nil
2019-04-20 00:18:06 +10:00
}
2020-09-05 10:57:13 +08:00
// Repository implements PayloadConvertor Repository method
func ( m * MSTeamsPayload ) Repository ( p * api . RepositoryPayload ) ( api . Payloader , error ) {
2019-04-20 00:18:06 +10:00
var title , url string
var color int
switch p . Action {
case api . HookRepoCreated :
title = fmt . Sprintf ( "[%s] Repository created" , p . Repository . FullName )
url = p . Repository . HTMLURL
2019-10-18 17:42:04 -05:00
color = greenColor
2019-04-20 00:18:06 +10:00
case api . HookRepoDeleted :
title = fmt . Sprintf ( "[%s] Repository deleted" , p . Repository . FullName )
2019-10-18 17:42:04 -05:00
color = yellowColor
2019-04-20 00:18:06 +10:00
}
2021-06-21 04:12:19 +02:00
return createMSTeamsPayload (
p . Repository ,
p . Sender ,
title ,
"" ,
url ,
color ,
nil ,
) , nil
2019-04-20 00:18:06 +10:00
}
2022-09-04 21:54:23 +02:00
// Wiki implements PayloadConvertor Wiki method
func ( m * MSTeamsPayload ) Wiki ( p * api . WikiPayload ) ( api . Payloader , error ) {
title , color , _ := getWikiPayloadInfo ( p , noneLinkFormatter , false )
return createMSTeamsPayload (
p . Repository ,
p . Sender ,
title ,
"" ,
p . Repository . HTMLURL + "/wiki/" + url . PathEscape ( p . Page ) ,
color ,
& MSTeamsFact { "Repository:" , p . Repository . FullName } ,
) , nil
}
2020-09-05 10:57:13 +08:00
// Release implements PayloadConvertor Release method
func ( m * MSTeamsPayload ) Release ( p * api . ReleasePayload ) ( api . Payloader , error ) {
2021-06-21 04:12:19 +02:00
title , color := getReleasePayloadInfo ( p , noneLinkFormatter , false )
return createMSTeamsPayload (
p . Repository ,
p . Sender ,
title ,
"" ,
2023-09-21 17:55:09 -05:00
p . Release . HTMLURL ,
2021-06-21 04:12:19 +02:00
color ,
& MSTeamsFact { "Tag:" , p . Release . TagName } ,
) , nil
}
2023-10-31 12:43:38 +08:00
func ( m * MSTeamsPayload ) Package ( p * api . PackagePayload ) ( api . Payloader , error ) {
title , color := getPackagePayloadInfo ( p , noneLinkFormatter , false )
return createMSTeamsPayload (
p . Repository ,
p . Sender ,
title ,
"" ,
p . Package . HTMLURL ,
color ,
& MSTeamsFact { "Package:" , p . Package . Name } ,
) , nil
}
2021-06-21 04:12:19 +02:00
// GetMSTeamsPayload converts a MSTeams webhook into a MSTeamsPayload
2023-01-01 16:23:15 +01:00
func GetMSTeamsPayload ( p api . Payloader , event webhook_module . HookEventType , _ string ) ( api . Payloader , error ) {
2021-06-21 04:12:19 +02:00
return convertPayloader ( new ( MSTeamsPayload ) , p , event )
}
func createMSTeamsPayload ( r * api . Repository , s * api . User , title , text , actionTarget string , color int , fact * MSTeamsFact ) * MSTeamsPayload {
2023-11-17 12:17:33 +01:00
facts := make ( [ ] MSTeamsFact , 0 , 2 )
if r != nil {
facts = append ( facts , MSTeamsFact {
2021-06-21 04:12:19 +02:00
Name : "Repository:" ,
Value : r . FullName ,
2023-11-17 12:17:33 +01:00
} )
2021-06-21 04:12:19 +02:00
}
if fact != nil {
facts = append ( facts , * fact )
}
2019-04-20 00:18:06 +10:00
return & MSTeamsPayload {
Type : "MessageCard" ,
Context : "https://schema.org/extensions" ,
ThemeColor : fmt . Sprintf ( "%x" , color ) ,
2021-06-21 04:12:19 +02:00
Title : title ,
Summary : title ,
2019-04-20 00:18:06 +10:00
Sections : [ ] MSTeamsSection {
{
2021-06-21 04:12:19 +02:00
ActivityTitle : s . FullName ,
ActivitySubtitle : s . UserName ,
ActivityImage : s . AvatarURL ,
Text : text ,
Facts : facts ,
2019-04-20 00:18:06 +10:00
} ,
} ,
PotentialAction : [ ] MSTeamsAction {
{
Type : "OpenUri" ,
Name : "View in Gitea" ,
Targets : [ ] MSTeamsActionTarget {
{
Os : "default" ,
2021-06-21 04:12:19 +02:00
URI : actionTarget ,
2019-04-20 00:18:06 +10:00
} ,
} ,
} ,
} ,
2021-06-21 04:12:19 +02:00
}
2019-04-20 00:18:06 +10:00
}