- Add OTLP/gRPC observability config and resource attributes - Instrument webhook and onboarding handlers with tracing and metrics - Add OpenTelemetry dependencies to go.mod/go.sum - Update NixOS modules to configure observability settings
253 lines
7.2 KiB
Go
253 lines
7.2 KiB
Go
package webhook
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"net/http"
|
|
"strings"
|
|
"time"
|
|
|
|
"go.opentelemetry.io/otel/attribute"
|
|
"go.opentelemetry.io/otel/codes"
|
|
"go.opentelemetry.io/otel/trace"
|
|
"maunium.net/go/mautrix/bridgev2"
|
|
"maunium.net/go/mautrix/event"
|
|
"maunium.net/go/mautrix/format"
|
|
"maunium.net/go/mautrix/id"
|
|
|
|
"sneeuwvlok/packages/arrtrix/pkg/observability"
|
|
)
|
|
|
|
const ArrWebhookPath = "/_arrtrix/webhook"
|
|
|
|
var (
|
|
ErrNoManagementRoom = errors.New("no management room configured")
|
|
ErrAmbiguousManagementRoom = errors.New("multiple management rooms configured")
|
|
)
|
|
|
|
type payload struct {
|
|
EventType string `json:"eventType"`
|
|
Movie *movie `json:"movie"`
|
|
MovieFile *movieFile `json:"movieFile"`
|
|
IsUpgrade bool `json:"isUpgrade"`
|
|
}
|
|
|
|
type movie struct {
|
|
Title string `json:"title"`
|
|
Year int `json:"year"`
|
|
ImdbID string `json:"imdbId"`
|
|
TmdbID int `json:"tmdbId"`
|
|
Path string `json:"path"`
|
|
}
|
|
|
|
type movieFile struct {
|
|
Quality string `json:"quality"`
|
|
RelativePath string `json:"relativePath"`
|
|
SceneName string `json:"sceneName"`
|
|
ReleaseGroup string `json:"releaseGroup"`
|
|
}
|
|
|
|
type roomResolver interface {
|
|
ResolveManagementRoom(context.Context) (id.RoomID, error)
|
|
}
|
|
|
|
type noticeSender interface {
|
|
SendNotice(context.Context, id.RoomID, string) error
|
|
}
|
|
|
|
type ArrHandler struct {
|
|
resolver roomResolver
|
|
sender noticeSender
|
|
}
|
|
|
|
func MountArr(router *http.ServeMux, bridge *bridgev2.Bridge) error {
|
|
if bridge == nil {
|
|
return fmt.Errorf("bridge is not initialized")
|
|
}
|
|
handler := &ArrHandler{
|
|
resolver: bridgeRoomResolver{bridge: bridge},
|
|
sender: bridgeNoticeSender{bridge: bridge},
|
|
}
|
|
router.Handle(fmt.Sprintf("POST %s", ArrWebhookPath), handler)
|
|
return nil
|
|
}
|
|
|
|
func (h *ArrHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
start := time.Now()
|
|
ctx, span := observability.StartSpan(r.Context(), "arrtrix.webhook.handle", trace.WithSpanKind(trace.SpanKindServer))
|
|
defer span.End()
|
|
|
|
statusCode := http.StatusAccepted
|
|
outcome := "ok"
|
|
eventType := ""
|
|
defer func() {
|
|
observability.RecordWebhook(ctx, eventType, outcome, statusCode, time.Since(start))
|
|
}()
|
|
|
|
var body payload
|
|
if err := json.NewDecoder(r.Body).Decode(&body); err != nil {
|
|
statusCode = http.StatusBadRequest
|
|
outcome = "invalid_payload"
|
|
span.RecordError(err)
|
|
span.SetStatus(codes.Error, err.Error())
|
|
http.Error(w, "invalid webhook payload", http.StatusBadRequest)
|
|
return
|
|
}
|
|
if strings.TrimSpace(body.EventType) == "" {
|
|
statusCode = http.StatusBadRequest
|
|
outcome = "missing_event_type"
|
|
span.SetStatus(codes.Error, "missing eventType")
|
|
http.Error(w, "missing eventType", http.StatusBadRequest)
|
|
return
|
|
}
|
|
eventType = body.EventType
|
|
span.SetAttributes(
|
|
attribute.String("arrtrix.webhook.event_type", body.EventType),
|
|
attribute.String("http.method", r.Method),
|
|
attribute.String("http.route", ArrWebhookPath),
|
|
)
|
|
|
|
roomID, err := h.resolver.ResolveManagementRoom(ctx)
|
|
if err != nil {
|
|
statusCode = http.StatusInternalServerError
|
|
outcome = "resolve_failed"
|
|
if errors.Is(err, ErrNoManagementRoom) || errors.Is(err, ErrAmbiguousManagementRoom) {
|
|
statusCode = http.StatusConflict
|
|
outcome = "routing_conflict"
|
|
}
|
|
span.RecordError(err)
|
|
span.SetStatus(codes.Error, err.Error())
|
|
http.Error(w, err.Error(), statusCode)
|
|
return
|
|
}
|
|
|
|
if err = h.sender.SendNotice(ctx, roomID, renderNotice(body)); err != nil {
|
|
statusCode = http.StatusBadGateway
|
|
outcome = "delivery_failed"
|
|
span.RecordError(err)
|
|
span.SetStatus(codes.Error, err.Error())
|
|
http.Error(w, "failed to deliver webhook", http.StatusBadGateway)
|
|
return
|
|
}
|
|
|
|
span.SetStatus(codes.Ok, "")
|
|
w.WriteHeader(statusCode)
|
|
}
|
|
|
|
type bridgeRoomResolver struct {
|
|
bridge *bridgev2.Bridge
|
|
}
|
|
|
|
func (r bridgeRoomResolver) ResolveManagementRoom(ctx context.Context) (id.RoomID, error) {
|
|
ctx, span := observability.StartSpan(ctx, "arrtrix.webhook.resolve_management_room")
|
|
defer span.End()
|
|
|
|
rows, err := r.bridge.DB.Query(ctx, `SELECT mxid, management_room FROM "user" WHERE bridge_id=$1 AND management_room IS NOT NULL AND management_room <> ''`, r.bridge.ID)
|
|
if err != nil {
|
|
span.RecordError(err)
|
|
span.SetStatus(codes.Error, err.Error())
|
|
return "", fmt.Errorf("failed to query management rooms: %w", err)
|
|
}
|
|
defer rows.Close()
|
|
|
|
var roomID id.RoomID
|
|
var owners []id.UserID
|
|
for rows.Next() {
|
|
var mxid, managementRoom string
|
|
if err = rows.Scan(&mxid, &managementRoom); err != nil {
|
|
span.RecordError(err)
|
|
span.SetStatus(codes.Error, err.Error())
|
|
return "", fmt.Errorf("failed to scan management room: %w", err)
|
|
}
|
|
owners = append(owners, id.UserID(mxid))
|
|
if roomID == "" {
|
|
roomID = id.RoomID(managementRoom)
|
|
}
|
|
}
|
|
if err = rows.Err(); err != nil {
|
|
span.RecordError(err)
|
|
span.SetStatus(codes.Error, err.Error())
|
|
return "", fmt.Errorf("failed to iterate management rooms: %w", err)
|
|
}
|
|
|
|
switch len(owners) {
|
|
case 0:
|
|
span.SetStatus(codes.Error, ErrNoManagementRoom.Error())
|
|
return "", ErrNoManagementRoom
|
|
case 1:
|
|
span.SetAttributes(attribute.Int("arrtrix.management_room.count", 1))
|
|
span.SetStatus(codes.Ok, "")
|
|
return roomID, nil
|
|
default:
|
|
span.SetAttributes(attribute.Int("arrtrix.management_room.count", len(owners)))
|
|
span.SetStatus(codes.Error, ErrAmbiguousManagementRoom.Error())
|
|
return "", fmt.Errorf("%w: %s", ErrAmbiguousManagementRoom, strings.Join(convertUserIDs(owners), ", "))
|
|
}
|
|
}
|
|
|
|
type bridgeNoticeSender struct {
|
|
bridge *bridgev2.Bridge
|
|
}
|
|
|
|
func (s bridgeNoticeSender) SendNotice(ctx context.Context, roomID id.RoomID, markdown string) error {
|
|
ctx, span := observability.StartSpan(ctx, "arrtrix.webhook.send_notice")
|
|
defer span.End()
|
|
span.SetAttributes(attribute.String("matrix.room_id", roomID.String()))
|
|
|
|
if err := s.bridge.Bot.EnsureJoined(ctx, roomID); err != nil {
|
|
span.RecordError(err)
|
|
span.SetStatus(codes.Error, err.Error())
|
|
return err
|
|
}
|
|
content := format.RenderMarkdown(markdown, true, false)
|
|
_, err := s.bridge.Bot.SendMessage(ctx, roomID, event.EventMessage, &event.Content{Parsed: &content}, nil)
|
|
if err != nil {
|
|
span.RecordError(err)
|
|
span.SetStatus(codes.Error, err.Error())
|
|
return err
|
|
}
|
|
span.SetStatus(codes.Ok, "")
|
|
return err
|
|
}
|
|
|
|
func renderNotice(body payload) string {
|
|
title := "Arr"
|
|
if body.Movie != nil {
|
|
title = body.Movie.Title
|
|
if body.Movie.Year != 0 {
|
|
title = fmt.Sprintf("%s (%d)", title, body.Movie.Year)
|
|
}
|
|
}
|
|
|
|
lines := []string{fmt.Sprintf("**Arr %s**", body.EventType)}
|
|
if title != "Arr" {
|
|
lines = append(lines, fmt.Sprintf("Movie: %s", title))
|
|
}
|
|
if body.MovieFile != nil && body.MovieFile.Quality != "" {
|
|
lines = append(lines, fmt.Sprintf("Quality: %s", body.MovieFile.Quality))
|
|
}
|
|
if body.MovieFile != nil && body.MovieFile.RelativePath != "" {
|
|
lines = append(lines, fmt.Sprintf("File: `%s`", body.MovieFile.RelativePath))
|
|
}
|
|
if body.EventType == "Download" {
|
|
lines = append(lines, fmt.Sprintf("Upgrade: %t", body.IsUpgrade))
|
|
}
|
|
if body.Movie != nil && body.Movie.ImdbID != "" {
|
|
lines = append(lines, fmt.Sprintf("IMDb: `%s`", body.Movie.ImdbID))
|
|
}
|
|
return strings.Join(lines, "\n")
|
|
}
|
|
|
|
func convertUserIDs(users []id.UserID) []string {
|
|
out := make([]string, len(users))
|
|
for i, user := range users {
|
|
out[i] = string(user)
|
|
}
|
|
return out
|
|
}
|
|
|
|
var _ roomResolver = bridgeRoomResolver{}
|
|
var _ noticeSender = bridgeNoticeSender{}
|
|
var _ http.Handler = (*ArrHandler)(nil)
|