mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 13:07:36 +03:00
Expose OpenSubsonic fields Genres, MusicBrainzId, Bpm, Comment (#2597)
* add Genres to subsonic responses * add genres in GetAlbum response * add musicBrainzId * add Bpm and Comment OpenSubsonic fields * remove omitempty on OpenSubsonic fields * add custom JSON marshalers to ensure genres attribute is non-nil * regenerate snapshots to capture now-mandatory fields
This commit is contained in:
parent
53607fe114
commit
7773522803
21 changed files with 118 additions and 63 deletions
|
@ -423,6 +423,7 @@ func (api *Router) buildAlbum(ctx context.Context, album *model.Album, mfs model
|
||||||
}
|
}
|
||||||
dir.Year = int32(album.MaxYear)
|
dir.Year = int32(album.MaxYear)
|
||||||
dir.Genre = album.Genre
|
dir.Genre = album.Genre
|
||||||
|
dir.Genres = itemGenresFromGenres(album.Genres)
|
||||||
dir.UserRating = int32(album.Rating)
|
dir.UserRating = int32(album.Rating)
|
||||||
if !album.CreatedAt.IsZero() {
|
if !album.CreatedAt.IsZero() {
|
||||||
dir.Created = &album.CreatedAt
|
dir.Created = &album.CreatedAt
|
||||||
|
@ -430,6 +431,7 @@ func (api *Router) buildAlbum(ctx context.Context, album *model.Album, mfs model
|
||||||
if album.Starred {
|
if album.Starred {
|
||||||
dir.Starred = &album.StarredAt
|
dir.Starred = &album.StarredAt
|
||||||
}
|
}
|
||||||
|
dir.MusicBrainzId = album.MbzAlbumID
|
||||||
dir.Song = childrenFromMediaFiles(ctx, mfs)
|
dir.Song = childrenFromMediaFiles(ctx, mfs)
|
||||||
return dir
|
return dir
|
||||||
}
|
}
|
||||||
|
|
|
@ -110,6 +110,7 @@ func toArtistID3(r *http.Request, a model.Artist) responses.ArtistID3 {
|
||||||
CoverArt: a.CoverArtID().String(),
|
CoverArt: a.CoverArtID().String(),
|
||||||
ArtistImageUrl: public.ImageURL(r, a.CoverArtID(), 600),
|
ArtistImageUrl: public.ImageURL(r, a.CoverArtID(), 600),
|
||||||
UserRating: int32(a.Rating),
|
UserRating: int32(a.Rating),
|
||||||
|
MusicBrainzId: a.MbzArtistID,
|
||||||
}
|
}
|
||||||
if a.Starred {
|
if a.Starred {
|
||||||
artist.Starred = &a.StarredAt
|
artist.Starred = &a.StarredAt
|
||||||
|
@ -151,6 +152,7 @@ func childFromMediaFile(ctx context.Context, mf model.MediaFile) responses.Child
|
||||||
child.Year = int32(mf.Year)
|
child.Year = int32(mf.Year)
|
||||||
child.Artist = mf.Artist
|
child.Artist = mf.Artist
|
||||||
child.Genre = mf.Genre
|
child.Genre = mf.Genre
|
||||||
|
child.Genres = itemGenresFromGenres(mf.Genres)
|
||||||
child.Track = int32(mf.TrackNumber)
|
child.Track = int32(mf.TrackNumber)
|
||||||
child.Duration = int32(mf.Duration)
|
child.Duration = int32(mf.Duration)
|
||||||
child.Size = mf.Size
|
child.Size = mf.Size
|
||||||
|
@ -184,6 +186,8 @@ func childFromMediaFile(ctx context.Context, mf model.MediaFile) responses.Child
|
||||||
child.TranscodedContentType = mime.TypeByExtension("." + format)
|
child.TranscodedContentType = mime.TypeByExtension("." + format)
|
||||||
}
|
}
|
||||||
child.BookmarkPosition = mf.BookmarkPosition
|
child.BookmarkPosition = mf.BookmarkPosition
|
||||||
|
child.Comment = mf.Comment
|
||||||
|
child.Bpm = int32(mf.Bpm)
|
||||||
return child
|
return child
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -217,6 +221,7 @@ func childFromAlbum(_ context.Context, al model.Album) responses.Child {
|
||||||
child.Artist = al.AlbumArtist
|
child.Artist = al.AlbumArtist
|
||||||
child.Year = int32(al.MaxYear)
|
child.Year = int32(al.MaxYear)
|
||||||
child.Genre = al.Genre
|
child.Genre = al.Genre
|
||||||
|
child.Genres = itemGenresFromGenres(al.Genres)
|
||||||
child.CoverArt = al.CoverArtID().String()
|
child.CoverArt = al.CoverArtID().String()
|
||||||
child.Created = &al.CreatedAt
|
child.Created = &al.CreatedAt
|
||||||
child.Parent = al.AlbumArtistID
|
child.Parent = al.AlbumArtistID
|
||||||
|
@ -241,3 +246,11 @@ func childrenFromAlbums(ctx context.Context, als model.Albums) []responses.Child
|
||||||
}
|
}
|
||||||
return children
|
return children
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func itemGenresFromGenres(genres model.Genres) []responses.ItemGenre {
|
||||||
|
itemGenres := make([]responses.ItemGenre, len(genres))
|
||||||
|
for i, g := range genres {
|
||||||
|
itemGenres[i] = responses.ItemGenre{Name: g.Name}
|
||||||
|
}
|
||||||
|
return itemGenres
|
||||||
|
}
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"albumList":{"album":[{"id":"1","isDir":false,"title":"title","isVideo":false}]}}
|
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"albumList":{"album":[{"id":"1","isDir":false,"title":"title","genres":[],"isVideo":false,"bpm":0,"comment":""}]}}
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><albumList><album id="1" isDir="false" title="title" isVideo="false"></album></albumList></subsonic-response>
|
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><albumList><album id="1" isDir="false" title="title" isVideo="false" bpm="0" comment=""></album></albumList></subsonic-response>
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"bookmarks":{"bookmark":[{"entry":{"id":"1","isDir":false,"title":"title","isVideo":false},"position":123,"username":"user2","comment":"a comment","created":"0001-01-01T00:00:00Z","changed":"0001-01-01T00:00:00Z"}]}}
|
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"bookmarks":{"bookmark":[{"entry":{"id":"1","isDir":false,"title":"title","genres":[],"isVideo":false,"bpm":0,"comment":""},"position":123,"username":"user2","comment":"a comment","created":"0001-01-01T00:00:00Z","changed":"0001-01-01T00:00:00Z"}]}}
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><bookmarks><bookmark position="123" username="user2" comment="a comment" created="0001-01-01T00:00:00Z" changed="0001-01-01T00:00:00Z"><entry id="1" isDir="false" title="title" isVideo="false"></entry></bookmark></bookmarks></subsonic-response>
|
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><bookmarks><bookmark position="123" username="user2" comment="a comment" created="0001-01-01T00:00:00Z" changed="0001-01-01T00:00:00Z"><entry id="1" isDir="false" title="title" isVideo="false" bpm="0" comment=""></entry></bookmark></bookmarks></subsonic-response>
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"directory":{"child":[{"id":"1","isDir":true,"title":"title","album":"album","artist":"artist","track":1,"year":1985,"genre":"Rock","coverArt":"1","size":8421341,"contentType":"audio/flac","suffix":"flac","starred":"2016-03-02T20:30:00Z","transcodedContentType":"audio/mpeg","transcodedSuffix":"mp3","duration":146,"bitRate":320,"isVideo":false}],"id":"1","name":"N"}}
|
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"directory":{"child":[{"id":"1","isDir":true,"title":"title","album":"album","artist":"artist","track":1,"year":1985,"genre":"Rock","genres":[],"coverArt":"1","size":8421341,"contentType":"audio/flac","suffix":"flac","starred":"2016-03-02T20:30:00Z","transcodedContentType":"audio/mpeg","transcodedSuffix":"mp3","duration":146,"bitRate":320,"isVideo":false,"bpm":0,"comment":""}],"id":"1","name":"N"}}
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><directory id="1" name="N"><child id="1" isDir="true" title="title" album="album" artist="artist" track="1" year="1985" genre="Rock" coverArt="1" size="8421341" contentType="audio/flac" suffix="flac" starred="2016-03-02T20:30:00Z" transcodedContentType="audio/mpeg" transcodedSuffix="mp3" duration="146" bitRate="320" isVideo="false"></child></directory></subsonic-response>
|
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><directory id="1" name="N"><child id="1" isDir="true" title="title" album="album" artist="artist" track="1" year="1985" genre="Rock" coverArt="1" size="8421341" contentType="audio/flac" suffix="flac" starred="2016-03-02T20:30:00Z" transcodedContentType="audio/mpeg" transcodedSuffix="mp3" duration="146" bitRate="320" isVideo="false" bpm="0" comment=""></child></directory></subsonic-response>
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"directory":{"child":[{"id":"1","isDir":false,"title":"title","isVideo":false}],"id":"1","name":"N"}}
|
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"directory":{"child":[{"id":"1","isDir":false,"title":"title","genres":[],"isVideo":false,"bpm":0,"comment":""}],"id":"1","name":"N"}}
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><directory id="1" name="N"><child id="1" isDir="false" title="title" isVideo="false"></child></directory></subsonic-response>
|
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><directory id="1" name="N"><child id="1" isDir="false" title="title" isVideo="false" bpm="0" comment=""></child></directory></subsonic-response>
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"playQueue":{"entry":[{"id":"1","isDir":false,"title":"title","isVideo":false}],"current":"111","position":243,"username":"user1","changed":"0001-01-01T00:00:00Z","changedBy":"a_client"}}
|
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"playQueue":{"entry":[{"id":"1","isDir":false,"title":"title","genres":[],"isVideo":false,"bpm":0,"comment":""}],"current":"111","position":243,"username":"user1","changed":"0001-01-01T00:00:00Z","changedBy":"a_client"}}
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><playQueue current="111" position="243" username="user1" changed="0001-01-01T00:00:00Z" changedBy="a_client"><entry id="1" isDir="false" title="title" isVideo="false"></entry></playQueue></subsonic-response>
|
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><playQueue current="111" position="243" username="user1" changed="0001-01-01T00:00:00Z" changedBy="a_client"><entry id="1" isDir="false" title="title" isVideo="false" bpm="0" comment=""></entry></playQueue></subsonic-response>
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"shares":{"share":[{"entry":[{"id":"1","isDir":false,"title":"title","album":"album","artist":"artist","duration":120,"isVideo":false},{"id":"2","isDir":false,"title":"title 2","album":"album","artist":"artist","duration":300,"isVideo":false}],"id":"ABC123","url":"http://localhost/p/ABC123","description":"Check it out!","username":"deluan","created":"0001-01-01T00:00:00Z","expires":"0001-01-01T00:00:00Z","lastVisited":"0001-01-01T00:00:00Z","visitCount":2}]}}
|
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"shares":{"share":[{"entry":[{"id":"1","isDir":false,"title":"title","album":"album","artist":"artist","genres":[],"duration":120,"isVideo":false,"bpm":0,"comment":""},{"id":"2","isDir":false,"title":"title 2","album":"album","artist":"artist","genres":[],"duration":300,"isVideo":false,"bpm":0,"comment":""}],"id":"ABC123","url":"http://localhost/p/ABC123","description":"Check it out!","username":"deluan","created":"0001-01-01T00:00:00Z","expires":"0001-01-01T00:00:00Z","lastVisited":"0001-01-01T00:00:00Z","visitCount":2}]}}
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><shares><share id="ABC123" url="http://localhost/p/ABC123" description="Check it out!" username="deluan" created="0001-01-01T00:00:00Z" expires="0001-01-01T00:00:00Z" lastVisited="0001-01-01T00:00:00Z" visitCount="2"><entry id="1" isDir="false" title="title" album="album" artist="artist" duration="120" isVideo="false"></entry><entry id="2" isDir="false" title="title 2" album="album" artist="artist" duration="300" isVideo="false"></entry></share></shares></subsonic-response>
|
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><shares><share id="ABC123" url="http://localhost/p/ABC123" description="Check it out!" username="deluan" created="0001-01-01T00:00:00Z" expires="0001-01-01T00:00:00Z" lastVisited="0001-01-01T00:00:00Z" visitCount="2"><entry id="1" isDir="false" title="title" album="album" artist="artist" duration="120" isVideo="false" bpm="0" comment=""></entry><entry id="2" isDir="false" title="title 2" album="album" artist="artist" duration="300" isVideo="false" bpm="0" comment=""></entry></share></shares></subsonic-response>
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"similarSongs":{"song":[{"id":"1","isDir":false,"title":"title","isVideo":false}]}}
|
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"similarSongs":{"song":[{"id":"1","isDir":false,"title":"title","genres":[],"isVideo":false,"bpm":0,"comment":""}]}}
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><similarSongs><song id="1" isDir="false" title="title" isVideo="false"></song></similarSongs></subsonic-response>
|
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><similarSongs><song id="1" isDir="false" title="title" isVideo="false" bpm="0" comment=""></song></similarSongs></subsonic-response>
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"similarSongs2":{"song":[{"id":"1","isDir":false,"title":"title","isVideo":false}]}}
|
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"similarSongs2":{"song":[{"id":"1","isDir":false,"title":"title","genres":[],"isVideo":false,"bpm":0,"comment":""}]}}
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><similarSongs2><song id="1" isDir="false" title="title" isVideo="false"></song></similarSongs2></subsonic-response>
|
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><similarSongs2><song id="1" isDir="false" title="title" isVideo="false" bpm="0" comment=""></song></similarSongs2></subsonic-response>
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"topSongs":{"song":[{"id":"1","isDir":false,"title":"title","isVideo":false}]}}
|
{"status":"ok","version":"1.8.0","type":"navidrome","serverVersion":"v0.0.0","openSubsonic":true,"topSongs":{"song":[{"id":"1","isDir":false,"title":"title","genres":[],"isVideo":false,"bpm":0,"comment":""}]}}
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><topSongs><song id="1" isDir="false" title="title" isVideo="false"></song></topSongs></subsonic-response>
|
<subsonic-response xmlns="http://subsonic.org/restapi" status="ok" version="1.8.0" type="navidrome" serverVersion="v0.0.0" openSubsonic="true"><topSongs><song id="1" isDir="false" title="title" isVideo="false" bpm="0" comment=""></song></topSongs></subsonic-response>
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
package responses
|
package responses
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"encoding/json"
|
||||||
"encoding/xml"
|
"encoding/xml"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
@ -106,42 +107,59 @@ type Indexes struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
type Child struct {
|
type Child struct {
|
||||||
Id string `xml:"id,attr" json:"id"`
|
Id string `xml:"id,attr" json:"id"`
|
||||||
Parent string `xml:"parent,attr,omitempty" json:"parent,omitempty"`
|
Parent string `xml:"parent,attr,omitempty" json:"parent,omitempty"`
|
||||||
IsDir bool `xml:"isDir,attr" json:"isDir"`
|
IsDir bool `xml:"isDir,attr" json:"isDir"`
|
||||||
Title string `xml:"title,attr,omitempty" json:"title,omitempty"`
|
Title string `xml:"title,attr,omitempty" json:"title,omitempty"`
|
||||||
Name string `xml:"name,attr,omitempty" json:"name,omitempty"`
|
Name string `xml:"name,attr,omitempty" json:"name,omitempty"`
|
||||||
Album string `xml:"album,attr,omitempty" json:"album,omitempty"`
|
Album string `xml:"album,attr,omitempty" json:"album,omitempty"`
|
||||||
Artist string `xml:"artist,attr,omitempty" json:"artist,omitempty"`
|
Artist string `xml:"artist,attr,omitempty" json:"artist,omitempty"`
|
||||||
Track int32 `xml:"track,attr,omitempty" json:"track,omitempty"`
|
Track int32 `xml:"track,attr,omitempty" json:"track,omitempty"`
|
||||||
Year int32 `xml:"year,attr,omitempty" json:"year,omitempty"`
|
Year int32 `xml:"year,attr,omitempty" json:"year,omitempty"`
|
||||||
Genre string `xml:"genre,attr,omitempty" json:"genre,omitempty"`
|
Genre string `xml:"genre,attr,omitempty" json:"genre,omitempty"`
|
||||||
CoverArt string `xml:"coverArt,attr,omitempty" json:"coverArt,omitempty"`
|
Genres []ItemGenre `xml:"genres" json:"genres"`
|
||||||
Size int64 `xml:"size,attr,omitempty" json:"size,omitempty"`
|
CoverArt string `xml:"coverArt,attr,omitempty" json:"coverArt,omitempty"`
|
||||||
ContentType string `xml:"contentType,attr,omitempty" json:"contentType,omitempty"`
|
Size int64 `xml:"size,attr,omitempty" json:"size,omitempty"`
|
||||||
Suffix string `xml:"suffix,attr,omitempty" json:"suffix,omitempty"`
|
ContentType string `xml:"contentType,attr,omitempty" json:"contentType,omitempty"`
|
||||||
Starred *time.Time `xml:"starred,attr,omitempty" json:"starred,omitempty"`
|
Suffix string `xml:"suffix,attr,omitempty" json:"suffix,omitempty"`
|
||||||
TranscodedContentType string `xml:"transcodedContentType,attr,omitempty" json:"transcodedContentType,omitempty"`
|
Starred *time.Time `xml:"starred,attr,omitempty" json:"starred,omitempty"`
|
||||||
TranscodedSuffix string `xml:"transcodedSuffix,attr,omitempty" json:"transcodedSuffix,omitempty"`
|
TranscodedContentType string `xml:"transcodedContentType,attr,omitempty" json:"transcodedContentType,omitempty"`
|
||||||
Duration int32 `xml:"duration,attr,omitempty" json:"duration,omitempty"`
|
TranscodedSuffix string `xml:"transcodedSuffix,attr,omitempty" json:"transcodedSuffix,omitempty"`
|
||||||
BitRate int32 `xml:"bitRate,attr,omitempty" json:"bitRate,omitempty"`
|
Duration int32 `xml:"duration,attr,omitempty" json:"duration,omitempty"`
|
||||||
Path string `xml:"path,attr,omitempty" json:"path,omitempty"`
|
BitRate int32 `xml:"bitRate,attr,omitempty" json:"bitRate,omitempty"`
|
||||||
PlayCount int64 `xml:"playCount,attr,omitempty" json:"playCount,omitempty"`
|
Path string `xml:"path,attr,omitempty" json:"path,omitempty"`
|
||||||
Played *time.Time `xml:"played,attr,omitempty" json:"played,omitempty"`
|
PlayCount int64 `xml:"playCount,attr,omitempty" json:"playCount,omitempty"`
|
||||||
DiscNumber int32 `xml:"discNumber,attr,omitempty" json:"discNumber,omitempty"`
|
Played *time.Time `xml:"played,attr,omitempty" json:"played,omitempty"`
|
||||||
Created *time.Time `xml:"created,attr,omitempty" json:"created,omitempty"`
|
DiscNumber int32 `xml:"discNumber,attr,omitempty" json:"discNumber,omitempty"`
|
||||||
AlbumId string `xml:"albumId,attr,omitempty" json:"albumId,omitempty"`
|
Created *time.Time `xml:"created,attr,omitempty" json:"created,omitempty"`
|
||||||
ArtistId string `xml:"artistId,attr,omitempty" json:"artistId,omitempty"`
|
AlbumId string `xml:"albumId,attr,omitempty" json:"albumId,omitempty"`
|
||||||
Type string `xml:"type,attr,omitempty" json:"type,omitempty"`
|
ArtistId string `xml:"artistId,attr,omitempty" json:"artistId,omitempty"`
|
||||||
UserRating int32 `xml:"userRating,attr,omitempty" json:"userRating,omitempty"`
|
Type string `xml:"type,attr,omitempty" json:"type,omitempty"`
|
||||||
SongCount int32 `xml:"songCount,attr,omitempty" json:"songCount,omitempty"`
|
UserRating int32 `xml:"userRating,attr,omitempty" json:"userRating,omitempty"`
|
||||||
IsVideo bool `xml:"isVideo,attr" json:"isVideo"`
|
SongCount int32 `xml:"songCount,attr,omitempty" json:"songCount,omitempty"`
|
||||||
BookmarkPosition int64 `xml:"bookmarkPosition,attr,omitempty" json:"bookmarkPosition,omitempty"`
|
IsVideo bool `xml:"isVideo,attr" json:"isVideo"`
|
||||||
|
BookmarkPosition int64 `xml:"bookmarkPosition,attr,omitempty" json:"bookmarkPosition,omitempty"`
|
||||||
|
Bpm int32 `xml:"bpm,attr" json:"bpm"`
|
||||||
|
Comment string `xml:"comment,attr" json:"comment"`
|
||||||
/*
|
/*
|
||||||
<xs:attribute name="averageRating" type="sub:AverageRating" use="optional"/> <!-- Added in 1.6.0 -->
|
<xs:attribute name="averageRating" type="sub:AverageRating" use="optional"/> <!-- Added in 1.6.0 -->
|
||||||
*/
|
*/
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c Child) MarshalJSON() ([]byte, error) {
|
||||||
|
// ensure fields that need custom default values are set properly
|
||||||
|
type Alias Child
|
||||||
|
a := struct {
|
||||||
|
Alias
|
||||||
|
}{
|
||||||
|
Alias: (Alias)(c),
|
||||||
|
}
|
||||||
|
if a.Genres == nil {
|
||||||
|
a.Genres = make([]ItemGenre, 0)
|
||||||
|
}
|
||||||
|
return json.Marshal(a)
|
||||||
|
}
|
||||||
|
|
||||||
type Songs struct {
|
type Songs struct {
|
||||||
Songs []Child `xml:"song" json:"song,omitempty"`
|
Songs []Child `xml:"song" json:"song,omitempty"`
|
||||||
}
|
}
|
||||||
|
@ -180,23 +198,40 @@ type ArtistID3 struct {
|
||||||
Starred *time.Time `xml:"starred,attr,omitempty" json:"starred,omitempty"`
|
Starred *time.Time `xml:"starred,attr,omitempty" json:"starred,omitempty"`
|
||||||
UserRating int32 `xml:"userRating,attr,omitempty" json:"userRating,omitempty"`
|
UserRating int32 `xml:"userRating,attr,omitempty" json:"userRating,omitempty"`
|
||||||
ArtistImageUrl string `xml:"artistImageUrl,attr,omitempty" json:"artistImageUrl,omitempty"`
|
ArtistImageUrl string `xml:"artistImageUrl,attr,omitempty" json:"artistImageUrl,omitempty"`
|
||||||
|
MusicBrainzId string `xml:"musicBrainzId,attr,omitempty" json:"musicBrainzId,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type AlbumID3 struct {
|
type AlbumID3 struct {
|
||||||
Id string `xml:"id,attr" json:"id"`
|
Id string `xml:"id,attr" json:"id"`
|
||||||
Name string `xml:"name,attr" json:"name"`
|
Name string `xml:"name,attr" json:"name"`
|
||||||
Artist string `xml:"artist,attr,omitempty" json:"artist,omitempty"`
|
Artist string `xml:"artist,attr,omitempty" json:"artist,omitempty"`
|
||||||
ArtistId string `xml:"artistId,attr,omitempty" json:"artistId,omitempty"`
|
ArtistId string `xml:"artistId,attr,omitempty" json:"artistId,omitempty"`
|
||||||
CoverArt string `xml:"coverArt,attr,omitempty" json:"coverArt,omitempty"`
|
CoverArt string `xml:"coverArt,attr,omitempty" json:"coverArt,omitempty"`
|
||||||
SongCount int32 `xml:"songCount,attr,omitempty" json:"songCount,omitempty"`
|
SongCount int32 `xml:"songCount,attr,omitempty" json:"songCount,omitempty"`
|
||||||
Duration int32 `xml:"duration,attr,omitempty" json:"duration,omitempty"`
|
Duration int32 `xml:"duration,attr,omitempty" json:"duration,omitempty"`
|
||||||
PlayCount int64 `xml:"playCount,attr,omitempty" json:"playCount,omitempty"`
|
PlayCount int64 `xml:"playCount,attr,omitempty" json:"playCount,omitempty"`
|
||||||
Played *time.Time `xml:"played,attr,omitempty" json:"played,omitempty"`
|
Played *time.Time `xml:"played,attr,omitempty" json:"played,omitempty"`
|
||||||
Created *time.Time `xml:"created,attr,omitempty" json:"created,omitempty"`
|
Created *time.Time `xml:"created,attr,omitempty" json:"created,omitempty"`
|
||||||
Starred *time.Time `xml:"starred,attr,omitempty" json:"starred,omitempty"`
|
Starred *time.Time `xml:"starred,attr,omitempty" json:"starred,omitempty"`
|
||||||
UserRating int32 `xml:"userRating,attr,omitempty" json:"userRating,omitempty"`
|
UserRating int32 `xml:"userRating,attr" json:"userRating"`
|
||||||
Year int32 `xml:"year,attr,omitempty" json:"year,omitempty"`
|
Year int32 `xml:"year,attr,omitempty" json:"year,omitempty"`
|
||||||
Genre string `xml:"genre,attr,omitempty" json:"genre,omitempty"`
|
Genre string `xml:"genre,attr,omitempty" json:"genre,omitempty"`
|
||||||
|
Genres []ItemGenre `xml:"genres" json:"genres"`
|
||||||
|
MusicBrainzId string `xml:"musicBrainzId,attr" json:"musicBrainzId"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a AlbumID3) MarshalJSON() ([]byte, error) {
|
||||||
|
// ensure fields that need custom default values are set properly
|
||||||
|
type Alias AlbumID3
|
||||||
|
x := struct {
|
||||||
|
Alias
|
||||||
|
}{
|
||||||
|
Alias: (Alias)(a),
|
||||||
|
}
|
||||||
|
if x.Genres == nil {
|
||||||
|
x.Genres = make([]ItemGenre, 0)
|
||||||
|
}
|
||||||
|
return json.Marshal(x)
|
||||||
}
|
}
|
||||||
|
|
||||||
type ArtistWithAlbumsID3 struct {
|
type ArtistWithAlbumsID3 struct {
|
||||||
|
@ -418,3 +453,8 @@ type JukeboxPlaylist struct {
|
||||||
Entry []Child `xml:"entry,omitempty" json:"entry,omitempty"`
|
Entry []Child `xml:"entry,omitempty" json:"entry,omitempty"`
|
||||||
}
|
}
|
||||||
type OpenSubsonicExtensions struct{}
|
type OpenSubsonicExtensions struct{}
|
||||||
|
|
||||||
|
// OpenSubsonic response type for multi-valued genres list
|
||||||
|
type ItemGenre struct {
|
||||||
|
Name string `xml:"name,attr" json:"name"`
|
||||||
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue