2014-05-05 20:52:25 -04:00
// Copyright 2014 The Gogs Authors. All rights reserved.
2017-05-29 09:17:15 +02:00
// Copyright 2017 The Gitea Authors. All rights reserved.
2014-05-05 20:52:25 -04:00
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package models
import (
2021-01-26 15:02:42 -06:00
"context"
2015-08-27 23:06:14 +08:00
"fmt"
2020-12-12 00:04:04 +08:00
"strings"
2014-06-08 04:45:34 -04:00
"time"
2014-05-05 20:52:25 -04:00
2021-09-19 19:49:59 +08:00
"code.gitea.io/gitea/models/db"
2021-07-25 00:03:58 +08:00
"code.gitea.io/gitea/modules/json"
2016-11-10 17:24:48 +01:00
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/setting"
2019-05-11 18:21:34 +08:00
api "code.gitea.io/gitea/modules/structs"
2019-08-15 22:46:21 +08:00
"code.gitea.io/gitea/modules/timeutil"
2021-08-12 14:43:08 +02:00
"code.gitea.io/gitea/modules/util"
2019-08-15 22:46:21 +08:00
2020-06-18 17:18:44 +08:00
gouuid "github.com/google/uuid"
2021-08-12 14:43:08 +02:00
"xorm.io/builder"
2014-05-05 20:52:25 -04:00
)
2016-11-22 07:42:52 +01:00
// HookContentType is the content type of a web hook
2014-06-08 04:45:34 -04:00
type HookContentType int
2014-05-05 20:52:25 -04:00
const (
2016-11-22 07:42:52 +01:00
// ContentTypeJSON is a JSON payload for web hooks
2016-11-07 21:58:22 +01:00
ContentTypeJSON HookContentType = iota + 1
2016-11-22 07:42:52 +01:00
// ContentTypeForm is an url-encoded form payload for web hook
2016-11-07 17:53:22 +01:00
ContentTypeForm
2014-05-05 20:52:25 -04:00
)
2014-11-13 12:57:00 -05:00
var hookContentTypes = map [ string ] HookContentType {
2016-11-07 21:58:22 +01:00
"json" : ContentTypeJSON ,
2016-11-07 17:53:22 +01:00
"form" : ContentTypeForm ,
2014-11-13 12:57:00 -05:00
}
// ToHookContentType returns HookContentType by given name.
func ToHookContentType ( name string ) HookContentType {
return hookContentTypes [ name ]
}
2021-01-26 15:02:42 -06:00
// HookTaskCleanupType is the type of cleanup to perform on hook_task
type HookTaskCleanupType int
const (
// OlderThan hook_task rows will be cleaned up by the age of the row
OlderThan HookTaskCleanupType = iota
// PerWebhook hook_task rows will be cleaned up by leaving the most recent deliveries for each webhook
PerWebhook
)
var hookTaskCleanupTypes = map [ string ] HookTaskCleanupType {
"OlderThan" : OlderThan ,
"PerWebhook" : PerWebhook ,
}
// ToHookTaskCleanupType returns HookTaskCleanupType by given name.
func ToHookTaskCleanupType ( name string ) HookTaskCleanupType {
return hookTaskCleanupTypes [ name ]
}
2016-11-22 07:42:52 +01:00
// Name returns the name of a given web hook's content type
2014-11-13 02:32:18 -05:00
func ( t HookContentType ) Name ( ) string {
switch t {
2016-11-07 21:58:22 +01:00
case ContentTypeJSON :
2014-11-13 02:32:18 -05:00
return "json"
2016-11-07 17:53:22 +01:00
case ContentTypeForm :
2014-11-13 02:32:18 -05:00
return "form"
}
return ""
}
2014-11-13 12:57:00 -05:00
// IsValidHookContentType returns true if given name is a valid hook content type.
func IsValidHookContentType ( name string ) bool {
_ , ok := hookContentTypes [ name ]
return ok
}
2016-11-22 07:42:52 +01:00
// HookEvents is a set of web hook events
2015-08-28 23:36:13 +08:00
type HookEvents struct {
2020-03-05 23:10:48 -06:00
Create bool ` json:"create" `
Delete bool ` json:"delete" `
Fork bool ` json:"fork" `
Issues bool ` json:"issues" `
IssueAssign bool ` json:"issue_assign" `
IssueLabel bool ` json:"issue_label" `
IssueMilestone bool ` json:"issue_milestone" `
IssueComment bool ` json:"issue_comment" `
Push bool ` json:"push" `
PullRequest bool ` json:"pull_request" `
PullRequestAssign bool ` json:"pull_request_assign" `
PullRequestLabel bool ` json:"pull_request_label" `
PullRequestMilestone bool ` json:"pull_request_milestone" `
PullRequestComment bool ` json:"pull_request_comment" `
PullRequestReview bool ` json:"pull_request_review" `
PullRequestSync bool ` json:"pull_request_sync" `
Repository bool ` json:"repository" `
Release bool ` json:"release" `
2015-08-28 23:36:13 +08:00
}
2014-06-08 04:45:34 -04:00
// HookEvent represents events that will delivery hook.
2014-05-05 20:52:25 -04:00
type HookEvent struct {
2019-09-09 08:48:21 +03:00
PushOnly bool ` json:"push_only" `
SendEverything bool ` json:"send_everything" `
ChooseEvents bool ` json:"choose_events" `
BranchFilter string ` json:"branch_filter" `
2015-08-28 23:36:13 +08:00
HookEvents ` json:"events" `
2014-05-05 20:52:25 -04:00
}
2021-06-27 21:21:09 +02:00
// HookType is the type of a webhook
type HookType = string
// Types of webhooks
const (
2021-07-23 12:41:27 +08:00
GITEA HookType = "gitea"
GOGS HookType = "gogs"
SLACK HookType = "slack"
DISCORD HookType = "discord"
DINGTALK HookType = "dingtalk"
TELEGRAM HookType = "telegram"
MSTEAMS HookType = "msteams"
FEISHU HookType = "feishu"
MATRIX HookType = "matrix"
WECHATWORK HookType = "wechatwork"
2021-06-27 21:21:09 +02:00
)
2016-11-22 07:42:52 +01:00
// HookStatus is the status of a web hook
2015-08-26 21:45:51 +08:00
type HookStatus int
2016-11-22 07:42:52 +01:00
// Possible statuses of a web hook
2015-08-26 21:45:51 +08:00
const (
2016-11-07 17:08:21 +01:00
HookStatusNone = iota
2016-11-07 16:37:32 +01:00
HookStatusSucceed
HookStatusFail
2015-08-26 21:45:51 +08:00
)
2014-06-08 04:45:34 -04:00
// Webhook represents a web hook object.
2014-05-05 20:52:25 -04:00
type Webhook struct {
2020-03-08 22:08:05 +00:00
ID int64 ` xorm:"pk autoincr" `
RepoID int64 ` xorm:"INDEX" ` // An ID of 0 indicates either a default or system webhook
OrgID int64 ` xorm:"INDEX" `
IsSystemWebhook bool
URL string ` xorm:"url TEXT" `
HTTPMethod string ` xorm:"http_method" `
ContentType HookContentType
Secret string ` xorm:"TEXT" `
Events string ` xorm:"TEXT" `
* HookEvent ` xorm:"-" `
2021-06-27 21:21:09 +02:00
IsActive bool ` xorm:"INDEX" `
Type HookType ` xorm:"VARCHAR(16) 'type'" `
Meta string ` xorm:"TEXT" ` // store hook-specific attributes
LastStatus HookStatus // Last delivery status
2016-03-09 19:53:30 -05:00
2019-08-15 22:46:21 +08:00
CreatedUnix timeutil . TimeStamp ` xorm:"INDEX created" `
UpdatedUnix timeutil . TimeStamp ` xorm:"INDEX updated" `
2014-05-05 20:52:25 -04:00
}
2021-09-19 19:49:59 +08:00
func init ( ) {
db . RegisterModel ( new ( Webhook ) )
db . RegisterModel ( new ( HookTask ) )
}
2017-10-02 00:52:35 +08:00
// AfterLoad updates the webhook object upon setting a column
func ( w * Webhook ) AfterLoad ( ) {
w . HookEvent = & HookEvent { }
if err := json . Unmarshal ( [ ] byte ( w . Events ) , w . HookEvent ) ; err != nil {
2019-04-02 08:48:31 +01:00
log . Error ( "Unmarshal[%d]: %v" , w . ID , err )
2014-05-05 20:52:25 -04:00
}
}
2015-08-27 23:06:14 +08:00
// History returns history of webhook by given conditions.
func ( w * Webhook ) History ( page int ) ( [ ] * HookTask , error ) {
return HookTasks ( w . ID , page )
}
2014-06-08 04:54:52 -04:00
// UpdateEvent handles conversion from HookEvent to Events.
2014-06-08 04:45:34 -04:00
func ( w * Webhook ) UpdateEvent ( ) error {
2014-05-05 21:36:08 -04:00
data , err := json . Marshal ( w . HookEvent )
2014-05-05 20:52:25 -04:00
w . Events = string ( data )
return err
}
2015-08-28 23:36:13 +08:00
// HasCreateEvent returns true if hook enabled create event.
func ( w * Webhook ) HasCreateEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . Create )
}
2018-05-16 22:01:55 +08:00
// HasDeleteEvent returns true if hook enabled delete event.
func ( w * Webhook ) HasDeleteEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . Delete )
}
// HasForkEvent returns true if hook enabled fork event.
func ( w * Webhook ) HasForkEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . Fork )
}
// HasIssuesEvent returns true if hook enabled issues event.
func ( w * Webhook ) HasIssuesEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . Issues )
}
2020-03-05 23:10:48 -06:00
// HasIssuesAssignEvent returns true if hook enabled issues assign event.
func ( w * Webhook ) HasIssuesAssignEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . IssueAssign )
}
// HasIssuesLabelEvent returns true if hook enabled issues label event.
func ( w * Webhook ) HasIssuesLabelEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . IssueLabel )
}
// HasIssuesMilestoneEvent returns true if hook enabled issues milestone event.
func ( w * Webhook ) HasIssuesMilestoneEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . IssueMilestone )
}
2018-05-16 22:01:55 +08:00
// HasIssueCommentEvent returns true if hook enabled issue_comment event.
func ( w * Webhook ) HasIssueCommentEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . IssueComment )
}
2014-12-06 20:22:48 -05:00
// HasPushEvent returns true if hook enabled push event.
2014-05-06 11:50:31 -04:00
func ( w * Webhook ) HasPushEvent ( ) bool {
2015-08-28 23:36:13 +08:00
return w . PushOnly || w . SendEverything ||
( w . ChooseEvents && w . HookEvents . Push )
2014-05-06 11:50:31 -04:00
}
2016-08-14 03:32:24 -07:00
// HasPullRequestEvent returns true if hook enabled pull request event.
func ( w * Webhook ) HasPullRequestEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . PullRequest )
}
2020-03-05 23:10:48 -06:00
// HasPullRequestAssignEvent returns true if hook enabled pull request assign event.
func ( w * Webhook ) HasPullRequestAssignEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . PullRequestAssign )
}
// HasPullRequestLabelEvent returns true if hook enabled pull request label event.
func ( w * Webhook ) HasPullRequestLabelEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . PullRequestLabel )
}
// HasPullRequestMilestoneEvent returns true if hook enabled pull request milestone event.
func ( w * Webhook ) HasPullRequestMilestoneEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . PullRequestMilestone )
}
// HasPullRequestCommentEvent returns true if hook enabled pull_request_comment event.
func ( w * Webhook ) HasPullRequestCommentEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . PullRequestComment )
}
// HasPullRequestApprovedEvent returns true if hook enabled pull request review event.
func ( w * Webhook ) HasPullRequestApprovedEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . PullRequestReview )
}
// HasPullRequestRejectedEvent returns true if hook enabled pull request review event.
func ( w * Webhook ) HasPullRequestRejectedEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . PullRequestReview )
}
// HasPullRequestReviewCommentEvent returns true if hook enabled pull request review event.
func ( w * Webhook ) HasPullRequestReviewCommentEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . PullRequestReview )
}
// HasPullRequestSyncEvent returns true if hook enabled pull request sync event.
func ( w * Webhook ) HasPullRequestSyncEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . PullRequestSync )
}
2018-05-16 22:01:55 +08:00
// HasReleaseEvent returns if hook enabled release event.
func ( w * Webhook ) HasReleaseEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . Release )
}
2017-09-03 01:20:24 -07:00
// HasRepositoryEvent returns if hook enabled repository event.
func ( w * Webhook ) HasRepositoryEvent ( ) bool {
return w . SendEverything ||
( w . ChooseEvents && w . HookEvents . Repository )
}
2019-11-02 06:51:22 +08:00
// EventCheckers returns event checkers
func ( w * Webhook ) EventCheckers ( ) [ ] struct {
Has func ( ) bool
Type HookEventType
2018-05-16 22:01:55 +08:00
} {
return [ ] struct {
2019-11-02 06:51:22 +08:00
Has func ( ) bool
Type HookEventType
2018-05-16 22:01:55 +08:00
} {
{ w . HasCreateEvent , HookEventCreate } ,
{ w . HasDeleteEvent , HookEventDelete } ,
{ w . HasForkEvent , HookEventFork } ,
{ w . HasPushEvent , HookEventPush } ,
{ w . HasIssuesEvent , HookEventIssues } ,
2020-03-05 23:10:48 -06:00
{ w . HasIssuesAssignEvent , HookEventIssueAssign } ,
{ w . HasIssuesLabelEvent , HookEventIssueLabel } ,
{ w . HasIssuesMilestoneEvent , HookEventIssueMilestone } ,
2018-05-16 22:01:55 +08:00
{ w . HasIssueCommentEvent , HookEventIssueComment } ,
{ w . HasPullRequestEvent , HookEventPullRequest } ,
2020-03-05 23:10:48 -06:00
{ w . HasPullRequestAssignEvent , HookEventPullRequestAssign } ,
{ w . HasPullRequestLabelEvent , HookEventPullRequestLabel } ,
{ w . HasPullRequestMilestoneEvent , HookEventPullRequestMilestone } ,
{ w . HasPullRequestCommentEvent , HookEventPullRequestComment } ,
{ w . HasPullRequestApprovedEvent , HookEventPullRequestReviewApproved } ,
{ w . HasPullRequestRejectedEvent , HookEventPullRequestReviewRejected } ,
{ w . HasPullRequestCommentEvent , HookEventPullRequestReviewComment } ,
{ w . HasPullRequestSyncEvent , HookEventPullRequestSync } ,
2018-05-16 22:01:55 +08:00
{ w . HasRepositoryEvent , HookEventRepository } ,
{ w . HasReleaseEvent , HookEventRelease } ,
}
}
2016-11-22 07:42:52 +01:00
// EventsArray returns an array of hook events
2015-08-29 11:49:59 +08:00
func ( w * Webhook ) EventsArray ( ) [ ] string {
2018-05-16 22:01:55 +08:00
events := make ( [ ] string , 0 , 7 )
2019-11-02 06:51:22 +08:00
for _ , c := range w . EventCheckers ( ) {
if c . Has ( ) {
events = append ( events , string ( c . Type ) )
2018-05-16 22:01:55 +08:00
}
2016-08-24 23:44:58 -04:00
}
2015-08-29 11:49:59 +08:00
return events
}
2014-06-08 04:45:34 -04:00
// CreateWebhook creates a new web hook.
2014-05-05 20:52:25 -04:00
func CreateWebhook ( w * Webhook ) error {
2021-09-23 16:45:36 +01:00
return createWebhook ( db . GetEngine ( db . DefaultContext ) , w )
2019-03-18 22:33:20 -04:00
}
2021-09-19 19:49:59 +08:00
func createWebhook ( e db . Engine , w * Webhook ) error {
2020-12-12 00:04:04 +08:00
w . Type = strings . TrimSpace ( w . Type )
2019-03-18 22:33:20 -04:00
_ , err := e . Insert ( w )
2014-05-05 20:52:25 -04:00
return err
}
2016-07-17 08:33:59 +08:00
// getWebhook uses argument bean as query condition,
// ID must be specified and do not assign unnecessary fields.
func getWebhook ( bean * Webhook ) ( * Webhook , error ) {
2021-09-23 16:45:36 +01:00
has , err := db . GetEngine ( db . DefaultContext ) . Get ( bean )
2014-05-05 21:36:08 -04:00
if err != nil {
return nil , err
} else if ! has {
2016-07-17 08:33:59 +08:00
return nil , ErrWebhookNotExist { bean . ID }
2014-05-05 21:36:08 -04:00
}
2016-07-17 08:33:59 +08:00
return bean , nil
}
2017-08-30 13:36:52 +08:00
// GetWebhookByID returns webhook of repository by given ID.
func GetWebhookByID ( id int64 ) ( * Webhook , error ) {
return getWebhook ( & Webhook {
ID : id ,
} )
}
2016-07-17 08:33:59 +08:00
// GetWebhookByRepoID returns webhook of repository by given ID.
func GetWebhookByRepoID ( repoID , id int64 ) ( * Webhook , error ) {
return getWebhook ( & Webhook {
ID : id ,
RepoID : repoID ,
} )
2014-05-05 21:36:08 -04:00
}
2016-07-16 01:02:55 +08:00
// GetWebhookByOrgID returns webhook of organization by given ID.
func GetWebhookByOrgID ( orgID , id int64 ) ( * Webhook , error ) {
2016-07-17 08:33:59 +08:00
return getWebhook ( & Webhook {
ID : id ,
OrgID : orgID ,
} )
2016-07-16 01:02:55 +08:00
}
2021-08-12 14:43:08 +02:00
// ListWebhookOptions are options to filter webhooks on ListWebhooksByOpts
type ListWebhookOptions struct {
2021-09-24 19:32:56 +08:00
db . ListOptions
2021-08-12 14:43:08 +02:00
RepoID int64
OrgID int64
IsActive util . OptionalBool
2017-09-03 01:20:24 -07:00
}
2021-08-12 14:43:08 +02:00
func ( opts * ListWebhookOptions ) toCond ( ) builder . Cond {
cond := builder . NewCond ( )
if opts . RepoID != 0 {
cond = cond . And ( builder . Eq { "webhook.repo_id" : opts . RepoID } )
}
if opts . OrgID != 0 {
cond = cond . And ( builder . Eq { "webhook.org_id" : opts . OrgID } )
}
if ! opts . IsActive . IsNone ( ) {
cond = cond . And ( builder . Eq { "webhook.is_active" : opts . IsActive . IsTrue ( ) } )
2020-01-24 19:00:29 +00:00
}
2021-08-12 14:43:08 +02:00
return cond
}
2020-01-24 19:00:29 +00:00
2021-09-19 19:49:59 +08:00
func listWebhooksByOpts ( e db . Engine , opts * ListWebhookOptions ) ( [ ] * Webhook , error ) {
2021-08-12 14:43:08 +02:00
sess := e . Where ( opts . toCond ( ) )
2020-01-24 19:00:29 +00:00
2021-08-12 14:43:08 +02:00
if opts . Page != 0 {
2021-09-24 19:32:56 +08:00
sess = db . SetSessionPagination ( sess , opts )
2021-08-12 14:43:08 +02:00
webhooks := make ( [ ] * Webhook , 0 , opts . PageSize )
err := sess . Find ( & webhooks )
return webhooks , err
}
2014-05-05 21:36:08 -04:00
2021-08-12 14:43:08 +02:00
webhooks := make ( [ ] * Webhook , 0 , 10 )
err := sess . Find ( & webhooks )
return webhooks , err
2017-09-03 01:20:24 -07:00
}
2021-08-12 14:43:08 +02:00
// ListWebhooksByOpts return webhooks based on options
func ListWebhooksByOpts ( opts * ListWebhookOptions ) ( [ ] * Webhook , error ) {
2021-09-23 16:45:36 +01:00
return listWebhooksByOpts ( db . GetEngine ( db . DefaultContext ) , opts )
2017-01-25 05:37:35 -05:00
}
2021-08-12 14:43:08 +02:00
// CountWebhooksByOpts count webhooks based on options and ignore pagination
func CountWebhooksByOpts ( opts * ListWebhookOptions ) ( int64 , error ) {
2021-09-23 16:45:36 +01:00
return db . GetEngine ( db . DefaultContext ) . Where ( opts . toCond ( ) ) . Count ( & Webhook { } )
2017-01-25 05:37:35 -05:00
}
2019-03-18 22:33:20 -04:00
// GetDefaultWebhooks returns all admin-default webhooks.
func GetDefaultWebhooks ( ) ( [ ] * Webhook , error ) {
2021-09-23 16:45:36 +01:00
return getDefaultWebhooks ( db . GetEngine ( db . DefaultContext ) )
2019-03-18 22:33:20 -04:00
}
2021-09-19 19:49:59 +08:00
func getDefaultWebhooks ( e db . Engine ) ( [ ] * Webhook , error ) {
2019-03-18 22:33:20 -04:00
webhooks := make ( [ ] * Webhook , 0 , 5 )
return webhooks , e .
2020-03-08 22:08:05 +00:00
Where ( "repo_id=? AND org_id=? AND is_system_webhook=?" , 0 , 0 , false ) .
Find ( & webhooks )
}
2021-01-15 01:24:03 +02:00
// GetSystemOrDefaultWebhook returns admin system or default webhook by given ID.
func GetSystemOrDefaultWebhook ( id int64 ) ( * Webhook , error ) {
2020-03-08 22:08:05 +00:00
webhook := & Webhook { ID : id }
2021-09-23 16:45:36 +01:00
has , err := db . GetEngine ( db . DefaultContext ) .
2021-01-15 01:24:03 +02:00
Where ( "repo_id=? AND org_id=?" , 0 , 0 ) .
2020-03-08 22:08:05 +00:00
Get ( webhook )
if err != nil {
return nil , err
} else if ! has {
return nil , ErrWebhookNotExist { id }
}
return webhook , nil
}
// GetSystemWebhooks returns all admin system webhooks.
func GetSystemWebhooks ( ) ( [ ] * Webhook , error ) {
2021-09-23 16:45:36 +01:00
return getSystemWebhooks ( db . GetEngine ( db . DefaultContext ) )
2020-03-08 22:08:05 +00:00
}
2021-09-19 19:49:59 +08:00
func getSystemWebhooks ( e db . Engine ) ( [ ] * Webhook , error ) {
2020-03-08 22:08:05 +00:00
webhooks := make ( [ ] * Webhook , 0 , 5 )
return webhooks , e .
Where ( "repo_id=? AND org_id=? AND is_system_webhook=?" , 0 , 0 , true ) .
2019-03-18 22:33:20 -04:00
Find ( & webhooks )
}
2014-06-08 04:45:34 -04:00
// UpdateWebhook updates information of webhook.
func UpdateWebhook ( w * Webhook ) error {
2021-09-23 16:45:36 +01:00
_ , err := db . GetEngine ( db . DefaultContext ) . ID ( w . ID ) . AllCols ( ) . Update ( w )
2014-06-08 04:45:34 -04:00
return err
}
2017-08-30 13:36:52 +08:00
// UpdateWebhookLastStatus updates last status of webhook.
func UpdateWebhookLastStatus ( w * Webhook ) error {
2021-09-23 16:45:36 +01:00
_ , err := db . GetEngine ( db . DefaultContext ) . ID ( w . ID ) . Cols ( "last_status" ) . Update ( w )
2017-08-30 13:36:52 +08:00
return err
}
2016-07-17 08:33:59 +08:00
// deleteWebhook uses argument bean as query condition,
// ID must be specified and do not assign unnecessary fields.
func deleteWebhook ( bean * Webhook ) ( err error ) {
2021-09-23 16:45:36 +01:00
sess := db . NewSession ( db . DefaultContext )
2017-06-21 03:57:05 +03:00
defer sess . Close ( )
2015-08-26 21:45:51 +08:00
if err = sess . Begin ( ) ; err != nil {
return err
}
2017-01-13 21:14:48 -05:00
if count , err := sess . Delete ( bean ) ; err != nil {
2015-08-26 21:45:51 +08:00
return err
2017-01-13 21:14:48 -05:00
} else if count == 0 {
return ErrWebhookNotExist { ID : bean . ID }
2016-07-17 08:33:59 +08:00
} else if _ , err = sess . Delete ( & HookTask { HookID : bean . ID } ) ; err != nil {
2015-08-26 21:45:51 +08:00
return err
}
return sess . Commit ( )
2014-05-05 21:36:08 -04:00
}
2014-06-08 04:45:34 -04:00
2016-07-17 08:33:59 +08:00
// DeleteWebhookByRepoID deletes webhook of repository by given ID.
2016-07-23 20:24:44 +08:00
func DeleteWebhookByRepoID ( repoID , id int64 ) error {
2016-07-17 08:33:59 +08:00
return deleteWebhook ( & Webhook {
ID : id ,
RepoID : repoID ,
} )
}
// DeleteWebhookByOrgID deletes webhook of organization by given ID.
2016-07-23 20:24:44 +08:00
func DeleteWebhookByOrgID ( orgID , id int64 ) error {
2016-07-17 08:33:59 +08:00
return deleteWebhook ( & Webhook {
ID : id ,
OrgID : orgID ,
} )
}
2020-03-08 22:08:05 +00:00
// DeleteDefaultSystemWebhook deletes an admin-configured default or system webhook (where Org and Repo ID both 0)
func DeleteDefaultSystemWebhook ( id int64 ) error {
2021-09-23 16:45:36 +01:00
sess := db . NewSession ( db . DefaultContext )
2019-03-18 22:33:20 -04:00
defer sess . Close ( )
if err := sess . Begin ( ) ; err != nil {
return err
}
count , err := sess .
Where ( "repo_id=? AND org_id=?" , 0 , 0 ) .
Delete ( & Webhook { ID : id } )
if err != nil {
return err
} else if count == 0 {
return ErrWebhookNotExist { ID : id }
}
if _ , err := sess . Delete ( & HookTask { HookID : id } ) ; err != nil {
return err
}
return sess . Commit ( )
}
// copyDefaultWebhooksToRepo creates copies of the default webhooks in a new repo
2021-09-19 19:49:59 +08:00
func copyDefaultWebhooksToRepo ( e db . Engine , repoID int64 ) error {
2019-03-18 22:33:20 -04:00
ws , err := getDefaultWebhooks ( e )
if err != nil {
return fmt . Errorf ( "GetDefaultWebhooks: %v" , err )
}
for _ , w := range ws {
w . ID = 0
w . RepoID = repoID
if err := createWebhook ( e , w ) ; err != nil {
return fmt . Errorf ( "CreateWebhook: %v" , err )
}
}
return nil
}
2014-06-08 04:45:34 -04:00
// ___ ___ __ ___________ __
// / | \ ____ ____ | | _\__ ___/____ _____| | __
// / ~ \/ _ \ / _ \| |/ / | | \__ \ / ___/ |/ /
// \ Y ( <_> | <_> ) < | | / __ \_\___ \| <
// \___|_ / \____/ \____/|__|_ \ |____| (____ /____ >__|_ \
// \/ \/ \/ \/ \/
2016-11-22 07:42:52 +01:00
// HookEventType is the type of an hook event
2014-08-09 15:40:10 -07:00
type HookEventType string
2016-11-22 07:42:52 +01:00
// Types of hook events
2014-08-09 15:40:10 -07:00
const (
2020-03-05 23:10:48 -06:00
HookEventCreate HookEventType = "create"
HookEventDelete HookEventType = "delete"
HookEventFork HookEventType = "fork"
HookEventPush HookEventType = "push"
HookEventIssues HookEventType = "issues"
HookEventIssueAssign HookEventType = "issue_assign"
HookEventIssueLabel HookEventType = "issue_label"
HookEventIssueMilestone HookEventType = "issue_milestone"
HookEventIssueComment HookEventType = "issue_comment"
HookEventPullRequest HookEventType = "pull_request"
HookEventPullRequestAssign HookEventType = "pull_request_assign"
HookEventPullRequestLabel HookEventType = "pull_request_label"
HookEventPullRequestMilestone HookEventType = "pull_request_milestone"
HookEventPullRequestComment HookEventType = "pull_request_comment"
HookEventPullRequestReviewApproved HookEventType = "pull_request_review_approved"
HookEventPullRequestReviewRejected HookEventType = "pull_request_review_rejected"
HookEventPullRequestReviewComment HookEventType = "pull_request_review_comment"
HookEventPullRequestSync HookEventType = "pull_request_sync"
HookEventRepository HookEventType = "repository"
HookEventRelease HookEventType = "release"
2014-08-09 15:40:10 -07:00
)
2020-03-05 23:10:48 -06:00
// Event returns the HookEventType as an event string
func ( h HookEventType ) Event ( ) string {
switch h {
case HookEventCreate :
return "create"
case HookEventDelete :
return "delete"
case HookEventFork :
return "fork"
case HookEventPush :
return "push"
case HookEventIssues , HookEventIssueAssign , HookEventIssueLabel , HookEventIssueMilestone :
return "issues"
case HookEventPullRequest , HookEventPullRequestAssign , HookEventPullRequestLabel , HookEventPullRequestMilestone ,
HookEventPullRequestSync :
return "pull_request"
case HookEventIssueComment , HookEventPullRequestComment :
return "issue_comment"
case HookEventPullRequestReviewApproved :
return "pull_request_approved"
case HookEventPullRequestReviewRejected :
return "pull_request_rejected"
case HookEventPullRequestReviewComment :
return "pull_request_comment"
case HookEventRepository :
return "repository"
case HookEventRelease :
return "release"
}
return ""
}
2015-08-27 23:06:14 +08:00
// HookRequest represents hook task request information.
type HookRequest struct {
2021-06-27 21:21:09 +02:00
URL string ` json:"url" `
HTTPMethod string ` json:"http_method" `
Headers map [ string ] string ` json:"headers" `
2015-08-27 23:06:14 +08:00
}
// HookResponse represents hook task response information.
type HookResponse struct {
Status int ` json:"status" `
Headers map [ string ] string ` json:"headers" `
Body string ` json:"body" `
}
2014-06-08 04:54:52 -04:00
// HookTask represents a hook task.
2014-06-08 04:45:34 -04:00
type HookTask struct {
2015-08-27 23:06:14 +08:00
ID int64 ` xorm:"pk autoincr" `
RepoID int64 ` xorm:"INDEX" `
HookID int64
UUID string
2015-08-28 23:36:13 +08:00
api . Payloader ` xorm:"-" `
2015-08-27 23:06:14 +08:00
PayloadContent string ` xorm:"TEXT" `
EventType HookEventType
IsDelivered bool
Delivered int64
DeliveredString string ` xorm:"-" `
// History info.
IsSucceed bool
RequestContent string ` xorm:"TEXT" `
RequestInfo * HookRequest ` xorm:"-" `
ResponseContent string ` xorm:"TEXT" `
ResponseInfo * HookResponse ` xorm:"-" `
}
2016-11-22 07:42:52 +01:00
// BeforeUpdate will be invoked by XORM before updating a record
// representing this object
2015-08-27 23:06:14 +08:00
func ( t * HookTask ) BeforeUpdate ( ) {
if t . RequestInfo != nil {
2016-11-22 07:42:52 +01:00
t . RequestContent = t . simpleMarshalJSON ( t . RequestInfo )
2015-08-27 23:06:14 +08:00
}
if t . ResponseInfo != nil {
2016-11-22 07:42:52 +01:00
t . ResponseContent = t . simpleMarshalJSON ( t . ResponseInfo )
2015-08-27 23:06:14 +08:00
}
}
2017-10-02 00:52:35 +08:00
// AfterLoad updates the webhook object upon setting a column
func ( t * HookTask ) AfterLoad ( ) {
t . DeliveredString = time . Unix ( 0 , t . Delivered ) . Format ( "2006-01-02 15:04:05 MST" )
if len ( t . RequestContent ) == 0 {
return
}
t . RequestInfo = & HookRequest { }
if err := json . Unmarshal ( [ ] byte ( t . RequestContent ) , t . RequestInfo ) ; err != nil {
2019-04-02 08:48:31 +01:00
log . Error ( "Unmarshal RequestContent[%d]: %v" , t . ID , err )
2018-05-23 14:12:02 +08:00
}
if len ( t . ResponseContent ) > 0 {
t . ResponseInfo = & HookResponse { }
if err := json . Unmarshal ( [ ] byte ( t . ResponseContent ) , t . ResponseInfo ) ; err != nil {
2019-04-02 08:48:31 +01:00
log . Error ( "Unmarshal ResponseContent[%d]: %v" , t . ID , err )
2018-05-23 14:12:02 +08:00
}
2015-08-27 23:06:14 +08:00
}
}
2016-11-22 07:42:52 +01:00
func ( t * HookTask ) simpleMarshalJSON ( v interface { } ) string {
2015-08-27 23:06:14 +08:00
p , err := json . Marshal ( v )
if err != nil {
2019-04-02 08:48:31 +01:00
log . Error ( "Marshal [%d]: %v" , t . ID , err )
2015-08-27 23:06:14 +08:00
}
return string ( p )
}
// HookTasks returns a list of hook tasks by given conditions.
func HookTasks ( hookID int64 , page int ) ( [ ] * HookTask , error ) {
tasks := make ( [ ] * HookTask , 0 , setting . Webhook . PagingNum )
2021-09-23 16:45:36 +01:00
return tasks , db . GetEngine ( db . DefaultContext ) .
2016-11-10 16:16:32 +01:00
Limit ( setting . Webhook . PagingNum , ( page - 1 ) * setting . Webhook . PagingNum ) .
Where ( "hook_id=?" , hookID ) .
Desc ( "id" ) .
Find ( & tasks )
2014-06-08 04:45:34 -04:00
}
// CreateHookTask creates a new hook task,
// it handles conversion from Payload to PayloadContent.
func CreateHookTask ( t * HookTask ) error {
2021-09-23 16:45:36 +01:00
return createHookTask ( db . GetEngine ( db . DefaultContext ) , t )
2017-09-03 01:20:24 -07:00
}
2021-09-19 19:49:59 +08:00
func createHookTask ( e db . Engine , t * HookTask ) error {
2015-08-28 23:36:13 +08:00
data , err := t . Payloader . JSONPayload ( )
2014-06-08 04:45:34 -04:00
if err != nil {
return err
}
2020-06-18 17:18:44 +08:00
t . UUID = gouuid . New ( ) . String ( )
2014-06-08 04:45:34 -04:00
t . PayloadContent = string ( data )
2017-09-03 01:20:24 -07:00
_ , err = e . Insert ( t )
2014-06-08 04:45:34 -04:00
return err
}
// UpdateHookTask updates information of hook task.
func UpdateHookTask ( t * HookTask ) error {
2021-09-23 16:45:36 +01:00
_ , err := db . GetEngine ( db . DefaultContext ) . ID ( t . ID ) . AllCols ( ) . Update ( t )
2014-06-08 04:45:34 -04:00
return err
}
2019-11-02 06:51:22 +08:00
// FindUndeliveredHookTasks represents find the undelivered hook tasks
func FindUndeliveredHookTasks ( ) ( [ ] * HookTask , error ) {
2015-07-25 21:32:04 +08:00
tasks := make ( [ ] * HookTask , 0 , 10 )
2021-09-23 16:45:36 +01:00
if err := db . GetEngine ( db . DefaultContext ) . Where ( "is_delivered=?" , false ) . Find ( & tasks ) ; err != nil {
2019-11-02 06:51:22 +08:00
return nil , err
2014-09-12 18:58:24 -04:00
}
2019-11-02 06:51:22 +08:00
return tasks , nil
2014-06-08 04:45:34 -04:00
}
2015-07-25 21:32:04 +08:00
2019-11-02 06:51:22 +08:00
// FindRepoUndeliveredHookTasks represents find the undelivered hook tasks of one repository
func FindRepoUndeliveredHookTasks ( repoID int64 ) ( [ ] * HookTask , error ) {
tasks := make ( [ ] * HookTask , 0 , 5 )
2021-09-23 16:45:36 +01:00
if err := db . GetEngine ( db . DefaultContext ) . Where ( "repo_id=? AND is_delivered=?" , repoID , false ) . Find ( & tasks ) ; err != nil {
2019-11-02 06:51:22 +08:00
return nil , err
2019-05-21 15:20:17 +08:00
}
2019-11-02 06:51:22 +08:00
return tasks , nil
2015-07-25 21:32:04 +08:00
}
2021-01-26 15:02:42 -06:00
// CleanupHookTaskTable deletes rows from hook_task as needed.
func CleanupHookTaskTable ( ctx context . Context , cleanupType HookTaskCleanupType , olderThan time . Duration , numberToKeep int ) error {
log . Trace ( "Doing: CleanupHookTaskTable" )
if cleanupType == OlderThan {
deleteOlderThan := time . Now ( ) . Add ( - olderThan ) . UnixNano ( )
2021-09-23 16:45:36 +01:00
deletes , err := db . GetEngine ( db . DefaultContext ) .
2021-01-26 15:02:42 -06:00
Where ( "is_delivered = ? and delivered < ?" , true , deleteOlderThan ) .
Delete ( new ( HookTask ) )
if err != nil {
return err
}
log . Trace ( "Deleted %d rows from hook_task" , deletes )
} else if cleanupType == PerWebhook {
hookIDs := make ( [ ] int64 , 0 , 10 )
2021-09-23 16:45:36 +01:00
err := db . GetEngine ( db . DefaultContext ) . Table ( "webhook" ) .
2021-01-26 15:02:42 -06:00
Where ( "id > 0" ) .
Cols ( "id" ) .
Find ( & hookIDs )
if err != nil {
return err
}
for _ , hookID := range hookIDs {
select {
case <- ctx . Done ( ) :
return ErrCancelledf ( "Before deleting hook_task records for hook id %d" , hookID )
default :
}
if err = deleteDeliveredHookTasksByWebhook ( hookID , numberToKeep ) ; err != nil {
return err
}
}
}
log . Trace ( "Finished: CleanupHookTaskTable" )
return nil
}
func deleteDeliveredHookTasksByWebhook ( hookID int64 , numberDeliveriesToKeep int ) error {
log . Trace ( "Deleting hook_task rows for webhook %d, keeping the most recent %d deliveries" , hookID , numberDeliveriesToKeep )
2021-03-15 02:52:12 +08:00
deliveryDates := make ( [ ] int64 , 0 , 10 )
2021-09-23 16:45:36 +01:00
err := db . GetEngine ( db . DefaultContext ) . Table ( "hook_task" ) .
2021-01-26 15:02:42 -06:00
Where ( "hook_task.hook_id = ? AND hook_task.is_delivered = ? AND hook_task.delivered is not null" , hookID , true ) .
Cols ( "hook_task.delivered" ) .
Join ( "INNER" , "webhook" , "hook_task.hook_id = webhook.id" ) .
OrderBy ( "hook_task.delivered desc" ) .
Limit ( 1 , int ( numberDeliveriesToKeep ) ) .
Find ( & deliveryDates )
if err != nil {
return err
}
if len ( deliveryDates ) > 0 {
2021-09-23 16:45:36 +01:00
deletes , err := db . GetEngine ( db . DefaultContext ) .
2021-01-26 15:02:42 -06:00
Where ( "hook_id = ? and is_delivered = ? and delivered <= ?" , hookID , true , deliveryDates [ 0 ] ) .
Delete ( new ( HookTask ) )
if err != nil {
return err
}
log . Trace ( "Deleted %d hook_task rows for webhook %d" , deletes , hookID )
} else {
log . Trace ( "No hook_task rows to delete for webhook %d" , hookID )
}
return nil
}