mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 13:07:36 +03:00
281 lines
7.2 KiB
Go
281 lines
7.2 KiB
Go
package subsonic
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"mime"
|
|
"net/http"
|
|
|
|
"github.com/deluan/navidrome/consts"
|
|
"github.com/deluan/navidrome/model"
|
|
"github.com/deluan/navidrome/model/request"
|
|
"github.com/deluan/navidrome/server/subsonic/engine"
|
|
"github.com/deluan/navidrome/server/subsonic/responses"
|
|
"github.com/deluan/navidrome/utils"
|
|
)
|
|
|
|
func newResponse() *responses.Subsonic {
|
|
return &responses.Subsonic{Status: "ok", Version: Version, Type: consts.AppName, ServerVersion: consts.Version()}
|
|
}
|
|
|
|
func requiredParamString(r *http.Request, param string, msg string) (string, error) {
|
|
p := utils.ParamString(r, param)
|
|
if p == "" {
|
|
return "", newError(responses.ErrorMissingParameter, msg)
|
|
}
|
|
return p, nil
|
|
}
|
|
|
|
func requiredParamStrings(r *http.Request, param string, msg string) ([]string, error) {
|
|
ps := utils.ParamStrings(r, param)
|
|
if len(ps) == 0 {
|
|
return nil, newError(responses.ErrorMissingParameter, msg)
|
|
}
|
|
return ps, nil
|
|
}
|
|
|
|
func requiredParamInt(r *http.Request, param string, msg string) (int, error) {
|
|
p := utils.ParamString(r, param)
|
|
if p == "" {
|
|
return 0, newError(responses.ErrorMissingParameter, msg)
|
|
}
|
|
return utils.ParamInt(r, param, 0), nil
|
|
}
|
|
|
|
type SubsonicError struct {
|
|
code int
|
|
messages []interface{}
|
|
}
|
|
|
|
func newError(code int, message ...interface{}) error {
|
|
return SubsonicError{
|
|
code: code,
|
|
messages: message,
|
|
}
|
|
}
|
|
|
|
func (e SubsonicError) Error() string {
|
|
var msg string
|
|
if len(e.messages) == 0 {
|
|
msg = responses.ErrorMsg(e.code)
|
|
} else {
|
|
msg = fmt.Sprintf(e.messages[0].(string), e.messages[1:]...)
|
|
}
|
|
return msg
|
|
}
|
|
|
|
func toAlbums(ctx context.Context, entries engine.Entries) []responses.Child {
|
|
children := make([]responses.Child, len(entries))
|
|
for i, entry := range entries {
|
|
children[i] = toAlbum(ctx, entry)
|
|
}
|
|
return children
|
|
}
|
|
|
|
func toAlbum(ctx context.Context, entry engine.Entry) responses.Child {
|
|
album := toChild(ctx, entry)
|
|
album.Name = album.Title
|
|
album.Title = ""
|
|
album.Parent = ""
|
|
album.Album = ""
|
|
album.AlbumId = ""
|
|
return album
|
|
}
|
|
|
|
func toArtists(entries engine.Entries) []responses.Artist {
|
|
artists := make([]responses.Artist, len(entries))
|
|
for i, entry := range entries {
|
|
artists[i] = responses.Artist{
|
|
Id: entry.Id,
|
|
Name: entry.Title,
|
|
AlbumCount: entry.AlbumCount,
|
|
}
|
|
if !entry.Starred.IsZero() {
|
|
artists[i].Starred = &entry.Starred
|
|
}
|
|
}
|
|
return artists
|
|
}
|
|
|
|
func toChildren(ctx context.Context, entries engine.Entries) []responses.Child {
|
|
children := make([]responses.Child, len(entries))
|
|
for i, entry := range entries {
|
|
children[i] = toChild(ctx, entry)
|
|
}
|
|
return children
|
|
}
|
|
|
|
func toChild(ctx context.Context, entry engine.Entry) responses.Child {
|
|
child := responses.Child{}
|
|
child.Id = entry.Id
|
|
child.Title = entry.Title
|
|
child.IsDir = entry.IsDir
|
|
child.Parent = entry.Parent
|
|
child.Album = entry.Album
|
|
child.Year = entry.Year
|
|
child.Artist = entry.Artist
|
|
child.Genre = entry.Genre
|
|
child.CoverArt = entry.CoverArt
|
|
child.Track = entry.Track
|
|
child.Duration = entry.Duration
|
|
child.Size = entry.Size
|
|
child.Suffix = entry.Suffix
|
|
child.BitRate = entry.BitRate
|
|
child.ContentType = entry.ContentType
|
|
if !entry.Starred.IsZero() {
|
|
child.Starred = &entry.Starred
|
|
}
|
|
child.Path = entry.Path
|
|
child.PlayCount = int64(entry.PlayCount)
|
|
child.DiscNumber = entry.DiscNumber
|
|
if !entry.Created.IsZero() {
|
|
child.Created = &entry.Created
|
|
}
|
|
child.AlbumId = entry.AlbumId
|
|
child.ArtistId = entry.ArtistId
|
|
child.Type = entry.Type
|
|
child.IsVideo = false
|
|
child.UserRating = entry.UserRating
|
|
child.SongCount = entry.SongCount
|
|
format, _ := getTranscoding(ctx)
|
|
if entry.Suffix != "" && format != "" && entry.Suffix != format {
|
|
child.TranscodedSuffix = format
|
|
child.TranscodedContentType = mime.TypeByExtension("." + format)
|
|
}
|
|
child.BookmarkPosition = entry.BookmarkPosition
|
|
return child
|
|
}
|
|
|
|
func toGenres(genres model.Genres) *responses.Genres {
|
|
response := make([]responses.Genre, len(genres))
|
|
for i, g := range genres {
|
|
response[i] = responses.Genre(g)
|
|
}
|
|
return &responses.Genres{Genre: response}
|
|
}
|
|
|
|
func getTranscoding(ctx context.Context) (format string, bitRate int) {
|
|
if trc, ok := request.TranscodingFrom(ctx); ok {
|
|
format = trc.TargetFormat
|
|
}
|
|
if plr, ok := request.PlayerFrom(ctx); ok {
|
|
bitRate = plr.MaxBitRate
|
|
}
|
|
return
|
|
}
|
|
|
|
// This seems to be duplicated, but it is an initial step into merging `engine` and the `subsonic` packages,
|
|
// In the future there won't be any conversion to/from `engine. Entry` anymore
|
|
func childFromMediaFile(ctx context.Context, mf model.MediaFile) responses.Child {
|
|
child := responses.Child{}
|
|
child.Id = mf.ID
|
|
child.Title = mf.Title
|
|
child.IsDir = false
|
|
child.Parent = mf.AlbumID
|
|
child.Album = mf.Album
|
|
child.Year = mf.Year
|
|
child.Artist = mf.Artist
|
|
child.Genre = mf.Genre
|
|
child.Track = mf.TrackNumber
|
|
child.Duration = int(mf.Duration)
|
|
child.Size = mf.Size
|
|
child.Suffix = mf.Suffix
|
|
child.BitRate = mf.BitRate
|
|
if mf.HasCoverArt {
|
|
child.CoverArt = mf.ID
|
|
} else {
|
|
child.CoverArt = "al-" + mf.AlbumID
|
|
}
|
|
child.ContentType = mf.ContentType()
|
|
child.Path = fmt.Sprintf("%s/%s/%s.%s", realArtistName(mf), mf.Album, mf.Title, mf.Suffix)
|
|
child.DiscNumber = mf.DiscNumber
|
|
child.Created = &mf.CreatedAt
|
|
child.AlbumId = mf.AlbumID
|
|
child.ArtistId = mf.ArtistID
|
|
child.Type = "music"
|
|
child.PlayCount = mf.PlayCount
|
|
if mf.Starred {
|
|
child.Starred = &mf.StarredAt
|
|
}
|
|
child.UserRating = mf.Rating
|
|
|
|
format, _ := getTranscoding(ctx)
|
|
if mf.Suffix != "" && format != "" && mf.Suffix != format {
|
|
child.TranscodedSuffix = format
|
|
child.TranscodedContentType = mime.TypeByExtension("." + format)
|
|
}
|
|
child.BookmarkPosition = mf.BookmarkPosition
|
|
return child
|
|
}
|
|
|
|
func realArtistName(mf model.MediaFile) string {
|
|
switch {
|
|
case mf.Compilation:
|
|
return consts.VariousArtists
|
|
case mf.AlbumArtist != "":
|
|
return mf.AlbumArtist
|
|
}
|
|
|
|
return mf.Artist
|
|
}
|
|
|
|
func childrenFromMediaFiles(ctx context.Context, mfs model.MediaFiles) []responses.Child {
|
|
children := make([]responses.Child, len(mfs))
|
|
for i, mf := range mfs {
|
|
children[i] = childFromMediaFile(ctx, mf)
|
|
}
|
|
return children
|
|
}
|
|
|
|
func childFromAlbum(ctx context.Context, al model.Album) responses.Child {
|
|
child := responses.Child{}
|
|
child.Id = al.ID
|
|
child.IsDir = true
|
|
child.Title = al.Name
|
|
child.Name = al.Name
|
|
child.Album = al.Name
|
|
child.Artist = al.AlbumArtist
|
|
child.Year = al.MaxYear
|
|
child.Genre = al.Genre
|
|
child.CoverArt = al.CoverArtId
|
|
child.Created = &al.CreatedAt
|
|
child.Parent = al.AlbumArtistID
|
|
child.ArtistId = al.AlbumArtistID
|
|
child.Duration = int(al.Duration)
|
|
child.SongCount = al.SongCount
|
|
if al.Starred {
|
|
child.Starred = &al.StarredAt
|
|
}
|
|
child.PlayCount = al.PlayCount
|
|
child.UserRating = al.Rating
|
|
return child
|
|
}
|
|
|
|
func childrenFromAlbums(ctx context.Context, als model.Albums) []responses.Child {
|
|
children := make([]responses.Child, len(als))
|
|
for i, al := range als {
|
|
children[i] = childFromAlbum(ctx, al)
|
|
}
|
|
return children
|
|
}
|
|
|
|
// TODO: Should the type be encoded in the ID?
|
|
func getEntityByID(ctx context.Context, ds model.DataStore, id string) (interface{}, error) {
|
|
ar, err := ds.Artist(ctx).Get(id)
|
|
if err == nil {
|
|
return ar, nil
|
|
}
|
|
al, err := ds.Album(ctx).Get(id)
|
|
if err == nil {
|
|
return al, nil
|
|
}
|
|
pls, err := ds.Playlist(ctx).Get(id)
|
|
if err == nil {
|
|
return pls, nil
|
|
}
|
|
mf, err := ds.MediaFile(ctx).Get(id)
|
|
if err == nil {
|
|
return mf, nil
|
|
}
|
|
return nil, err
|
|
}
|