mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-03 12:37:37 +03:00
Add local agent, only for images
This commit is contained in:
parent
387acc5f63
commit
bf461473ef
15 changed files with 134 additions and 76 deletions
|
@ -29,9 +29,11 @@ const (
|
||||||
DevInitialUserName = "admin"
|
DevInitialUserName = "admin"
|
||||||
DevInitialName = "Dev Admin"
|
DevInitialName = "Dev Admin"
|
||||||
|
|
||||||
URLPathUI = "/app"
|
URLPathUI = "/app"
|
||||||
URLPathNativeAPI = "/api"
|
URLPathNativeAPI = "/api"
|
||||||
URLPathSubsonicAPI = "/rest"
|
URLPathSubsonicAPI = "/rest"
|
||||||
|
URLPathPublic = "/p"
|
||||||
|
URLPathPublicImages = URLPathPublic + "/img"
|
||||||
|
|
||||||
// DefaultUILoginBackgroundURL uses Navidrome curated background images collection,
|
// DefaultUILoginBackgroundURL uses Navidrome curated background images collection,
|
||||||
// available at https://unsplash.com/collections/20072696/navidrome
|
// available at https://unsplash.com/collections/20072696/navidrome
|
||||||
|
@ -46,13 +48,15 @@ const (
|
||||||
|
|
||||||
ServerReadHeaderTimeout = 3 * time.Second
|
ServerReadHeaderTimeout = 3 * time.Second
|
||||||
|
|
||||||
ArtistInfoTimeToLive = 24 * time.Hour
|
ArtistInfoTimeToLive = time.Second
|
||||||
|
//ArtistInfoTimeToLive = 24 * time.Hour
|
||||||
|
|
||||||
I18nFolder = "i18n"
|
I18nFolder = "i18n"
|
||||||
SkipScanFile = ".ndignore"
|
SkipScanFile = ".ndignore"
|
||||||
|
|
||||||
PlaceholderAlbumArt = "placeholder.png"
|
PlaceholderArtistArt = "artist-placeholder.webp"
|
||||||
PlaceholderAvatar = "logo-192x192.png"
|
PlaceholderAlbumArt = "placeholder.png"
|
||||||
|
PlaceholderAvatar = "logo-192x192.png"
|
||||||
|
|
||||||
DefaultUIVolume = 100
|
DefaultUIVolume = 100
|
||||||
|
|
||||||
|
|
|
@ -9,4 +9,4 @@ A new agent must comply with these simple implementation rules:
|
||||||
|
|
||||||
For an agent to be used it needs to be listed in the `Agents` config option (default is `"lastfm,spotify"`). The order dictates the priority of the agents
|
For an agent to be used it needs to be listed in the `Agents` config option (default is `"lastfm,spotify"`). The order dictates the priority of the agents
|
||||||
|
|
||||||
For a simple Agent example, look at the [placeholders](placeholders.go) agent source code.
|
For a simple Agent example, look at the [local_agent](local_agent.go) agent source code.
|
||||||
|
|
|
@ -22,7 +22,7 @@ func New(ds model.DataStore) *Agents {
|
||||||
if conf.Server.Agents != "" {
|
if conf.Server.Agents != "" {
|
||||||
order = strings.Split(conf.Server.Agents, ",")
|
order = strings.Split(conf.Server.Agents, ",")
|
||||||
}
|
}
|
||||||
order = append(order, PlaceholderAgentName)
|
order = append(order, LocalAgentName)
|
||||||
var res []Interface
|
var res []Interface
|
||||||
for _, name := range order {
|
for _, name := range order {
|
||||||
init, ok := Map[name]
|
init, ok := Map[name]
|
||||||
|
|
49
core/agents/local_agent.go
Normal file
49
core/agents/local_agent.go
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
package agents
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"path/filepath"
|
||||||
|
|
||||||
|
"github.com/navidrome/navidrome/consts"
|
||||||
|
"github.com/navidrome/navidrome/core/artwork"
|
||||||
|
"github.com/navidrome/navidrome/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
const LocalAgentName = "local"
|
||||||
|
|
||||||
|
const (
|
||||||
|
localBiography = "Biography not available"
|
||||||
|
)
|
||||||
|
|
||||||
|
type localAgent struct{}
|
||||||
|
|
||||||
|
func localsConstructor(_ model.DataStore) Interface {
|
||||||
|
return &localAgent{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *localAgent) AgentName() string {
|
||||||
|
return LocalAgentName
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *localAgent) GetBiography(ctx context.Context, id, name, mbid string) (string, error) {
|
||||||
|
return localBiography, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *localAgent) GetImages(_ context.Context, id, name, mbid string) ([]ArtistImage, error) {
|
||||||
|
return []ArtistImage{
|
||||||
|
p.artistImage(id, 300),
|
||||||
|
p.artistImage(id, 174),
|
||||||
|
p.artistImage(id, 64),
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *localAgent) artistImage(id string, size int) ArtistImage {
|
||||||
|
return ArtistImage{
|
||||||
|
filepath.Join(consts.URLPathPublicImages, artwork.Public(model.NewArtworkID(model.KindArtistArtwork, id), size)),
|
||||||
|
size,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
Register(LocalAgentName, localsConstructor)
|
||||||
|
}
|
|
@ -1,43 +0,0 @@
|
||||||
package agents
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
|
|
||||||
"github.com/navidrome/navidrome/consts"
|
|
||||||
"github.com/navidrome/navidrome/model"
|
|
||||||
)
|
|
||||||
|
|
||||||
const PlaceholderAgentName = "placeholder"
|
|
||||||
|
|
||||||
const (
|
|
||||||
placeholderArtistImageSmallUrl = consts.URLPathUI + "/artist-placeholder.webp"
|
|
||||||
placeholderArtistImageMediumUrl = consts.URLPathUI + "/artist-placeholder.webp"
|
|
||||||
placeholderArtistImageLargeUrl = consts.URLPathUI + "/artist-placeholder.webp"
|
|
||||||
placeholderBiography = "Biography not available"
|
|
||||||
)
|
|
||||||
|
|
||||||
type placeholderAgent struct{}
|
|
||||||
|
|
||||||
func placeholdersConstructor(_ model.DataStore) Interface {
|
|
||||||
return &placeholderAgent{}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *placeholderAgent) AgentName() string {
|
|
||||||
return PlaceholderAgentName
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *placeholderAgent) GetBiography(ctx context.Context, id, name, mbid string) (string, error) {
|
|
||||||
return placeholderBiography, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (p *placeholderAgent) GetImages(ctx context.Context, id, name, mbid string) ([]ArtistImage, error) {
|
|
||||||
return []ArtistImage{
|
|
||||||
{placeholderArtistImageLargeUrl, 300},
|
|
||||||
{placeholderArtistImageMediumUrl, 174},
|
|
||||||
{placeholderArtistImageSmallUrl, 64},
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
Register(PlaceholderAgentName, placeholdersConstructor)
|
|
||||||
}
|
|
|
@ -7,6 +7,7 @@ import (
|
||||||
"io"
|
"io"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/navidrome/navidrome/core/auth"
|
||||||
"github.com/navidrome/navidrome/core/ffmpeg"
|
"github.com/navidrome/navidrome/core/ffmpeg"
|
||||||
"github.com/navidrome/navidrome/log"
|
"github.com/navidrome/navidrome/log"
|
||||||
"github.com/navidrome/navidrome/model"
|
"github.com/navidrome/navidrome/model"
|
||||||
|
@ -73,6 +74,9 @@ func (a *artwork) getArtworkId(ctx context.Context, id string) (model.ArtworkID,
|
||||||
return model.ArtworkID{}, err
|
return model.ArtworkID{}, err
|
||||||
}
|
}
|
||||||
switch e := entity.(type) {
|
switch e := entity.(type) {
|
||||||
|
case *model.Artist:
|
||||||
|
artID = model.NewArtworkID(model.KindArtistArtwork, e.ID)
|
||||||
|
log.Trace(ctx, "ID is for an Artist", "id", id, "name", e.Name, "artist", e.Name)
|
||||||
case *model.Album:
|
case *model.Album:
|
||||||
artID = model.NewArtworkID(model.KindAlbumArtwork, e.ID)
|
artID = model.NewArtworkID(model.KindAlbumArtwork, e.ID)
|
||||||
log.Trace(ctx, "ID is for an Album", "id", id, "name", e.Name, "artist", e.AlbumArtist)
|
log.Trace(ctx, "ID is for an Album", "id", id, "name", e.Name, "artist", e.AlbumArtist)
|
||||||
|
@ -93,6 +97,8 @@ func (a *artwork) getArtworkReader(ctx context.Context, artID model.ArtworkID, s
|
||||||
artReader, err = resizedFromOriginal(ctx, a, artID, size)
|
artReader, err = resizedFromOriginal(ctx, a, artID, size)
|
||||||
} else {
|
} else {
|
||||||
switch artID.Kind {
|
switch artID.Kind {
|
||||||
|
case model.KindArtistArtwork:
|
||||||
|
artReader, err = newArtistReader(ctx, a, artID)
|
||||||
case model.KindAlbumArtwork:
|
case model.KindAlbumArtwork:
|
||||||
artReader, err = newAlbumArtworkReader(ctx, a, artID)
|
artReader, err = newAlbumArtworkReader(ctx, a, artID)
|
||||||
case model.KindMediaFileArtwork:
|
case model.KindMediaFileArtwork:
|
||||||
|
@ -105,3 +111,11 @@ func (a *artwork) getArtworkReader(ctx context.Context, artID model.ArtworkID, s
|
||||||
}
|
}
|
||||||
return artReader, err
|
return artReader, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func Public(artID model.ArtworkID, size int) string {
|
||||||
|
token, _ := auth.CreatePublicToken(map[string]any{
|
||||||
|
"id": artID.String(),
|
||||||
|
"size": size,
|
||||||
|
})
|
||||||
|
return token
|
||||||
|
}
|
||||||
|
|
|
@ -37,7 +37,7 @@ func (a *albumArtworkReader) LastUpdated() time.Time {
|
||||||
|
|
||||||
func (a *albumArtworkReader) Reader(ctx context.Context) (io.ReadCloser, string, error) {
|
func (a *albumArtworkReader) Reader(ctx context.Context) (io.ReadCloser, string, error) {
|
||||||
var ff = fromCoverArtPriority(ctx, a.a.ffmpeg, conf.Server.CoverArtPriority, a.album)
|
var ff = fromCoverArtPriority(ctx, a.a.ffmpeg, conf.Server.CoverArtPriority, a.album)
|
||||||
ff = append(ff, fromPlaceholder())
|
ff = append(ff, fromAlbumPlaceholder())
|
||||||
return selectImageReader(ctx, a.artID, ff...)
|
return selectImageReader(ctx, a.artID, ff...)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
35
core/artwork/reader_artist.go
Normal file
35
core/artwork/reader_artist.go
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
package artwork
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/navidrome/navidrome/conf"
|
||||||
|
"github.com/navidrome/navidrome/consts"
|
||||||
|
"github.com/navidrome/navidrome/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
type artistReader struct {
|
||||||
|
artID model.ArtworkID
|
||||||
|
}
|
||||||
|
|
||||||
|
func newArtistReader(_ context.Context, _ *artwork, artID model.ArtworkID) (*artistReader, error) {
|
||||||
|
a := &artistReader{
|
||||||
|
artID: artID,
|
||||||
|
}
|
||||||
|
return a, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *artistReader) LastUpdated() time.Time {
|
||||||
|
return consts.ServerStart // Invalidate cached placeholder every server start
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *artistReader) Key() string {
|
||||||
|
return fmt.Sprintf("placeholder.%d.0.%d", a.LastUpdated().UnixMilli(), conf.Server.CoverJpegQuality)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *artistReader) Reader(ctx context.Context) (io.ReadCloser, string, error) {
|
||||||
|
return selectImageReader(ctx, a.artID, fromArtistPlaceholder())
|
||||||
|
}
|
|
@ -31,5 +31,5 @@ func (a *emptyIDReader) Key() string {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *emptyIDReader) Reader(ctx context.Context) (io.ReadCloser, string, error) {
|
func (a *emptyIDReader) Reader(ctx context.Context) (io.ReadCloser, string, error) {
|
||||||
return selectImageReader(ctx, a.artID, fromPlaceholder())
|
return selectImageReader(ctx, a.artID, fromAlbumPlaceholder())
|
||||||
}
|
}
|
||||||
|
|
|
@ -53,13 +53,3 @@ func (a *mediafileArtworkReader) Reader(ctx context.Context) (io.ReadCloser, str
|
||||||
ff = append(ff, fromAlbum(ctx, a.a, a.mediafile.AlbumCoverArtID()))
|
ff = append(ff, fromAlbum(ctx, a.a, a.mediafile.AlbumCoverArtID()))
|
||||||
return selectImageReader(ctx, a.artID, ff...)
|
return selectImageReader(ctx, a.artID, ff...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func fromAlbum(ctx context.Context, a *artwork, id model.ArtworkID) sourceFunc {
|
|
||||||
return func() (io.ReadCloser, string, error) {
|
|
||||||
r, _, err := a.Get(ctx, id.String(), 0)
|
|
||||||
if err != nil {
|
|
||||||
return nil, "", err
|
|
||||||
}
|
|
||||||
return r, id.String(), nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -49,7 +49,7 @@ func (a *playlistArtworkReader) Reader(ctx context.Context) (io.ReadCloser, stri
|
||||||
if err == nil {
|
if err == nil {
|
||||||
ff = append(ff, a.fromGeneratedTile(ctx, pl.Tracks))
|
ff = append(ff, a.fromGeneratedTile(ctx, pl.Tracks))
|
||||||
}
|
}
|
||||||
ff = append(ff, fromPlaceholder())
|
ff = append(ff, fromAlbumPlaceholder())
|
||||||
return selectImageReader(ctx, a.artID, ff...)
|
return selectImageReader(ctx, a.artID, ff...)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -112,9 +112,26 @@ func fromFFmpegTag(ctx context.Context, ffmpeg ffmpeg.FFmpeg, path string) sourc
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func fromPlaceholder() sourceFunc {
|
func fromAlbum(ctx context.Context, a *artwork, id model.ArtworkID) sourceFunc {
|
||||||
|
return func() (io.ReadCloser, string, error) {
|
||||||
|
r, _, err := a.Get(ctx, id.String(), 0)
|
||||||
|
if err != nil {
|
||||||
|
return nil, "", err
|
||||||
|
}
|
||||||
|
return r, id.String(), nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func fromAlbumPlaceholder() sourceFunc {
|
||||||
return func() (io.ReadCloser, string, error) {
|
return func() (io.ReadCloser, string, error) {
|
||||||
r, _ := resources.FS().Open(consts.PlaceholderAlbumArt)
|
r, _ := resources.FS().Open(consts.PlaceholderAlbumArt)
|
||||||
return r, consts.PlaceholderAlbumArt, nil
|
return r, consts.PlaceholderAlbumArt, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func fromArtistPlaceholder() sourceFunc {
|
||||||
|
return func() (io.ReadCloser, string, error) {
|
||||||
|
r, _ := resources.FS().Open(consts.PlaceholderArtistArt)
|
||||||
|
return r, consts.PlaceholderArtistArt, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -12,13 +12,15 @@ type Kind struct{ prefix string }
|
||||||
|
|
||||||
var (
|
var (
|
||||||
KindMediaFileArtwork = Kind{"mf"}
|
KindMediaFileArtwork = Kind{"mf"}
|
||||||
|
KindArtistArtwork = Kind{"ar"}
|
||||||
KindAlbumArtwork = Kind{"al"}
|
KindAlbumArtwork = Kind{"al"}
|
||||||
KindPlaylistArtwork = Kind{"pl"}
|
KindPlaylistArtwork = Kind{"pl"}
|
||||||
)
|
)
|
||||||
|
|
||||||
var artworkKindList = []string{
|
var artworkKindList = []string{
|
||||||
KindAlbumArtwork.prefix,
|
|
||||||
KindMediaFileArtwork.prefix,
|
KindMediaFileArtwork.prefix,
|
||||||
|
KindArtistArtwork.prefix,
|
||||||
|
KindAlbumArtwork.prefix,
|
||||||
KindPlaylistArtwork.prefix,
|
KindPlaylistArtwork.prefix,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Before Width: | Height: | Size: 546 B After Width: | Height: | Size: 546 B |
|
@ -3,7 +3,6 @@ package public
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
|
||||||
"io"
|
"io"
|
||||||
"net/http"
|
"net/http"
|
||||||
"time"
|
"time"
|
||||||
|
@ -27,15 +26,6 @@ func New(artwork artwork.Artwork) *Router {
|
||||||
p := &Router{artwork: artwork}
|
p := &Router{artwork: artwork}
|
||||||
p.Handler = p.routes()
|
p.Handler = p.routes()
|
||||||
|
|
||||||
t, err := auth.CreatePublicToken(map[string]any{
|
|
||||||
"id": "al-ee07551e7371500da55e23ae8520f1d8",
|
|
||||||
"size": 300,
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
fmt.Println("!!!!!!!!!!!!!!!!!", t, "!!!!!!!!!!!!!!!!")
|
|
||||||
|
|
||||||
return p
|
return p
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue