mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-14 19:20:37 +03:00
146 lines
3.9 KiB
Go
146 lines
3.9 KiB
Go
package persistence
|
|
|
|
import (
|
|
"context"
|
|
"sort"
|
|
"strings"
|
|
|
|
. "github.com/Masterminds/squirrel"
|
|
"github.com/astaxie/beego/orm"
|
|
"github.com/deluan/navidrome/conf"
|
|
"github.com/deluan/navidrome/model"
|
|
"github.com/deluan/navidrome/utils"
|
|
"github.com/deluan/rest"
|
|
)
|
|
|
|
type artistRepository struct {
|
|
sqlRepository
|
|
indexGroups utils.IndexGroups
|
|
}
|
|
|
|
func NewArtistRepository(ctx context.Context, o orm.Ormer) model.ArtistRepository {
|
|
r := &artistRepository{}
|
|
r.ctx = ctx
|
|
r.ormer = o
|
|
r.indexGroups = utils.ParseIndexGroups(conf.Server.IndexGroups)
|
|
r.tableName = "media_file"
|
|
return r
|
|
}
|
|
|
|
func (r *artistRepository) selectArtist(options ...model.QueryOptions) SelectBuilder {
|
|
// FIXME Handle AlbumArtist/Various Artists...
|
|
return r.newSelectWithAnnotation(model.ArtistItemType, "album_id", options...).
|
|
Columns("artist_id as id", "artist as name", "count(distinct album_id) as album_count").
|
|
GroupBy("artist_id").Where(Eq{"compilation": false})
|
|
}
|
|
|
|
func (r *artistRepository) CountAll(options ...model.QueryOptions) (int64, error) {
|
|
sel := r.selectArtist(options...).Where(Eq{"compilation": false})
|
|
return r.count(sel, options...)
|
|
}
|
|
|
|
func (r *artistRepository) Exists(id string) (bool, error) {
|
|
return r.exists(Select().Where(Eq{"artist_id": id}))
|
|
}
|
|
|
|
func (r *artistRepository) getIndexKey(a *model.Artist) string {
|
|
name := strings.ToLower(utils.NoArticle(a.Name))
|
|
for k, v := range r.indexGroups {
|
|
key := strings.ToLower(k)
|
|
if strings.HasPrefix(name, key) {
|
|
return v
|
|
}
|
|
}
|
|
return "#"
|
|
}
|
|
|
|
func (r *artistRepository) Put(a *model.Artist) error {
|
|
return nil
|
|
}
|
|
|
|
func (r *artistRepository) Get(id string) (*model.Artist, error) {
|
|
sel := Select("artist_id as id", "artist as name", "count(distinct album_id) as album_count").
|
|
From("media_file").GroupBy("artist_id").Where(Eq{"artist_id": id})
|
|
var res model.Artist
|
|
err := r.queryOne(sel, &res)
|
|
return &res, err
|
|
}
|
|
|
|
func (r *artistRepository) GetAll(options ...model.QueryOptions) (model.Artists, error) {
|
|
sel := r.selectArtist(options...)
|
|
var res model.Artists
|
|
err := r.queryAll(sel, &res)
|
|
return res, err
|
|
}
|
|
|
|
// TODO Cache the index (recalculate when there are changes to the DB)
|
|
func (r *artistRepository) GetIndex() (model.ArtistIndexes, error) {
|
|
sq := Select("artist_id as id", "artist as name", "count(distinct album_id) as album_count").
|
|
From("media_file").GroupBy("artist_id").OrderBy("name")
|
|
var all model.Artists
|
|
// TODO Paginate
|
|
err := r.queryAll(sq, &all)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
fullIdx := make(map[string]*model.ArtistIndex)
|
|
for _, a := range all {
|
|
ax := r.getIndexKey(&a)
|
|
idx, ok := fullIdx[ax]
|
|
if !ok {
|
|
idx = &model.ArtistIndex{ID: ax}
|
|
fullIdx[ax] = idx
|
|
}
|
|
idx.Artists = append(idx.Artists, model.Artist(a))
|
|
}
|
|
var result model.ArtistIndexes
|
|
for _, idx := range fullIdx {
|
|
result = append(result, *idx)
|
|
}
|
|
sort.Slice(result, func(i, j int) bool {
|
|
return result[i].ID < result[j].ID
|
|
})
|
|
return result, nil
|
|
}
|
|
|
|
func (r *artistRepository) Refresh(ids ...string) error {
|
|
return nil
|
|
}
|
|
|
|
func (r *artistRepository) GetStarred(options ...model.QueryOptions) (model.Artists, error) {
|
|
return nil, nil // TODO
|
|
}
|
|
|
|
func (r *artistRepository) PurgeEmpty() error {
|
|
return nil
|
|
}
|
|
|
|
func (r *artistRepository) Search(q string, offset int, size int) (model.Artists, error) {
|
|
return nil, nil // TODO
|
|
}
|
|
|
|
func (r *artistRepository) Count(options ...rest.QueryOptions) (int64, error) {
|
|
return r.CountAll(r.parseRestOptions(options...)) // TODO Don't apply pagination!
|
|
}
|
|
|
|
func (r *artistRepository) Read(id string) (interface{}, error) {
|
|
return r.Get(id)
|
|
}
|
|
|
|
func (r *artistRepository) ReadAll(options ...rest.QueryOptions) (interface{}, error) {
|
|
return r.GetAll(r.parseRestOptions(options...))
|
|
}
|
|
|
|
func (r *artistRepository) EntityName() string {
|
|
return "artist"
|
|
}
|
|
|
|
func (r *artistRepository) NewInstance() interface{} {
|
|
return &model.Artist{}
|
|
}
|
|
|
|
var _ model.ArtistRepository = (*artistRepository)(nil)
|
|
var _ model.ArtistRepository = (*artistRepository)(nil)
|
|
var _ model.ResourceRepository = (*artistRepository)(nil)
|