navidrome/persistence/mediafile_repository.go

160 lines
4.4 KiB
Go
Raw Normal View History

package persistence
2020-01-13 00:36:19 +01:00
import (
"context"
2020-01-31 23:56:02 +01:00
"os"
"strings"
2020-01-13 00:36:19 +01:00
. "github.com/Masterminds/squirrel"
2020-01-13 00:36:19 +01:00
"github.com/astaxie/beego/orm"
2020-01-31 23:56:02 +01:00
"github.com/deluan/navidrome/log"
2020-01-24 01:44:08 +01:00
"github.com/deluan/navidrome/model"
"github.com/deluan/rest"
2020-01-13 00:36:19 +01:00
)
type mediaFileRepository struct {
sqlRepository
2020-01-13 00:36:19 +01:00
}
func NewMediaFileRepository(ctx context.Context, o orm.Ormer) *mediaFileRepository {
2020-01-13 00:36:19 +01:00
r := &mediaFileRepository{}
r.ctx = ctx
r.ormer = o
2020-01-13 06:04:11 +01:00
r.tableName = "media_file"
2020-02-05 20:12:13 +01:00
r.sortMappings = map[string]string{
"artist": "artist asc, album asc, disc_number asc, track_number asc",
"album": "album asc, disc_number asc, track_number asc",
}
r.filterMappings = map[string]filterFunc{
"title": fullTextFilter,
}
2020-01-13 00:36:19 +01:00
return r
}
func (r mediaFileRepository) CountAll(options ...model.QueryOptions) (int64, error) {
return r.count(Select(), options...)
2020-01-13 00:36:19 +01:00
}
func (r mediaFileRepository) Exists(id string) (bool, error) {
return r.exists(Select().Where(Eq{"id": id}))
}
func (r mediaFileRepository) Put(m *model.MediaFile) error {
m.FullText = r.getFullText(m.Title, m.Album, m.Artist, m.AlbumArtist)
_, err := r.put(m.ID, m)
return err
2020-01-13 00:36:19 +01:00
}
func (r mediaFileRepository) selectMediaFile(options ...model.QueryOptions) SelectBuilder {
2020-02-01 03:09:23 +01:00
return r.newSelectWithAnnotation("media_file.id", options...).Columns("media_file.*")
2020-01-13 00:36:19 +01:00
}
func (r mediaFileRepository) Get(id string) (*model.MediaFile, error) {
sel := r.selectMediaFile().Where(Eq{"id": id})
var res model.MediaFile
err := r.queryOne(sel, &res)
return &res, err
2020-01-13 00:36:19 +01:00
}
func (r mediaFileRepository) GetAll(options ...model.QueryOptions) (model.MediaFiles, error) {
sq := r.selectMediaFile(options...)
2020-01-31 22:47:13 +01:00
res := model.MediaFiles{}
err := r.queryAll(sq, &res)
return res, err
2020-01-16 21:56:24 +01:00
}
func (r mediaFileRepository) FindByAlbum(albumId string) (model.MediaFiles, error) {
sel := r.selectMediaFile().Where(Eq{"album_id": albumId}).OrderBy("disc_number", "track_number")
2020-01-31 22:47:13 +01:00
res := model.MediaFiles{}
err := r.queryAll(sel, &res)
return res, err
2020-01-16 22:53:48 +01:00
}
func (r mediaFileRepository) FindByPath(path string) (model.MediaFiles, error) {
sel := r.selectMediaFile().Where(Like{"path": path + "%"})
2020-01-31 22:47:13 +01:00
res := model.MediaFiles{}
err := r.queryAll(sel, &res)
2020-01-31 23:56:02 +01:00
if err != nil {
return nil, err
}
// Only return mediafiles that are direct child of requested path
filtered := model.MediaFiles{}
path = strings.ToLower(path) + string(os.PathSeparator)
for _, mf := range res {
filename := strings.TrimPrefix(strings.ToLower(mf.Path), path)
if len(strings.Split(filename, string(os.PathSeparator))) > 1 {
continue
}
filtered = append(filtered, mf)
}
return filtered, nil
}
func (r mediaFileRepository) GetStarred(options ...model.QueryOptions) (model.MediaFiles, error) {
sq := r.selectMediaFile(options...).Where("starred = true")
2020-01-31 22:47:13 +01:00
starred := model.MediaFiles{}
err := r.queryAll(sq, &starred)
return starred, err
}
// TODO Keep order when paginating
func (r mediaFileRepository) GetRandom(options ...model.QueryOptions) (model.MediaFiles, error) {
sq := r.selectMediaFile(options...)
sq = sq.OrderBy("RANDOM()")
2020-01-31 22:47:13 +01:00
results := model.MediaFiles{}
err := r.queryAll(sq, &results)
return results, err
}
func (r mediaFileRepository) Delete(id string) error {
return r.delete(Eq{"id": id})
}
func (r mediaFileRepository) DeleteByPath(path string) error {
2020-01-31 23:56:02 +01:00
filtered, err := r.FindByPath(path)
if err != nil {
return err
}
if len(filtered) == 0 {
return nil
}
ids := make([]string, len(filtered))
for i, mf := range filtered {
ids[i] = mf.ID
}
log.Debug(r.ctx, "Deleting mediafiles by path", "path", path, "totalDeleted", len(ids))
del := Delete(r.tableName).Where(Eq{"id": ids})
_, err = r.executeSQL(del)
return err
}
func (r mediaFileRepository) Search(q string, offset int, size int) (model.MediaFiles, error) {
2020-01-31 22:47:13 +01:00
results := model.MediaFiles{}
err := r.doSearch(q, offset, size, &results, "title")
return results, err
}
func (r mediaFileRepository) Count(options ...rest.QueryOptions) (int64, error) {
return r.CountAll(r.parseRestOptions(options...))
}
func (r mediaFileRepository) Read(id string) (interface{}, error) {
return r.Get(id)
}
func (r mediaFileRepository) ReadAll(options ...rest.QueryOptions) (interface{}, error) {
return r.GetAll(r.parseRestOptions(options...))
}
func (r mediaFileRepository) EntityName() string {
return "mediafile"
}
func (r mediaFileRepository) NewInstance() interface{} {
return model.MediaFile{}
}
2020-01-15 04:22:34 +01:00
var _ model.MediaFileRepository = (*mediaFileRepository)(nil)
var _ model.ResourceRepository = (*mediaFileRepository)(nil)