mirror of
https://github.com/Luzifer/twitch-bot.git
synced 2024-12-21 04:11:16 +00:00
Knut Ahlers
f684abc29f
which might be a false-negative and re-added later Signed-off-by: Knut Ahlers <knut@ahlers.me>
166 lines
4.7 KiB
Go
166 lines
4.7 KiB
Go
// Package customevent contains an actor and database modules to create
|
|
// custom (timed) events
|
|
package customevent
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/gorilla/mux"
|
|
"github.com/pkg/errors"
|
|
"gorm.io/gorm"
|
|
|
|
"github.com/Luzifer/twitch-bot/v3/pkg/database"
|
|
"github.com/Luzifer/twitch-bot/v3/plugins"
|
|
)
|
|
|
|
const actorName = "customevent"
|
|
|
|
var (
|
|
db database.Connector
|
|
eventCreatorFunc plugins.EventHandlerFunc
|
|
formatMessage plugins.MsgFormatter
|
|
mc *memoryCache
|
|
|
|
ptrStringEmpty = func(s string) *string { return &s }("")
|
|
)
|
|
|
|
// Register provides the plugins.RegisterFunc
|
|
func Register(args plugins.RegistrationArguments) (err error) {
|
|
db = args.GetDatabaseConnector()
|
|
if err = db.DB().AutoMigrate(&storedCustomEvent{}); err != nil {
|
|
return errors.Wrap(err, "applying schema migration")
|
|
}
|
|
|
|
args.RegisterCopyDatabaseFunc("custom_event", func(src, target *gorm.DB) error {
|
|
return database.CopyObjects(src, target, &storedCustomEvent{})
|
|
})
|
|
|
|
mc = &memoryCache{dbc: db}
|
|
|
|
eventCreatorFunc = args.CreateEvent
|
|
formatMessage = args.FormatMessage
|
|
|
|
args.RegisterActor(actorName, func() plugins.Actor { return &actor{} })
|
|
|
|
args.RegisterActorDocumentation(plugins.ActionDocumentation{
|
|
Description: "Create a custom event",
|
|
Name: "Custom Event",
|
|
Type: actorName,
|
|
|
|
Fields: []plugins.ActionDocumentationField{
|
|
{
|
|
Default: "{}",
|
|
Description: "JSON representation of fields in the event (`map[string]any`)",
|
|
Key: "fields",
|
|
Name: "Fields",
|
|
Optional: false,
|
|
SupportTemplate: true,
|
|
Type: plugins.ActionDocumentationFieldTypeString,
|
|
},
|
|
{
|
|
Default: "",
|
|
Description: "Time until the event is triggered (must be valid duration like 1h, 1h1m, 10s, ...)",
|
|
Key: "schedule_in",
|
|
Name: "Schedule In",
|
|
Optional: true,
|
|
SupportTemplate: true,
|
|
Type: plugins.ActionDocumentationFieldTypeString,
|
|
},
|
|
},
|
|
})
|
|
|
|
if err = args.RegisterAPIRoute(plugins.HTTPRouteRegistrationArgs{
|
|
Description: "Creates an `custom` event containing the fields provided in the request body",
|
|
HandlerFunc: handleCreateEvent,
|
|
Method: http.MethodPost,
|
|
Module: "customevent",
|
|
Name: "Create custom event",
|
|
Path: "/{channel}",
|
|
QueryParams: []plugins.HTTPRouteParamDocumentation{
|
|
{
|
|
Description: "Time until the event is triggered (must be valid duration like 1h, 1h1m, 10s, ...)",
|
|
Name: "schedule_in",
|
|
Required: false,
|
|
Type: "duration",
|
|
},
|
|
},
|
|
RequiresWriteAuth: true,
|
|
ResponseType: plugins.HTTPRouteResponseTypeNo200,
|
|
RouteParams: []plugins.HTTPRouteParamDocumentation{
|
|
{
|
|
Description: "Channel to create the event in",
|
|
Name: "channel",
|
|
},
|
|
},
|
|
}); err != nil {
|
|
return fmt.Errorf("registering API route: %w", err)
|
|
}
|
|
|
|
for schedule, fn := range map[string]func(){
|
|
fmt.Sprintf("@every %s", cleanupTimeout): scheduleCleanup,
|
|
"* * * * * *": scheduleSend,
|
|
} {
|
|
if _, err := args.RegisterCron(schedule, fn); err != nil {
|
|
return errors.Wrap(err, "registering cron function")
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func handleCreateEvent(w http.ResponseWriter, r *http.Request) {
|
|
channel := mux.Vars(r)["channel"]
|
|
|
|
if channel == "" {
|
|
http.Error(w, errors.New("missing channel").Error(), http.StatusBadRequest)
|
|
return
|
|
}
|
|
channel = "#" + strings.TrimLeft(channel, "#") // Sanitize
|
|
|
|
if err := triggerOrStoreEvent(channel, r.Body, r.FormValue("schedule_in")); err != nil {
|
|
http.Error(w, errors.Wrap(err, "creating event").Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
w.WriteHeader(http.StatusNoContent)
|
|
}
|
|
|
|
func parseEvent(channel string, fieldData io.Reader) (*plugins.FieldCollection, error) {
|
|
payload := make(map[string]any)
|
|
|
|
if err := json.NewDecoder(fieldData).Decode(&payload); err != nil {
|
|
return nil, errors.Wrap(err, "parsing event payload")
|
|
}
|
|
|
|
fields := plugins.FieldCollectionFromData(payload)
|
|
fields.Set("channel", "#"+strings.TrimLeft(channel, "#"))
|
|
|
|
return fields, nil
|
|
}
|
|
|
|
func triggerOrStoreEvent(channel string, fieldData io.Reader, rawDelay string) error {
|
|
fields, err := parseEvent(channel, fieldData)
|
|
if err != nil {
|
|
return errors.Wrap(err, "parsing fields")
|
|
}
|
|
|
|
if delay, err := time.ParseDuration(rawDelay); err == nil && delay > 0 {
|
|
// Delay set, store for later triggering
|
|
if err = storeEvent(db, time.Now().Add(delay).UTC(), channel, fields); err != nil {
|
|
return errors.Wrap(err, "storing event")
|
|
}
|
|
return errors.Wrap(mc.Refresh(), "refreshing memory cache")
|
|
}
|
|
|
|
// No delay, trigger instantly
|
|
if err := eventCreatorFunc("custom", fields); err != nil {
|
|
return errors.Wrap(err, "creating event")
|
|
}
|
|
|
|
return nil
|
|
}
|