Files
stream.api/internal/service/service_render.go

226 lines
6.2 KiB
Go

package service
import (
"context"
"encoding/json"
"errors"
"strings"
"github.com/google/uuid"
"gorm.io/gorm"
"stream.api/internal/database/model"
"stream.api/internal/dto"
)
type AgentRuntime interface {
ListAgentsWithStats() []*dto.AgentWithStats
SendCommand(agentID string, cmd string) bool
}
var (
ErrUserNotFound = errors.New("user not found")
ErrAdTemplateNotFound = errors.New("ad template not found")
ErrJobServiceUnavailable = errors.New("job service is unavailable")
)
type Service struct {
db *gorm.DB
jobService *JobService
}
type CreateVideoInput struct {
UserID string
Title string
Description *string
URL string
Size int64
Duration int32
Format string
AdTemplateID *string
}
type CreateVideoResult struct {
Video *model.Video
Job model.Job
}
func NewService(db *gorm.DB, jobService *JobService) *Service {
return &Service{db: db, jobService: jobService}
}
func (s *Service) JobService() *JobService {
if s == nil {
return nil
}
return s.jobService
}
func (s *Service) CreateVideo(ctx context.Context, input CreateVideoInput) (*CreateVideoResult, error) {
if s == nil || s.db == nil {
return nil, gorm.ErrInvalidDB
}
userID := strings.TrimSpace(input.UserID)
if userID == "" {
return nil, ErrUserNotFound
}
var user model.User
if err := s.db.WithContext(ctx).Where("id = ?", userID).First(&user).Error; err != nil {
if errors.Is(err, gorm.ErrRecordNotFound) {
return nil, ErrUserNotFound
}
return nil, err
}
title := strings.TrimSpace(input.Title)
videoURL := strings.TrimSpace(input.URL)
format := strings.TrimSpace(input.Format)
statusValue := "processing"
processingStatus := "PENDING"
storageType := detectStorageType(videoURL)
video := &model.Video{
ID: uuid.NewString(),
UserID: user.ID,
Name: title,
Title: title,
Description: nullableTrimmedString(input.Description),
URL: videoURL,
Size: input.Size,
Duration: input.Duration,
Format: format,
Status: model.StringPtr(statusValue),
ProcessingStatus: model.StringPtr(processingStatus),
StorageType: model.StringPtr(storageType),
}
if err := s.db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
if err := tx.Create(video).Error; err != nil {
return err
}
if err := tx.Model(&model.User{}).Where("id = ?", user.ID).UpdateColumn("storage_used", gorm.Expr("storage_used + ?", video.Size)).Error; err != nil {
return err
}
return saveVideoAdConfig(ctx, tx, video, user.ID, input.AdTemplateID)
}); err != nil {
return nil, err
}
if s.jobService == nil {
_ = markVideoJobFailed(ctx, s.db, video.ID)
return nil, ErrJobServiceUnavailable
}
jobPayload, err := buildJobPayload(video.ID, user.ID, videoURL, format)
if err != nil {
_ = markVideoJobFailed(ctx, s.db, video.ID)
return nil, err
}
job, err := s.jobService.CreateJob(ctx, user.ID, video.ID, title, jobPayload, 0, 0)
if err != nil {
_ = markVideoJobFailed(ctx, s.db, video.ID)
return nil, err
}
return &CreateVideoResult{Video: video, Job: *job}, nil
}
func (s *Service) ListJobs(ctx context.Context, offset, limit int) (*dto.PaginatedJobs, error) {
if s == nil || s.jobService == nil {
return nil, ErrJobServiceUnavailable
}
return s.jobService.ListJobs(ctx, offset, limit)
}
func (s *Service) ListJobsByAgent(ctx context.Context, agentID string, offset, limit int) (*dto.PaginatedJobs, error) {
if s == nil || s.jobService == nil {
return nil, ErrJobServiceUnavailable
}
return s.jobService.ListJobsByAgent(ctx, agentID, offset, limit)
}
func (s *Service) ListJobsByCursor(ctx context.Context, agentID string, cursor string, pageSize int) (*dto.PaginatedJobs, error) {
if s == nil || s.jobService == nil {
return nil, ErrJobServiceUnavailable
}
return s.jobService.ListJobsByCursor(ctx, agentID, cursor, pageSize)
}
func (s *Service) GetJob(ctx context.Context, id string) (*model.Job, error) {
if s == nil || s.jobService == nil {
return nil, ErrJobServiceUnavailable
}
return s.jobService.GetJob(ctx, id)
}
func (s *Service) CreateJob(ctx context.Context, userID string, videoID string, name string, config []byte, priority int, timeLimit int64) (*model.Job, error) {
if s == nil || s.jobService == nil {
return nil, ErrJobServiceUnavailable
}
return s.jobService.CreateJob(ctx, userID, videoID, name, config, priority, timeLimit)
}
func (s *Service) CancelJob(ctx context.Context, id string) error {
if s == nil || s.jobService == nil {
return ErrJobServiceUnavailable
}
return s.jobService.CancelJob(ctx, id)
}
func (s *Service) RetryJob(ctx context.Context, id string) (*model.Job, error) {
if s == nil || s.jobService == nil {
return nil, ErrJobServiceUnavailable
}
return s.jobService.RetryJob(ctx, id)
}
func buildJobPayload(videoID, userID, videoURL, format string) ([]byte, error) {
return json.Marshal(map[string]any{
"video_id": videoID,
"user_id": userID,
"input_url": videoURL,
"source_url": videoURL,
"format": format,
})
}
func saveVideoAdConfig(ctx context.Context, tx *gorm.DB, video *model.Video, userID string, adTemplateID *string) error {
if video == nil || adTemplateID == nil {
return nil
}
trimmed := strings.TrimSpace(*adTemplateID)
if trimmed == "" {
if err := tx.WithContext(ctx).Model(&model.Video{}).Where("id = ?", video.ID).Update("ad_id", nil).Error; err != nil {
return err
}
video.AdID = nil
return nil
}
var template model.AdTemplate
if err := tx.WithContext(ctx).Select("id").Where("id = ? AND user_id = ?", trimmed, userID).First(&template).Error; err != nil {
if errors.Is(err, gorm.ErrRecordNotFound) {
return ErrAdTemplateNotFound
}
return err
}
if err := tx.WithContext(ctx).Model(&model.Video{}).Where("id = ?", video.ID).Update("ad_id", template.ID).Error; err != nil {
return err
}
video.AdID = &template.ID
return nil
}
func markVideoJobFailed(ctx context.Context, db *gorm.DB, videoID string) error {
if db == nil {
return nil
}
return db.WithContext(ctx).
Model(&model.Video{}).
Where("id = ?", strings.TrimSpace(videoID)).
Updates(map[string]any{"status": "failed", "processing_status": "FAILED"}).Error
}