bench-forgejo/models/webhook.go

317 lines
7.8 KiB
Go
Raw Normal View History

2014-05-06 06:22:25 +05:30
// 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"
2014-05-06 07:06:08 +05:30
"errors"
2014-08-24 18:29:47 +05:30
"io/ioutil"
2014-06-08 14:15:34 +05:30
"time"
2014-05-06 06:22:25 +05:30
2014-06-08 14:15:34 +05:30
"github.com/gogits/gogs/modules/httplib"
2014-05-06 06:22:25 +05:30
"github.com/gogits/gogs/modules/log"
2014-06-08 14:15:34 +05:30
"github.com/gogits/gogs/modules/setting"
2014-08-10 04:10:10 +05:30
"github.com/gogits/gogs/modules/uuid"
2014-05-06 06:22:25 +05:30
)
2014-05-06 07:06:08 +05:30
var (
ErrWebhookNotExist = errors.New("Webhook does not exist")
)
2014-06-08 14:15:34 +05:30
type HookContentType int
2014-05-06 06:22:25 +05:30
const (
2014-06-08 14:15:34 +05:30
JSON HookContentType = iota + 1
FORM
2014-05-06 06:22:25 +05:30
)
2014-06-08 14:15:34 +05:30
// HookEvent represents events that will delivery hook.
2014-05-06 06:22:25 +05:30
type HookEvent struct {
PushOnly bool `json:"push_only"`
}
2014-06-08 14:15:34 +05:30
// Webhook represents a web hook object.
2014-05-06 06:22:25 +05:30
type Webhook struct {
2014-08-24 18:29:47 +05:30
Id int64
RepoId int64
Url string `xorm:"TEXT"`
ContentType HookContentType
Secret string `xorm:"TEXT"`
Events string `xorm:"TEXT"`
*HookEvent `xorm:"-"`
IsSsl bool
IsActive bool
HookTaskType HookTaskType
Meta string `xorm:"TEXT"` // store hook-specific attributes
2014-09-04 16:47:00 +05:30
OrgId int64
2014-05-06 06:22:25 +05:30
}
2014-06-08 14:24:52 +05:30
// GetEvent handles conversion from Events to HookEvent.
2014-05-06 07:06:08 +05:30
func (w *Webhook) GetEvent() {
w.HookEvent = &HookEvent{}
if err := json.Unmarshal([]byte(w.Events), w.HookEvent); err != nil {
2014-07-26 09:54:27 +05:30
log.Error(4, "webhook.GetEvent(%d): %v", w.Id, err)
2014-05-06 06:22:25 +05:30
}
}
2014-08-24 18:29:47 +05:30
func (w *Webhook) GetSlackHook() *Slack {
s := &Slack{}
if err := json.Unmarshal([]byte(w.Meta), s); err != nil {
log.Error(4, "webhook.GetSlackHook(%d): %v", w.Id, err)
}
return s
}
2014-06-08 14:24:52 +05:30
// UpdateEvent handles conversion from HookEvent to Events.
2014-06-08 14:15:34 +05:30
func (w *Webhook) UpdateEvent() error {
2014-05-06 07:06:08 +05:30
data, err := json.Marshal(w.HookEvent)
2014-05-06 06:22:25 +05:30
w.Events = string(data)
return err
}
2014-06-08 14:24:52 +05:30
// HasPushEvent returns true if hook enbaled push event.
2014-05-06 21:20:31 +05:30
func (w *Webhook) HasPushEvent() bool {
if w.PushOnly {
return true
}
return false
}
2014-06-08 14:15:34 +05:30
// CreateWebhook creates a new web hook.
2014-05-06 06:22:25 +05:30
func CreateWebhook(w *Webhook) error {
2014-06-21 10:21:41 +05:30
_, err := x.Insert(w)
2014-05-06 06:22:25 +05:30
return err
}
2014-05-06 07:06:08 +05:30
// GetWebhookById returns webhook by given ID.
func GetWebhookById(hookId int64) (*Webhook, error) {
w := &Webhook{Id: hookId}
2014-06-21 10:21:41 +05:30
has, err := x.Get(w)
2014-05-06 07:06:08 +05:30
if err != nil {
return nil, err
} else if !has {
return nil, ErrWebhookNotExist
}
return w, nil
}
2014-05-06 21:20:31 +05:30
// GetActiveWebhooksByRepoId returns all active webhooks of repository.
func GetActiveWebhooksByRepoId(repoId int64) (ws []*Webhook, err error) {
2014-09-09 19:47:35 +05:30
err = x.Where("repo_id=?", repoId).And("is_active=?", true).Find(&ws)
2014-05-06 21:20:31 +05:30
return ws, err
}
2014-05-06 06:22:25 +05:30
// GetWebhooksByRepoId returns all webhooks of repository.
func GetWebhooksByRepoId(repoId int64) (ws []*Webhook, err error) {
2014-06-21 10:21:41 +05:30
err = x.Find(&ws, &Webhook{RepoId: repoId})
2014-05-06 06:22:25 +05:30
return ws, err
}
2014-05-06 07:06:08 +05:30
2014-06-08 14:15:34 +05:30
// UpdateWebhook updates information of webhook.
func UpdateWebhook(w *Webhook) error {
_, err := x.Id(w.Id).AllCols().Update(w)
2014-06-08 14:15:34 +05:30
return err
}
2014-05-06 07:06:08 +05:30
// DeleteWebhook deletes webhook of repository.
func DeleteWebhook(hookId int64) error {
2014-06-21 10:21:41 +05:30
_, err := x.Delete(&Webhook{Id: hookId})
2014-05-06 07:06:08 +05:30
return err
}
2014-06-08 14:15:34 +05:30
2014-09-04 16:47:00 +05:30
// GetWebhooksByOrgId returns all webhooks for an organization.
func GetWebhooksByOrgId(orgId int64) (ws []*Webhook, err error) {
err = x.Find(&ws, &Webhook{OrgId: orgId})
return ws, err
}
// GetActiveWebhooksByOrgId returns all active webhooks for an organization.
func GetActiveWebhooksByOrgId(orgId int64) (ws []*Webhook, err error) {
2014-09-09 19:47:35 +05:30
err = x.Where("org_id=?", orgId).And("is_active=?", true).Find(&ws)
2014-09-04 16:47:00 +05:30
return ws, err
}
2014-06-08 14:15:34 +05:30
// ___ ___ __ ___________ __
// / | \ ____ ____ | | _\__ ___/____ _____| | __
// / ~ \/ _ \ / _ \| |/ / | | \__ \ / ___/ |/ /
// \ Y ( <_> | <_> ) < | | / __ \_\___ \| <
// \___|_ / \____/ \____/|__|_ \ |____| (____ /____ >__|_ \
// \/ \/ \/ \/ \/
type HookTaskType int
const (
2014-08-24 18:29:47 +05:30
GOGS HookTaskType = iota + 1
SLACK
2014-06-08 14:15:34 +05:30
)
2014-08-10 04:10:10 +05:30
type HookEventType string
const (
PUSH HookEventType = "push"
)
2014-06-08 14:15:34 +05:30
type PayloadAuthor struct {
Name string `json:"name"`
Email string `json:"email"`
UserName string `json:"username"`
2014-06-08 14:15:34 +05:30
}
type PayloadCommit struct {
Id string `json:"id"`
Message string `json:"message"`
Url string `json:"url"`
Author *PayloadAuthor `json:"author"`
}
type PayloadRepo struct {
Id int64 `json:"id"`
Name string `json:"name"`
Url string `json:"url"`
Description string `json:"description"`
Website string `json:"website"`
Watchers int `json:"watchers"`
Owner *PayloadAuthor `json:"owner"`
2014-06-08 14:15:34 +05:30
Private bool `json:"private"`
}
2014-08-24 18:29:47 +05:30
type BasePayload interface {
GetJSONPayload() ([]byte, error)
}
2014-06-08 14:24:52 +05:30
// Payload represents a payload information of hook.
2014-06-08 14:15:34 +05:30
type Payload struct {
2014-08-26 17:50:18 +05:30
Secret string `json:"secret"`
Ref string `json:"ref"`
Commits []*PayloadCommit `json:"commits"`
Repo *PayloadRepo `json:"repository"`
Pusher *PayloadAuthor `json:"pusher"`
Before string `json:"before"`
After string `json:"after"`
CompareUrl string `json:"compare_url"`
2014-06-08 14:15:34 +05:30
}
2014-08-24 18:29:47 +05:30
func (p Payload) GetJSONPayload() ([]byte, error) {
data, err := json.Marshal(p)
if err != nil {
return []byte{}, err
}
return data, nil
}
2014-06-08 14:24:52 +05:30
// HookTask represents a hook task.
2014-06-08 14:15:34 +05:30
type HookTask struct {
Id int64
2014-08-10 04:10:10 +05:30
Uuid string
2014-06-08 14:24:52 +05:30
Type HookTaskType
2014-06-08 14:15:34 +05:30
Url string
2014-08-24 18:29:47 +05:30
BasePayload `xorm:"-"`
2014-06-08 14:15:34 +05:30
PayloadContent string `xorm:"TEXT"`
ContentType HookContentType
2014-08-10 04:10:10 +05:30
EventType HookEventType
2014-06-08 14:15:34 +05:30
IsSsl bool
2014-08-24 18:29:47 +05:30
IsDelivered bool
2014-08-10 04:10:10 +05:30
IsSucceed bool
2014-06-08 14:15:34 +05:30
}
// CreateHookTask creates a new hook task,
// it handles conversion from Payload to PayloadContent.
func CreateHookTask(t *HookTask) error {
2014-08-24 18:29:47 +05:30
data, err := t.BasePayload.GetJSONPayload()
2014-06-08 14:15:34 +05:30
if err != nil {
return err
}
2014-08-10 04:10:10 +05:30
t.Uuid = uuid.NewV4().String()
2014-06-08 14:15:34 +05:30
t.PayloadContent = string(data)
2014-06-21 10:21:41 +05:30
_, err = x.Insert(t)
2014-06-08 14:15:34 +05:30
return err
}
// UpdateHookTask updates information of hook task.
func UpdateHookTask(t *HookTask) error {
2014-09-09 18:07:34 +05:30
_, err := x.Id(t.Id).AllCols().Update(t)
2014-06-08 14:15:34 +05:30
return err
}
var (
// Prevent duplicate deliveries.
// This happens with massive hook tasks cannot finish delivering
// before next shooting starts.
isShooting = false
)
2014-06-08 14:15:34 +05:30
// DeliverHooks checks and delivers undelivered hooks.
// FIXME: maybe can use goroutine to shoot a number of them at same time?
2014-06-08 14:15:34 +05:30
func DeliverHooks() {
if isShooting {
return
}
isShooting = true
defer func() { isShooting = false }()
tasks := make([]*HookTask, 0, 10)
2014-06-08 14:15:34 +05:30
timeout := time.Duration(setting.WebhookDeliverTimeout) * time.Second
2014-08-24 18:29:47 +05:30
x.Where("is_delivered=?", false).Iterate(new(HookTask),
2014-06-08 14:15:34 +05:30
func(idx int, bean interface{}) error {
t := bean.(*HookTask)
2014-08-10 04:10:10 +05:30
req := httplib.Post(t.Url).SetTimeout(timeout, timeout).
Header("X-Gogs-Delivery", t.Uuid).
Header("X-Gogs-Event", string(t.EventType))
switch t.ContentType {
case JSON:
req = req.Header("Content-Type", "application/json").Body(t.PayloadContent)
case FORM:
req.Param("payload", t.PayloadContent)
2014-06-08 14:15:34 +05:30
}
2014-08-24 18:29:47 +05:30
t.IsDelivered = true
2014-08-10 04:10:10 +05:30
// FIXME: record response.
2014-08-24 18:29:47 +05:30
switch t.Type {
case GOGS:
{
if _, err := req.Response(); err != nil {
log.Error(4, "Delivery: %v", err)
} else {
t.IsSucceed = true
}
}
case SLACK:
{
if res, err := req.Response(); err != nil {
log.Error(4, "Delivery: %v", err)
} else {
defer res.Body.Close()
contents, err := ioutil.ReadAll(res.Body)
if err != nil {
log.Error(4, "%s", err)
} else {
if string(contents) != "ok" {
log.Error(4, "slack failed with: %s", string(contents))
} else {
t.IsSucceed = true
}
}
}
}
2014-08-10 04:10:10 +05:30
}
tasks = append(tasks, t)
2014-06-08 14:15:34 +05:30
if t.IsSucceed {
log.Trace("Hook delivered(%s): %s", t.Uuid, t.PayloadContent)
}
2014-06-08 14:15:34 +05:30
return nil
})
// Update hook task status.
for _, t := range tasks {
if err := UpdateHookTask(t); err != nil {
log.Error(4, "UpdateHookTask(%d): %v", t.Id, err)
}
}
2014-06-08 14:15:34 +05:30
}