Files
2026-04-24 10:45:19 -07:00

154 lines
3.9 KiB
Go

package service
import (
"context"
"database/sql"
"encoding/json"
"fmt"
"log/slog"
"time"
"github.com/TopherMayor/unified-media-manager/internal/db"
)
type ActivityEvent struct {
ID int64 `json:"id"`
EventType string `json:"event_type"`
MediaID *int64 `json:"media_id,omitempty"`
MediaType *string `json:"media_type,omitempty"`
Title string `json:"title"`
Description *string `json:"description,omitempty"`
Data json.RawMessage `json:"data"`
CreatedAt time.Time `json:"created_at"`
}
type ActivityFilters struct {
EventType string
MediaID *int64
MediaType string
Page int
PageSize int
}
type LogEntry struct {
EventType string
MediaID *int64
MediaType *string
Title string
Description *string
Data json.RawMessage
}
type ActivityService struct {
db *db.DB
}
func NewActivityService(database *db.DB) *ActivityService {
return &ActivityService{db: database}
}
const activityColumns = `id, event_type, media_id, media_type, title, description, data, created_at`
func scanActivityEvent(scanner interface{ Scan(...interface{}) error }) (*ActivityEvent, error) {
var event ActivityEvent
var mediaID sql.NullInt64
var mediaType sql.NullString
var description sql.NullString
var data []byte
err := scanner.Scan(&event.ID, &event.EventType, &mediaID, &mediaType,
&event.Title, &description, &data, &event.CreatedAt)
if err != nil {
return nil, err
}
if mediaID.Valid {
event.MediaID = &mediaID.Int64
}
if mediaType.Valid {
event.MediaType = &mediaType.String
}
if description.Valid {
event.Description = &description.String
}
if data != nil {
event.Data = json.RawMessage(data)
}
return &event, nil
}
func (s *ActivityService) Log(ctx context.Context, entry LogEntry) (int64, error) {
data := entry.Data
if data == nil {
data = json.RawMessage("{}")
}
var id int64
err := s.db.Pool.QueryRow(ctx,
`INSERT INTO activity_events (event_type, media_id, media_type, title, description, data)
VALUES ($1, $2, $3, $4, $5, $6) RETURNING id`,
entry.EventType, entry.MediaID, entry.MediaType, entry.Title, entry.Description, data).Scan(&id)
if err != nil {
return 0, fmt.Errorf("insert activity event: %w", err)
}
return id, nil
}
func (s *ActivityService) LogAsync(entry LogEntry) {
go func() {
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel()
if _, err := s.Log(ctx, entry); err != nil {
slog.Error("failed to log activity event async", "error", err, "event_type", entry.EventType, "title", entry.Title)
}
}()
}
func (s *ActivityService) List(ctx context.Context, filters ActivityFilters) ([]ActivityEvent, int, error) {
qb := NewQueryBuilder(1)
if filters.EventType != "" {
qb.Add("event_type = $%d", filters.EventType)
}
if filters.MediaID != nil {
qb.Add("media_id = $%d", *filters.MediaID)
if filters.MediaType != "" {
qb.Add("media_type = $%d", filters.MediaType)
}
}
where := qb.Where()
var total int
countQuery := fmt.Sprintf("SELECT COUNT(*) FROM activity_events%s", where)
if err := s.db.Pool.QueryRow(ctx, countQuery, qb.Args()...).Scan(&total); err != nil {
return nil, 0, fmt.Errorf("count activity events: %w", err)
}
offset := (filters.Page - 1) * filters.PageSize
dataQuery := fmt.Sprintf(
"SELECT %s FROM activity_events%s ORDER BY created_at DESC LIMIT $%d OFFSET $%d",
activityColumns, where, qb.Idx(), qb.Idx()+1)
args := append(qb.Args(), filters.PageSize, offset)
rows, err := s.db.Pool.Query(ctx, dataQuery, args...)
if err != nil {
return nil, 0, fmt.Errorf("list activity events: %w", err)
}
defer rows.Close()
var events []ActivityEvent
for rows.Next() {
event, err := scanActivityEvent(rows)
if err != nil {
slog.Error("failed to scan activity event", "error", err)
continue
}
events = append(events, *event)
}
return events, total, nil
}