Files
casdoor/object/webhook_event.go

281 lines
7.6 KiB
Go

// Copyright 2026 The Casdoor Authors. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package object
import (
"fmt"
"github.com/casdoor/casdoor/util"
"github.com/xorm-io/core"
"github.com/xorm-io/xorm"
)
// WebhookEventStatus represents the delivery status of a webhook event
type WebhookEventStatus string
const (
WebhookEventStatusPending WebhookEventStatus = "pending"
WebhookEventStatusSuccess WebhookEventStatus = "success"
WebhookEventStatusFailed WebhookEventStatus = "failed"
WebhookEventStatusRetrying WebhookEventStatus = "retrying"
)
// WebhookEvent represents a webhook delivery event with retry and replay capability
type WebhookEvent struct {
Owner string `xorm:"varchar(100) notnull pk" json:"owner"`
Name string `xorm:"varchar(100) notnull pk" json:"name"`
CreatedTime string `xorm:"varchar(100)" json:"createdTime"`
UpdatedTime string `xorm:"varchar(100)" json:"updatedTime"`
WebhookName string `xorm:"varchar(200) index" json:"webhookName"`
Organization string `xorm:"varchar(100) index" json:"organization"`
EventType string `xorm:"varchar(100)" json:"eventType"`
Status WebhookEventStatus `xorm:"varchar(50) index" json:"status"`
// Payload stores the event data (Record)
Payload string `xorm:"mediumtext" json:"payload"`
// Extended user data if applicable
ExtendedUser string `xorm:"mediumtext" json:"extendedUser"`
// Delivery tracking
AttemptCount int `xorm:"int default 0" json:"attemptCount"`
MaxRetries int `xorm:"int default 3" json:"maxRetries"`
NextRetryTime string `xorm:"varchar(100)" json:"nextRetryTime"`
// Last delivery response
LastStatusCode int `xorm:"int" json:"lastStatusCode"`
LastResponse string `xorm:"mediumtext" json:"lastResponse"`
LastError string `xorm:"mediumtext" json:"lastError"`
}
func GetWebhookEvent(id string) (*WebhookEvent, error) {
owner, name, err := util.GetOwnerAndNameFromIdWithError(id)
if err != nil {
return nil, err
}
return getWebhookEvent(owner, name)
}
func getWebhookEvent(owner string, name string) (*WebhookEvent, error) {
if owner == "" || name == "" {
return nil, nil
}
event := WebhookEvent{Owner: owner, Name: name}
existed, err := ormer.Engine.Get(&event)
if err != nil {
return &event, err
}
if existed {
return &event, nil
}
return nil, nil
}
func getWebhookEventSession(owner, organization, webhookName string, status WebhookEventStatus, offset, limit int, sortField, sortOrder string) *xorm.Session {
session := ormer.Engine.Prepare()
if owner != "" {
session = session.Where("owner = ?", owner)
}
if organization != "" {
session = session.Where("organization = ?", organization)
}
if webhookName != "" {
session = session.Where("webhook_name = ?", webhookName)
}
if status != "" {
session = session.Where("status = ?", status)
}
if offset > 0 {
session = session.Limit(limit, offset)
} else if limit > 0 {
session = session.Limit(limit)
}
if sortField == "" || sortOrder == "" {
sortField = "created_time"
}
if sortOrder == "ascend" {
session = session.Asc(util.SnakeString(sortField))
} else {
session = session.Desc(util.SnakeString(sortField))
}
return session
}
func GetWebhookEvents(owner, organization, webhookName string, status WebhookEventStatus, offset, limit int, sortField, sortOrder string) ([]*WebhookEvent, error) {
events := []*WebhookEvent{}
session := getWebhookEventSession(owner, organization, webhookName, status, offset, limit, sortField, sortOrder)
err := session.Find(&events)
if err != nil {
return nil, err
}
return events, nil
}
func GetWebhookEventCount(owner, organization, webhookName string, status WebhookEventStatus) (int64, error) {
session := ormer.Engine.Where("1 = 1")
if owner != "" {
session = session.Where("owner = ?", owner)
}
if organization != "" {
session = session.Where("organization = ?", organization)
}
if webhookName != "" {
session = session.Where("webhook_name = ?", webhookName)
}
if status != "" {
session = session.Where("status = ?", status)
}
return session.Count(&WebhookEvent{})
}
func GetPendingWebhookEvents(limit int) ([]*WebhookEvent, error) {
events := []*WebhookEvent{}
currentTime := util.GetCurrentTime()
err := ormer.Engine.
Where("status = ? OR status = ?", WebhookEventStatusPending, WebhookEventStatusRetrying).
And("(next_retry_time = '' OR next_retry_time <= ?)", currentTime).
Asc("created_time").
Limit(limit).
Find(&events)
if err != nil {
return nil, err
}
return events, nil
}
func AddWebhookEvent(event *WebhookEvent) (bool, error) {
if event.Name == "" {
event.Name = util.GenerateId()
}
if event.CreatedTime == "" {
event.CreatedTime = util.GetCurrentTime()
}
if event.UpdatedTime == "" {
event.UpdatedTime = util.GetCurrentTime()
}
if event.Status == "" {
event.Status = WebhookEventStatusPending
}
affected, err := ormer.Engine.Insert(event)
if err != nil {
return false, err
}
return affected != 0, nil
}
func UpdateWebhookEvent(id string, event *WebhookEvent) (bool, error) {
owner, name, err := util.GetOwnerAndNameFromIdWithError(id)
if err != nil {
return false, err
}
event.UpdatedTime = util.GetCurrentTime()
affected, err := ormer.Engine.ID(core.PK{owner, name}).AllCols().Update(event)
if err != nil {
return false, err
}
return affected != 0, nil
}
func UpdateWebhookEventStatus(event *WebhookEvent, status WebhookEventStatus, statusCode int, response string, err error) (bool, error) {
event.Status = status
event.LastStatusCode = statusCode
event.LastResponse = response
event.UpdatedTime = util.GetCurrentTime()
if status != WebhookEventStatusRetrying {
event.NextRetryTime = ""
}
if err != nil {
event.LastError = err.Error()
} else {
event.LastError = ""
}
affected, dbErr := ormer.Engine.ID(core.PK{event.Owner, event.Name}).
Cols("status", "last_status_code", "last_response", "last_error", "updated_time", "attempt_count", "max_retries", "next_retry_time").
Update(event)
if dbErr != nil {
return false, dbErr
}
return affected != 0, nil
}
func DeleteWebhookEvent(event *WebhookEvent) (bool, error) {
affected, err := ormer.Engine.ID(core.PK{event.Owner, event.Name}).Delete(&WebhookEvent{})
if err != nil {
return false, err
}
return affected != 0, nil
}
func (e *WebhookEvent) GetId() string {
return fmt.Sprintf("%s/%s", e.Owner, e.Name)
}
// CreateWebhookEventFromRecord creates a webhook event from a record
func CreateWebhookEventFromRecord(webhook *Webhook, record *Record, extendedUser *User) (*WebhookEvent, error) {
maxRetries := webhook.MaxRetries
if maxRetries <= 0 {
maxRetries = 3
}
event := &WebhookEvent{
Owner: webhook.Owner,
Name: util.GenerateId(),
CreatedTime: util.GetCurrentTime(),
UpdatedTime: util.GetCurrentTime(),
WebhookName: webhook.GetId(),
Organization: record.Organization,
EventType: record.Action,
Status: WebhookEventStatusPending,
Payload: util.StructToJson(record),
AttemptCount: 0,
MaxRetries: maxRetries,
}
if extendedUser != nil {
event.ExtendedUser = util.StructToJson(extendedUser)
}
_, err := AddWebhookEvent(event)
if err != nil {
return nil, err
}
return event, nil
}