package service import ( "context" "regexp" "time" "watch-party-backend/internal/core/episode" ) type EpisodeService interface { GetCurrent(ctx context.Context) (episode.Episode, error) SetCurrent(ctx context.Context, id int64, start string) (episode.Episode, error) MoveToArchive(ctx context.Context, ids []int64) (episode.MoveResult, error) ListAll(ctx context.Context) ([]episode.Episode, error) Delete(ctx context.Context, id int64) error } type episodeService struct { repo episode.Repository } func NewEpisodeService(r episode.Repository) EpisodeService { return &episodeService{repo: r} } func (s *episodeService) ListAll(ctx context.Context) ([]episode.Episode, error) { c, cancel := context.WithTimeout(ctx, 5*time.Second) defer cancel() return s.repo.ListAll(c) } func (s *episodeService) GetCurrent(ctx context.Context) (episode.Episode, error) { c, cancel := context.WithTimeout(ctx, 3*time.Second) defer cancel() return s.repo.GetCurrent(c) } var hhmmss = regexp.MustCompile(`^(?:[01]\d|2[0-3]):[0-5]\d:[0-5]\d$`) func (s *episodeService) SetCurrent(ctx context.Context, id int64, start string) (episode.Episode, error) { if !hhmmss.MatchString(start) { return episode.Episode{}, episode.ErrInvalidStartTime } c, cancel := context.WithTimeout(ctx, 5*time.Second) defer cancel() if err := s.repo.SetCurrent(c, id, start); err != nil { return episode.Episode{}, err } return s.repo.GetCurrent(c) } func (s *episodeService) MoveToArchive(ctx context.Context, ids []int64) (episode.MoveResult, error) { uniq := make([]int64, 0, len(ids)) seen := make(map[int64]struct{}, len(ids)) for _, id := range ids { if _, ok := seen[id]; !ok { seen[id] = struct{}{} uniq = append(uniq, id) } } if len(uniq) == 0 { return episode.MoveResult{}, episode.ErrEmptyIDs } c, cancel := context.WithTimeout(ctx, 10*time.Second) defer cancel() return s.repo.MoveToArchive(c, uniq) } func (s *episodeService) Delete(ctx context.Context, id int64) error { c, cancel := context.WithTimeout(ctx, 5*time.Second) defer cancel() return s.repo.Delete(c, id) }