David Ashby
f9d1cf744e
All checks were successful
ci/woodpecker/push/woodpecker Pipeline was successful
183 lines
5.1 KiB
Go
183 lines
5.1 KiB
Go
package database
|
|
|
|
import (
|
|
"context"
|
|
"encoding/gob"
|
|
"errors"
|
|
"fmt"
|
|
"log"
|
|
"os"
|
|
"strconv"
|
|
"sync"
|
|
"time"
|
|
|
|
"git.yetaga.in/alazyreader/library/media"
|
|
"github.com/irlndts/go-discogs"
|
|
)
|
|
|
|
type DiscogsCache struct {
|
|
authToken string
|
|
m sync.Mutex
|
|
cache []media.Record
|
|
maxCacheAge time.Duration
|
|
lastRefresh time.Time
|
|
client discogs.Discogs
|
|
username string
|
|
persistence bool
|
|
persistFile string
|
|
}
|
|
|
|
type persistence struct {
|
|
CachedRecordSlice []media.Record
|
|
LastRefresh time.Time
|
|
}
|
|
|
|
func NewDiscogsCache(token string, maxCacheAge time.Duration, username string, persist bool, persistFile string) (*DiscogsCache, error) {
|
|
client, err := discogs.New(&discogs.Options{
|
|
UserAgent: "library.yetaga.in personal collection cache",
|
|
Token: token,
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
cache := &DiscogsCache{
|
|
authToken: token,
|
|
client: client,
|
|
maxCacheAge: maxCacheAge,
|
|
username: username,
|
|
persistence: persist,
|
|
persistFile: persistFile,
|
|
}
|
|
if cache.persistence && cache.persistFile != "" {
|
|
cache.cache, cache.lastRefresh, err = cache.loadRecordsFromFS(context.Background())
|
|
if err != nil {
|
|
return nil, fmt.Errorf("cache load failed: %w", err)
|
|
}
|
|
if time.Now().After(cache.lastRefresh.Add(cache.maxCacheAge)) {
|
|
log.Printf("cache expired, running refresh...")
|
|
go func() {
|
|
err := cache.FlushCache(context.Background())
|
|
if err != nil {
|
|
log.Printf("error loading discogs content: %v", err)
|
|
}
|
|
}()
|
|
}
|
|
}
|
|
return cache, nil
|
|
}
|
|
|
|
func (c *DiscogsCache) FlushCache(ctx context.Context) error {
|
|
c.m.Lock()
|
|
defer c.m.Unlock()
|
|
records, err := c.fetchRecords(ctx, nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return c.saveRecordsToCache(ctx, records)
|
|
}
|
|
|
|
func (c *DiscogsCache) GetAllRecords(ctx context.Context) ([]media.Record, error) {
|
|
c.m.Lock()
|
|
defer c.m.Unlock()
|
|
if time.Now().After(c.lastRefresh.Add(c.maxCacheAge)) {
|
|
records, err := c.fetchRecords(ctx, nil)
|
|
if err != nil {
|
|
return c.cache, err
|
|
}
|
|
err = c.saveRecordsToCache(ctx, records)
|
|
return c.cache, err
|
|
}
|
|
return c.cache, nil
|
|
}
|
|
|
|
func (c *DiscogsCache) loadRecordsFromFS(ctx context.Context) ([]media.Record, time.Time, error) {
|
|
p := &persistence{}
|
|
f, err := os.Open(c.persistFile)
|
|
if errors.Is(err, os.ErrNotExist) {
|
|
log.Printf("%s not found, skipping file load...", c.persistFile)
|
|
return nil, time.Time{}, nil
|
|
}
|
|
if err != nil {
|
|
return nil, time.Time{}, fmt.Errorf("error opening cache file %s: %w", c.persistFile, err)
|
|
}
|
|
err = gob.NewDecoder(f).Decode(p)
|
|
if err != nil {
|
|
return nil, time.Time{}, fmt.Errorf("error readhing from cache file %s: %w", c.persistFile, err)
|
|
}
|
|
log.Printf("loaded %d records from %s", len(p.CachedRecordSlice), c.persistFile)
|
|
return p.CachedRecordSlice, p.LastRefresh, nil
|
|
}
|
|
|
|
func (c *DiscogsCache) saveRecordsToCache(ctx context.Context, records []media.Record) error {
|
|
c.cache = records
|
|
c.lastRefresh = time.Now()
|
|
if c.persistence && c.persistFile != "" {
|
|
p := persistence{
|
|
CachedRecordSlice: c.cache,
|
|
LastRefresh: c.lastRefresh,
|
|
}
|
|
f, err := os.OpenFile(c.persistFile, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0755)
|
|
if err != nil {
|
|
return fmt.Errorf("error opening cache file %s: %w", c.persistFile, err)
|
|
}
|
|
err = gob.NewEncoder(f).Encode(p)
|
|
if err != nil {
|
|
return fmt.Errorf("error writing to cache file %s: %w", c.persistFile, err)
|
|
}
|
|
log.Printf("wrote %d records to %s", len(c.cache), c.persistFile)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (c *DiscogsCache) fetchRecords(ctx context.Context, pagination *discogs.Pagination) ([]media.Record, error) {
|
|
records := []media.Record{}
|
|
if pagination == nil {
|
|
pagination = getPagination(1)
|
|
}
|
|
log.Printf("calling discogs API, page %v", pagination.Page)
|
|
coll, err := c.client.CollectionItemsByFolder(c.username, 0, pagination)
|
|
if err != nil {
|
|
return records, fmt.Errorf("error loading collection: %w", err)
|
|
}
|
|
log.Printf("length: %v, first item in list: %s", len(coll.Items), coll.Items[0].BasicInformation.Title)
|
|
for i := range coll.Items {
|
|
records = append(records, collectionItemToRecord(&coll.Items[i]))
|
|
}
|
|
// recurse down the list
|
|
if coll.Pagination.URLs.Next != "" {
|
|
coll, err := c.fetchRecords(ctx, getPagination(pagination.Page+1))
|
|
if err != nil {
|
|
return records, err
|
|
}
|
|
records = append(records, coll...)
|
|
}
|
|
return records, nil
|
|
}
|
|
|
|
func getPagination(page int) *discogs.Pagination {
|
|
return &discogs.Pagination{Page: page, Sort: "added", SortOrder: "asc", PerPage: 100}
|
|
}
|
|
|
|
func collectionItemToRecord(item *discogs.CollectionItemSource) media.Record {
|
|
artists := []string{}
|
|
for _, artist := range item.BasicInformation.Artists {
|
|
artists = append(artists, artist.Name)
|
|
}
|
|
year := strconv.Itoa(item.BasicInformation.Year)
|
|
if year == "0" {
|
|
year = ""
|
|
}
|
|
return media.Record{
|
|
ID: item.ID,
|
|
AlbumName: item.BasicInformation.Title,
|
|
Artists: artists,
|
|
Identifier: item.BasicInformation.Labels[0].Catno,
|
|
Format: item.BasicInformation.Formats[0].Name,
|
|
Genre: item.BasicInformation.Genres[0],
|
|
Label: item.BasicInformation.Labels[0].Name,
|
|
Year: year,
|
|
CoverURL: item.BasicInformation.CoverImage,
|
|
DiscogsURL: fmt.Sprintf("https://www.discogs.com/release/%v", item.ID),
|
|
}
|
|
}
|