2020-03-28 14:09:55 +01:00
// Copyright 2020 The Gitea Authors. All rights reserved.
2022-11-27 13:20:29 -05:00
// SPDX-License-Identifier: MIT
2020-03-28 14:09:55 +01:00
package webhook
import (
2020-07-31 00:04:19 +02:00
"crypto/sha1"
2022-11-03 19:23:20 +01:00
"encoding/hex"
2020-03-28 14:09:55 +01:00
"errors"
"fmt"
"html"
2021-11-16 18:18:25 +00:00
"net/url"
2020-03-28 14:09:55 +01:00
"regexp"
"strings"
2021-11-10 13:13:16 +08:00
webhook_model "code.gitea.io/gitea/models/webhook"
2020-03-28 14:09:55 +01:00
"code.gitea.io/gitea/modules/git"
2021-07-25 00:03:58 +08:00
"code.gitea.io/gitea/modules/json"
2020-03-28 14:09:55 +01:00
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/setting"
api "code.gitea.io/gitea/modules/structs"
2021-11-16 18:18:25 +00:00
"code.gitea.io/gitea/modules/util"
2020-03-28 14:09:55 +01:00
)
const matrixPayloadSizeLimit = 1024 * 64
// MatrixMeta contains the Matrix metadata
type MatrixMeta struct {
HomeserverURL string ` json:"homeserver_url" `
Room string ` json:"room_id" `
MessageType int ` json:"message_type" `
}
var messageTypeText = map [ int ] string {
1 : "m.notice" ,
2 : "m.text" ,
}
// GetMatrixHook returns Matrix metadata
2021-11-10 13:13:16 +08:00
func GetMatrixHook ( w * webhook_model . Webhook ) * MatrixMeta {
2020-03-28 14:09:55 +01:00
s := & MatrixMeta { }
if err := json . Unmarshal ( [ ] byte ( w . Meta ) , s ) ; err != nil {
log . Error ( "webhook.GetMatrixHook(%d): %v" , w . ID , err )
}
return s
}
2022-11-03 19:23:20 +01:00
var _ PayloadConvertor = & MatrixPayload { }
2020-03-28 14:09:55 +01:00
2022-11-03 19:23:20 +01:00
// MatrixPayload contains payload for a Matrix room
type MatrixPayload struct {
2020-03-28 14:09:55 +01:00
Body string ` json:"body" `
MsgType string ` json:"msgtype" `
Format string ` json:"format" `
FormattedBody string ` json:"formatted_body" `
Commits [ ] * api . PayloadCommit ` json:"io.gitea.commits,omitempty" `
}
2022-11-03 19:23:20 +01:00
// JSONPayload Marshals the MatrixPayload to json
func ( m * MatrixPayload ) JSONPayload ( ) ( [ ] byte , error ) {
2020-09-05 10:57:13 +08:00
data , err := json . MarshalIndent ( m , "" , " " )
2020-03-28 14:09:55 +01:00
if err != nil {
return [ ] byte { } , err
}
return data , nil
}
// MatrixLinkFormatter creates a link compatible with Matrix
2021-12-20 05:41:31 +01:00
func MatrixLinkFormatter ( url , text string ) string {
2020-03-28 14:09:55 +01:00
return fmt . Sprintf ( ` <a href="%s">%s</a> ` , html . EscapeString ( url ) , html . EscapeString ( text ) )
}
// MatrixLinkToRef Matrix-formatter link to a repo ref
func MatrixLinkToRef ( repoURL , ref string ) string {
refName := git . RefEndName ( ref )
switch {
case strings . HasPrefix ( ref , git . BranchPrefix ) :
2021-11-16 18:18:25 +00:00
return MatrixLinkFormatter ( repoURL + "/src/branch/" + util . PathEscapeSegments ( refName ) , refName )
2020-03-28 14:09:55 +01:00
case strings . HasPrefix ( ref , git . TagPrefix ) :
2021-11-16 18:18:25 +00:00
return MatrixLinkFormatter ( repoURL + "/src/tag/" + util . PathEscapeSegments ( refName ) , refName )
2020-03-28 14:09:55 +01:00
default :
2021-11-16 18:18:25 +00:00
return MatrixLinkFormatter ( repoURL + "/src/commit/" + util . PathEscapeSegments ( refName ) , refName )
2020-03-28 14:09:55 +01:00
}
}
2020-09-05 10:57:13 +08:00
// Create implements PayloadConvertor Create method
2022-11-03 19:23:20 +01:00
func ( m * MatrixPayload ) Create ( p * api . CreatePayload ) ( api . Payloader , error ) {
2020-03-28 14:09:55 +01:00
repoLink := MatrixLinkFormatter ( p . Repo . HTMLURL , p . Repo . FullName )
refLink := MatrixLinkToRef ( p . Repo . HTMLURL , p . Ref )
text := fmt . Sprintf ( "[%s:%s] %s created by %s" , repoLink , refLink , p . RefType , p . Sender . UserName )
2022-11-03 19:23:20 +01:00
return getMatrixPayload ( text , nil , m . MsgType ) , nil
2020-03-28 14:09:55 +01:00
}
2020-09-05 10:57:13 +08:00
// Delete composes Matrix payload for delete a branch or tag.
2022-11-03 19:23:20 +01:00
func ( m * MatrixPayload ) Delete ( p * api . DeletePayload ) ( api . Payloader , error ) {
2020-03-28 14:09:55 +01:00
refName := git . RefEndName ( p . Ref )
repoLink := MatrixLinkFormatter ( p . Repo . HTMLURL , p . Repo . FullName )
text := fmt . Sprintf ( "[%s:%s] %s deleted by %s" , repoLink , refName , p . RefType , p . Sender . UserName )
2022-11-03 19:23:20 +01:00
return getMatrixPayload ( text , nil , m . MsgType ) , nil
2020-03-28 14:09:55 +01:00
}
2020-09-05 10:57:13 +08:00
// Fork composes Matrix payload for forked by a repository.
2022-11-03 19:23:20 +01:00
func ( m * MatrixPayload ) Fork ( p * api . ForkPayload ) ( api . Payloader , error ) {
2020-03-28 14:09:55 +01:00
baseLink := MatrixLinkFormatter ( p . Forkee . HTMLURL , p . Forkee . FullName )
forkLink := MatrixLinkFormatter ( p . Repo . HTMLURL , p . Repo . FullName )
text := fmt . Sprintf ( "%s is forked to %s" , baseLink , forkLink )
2022-11-03 19:23:20 +01:00
return getMatrixPayload ( text , nil , m . MsgType ) , nil
2020-03-28 14:09:55 +01:00
}
2020-09-05 10:57:13 +08:00
// Issue implements PayloadConvertor Issue method
2022-11-03 19:23:20 +01:00
func ( m * MatrixPayload ) Issue ( p * api . IssuePayload ) ( api . Payloader , error ) {
2020-03-28 14:09:55 +01:00
text , _ , _ , _ := getIssuesPayloadInfo ( p , MatrixLinkFormatter , true )
2022-11-03 19:23:20 +01:00
return getMatrixPayload ( text , nil , m . MsgType ) , nil
2020-03-28 14:09:55 +01:00
}
2020-09-05 10:57:13 +08:00
// IssueComment implements PayloadConvertor IssueComment method
2022-11-03 19:23:20 +01:00
func ( m * MatrixPayload ) IssueComment ( p * api . IssueCommentPayload ) ( api . Payloader , error ) {
2020-03-28 14:09:55 +01:00
text , _ , _ := getIssueCommentPayloadInfo ( p , MatrixLinkFormatter , true )
2022-11-03 19:23:20 +01:00
return getMatrixPayload ( text , nil , m . MsgType ) , nil
2020-03-28 14:09:55 +01:00
}
2022-09-04 21:54:23 +02:00
// Wiki implements PayloadConvertor Wiki method
2022-11-03 19:23:20 +01:00
func ( m * MatrixPayload ) Wiki ( p * api . WikiPayload ) ( api . Payloader , error ) {
2022-09-04 21:54:23 +02:00
text , _ , _ := getWikiPayloadInfo ( p , MatrixLinkFormatter , true )
2022-11-03 19:23:20 +01:00
return getMatrixPayload ( text , nil , m . MsgType ) , nil
2022-09-04 21:54:23 +02:00
}
2020-09-05 10:57:13 +08:00
// Release implements PayloadConvertor Release method
2022-11-03 19:23:20 +01:00
func ( m * MatrixPayload ) Release ( p * api . ReleasePayload ) ( api . Payloader , error ) {
2020-03-28 14:09:55 +01:00
text , _ := getReleasePayloadInfo ( p , MatrixLinkFormatter , true )
2022-11-03 19:23:20 +01:00
return getMatrixPayload ( text , nil , m . MsgType ) , nil
2020-03-28 14:09:55 +01:00
}
2020-09-05 10:57:13 +08:00
// Push implements PayloadConvertor Push method
2022-11-03 19:23:20 +01:00
func ( m * MatrixPayload ) Push ( p * api . PushPayload ) ( api . Payloader , error ) {
2020-03-28 14:09:55 +01:00
var commitDesc string
2022-10-16 18:22:34 +02:00
if p . TotalCommits == 1 {
2020-03-28 14:09:55 +01:00
commitDesc = "1 commit"
} else {
2022-10-16 18:22:34 +02:00
commitDesc = fmt . Sprintf ( "%d commits" , p . TotalCommits )
2020-03-28 14:09:55 +01:00
}
repoLink := MatrixLinkFormatter ( p . Repo . HTMLURL , p . Repo . FullName )
branchLink := MatrixLinkToRef ( p . Repo . HTMLURL , p . Ref )
text := fmt . Sprintf ( "[%s] %s pushed %s to %s:<br>" , repoLink , p . Pusher . UserName , commitDesc , branchLink )
// for each commit, generate a new line text
for i , commit := range p . Commits {
text += fmt . Sprintf ( "%s: %s - %s" , MatrixLinkFormatter ( commit . URL , commit . ID [ : 7 ] ) , commit . Message , commit . Author . Name )
// add linebreak to each commit but the last
if i < len ( p . Commits ) - 1 {
text += "<br>"
}
}
2022-11-03 19:23:20 +01:00
return getMatrixPayload ( text , p . Commits , m . MsgType ) , nil
2020-03-28 14:09:55 +01:00
}
2020-09-05 10:57:13 +08:00
// PullRequest implements PayloadConvertor PullRequest method
2022-11-03 19:23:20 +01:00
func ( m * MatrixPayload ) PullRequest ( p * api . PullRequestPayload ) ( api . Payloader , error ) {
2020-03-28 14:09:55 +01:00
text , _ , _ , _ := getPullRequestPayloadInfo ( p , MatrixLinkFormatter , true )
2022-11-03 19:23:20 +01:00
return getMatrixPayload ( text , nil , m . MsgType ) , nil
2020-03-28 14:09:55 +01:00
}
2020-09-05 10:57:13 +08:00
// Review implements PayloadConvertor Review method
2022-11-03 19:23:20 +01:00
func ( m * MatrixPayload ) Review ( p * api . PullRequestPayload , event webhook_model . HookEventType ) ( api . Payloader , error ) {
2021-11-16 18:18:25 +00:00
senderLink := MatrixLinkFormatter ( setting . AppURL + url . PathEscape ( p . Sender . UserName ) , p . Sender . UserName )
2020-03-28 14:09:55 +01:00
title := fmt . Sprintf ( "#%d %s" , p . Index , p . PullRequest . Title )
2022-10-07 23:06:34 +02:00
titleLink := MatrixLinkFormatter ( p . PullRequest . URL , title )
2020-03-28 14:09:55 +01:00
repoLink := MatrixLinkFormatter ( p . Repository . HTMLURL , p . Repository . FullName )
var text string
switch p . Action {
case api . HookIssueReviewed :
action , err := parseHookPullRequestEventType ( event )
if err != nil {
return nil , err
}
2022-10-07 23:06:34 +02:00
text = fmt . Sprintf ( "[%s] Pull request review %s: %s by %s" , repoLink , action , titleLink , senderLink )
2020-03-28 14:09:55 +01:00
}
2022-11-03 19:23:20 +01:00
return getMatrixPayload ( text , nil , m . MsgType ) , nil
2020-03-28 14:09:55 +01:00
}
2020-09-05 10:57:13 +08:00
// Repository implements PayloadConvertor Repository method
2022-11-03 19:23:20 +01:00
func ( m * MatrixPayload ) Repository ( p * api . RepositoryPayload ) ( api . Payloader , error ) {
2020-03-28 14:09:55 +01:00
senderLink := MatrixLinkFormatter ( setting . AppURL + p . Sender . UserName , p . Sender . UserName )
repoLink := MatrixLinkFormatter ( p . Repository . HTMLURL , p . Repository . FullName )
var text string
switch p . Action {
case api . HookRepoCreated :
text = fmt . Sprintf ( "[%s] Repository created by %s" , repoLink , senderLink )
case api . HookRepoDeleted :
text = fmt . Sprintf ( "[%s] Repository deleted by %s" , repoLink , senderLink )
}
2022-11-03 19:23:20 +01:00
return getMatrixPayload ( text , nil , m . MsgType ) , nil
2020-03-28 14:09:55 +01:00
}
2022-11-03 19:23:20 +01:00
// GetMatrixPayload converts a Matrix webhook into a MatrixPayload
2021-11-10 13:13:16 +08:00
func GetMatrixPayload ( p api . Payloader , event webhook_model . HookEventType , meta string ) ( api . Payloader , error ) {
2022-11-03 19:23:20 +01:00
s := new ( MatrixPayload )
2020-03-28 14:09:55 +01:00
matrix := & MatrixMeta { }
if err := json . Unmarshal ( [ ] byte ( meta ) , & matrix ) ; err != nil {
return s , errors . New ( "GetMatrixPayload meta json:" + err . Error ( ) )
}
2020-09-05 10:57:13 +08:00
s . MsgType = messageTypeText [ matrix . MessageType ]
2020-03-28 14:09:55 +01:00
2020-09-05 10:57:13 +08:00
return convertPayloader ( s , p , event )
2020-03-28 14:09:55 +01:00
}
2022-11-03 19:23:20 +01:00
func getMatrixPayload ( text string , commits [ ] * api . PayloadCommit , msgType string ) * MatrixPayload {
p := MatrixPayload { }
2020-03-28 14:09:55 +01:00
p . FormattedBody = text
p . Body = getMessageBody ( text )
p . Format = "org.matrix.custom.html"
2020-09-05 10:57:13 +08:00
p . MsgType = msgType
2020-03-28 14:09:55 +01:00
p . Commits = commits
return & p
}
var urlRegex = regexp . MustCompile ( ` <a [^>]*?href="([^">]*?)">(.*?)</a> ` )
func getMessageBody ( htmlText string ) string {
htmlText = urlRegex . ReplaceAllString ( htmlText , "[$2]($1)" )
htmlText = strings . ReplaceAll ( htmlText , "<br>" , "\n" )
return htmlText
}
2022-11-03 19:23:20 +01:00
// getMatrixTxnID computes the transaction ID to ensure idempotency
func getMatrixTxnID ( payload [ ] byte ) ( string , error ) {
2020-03-28 14:09:55 +01:00
if len ( payload ) >= matrixPayloadSizeLimit {
2022-11-03 19:23:20 +01:00
return "" , fmt . Errorf ( "getMatrixTxnID: payload size %d > %d" , len ( payload ) , matrixPayloadSizeLimit )
2020-03-28 14:09:55 +01:00
}
2020-07-31 00:04:19 +02:00
h := sha1 . New ( )
_ , err := h . Write ( payload )
if err != nil {
return "" , err
}
2022-11-03 19:23:20 +01:00
return hex . EncodeToString ( h . Sum ( nil ) ) , nil
2020-07-31 00:04:19 +02:00
}