153 lines
3.7 KiB
Go
153 lines
3.7 KiB
Go
package jikan
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"log"
|
|
"strings"
|
|
"time"
|
|
|
|
"mal/internal/watchorder"
|
|
)
|
|
|
|
const chiakiWatchOrderURL = "https://chiaki.site/?/tools/watch_order/id/%d"
|
|
const watchOrderCacheTTL = time.Hour * 24
|
|
const maxWatchOrderEntries = 120
|
|
|
|
func watchOrderTypeLabel(value string) string {
|
|
normalized := strings.ToLower(strings.TrimSpace(value))
|
|
switch normalized {
|
|
case "tv":
|
|
return "TV"
|
|
case "movie":
|
|
return "Movie"
|
|
default:
|
|
return strings.TrimSpace(value)
|
|
}
|
|
}
|
|
|
|
func isAllowedWatchOrderType(value string) bool {
|
|
normalized := strings.ToLower(strings.TrimSpace(value))
|
|
return normalized == "tv" || normalized == "movie"
|
|
}
|
|
|
|
func relationCacheKey(id int) string {
|
|
return fmt.Sprintf("relations:watch-order:%d", id)
|
|
}
|
|
|
|
func (c *Client) getWatchOrder(ctx context.Context, id int) (watchorder.WatchOrderResult, error) {
|
|
cacheKey := relationCacheKey(id)
|
|
|
|
var cached watchorder.WatchOrderResult
|
|
if c.getCache(ctx, cacheKey, &cached) {
|
|
return cached, nil
|
|
}
|
|
|
|
watchOrderURL := fmt.Sprintf(chiakiWatchOrderURL, id)
|
|
requestCtx, cancel := context.WithTimeout(ctx, 10*time.Second)
|
|
defer cancel()
|
|
|
|
result, err := watchorder.FetchWatchOrder(requestCtx, c.httpClient, watchOrderURL)
|
|
if err != nil {
|
|
var statusError *watchorder.HTTPStatusError
|
|
if errors.Is(err, watchorder.ErrWatchOrderMarkupNotFound) {
|
|
log.Printf("relations: watch-order markup missing for %d (%s): %v", id, watchOrderURL, err)
|
|
} else if errors.As(err, &statusError) {
|
|
log.Printf(
|
|
"relations: watch-order http error for %d (%s): status=%d server=%q cf_ray=%q location=%q content_type=%q body=%q",
|
|
id,
|
|
watchOrderURL,
|
|
statusError.StatusCode,
|
|
statusError.Server,
|
|
statusError.CFRay,
|
|
statusError.Location,
|
|
statusError.ContentType,
|
|
statusError.BodyPreview,
|
|
)
|
|
} else {
|
|
log.Printf("relations: watch-order fetch failed for %d (%s): %v", id, watchOrderURL, err)
|
|
}
|
|
return watchorder.WatchOrderResult{}, err
|
|
}
|
|
|
|
c.setCache(ctx, cacheKey, result, watchOrderCacheTTL)
|
|
return result, nil
|
|
}
|
|
|
|
func (c *Client) currentOnlyRelation(ctx context.Context, id int) ([]RelationEntry, error) {
|
|
currentAnime, err := c.GetAnimeByID(ctx, id)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return []RelationEntry{{
|
|
Anime: currentAnime,
|
|
Relation: "Current",
|
|
IsCurrent: true,
|
|
IsExtra: false,
|
|
}}, nil
|
|
}
|
|
|
|
func (c *Client) GetFullRelations(ctx context.Context, id int) ([]RelationEntry, error) {
|
|
result, err := c.getWatchOrder(ctx, id)
|
|
if err != nil {
|
|
log.Printf("relations: using current-only fallback for %d: %v", id, err)
|
|
return c.currentOnlyRelation(ctx, id)
|
|
}
|
|
|
|
seen := make(map[int]bool)
|
|
relations := make([]RelationEntry, 0, len(result.WatchOrder)+1)
|
|
|
|
for _, watchOrderEntry := range result.WatchOrder {
|
|
if len(relations) >= maxWatchOrderEntries {
|
|
break
|
|
}
|
|
|
|
if !isAllowedWatchOrderType(watchOrderEntry.Type) {
|
|
continue
|
|
}
|
|
|
|
if seen[watchOrderEntry.ID] {
|
|
continue
|
|
}
|
|
|
|
anime, err := c.GetAnimeByID(ctx, watchOrderEntry.ID)
|
|
if err != nil {
|
|
log.Printf("relations: skipping related anime %d for root %d: %v", watchOrderEntry.ID, id, err)
|
|
continue
|
|
}
|
|
|
|
seen[watchOrderEntry.ID] = true
|
|
relations = append(relations, RelationEntry{
|
|
Anime: anime,
|
|
Relation: watchOrderTypeLabel(watchOrderEntry.Type),
|
|
IsCurrent: watchOrderEntry.ID == id,
|
|
IsExtra: false,
|
|
})
|
|
if watchOrderEntry.ID == id {
|
|
relations[len(relations)-1].Relation = "Current"
|
|
}
|
|
}
|
|
|
|
if !seen[id] {
|
|
currentAnime, err := c.GetAnimeByID(ctx, id)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
relations = append([]RelationEntry{{
|
|
Anime: currentAnime,
|
|
Relation: "Current",
|
|
IsCurrent: true,
|
|
IsExtra: false,
|
|
}}, relations...)
|
|
}
|
|
|
|
if len(relations) == 0 {
|
|
return c.currentOnlyRelation(ctx, id)
|
|
}
|
|
|
|
return relations, nil
|
|
}
|