package engine import ( "context" "fmt" "strconv" "time" "github.com/cloudsonic/sonic-server/domain" "github.com/cloudsonic/sonic-server/log" "github.com/cloudsonic/sonic-server/utils" ) type Browser interface { MediaFolders() (domain.MediaFolders, error) Indexes(ifModifiedSince time.Time) (domain.ArtistIndexes, time.Time, error) Directory(ctx context.Context, id string) (*DirectoryInfo, error) Artist(ctx context.Context, id string) (*DirectoryInfo, error) Album(ctx context.Context, id string) (*DirectoryInfo, error) GetSong(id string) (*Entry, error) } func NewBrowser(pr PropertyRepository, fr domain.MediaFolderRepository, ir domain.ArtistIndexRepository, ar domain.ArtistRepository, alr domain.AlbumRepository, mr domain.MediaFileRepository) Browser { return &browser{pr, fr, ir, ar, alr, mr} } type browser struct { propRepo PropertyRepository folderRepo domain.MediaFolderRepository indexRepo domain.ArtistIndexRepository artistRepo domain.ArtistRepository albumRepo domain.AlbumRepository mfileRepo domain.MediaFileRepository } func (b *browser) MediaFolders() (domain.MediaFolders, error) { return b.folderRepo.GetAll() } func (b *browser) Indexes(ifModifiedSince time.Time) (domain.ArtistIndexes, time.Time, error) { l, err := b.propRepo.DefaultGet(PropLastScan, "-1") ms, _ := strconv.ParseInt(l, 10, 64) lastModified := utils.ToTime(ms) if err != nil { return nil, time.Time{}, fmt.Errorf("error retrieving LastScan property: %v", err) } if lastModified.After(ifModifiedSince) { indexes, err := b.indexRepo.GetAll() return indexes, lastModified, err } return nil, lastModified, nil } type DirectoryInfo struct { Id string Name string Entries Entries Parent string Starred time.Time PlayCount int32 UserRating int AlbumCount int CoverArt string Artist string ArtistId string SongCount int Duration int Created time.Time Year int Genre string } func (b *browser) Artist(ctx context.Context, id string) (*DirectoryInfo, error) { a, albums, err := b.retrieveArtist(id) if err != nil { return nil, err } log.Debug(ctx, "Found Artist", "id", id, "name", a.Name) return b.buildArtistDir(a, albums), nil } func (b *browser) Album(ctx context.Context, id string) (*DirectoryInfo, error) { al, tracks, err := b.retrieveAlbum(id) if err != nil { return nil, err } log.Debug(ctx, "Found Album", "id", id, "name", al.Name) return b.buildAlbumDir(al, tracks), nil } func (b *browser) Directory(ctx context.Context, id string) (*DirectoryInfo, error) { switch { case b.isArtist(ctx, id): return b.Artist(ctx, id) case b.isAlbum(ctx, id): return b.Album(ctx, id) default: log.Debug(ctx, "Directory not found", "id", id) return nil, domain.ErrNotFound } } func (b *browser) GetSong(id string) (*Entry, error) { mf, err := b.mfileRepo.Get(id) if err != nil { return nil, err } entry := FromMediaFile(mf) return &entry, nil } func (b *browser) buildArtistDir(a *domain.Artist, albums domain.Albums) *DirectoryInfo { dir := &DirectoryInfo{ Id: a.Id, Name: a.Name, AlbumCount: a.AlbumCount, } dir.Entries = make(Entries, len(albums)) for i, al := range albums { dir.Entries[i] = FromAlbum(&al) dir.PlayCount += int32(al.PlayCount) } return dir } func (b *browser) buildAlbumDir(al *domain.Album, tracks domain.MediaFiles) *DirectoryInfo { dir := &DirectoryInfo{ Id: al.Id, Name: al.Name, Parent: al.ArtistId, PlayCount: int32(al.PlayCount), UserRating: al.Rating, Starred: al.StarredAt, Artist: al.Artist, ArtistId: al.ArtistId, SongCount: al.SongCount, Duration: al.Duration, Created: al.CreatedAt, Year: al.Year, Genre: al.Genre, CoverArt: al.CoverArtId, } dir.Entries = make(Entries, len(tracks)) for i, mf := range tracks { dir.Entries[i] = FromMediaFile(&mf) } return dir } func (b *browser) isArtist(ctx context.Context, id string) bool { found, err := b.artistRepo.Exists(id) if err != nil { log.Debug(ctx, "Error searching for Artist", "id", id, err) return false } return found } func (b *browser) isAlbum(ctx context.Context, id string) bool { found, err := b.albumRepo.Exists(id) if err != nil { log.Debug(ctx, "Error searching for Album", "id", id, err) return false } return found } func (b *browser) retrieveArtist(id string) (a *domain.Artist, as domain.Albums, err error) { a, err = b.artistRepo.Get(id) if err != nil { err = fmt.Errorf("Error reading Artist %s from DB: %v", id, err) return } if as, err = b.albumRepo.FindByArtist(id); err != nil { err = fmt.Errorf("Error reading %s's albums from DB: %v", a.Name, err) } return } func (b *browser) retrieveAlbum(id string) (al *domain.Album, mfs domain.MediaFiles, err error) { al, err = b.albumRepo.Get(id) if err != nil { err = fmt.Errorf("Error reading Album %s from DB: %v", id, err) return } if mfs, err = b.mfileRepo.FindByAlbum(id); err != nil { err = fmt.Errorf("Error reading %s's tracks from DB: %v", al.Name, err) } return }