104 lines
2.4 KiB
Go
104 lines
2.4 KiB
Go
package jikan
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net/url"
|
|
)
|
|
|
|
func (c *Client) Search(ctx context.Context, query string, page int) (SearchResult, error) {
|
|
return c.search(ctx, query, page, 0)
|
|
}
|
|
|
|
func (c *Client) SearchWithLimit(ctx context.Context, query string, page int, limit int) (SearchResult, error) {
|
|
return c.search(ctx, query, page, limit)
|
|
}
|
|
|
|
func (c *Client) search(ctx context.Context, query string, page int, limit int) (SearchResult, error) {
|
|
if query == "" {
|
|
return SearchResult{}, nil
|
|
}
|
|
if page < 1 {
|
|
page = 1
|
|
}
|
|
if limit < 0 {
|
|
limit = 0
|
|
}
|
|
|
|
cacheKey := fmt.Sprintf("search:%s:%d:%d", query, page, limit)
|
|
var cached SearchResult
|
|
if c.getCache(ctx, cacheKey, &cached) {
|
|
return cached, nil
|
|
}
|
|
|
|
var stale SearchResult
|
|
hasStale := c.getStaleCache(ctx, cacheKey, &stale)
|
|
|
|
var result SearchResponse
|
|
reqURL := fmt.Sprintf("%s/anime?q=%s&page=%d", c.baseURL, url.QueryEscape(query), page)
|
|
if limit > 0 {
|
|
reqURL = fmt.Sprintf("%s&limit=%d", reqURL, limit)
|
|
}
|
|
|
|
if err := c.fetchWithRetry(ctx, reqURL, &result); err != nil {
|
|
if limit > 0 && IsRetryableError(err) {
|
|
fallbackURL := fmt.Sprintf("%s/anime?q=%s&page=%d", c.baseURL, url.QueryEscape(query), page)
|
|
if fallbackErr := c.fetchWithRetry(ctx, fallbackURL, &result); fallbackErr == nil {
|
|
res := SearchResult{
|
|
Animes: result.Data,
|
|
HasNextPage: result.Pagination.HasNextPage,
|
|
}
|
|
c.setCache(ctx, cacheKey, res, shortCacheTTL)
|
|
return res, nil
|
|
}
|
|
}
|
|
|
|
if hasStale {
|
|
return stale, nil
|
|
}
|
|
|
|
return SearchResult{}, err
|
|
}
|
|
|
|
res := SearchResult{
|
|
Animes: result.Data,
|
|
HasNextPage: result.Pagination.HasNextPage,
|
|
}
|
|
|
|
c.setCache(ctx, cacheKey, res, shortCacheTTL)
|
|
return res, nil
|
|
}
|
|
|
|
func (c *Client) GetTopAnime(ctx context.Context, page int) (TopAnimeResult, error) {
|
|
if page < 1 {
|
|
page = 1
|
|
}
|
|
cacheKey := fmt.Sprintf("top:%d", page)
|
|
var cached TopAnimeResult
|
|
if c.getCache(ctx, cacheKey, &cached) {
|
|
return cached, nil
|
|
}
|
|
|
|
var stale TopAnimeResult
|
|
hasStale := c.getStaleCache(ctx, cacheKey, &stale)
|
|
|
|
var result TopAnimeResponse
|
|
reqURL := fmt.Sprintf("%s/top/anime?page=%d", c.baseURL, page)
|
|
|
|
if err := c.fetchWithRetry(ctx, reqURL, &result); err != nil {
|
|
if hasStale {
|
|
return stale, nil
|
|
}
|
|
|
|
return TopAnimeResult{}, err
|
|
}
|
|
|
|
res := TopAnimeResult{
|
|
Animes: result.Data,
|
|
HasNextPage: result.Pagination.HasNextPage,
|
|
}
|
|
|
|
c.setCache(ctx, cacheKey, res, shortCacheTTL)
|
|
return res, nil
|
|
}
|