mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-03 20:47:35 +03:00
fix(server): remove misleading "Agent not available" warning.
Signed-off-by: Deluan <deluan@navidrome.org>
This commit is contained in:
parent
dc4e091622
commit
a04167672c
8 changed files with 97 additions and 35 deletions
|
@ -24,15 +24,23 @@ func New(ds model.DataStore) *Agents {
|
||||||
}
|
}
|
||||||
order = append(order, LocalAgentName)
|
order = append(order, LocalAgentName)
|
||||||
var res []Interface
|
var res []Interface
|
||||||
|
var enabled []string
|
||||||
for _, name := range order {
|
for _, name := range order {
|
||||||
init, ok := Map[name]
|
init, ok := Map[name]
|
||||||
if !ok {
|
if !ok {
|
||||||
log.Error("Agent not available. Check configuration", "name", name)
|
log.Error("Invalid agent. Check `Agents` configuration", "name", name, "conf", conf.Server.Agents)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
agent := init(ds)
|
||||||
|
if agent == nil {
|
||||||
|
log.Debug("Agent not available. Missing configuration?", "name", name)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
enabled = append(enabled, name)
|
||||||
res = append(res, init(ds))
|
res = append(res, init(ds))
|
||||||
}
|
}
|
||||||
|
log.Debug("List of agents enabled", "names", enabled)
|
||||||
|
|
||||||
return &Agents{ds: ds, agents: res}
|
return &Agents{ds: ds, agents: res}
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,6 +7,7 @@ import (
|
||||||
"github.com/navidrome/navidrome/consts"
|
"github.com/navidrome/navidrome/consts"
|
||||||
"github.com/navidrome/navidrome/model"
|
"github.com/navidrome/navidrome/model"
|
||||||
"github.com/navidrome/navidrome/tests"
|
"github.com/navidrome/navidrome/tests"
|
||||||
|
"github.com/navidrome/navidrome/utils/slice"
|
||||||
|
|
||||||
"github.com/navidrome/navidrome/conf"
|
"github.com/navidrome/navidrome/conf"
|
||||||
. "github.com/onsi/ginkgo/v2"
|
. "github.com/onsi/ginkgo/v2"
|
||||||
|
@ -44,19 +45,21 @@ var _ = Describe("Agents", func() {
|
||||||
var mock *mockAgent
|
var mock *mockAgent
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
mock = &mockAgent{}
|
mock = &mockAgent{}
|
||||||
Register("fake", func(ds model.DataStore) Interface {
|
Register("fake", func(model.DataStore) Interface { return mock })
|
||||||
return mock
|
Register("disabled", func(model.DataStore) Interface { return nil })
|
||||||
})
|
Register("empty", func(model.DataStore) Interface { return &emptyAgent{} })
|
||||||
Register("empty", func(ds model.DataStore) Interface {
|
conf.Server.Agents = "empty,fake,disabled"
|
||||||
return struct {
|
|
||||||
Interface
|
|
||||||
}{}
|
|
||||||
})
|
|
||||||
conf.Server.Agents = "empty,fake"
|
|
||||||
ag = New(ds)
|
ag = New(ds)
|
||||||
Expect(ag.AgentName()).To(Equal("agents"))
|
Expect(ag.AgentName()).To(Equal("agents"))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("does not register disabled agents", func() {
|
||||||
|
ags := slice.Map(ag.agents, func(a Interface) string { return a.AgentName() })
|
||||||
|
// local agent is always appended to the end of the agents list
|
||||||
|
Expect(ags).To(HaveExactElements("empty", "fake", "local"))
|
||||||
|
Expect(ags).ToNot(ContainElement("disabled"))
|
||||||
|
})
|
||||||
|
|
||||||
Describe("GetArtistMBID", func() {
|
Describe("GetArtistMBID", func() {
|
||||||
It("returns on first match", func() {
|
It("returns on first match", func() {
|
||||||
Expect(ag.GetArtistMBID(ctx, "123", "test")).To(Equal("mbid"))
|
Expect(ag.GetArtistMBID(ctx, "123", "test")).To(Equal("mbid"))
|
||||||
|
@ -344,3 +347,11 @@ func (a *mockAgent) GetAlbumInfo(ctx context.Context, name, artist, mbid string)
|
||||||
},
|
},
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type emptyAgent struct {
|
||||||
|
Interface
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *emptyAgent) AgentName() string {
|
||||||
|
return "empty"
|
||||||
|
}
|
||||||
|
|
|
@ -42,6 +42,9 @@ type lastfmAgent struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
func lastFMConstructor(ds model.DataStore) *lastfmAgent {
|
func lastFMConstructor(ds model.DataStore) *lastfmAgent {
|
||||||
|
if !conf.Server.LastFM.Enabled || conf.Server.LastFM.ApiKey == "" || conf.Server.LastFM.Secret == "" {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
l := &lastfmAgent{
|
l := &lastfmAgent{
|
||||||
ds: ds,
|
ds: ds,
|
||||||
lang: conf.Server.LastFM.Language,
|
lang: conf.Server.LastFM.Language,
|
||||||
|
@ -340,15 +343,19 @@ func (l *lastfmAgent) IsAuthorized(ctx context.Context, userId string) bool {
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
conf.AddHook(func() {
|
conf.AddHook(func() {
|
||||||
if conf.Server.LastFM.Enabled {
|
agents.Register(lastFMAgentName, func(ds model.DataStore) agents.Interface {
|
||||||
if conf.Server.LastFM.ApiKey != "" && conf.Server.LastFM.Secret != "" {
|
a := lastFMConstructor(ds)
|
||||||
agents.Register(lastFMAgentName, func(ds model.DataStore) agents.Interface {
|
if a != nil {
|
||||||
return lastFMConstructor(ds)
|
return a
|
||||||
})
|
|
||||||
scrobbler.Register(lastFMAgentName, func(ds model.DataStore) scrobbler.Scrobbler {
|
|
||||||
return lastFMConstructor(ds)
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
}
|
return nil
|
||||||
|
})
|
||||||
|
scrobbler.Register(lastFMAgentName, func(ds model.DataStore) scrobbler.Scrobbler {
|
||||||
|
a := lastFMConstructor(ds)
|
||||||
|
if a != nil {
|
||||||
|
return a
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,6 +11,7 @@ import (
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/navidrome/navidrome/conf"
|
"github.com/navidrome/navidrome/conf"
|
||||||
|
"github.com/navidrome/navidrome/conf/configtest"
|
||||||
"github.com/navidrome/navidrome/core/agents"
|
"github.com/navidrome/navidrome/core/agents"
|
||||||
"github.com/navidrome/navidrome/core/scrobbler"
|
"github.com/navidrome/navidrome/core/scrobbler"
|
||||||
"github.com/navidrome/navidrome/model"
|
"github.com/navidrome/navidrome/model"
|
||||||
|
@ -30,16 +31,38 @@ var _ = Describe("lastfmAgent", func() {
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
ds = &tests.MockDataStore{}
|
ds = &tests.MockDataStore{}
|
||||||
ctx = context.Background()
|
ctx = context.Background()
|
||||||
|
DeferCleanup(configtest.SetupConfig())
|
||||||
|
conf.Server.LastFM.Enabled = true
|
||||||
|
conf.Server.LastFM.ApiKey = "123"
|
||||||
|
conf.Server.LastFM.Secret = "secret"
|
||||||
})
|
})
|
||||||
Describe("lastFMConstructor", func() {
|
Describe("lastFMConstructor", func() {
|
||||||
It("uses configured api key and language", func() {
|
When("Agent is properly configured", func() {
|
||||||
conf.Server.LastFM.ApiKey = "123"
|
It("uses configured api key and language", func() {
|
||||||
conf.Server.LastFM.Secret = "secret"
|
conf.Server.LastFM.Language = "pt"
|
||||||
conf.Server.LastFM.Language = "pt"
|
agent := lastFMConstructor(ds)
|
||||||
agent := lastFMConstructor(ds)
|
Expect(agent.apiKey).To(Equal("123"))
|
||||||
Expect(agent.apiKey).To(Equal("123"))
|
Expect(agent.secret).To(Equal("secret"))
|
||||||
Expect(agent.secret).To(Equal("secret"))
|
Expect(agent.lang).To(Equal("pt"))
|
||||||
Expect(agent.lang).To(Equal("pt"))
|
})
|
||||||
|
})
|
||||||
|
When("Agent is disabled", func() {
|
||||||
|
It("returns nil", func() {
|
||||||
|
conf.Server.LastFM.Enabled = false
|
||||||
|
Expect(lastFMConstructor(ds)).To(BeNil())
|
||||||
|
})
|
||||||
|
})
|
||||||
|
When("ApiKey is empty", func() {
|
||||||
|
It("returns nil", func() {
|
||||||
|
conf.Server.LastFM.ApiKey = ""
|
||||||
|
Expect(lastFMConstructor(ds)).To(BeNil())
|
||||||
|
})
|
||||||
|
})
|
||||||
|
When("Secret is empty", func() {
|
||||||
|
It("returns nil", func() {
|
||||||
|
conf.Server.LastFM.Secret = ""
|
||||||
|
Expect(lastFMConstructor(ds)).To(BeNil())
|
||||||
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
|
@ -27,6 +27,9 @@ type spotifyAgent struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
func spotifyConstructor(ds model.DataStore) agents.Interface {
|
func spotifyConstructor(ds model.DataStore) agents.Interface {
|
||||||
|
if conf.Server.Spotify.ID == "" || conf.Server.Spotify.Secret == "" {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
l := &spotifyAgent{
|
l := &spotifyAgent{
|
||||||
ds: ds,
|
ds: ds,
|
||||||
id: conf.Server.Spotify.ID,
|
id: conf.Server.Spotify.ID,
|
||||||
|
@ -88,8 +91,6 @@ func (s *spotifyAgent) searchArtist(ctx context.Context, name string) (*Artist,
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
conf.AddHook(func() {
|
conf.AddHook(func() {
|
||||||
if conf.Server.Spotify.ID != "" && conf.Server.Spotify.Secret != "" {
|
agents.Register(spotifyAgentName, spotifyConstructor)
|
||||||
agents.Register(spotifyAgentName, spotifyConstructor)
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -53,13 +53,20 @@ func newPlayTracker(ds model.DataStore, broker events.Broker) *playTracker {
|
||||||
m := cache.NewSimpleCache[string, NowPlayingInfo]()
|
m := cache.NewSimpleCache[string, NowPlayingInfo]()
|
||||||
p := &playTracker{ds: ds, playMap: m, broker: broker}
|
p := &playTracker{ds: ds, playMap: m, broker: broker}
|
||||||
p.scrobblers = make(map[string]Scrobbler)
|
p.scrobblers = make(map[string]Scrobbler)
|
||||||
|
var enabled []string
|
||||||
for name, constructor := range constructors {
|
for name, constructor := range constructors {
|
||||||
s := constructor(ds)
|
s := constructor(ds)
|
||||||
|
if s == nil {
|
||||||
|
log.Debug("Scrobbler not available. Missing configuration?", "name", name)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
enabled = append(enabled, name)
|
||||||
if conf.Server.DevEnableBufferedScrobble {
|
if conf.Server.DevEnableBufferedScrobble {
|
||||||
s = newBufferedScrobbler(ds, s, name)
|
s = newBufferedScrobbler(ds, s, name)
|
||||||
}
|
}
|
||||||
p.scrobblers[name] = s
|
p.scrobblers[name] = s
|
||||||
}
|
}
|
||||||
|
log.Debug("List of scrobblers enabled", "names", enabled)
|
||||||
return p
|
return p
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -35,9 +35,12 @@ var _ = Describe("PlayTracker", func() {
|
||||||
ctx = request.WithPlayer(ctx, model.Player{ScrobbleEnabled: true})
|
ctx = request.WithPlayer(ctx, model.Player{ScrobbleEnabled: true})
|
||||||
ds = &tests.MockDataStore{}
|
ds = &tests.MockDataStore{}
|
||||||
fake = fakeScrobbler{Authorized: true}
|
fake = fakeScrobbler{Authorized: true}
|
||||||
Register("fake", func(ds model.DataStore) Scrobbler {
|
Register("fake", func(model.DataStore) Scrobbler {
|
||||||
return &fake
|
return &fake
|
||||||
})
|
})
|
||||||
|
Register("disabled", func(model.DataStore) Scrobbler {
|
||||||
|
return nil
|
||||||
|
})
|
||||||
tracker = newPlayTracker(ds, events.GetBroker())
|
tracker = newPlayTracker(ds, events.GetBroker())
|
||||||
|
|
||||||
track = model.MediaFile{
|
track = model.MediaFile{
|
||||||
|
@ -61,6 +64,11 @@ var _ = Describe("PlayTracker", func() {
|
||||||
_ = ds.Album(ctx).(*tests.MockAlbumRepo).Put(&album)
|
_ = ds.Album(ctx).(*tests.MockAlbumRepo).Put(&album)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("does not register disabled scrobblers", func() {
|
||||||
|
Expect(tracker.(*playTracker).scrobblers).To(HaveKey("fake"))
|
||||||
|
Expect(tracker.(*playTracker).scrobblers).ToNot(HaveKey("disabled"))
|
||||||
|
})
|
||||||
|
|
||||||
Describe("NowPlaying", func() {
|
Describe("NowPlaying", func() {
|
||||||
It("sends track to agent", func() {
|
It("sends track to agent", func() {
|
||||||
err := tracker.NowPlaying(ctx, "player-1", "player-one", "123")
|
err := tracker.NowPlaying(ctx, "player-1", "player-one", "123")
|
||||||
|
|
|
@ -18,6 +18,7 @@ var _ = Describe("ArtistRepository", func() {
|
||||||
var repo model.ArtistRepository
|
var repo model.ArtistRepository
|
||||||
|
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
|
DeferCleanup(configtest.SetupConfig())
|
||||||
ctx := log.NewContext(context.TODO())
|
ctx := log.NewContext(context.TODO())
|
||||||
ctx = request.WithUser(ctx, model.User{ID: "userid"})
|
ctx = request.WithUser(ctx, model.User{ID: "userid"})
|
||||||
repo = NewArtistRepository(ctx, GetDBXBuilder())
|
repo = NewArtistRepository(ctx, GetDBXBuilder())
|
||||||
|
@ -51,7 +52,6 @@ var _ = Describe("ArtistRepository", func() {
|
||||||
r := artistRepository{indexGroups: utils.ParseIndexGroups(conf.Server.IndexGroups)}
|
r := artistRepository{indexGroups: utils.ParseIndexGroups(conf.Server.IndexGroups)}
|
||||||
When("PreferSortTags is false", func() {
|
When("PreferSortTags is false", func() {
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
DeferCleanup(configtest.SetupConfig)
|
|
||||||
conf.Server.PreferSortTags = false
|
conf.Server.PreferSortTags = false
|
||||||
})
|
})
|
||||||
It("returns the OrderArtistName key is SortArtistName is empty", func() {
|
It("returns the OrderArtistName key is SortArtistName is empty", func() {
|
||||||
|
@ -68,7 +68,6 @@ var _ = Describe("ArtistRepository", func() {
|
||||||
})
|
})
|
||||||
When("PreferSortTags is true", func() {
|
When("PreferSortTags is true", func() {
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
DeferCleanup(configtest.SetupConfig)
|
|
||||||
conf.Server.PreferSortTags = true
|
conf.Server.PreferSortTags = true
|
||||||
})
|
})
|
||||||
It("returns the SortArtistName key if it is not empty", func() {
|
It("returns the SortArtistName key if it is not empty", func() {
|
||||||
|
@ -87,7 +86,6 @@ var _ = Describe("ArtistRepository", func() {
|
||||||
Describe("GetIndex", func() {
|
Describe("GetIndex", func() {
|
||||||
When("PreferSortTags is true", func() {
|
When("PreferSortTags is true", func() {
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
DeferCleanup(configtest.SetupConfig())
|
|
||||||
conf.Server.PreferSortTags = true
|
conf.Server.PreferSortTags = true
|
||||||
})
|
})
|
||||||
It("returns the index when PreferSortTags is true and SortArtistName is not empty", func() {
|
It("returns the index when PreferSortTags is true and SortArtistName is not empty", func() {
|
||||||
|
@ -128,7 +126,6 @@ var _ = Describe("ArtistRepository", func() {
|
||||||
|
|
||||||
When("PreferSortTags is false", func() {
|
When("PreferSortTags is false", func() {
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
DeferCleanup(configtest.SetupConfig())
|
|
||||||
conf.Server.PreferSortTags = false
|
conf.Server.PreferSortTags = false
|
||||||
})
|
})
|
||||||
It("returns the index when SortArtistName is NOT empty", func() {
|
It("returns the index when SortArtistName is NOT empty", func() {
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue