Add config option to show album participations under artists in Subsonic clients

This commit is contained in:
Deluan 2023-01-18 14:20:06 -05:00
parent 8ae0bcb459
commit 136d5f9a83
3 changed files with 75 additions and 69 deletions

View file

@ -38,11 +38,12 @@ type configOptions struct {
AutoImportPlaylists bool AutoImportPlaylists bool
PlaylistsPath string PlaylistsPath string
AutoTranscodeDownload bool AutoTranscodeDownload bool
DefaultDownsamplingFormat string
SearchFullString bool SearchFullString bool
RecentlyAddedByModTime bool RecentlyAddedByModTime bool
IgnoredArticles string IgnoredArticles string
IndexGroups string IndexGroups string
SubsonicArtistParticipations bool
ProbeCommand string ProbeCommand string
CoverArtPriority string CoverArtPriority string
CoverJpegQuality int CoverJpegQuality int
@ -54,6 +55,7 @@ type configOptions struct {
DefaultTheme string DefaultTheme string
DefaultLanguage string DefaultLanguage string
DefaultUIVolume int DefaultUIVolume int
EnableReplayGain bool
EnableCoverAnimation bool EnableCoverAnimation bool
GATrackingID string GATrackingID string
EnableLogRedacting bool EnableLogRedacting bool
@ -63,8 +65,6 @@ type configOptions struct {
ReverseProxyUserHeader string ReverseProxyUserHeader string
ReverseProxyWhitelist string ReverseProxyWhitelist string
Prometheus prometheusOptions Prometheus prometheusOptions
EnableReplayGain bool
Scanner scannerOptions Scanner scannerOptions
Agents string Agents string
@ -72,9 +72,6 @@ type configOptions struct {
Spotify spotifyOptions Spotify spotifyOptions
ListenBrainz listenBrainzOptions ListenBrainz listenBrainzOptions
//test downsampling
DefaultDownsamplingFormat string
// DevFlags. These are used to enable/disable debugging and incomplete features // DevFlags. These are used to enable/disable debugging and incomplete features
DevLogSourceLine bool DevLogSourceLine bool
DevLogLevels map[string]string DevLogLevels map[string]string
@ -236,12 +233,12 @@ func init() {
viper.SetDefault("enableexternalservices", true) viper.SetDefault("enableexternalservices", true)
viper.SetDefault("enableMediaFileCoverArt", true) viper.SetDefault("enableMediaFileCoverArt", true)
viper.SetDefault("autotranscodedownload", false) viper.SetDefault("autotranscodedownload", false)
viper.SetDefault("defaultdownsamplingformat", consts.DefaultDownsamplingFormat)
// Config options only valid for file/env configuration
viper.SetDefault("searchfullstring", false) viper.SetDefault("searchfullstring", false)
viper.SetDefault("recentlyaddedbymodtime", false) viper.SetDefault("recentlyaddedbymodtime", false)
viper.SetDefault("ignoredarticles", "The El La Los Las Le Les Os As O A") viper.SetDefault("ignoredarticles", "The El La Los Las Le Les Os As O A")
viper.SetDefault("indexgroups", "A B C D E F G H I J K L M N O P Q R S T U V W X-Z(XYZ) [Unknown]([)") viper.SetDefault("indexgroups", "A B C D E F G H I J K L M N O P Q R S T U V W X-Z(XYZ) [Unknown]([)")
viper.SetDefault("subsonicartistparticipations", false)
viper.SetDefault("probecommand", "ffmpeg %s -f ffmetadata") viper.SetDefault("probecommand", "ffmpeg %s -f ffmetadata")
viper.SetDefault("coverartpriority", "cover.*, folder.*, front.*, embedded, external") viper.SetDefault("coverartpriority", "cover.*, folder.*, front.*, embedded, external")
viper.SetDefault("coverjpegquality", 75) viper.SetDefault("coverjpegquality", 75)
@ -253,6 +250,7 @@ func init() {
viper.SetDefault("defaulttheme", "Dark") viper.SetDefault("defaulttheme", "Dark")
viper.SetDefault("defaultlanguage", "") viper.SetDefault("defaultlanguage", "")
viper.SetDefault("defaultuivolume", consts.DefaultUIVolume) viper.SetDefault("defaultuivolume", consts.DefaultUIVolume)
viper.SetDefault("enablereplaygain", false)
viper.SetDefault("enablecoveranimation", true) viper.SetDefault("enablecoveranimation", true)
viper.SetDefault("gatrackingid", "") viper.SetDefault("gatrackingid", "")
viper.SetDefault("enablelogredacting", true) viper.SetDefault("enablelogredacting", true)
@ -266,8 +264,6 @@ func init() {
viper.SetDefault("prometheus.enabled", false) viper.SetDefault("prometheus.enabled", false)
viper.SetDefault("prometheus.metricspath", "/metrics") viper.SetDefault("prometheus.metricspath", "/metrics")
viper.SetDefault("enablereplaygain", false)
viper.SetDefault("scanner.extractor", consts.DefaultScannerExtractor) viper.SetDefault("scanner.extractor", consts.DefaultScannerExtractor)
viper.SetDefault("scanner.genreseparators", ";/,") viper.SetDefault("scanner.genreseparators", ";/,")
@ -281,8 +277,6 @@ func init() {
viper.SetDefault("listenbrainz.enabled", true) viper.SetDefault("listenbrainz.enabled", true)
viper.SetDefault("listenbrainz.baseurl", "https://api.listenbrainz.org/1/") viper.SetDefault("listenbrainz.baseurl", "https://api.listenbrainz.org/1/")
viper.SetDefault("defaultdownsamplingformat", consts.DefaultDownsamplingFormat)
// DevFlags. These are used to enable/disable debugging and incomplete features // DevFlags. These are used to enable/disable debugging and incomplete features
viper.SetDefault("devlogsourceline", false) viper.SetDefault("devlogsourceline", false)
viper.SetDefault("devautocreateadminpassword", "") viper.SetDefault("devautocreateadminpassword", "")

View file

@ -141,15 +141,12 @@ func (api *Router) GetArtist(r *http.Request) (*responses.Subsonic, error) {
return nil, err return nil, err
} }
albums, err := api.ds.Album(ctx).GetAllWithoutGenres(filter.AlbumsByArtistID(id))
if err != nil {
log.Error(ctx, "Error retrieving albums by artist", "id", id, "name", artist.Name, err)
return nil, err
}
response := newResponse() response := newResponse()
response.ArtistWithAlbumsID3 = api.buildArtist(r, artist, albums) response.ArtistWithAlbumsID3, err = api.buildArtist(r, artist)
return response, nil if err != nil {
log.Error(ctx, "Error retrieving albums by artist", "id", artist.ID, "name", artist.Name, err)
}
return response, err
} }
func (api *Router) GetAlbum(r *http.Request) (*responses.Subsonic, error) { func (api *Router) GetAlbum(r *http.Request) (*responses.Subsonic, error) {
@ -370,11 +367,18 @@ func (api *Router) buildArtistDirectory(ctx context.Context, artist *model.Artis
return dir, nil return dir, nil
} }
func (api *Router) buildArtist(r *http.Request, artist *model.Artist, albums model.Albums) *responses.ArtistWithAlbumsID3 { func (api *Router) buildArtist(r *http.Request, artist *model.Artist) (*responses.ArtistWithAlbumsID3, error) {
ctx := r.Context()
a := &responses.ArtistWithAlbumsID3{} a := &responses.ArtistWithAlbumsID3{}
a.ArtistID3 = toArtistID3(r, *artist) a.ArtistID3 = toArtistID3(r, *artist)
albums, err := api.ds.Album(ctx).GetAllWithoutGenres(filter.AlbumsByArtistID(artist.ID))
if err != nil {
return nil, err
}
a.Album = childrenFromAlbums(r.Context(), albums) a.Album = childrenFromAlbums(r.Context(), albums)
return a return a, nil
} }
func (api *Router) buildAlbumDirectory(ctx context.Context, album *model.Album) (*responses.Directory, error) { func (api *Router) buildAlbumDirectory(ctx context.Context, album *model.Album) (*responses.Directory, error) {

View file

@ -1,9 +1,11 @@
package filter package filter
import ( import (
"fmt"
"time" "time"
"github.com/Masterminds/squirrel" "github.com/Masterminds/squirrel"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/model" "github.com/navidrome/navidrome/model"
) )
@ -49,9 +51,15 @@ func AlbumsByGenre(genre string) Options {
} }
func AlbumsByArtistID(artistId string) Options { func AlbumsByArtistID(artistId string) Options {
var filters squirrel.Sqlizer
if conf.Server.SubsonicArtistParticipations {
filters = squirrel.Like{"all_artist_ids": fmt.Sprintf("%%%s%%", artistId)}
} else {
filters = squirrel.Eq{"album_artist_id": artistId}
}
return Options{ return Options{
Sort: "max_year", Sort: "max_year",
Filters: squirrel.Eq{"album_artist_id": artistId}, Filters: filters,
} }
} }