mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-05 05:27:37 +03:00
Import smart playlists (extension .nsp)
This commit is contained in:
parent
21da1df4ea
commit
1a96e9fe65
12 changed files with 242 additions and 110 deletions
|
@ -1,7 +1,8 @@
|
||||||
// Code generated by Wire. DO NOT EDIT.
|
// Code generated by Wire. DO NOT EDIT.
|
||||||
|
|
||||||
//go:generate go run github.com/google/wire/cmd/wire
|
//go:generate go run github.com/google/wire/cmd/wire
|
||||||
//+build !wireinject
|
//go:build !wireinject
|
||||||
|
// +build !wireinject
|
||||||
|
|
||||||
package cmd
|
package cmd
|
||||||
|
|
||||||
|
@ -69,11 +70,12 @@ func CreateLastFMRouter() *lastfm.Router {
|
||||||
func createScanner() scanner.Scanner {
|
func createScanner() scanner.Scanner {
|
||||||
sqlDB := db.Db()
|
sqlDB := db.Db()
|
||||||
dataStore := persistence.New(sqlDB)
|
dataStore := persistence.New(sqlDB)
|
||||||
|
playlists := core.NewPlaylists(dataStore)
|
||||||
artworkCache := core.GetImageCache()
|
artworkCache := core.GetImageCache()
|
||||||
artwork := core.NewArtwork(dataStore, artworkCache)
|
artwork := core.NewArtwork(dataStore, artworkCache)
|
||||||
cacheWarmer := core.NewCacheWarmer(artwork, artworkCache)
|
cacheWarmer := core.NewCacheWarmer(artwork, artworkCache)
|
||||||
broker := events.GetBroker()
|
broker := events.GetBroker()
|
||||||
scannerScanner := scanner.New(dataStore, cacheWarmer, broker)
|
scannerScanner := scanner.New(dataStore, playlists, cacheWarmer, broker)
|
||||||
return scannerScanner
|
return scannerScanner
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,68 +1,76 @@
|
||||||
package scanner
|
package core
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
"bytes"
|
"bytes"
|
||||||
"context"
|
"context"
|
||||||
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io"
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/mattn/go-zglob"
|
|
||||||
"github.com/navidrome/navidrome/conf"
|
|
||||||
"github.com/navidrome/navidrome/log"
|
"github.com/navidrome/navidrome/log"
|
||||||
"github.com/navidrome/navidrome/model"
|
"github.com/navidrome/navidrome/model"
|
||||||
"github.com/navidrome/navidrome/model/request"
|
"github.com/navidrome/navidrome/model/request"
|
||||||
"github.com/navidrome/navidrome/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type playlistSync struct {
|
type Playlists interface {
|
||||||
|
ImportFile(ctx context.Context, dir string, fname string) (*model.Playlist, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type playlists struct {
|
||||||
ds model.DataStore
|
ds model.DataStore
|
||||||
rootFolder string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func newPlaylistSync(ds model.DataStore, rootFolder string) *playlistSync {
|
func NewPlaylists(ds model.DataStore) Playlists {
|
||||||
return &playlistSync{ds: ds, rootFolder: rootFolder}
|
return &playlists{ds: ds}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *playlistSync) processPlaylists(ctx context.Context, dir string) int64 {
|
func IsPlaylist(filePath string) bool {
|
||||||
if !s.inPlaylistsPath(dir) {
|
extension := strings.ToLower(filepath.Ext(filePath))
|
||||||
return 0
|
return extension == ".m3u" || extension == ".m3u8" || extension == ".nsp"
|
||||||
}
|
}
|
||||||
var count int64
|
|
||||||
files, err := os.ReadDir(dir)
|
func (s *playlists) ImportFile(ctx context.Context, dir string, fname string) (*model.Playlist, error) {
|
||||||
|
pls, err := s.parsePlaylist(ctx, fname, dir)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error(ctx, "Error reading files", "dir", dir, err)
|
log.Error(ctx, "Error parsing playlist", "playlist", fname, err)
|
||||||
return count
|
return nil, err
|
||||||
}
|
|
||||||
for _, f := range files {
|
|
||||||
if !utils.IsPlaylist(f.Name()) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
pls, err := s.parsePlaylist(ctx, f.Name(), dir)
|
|
||||||
if err != nil {
|
|
||||||
log.Error(ctx, "Error parsing playlist", "playlist", f.Name(), err)
|
|
||||||
continue
|
|
||||||
}
|
}
|
||||||
log.Debug("Found playlist", "name", pls.Name, "lastUpdated", pls.UpdatedAt, "path", pls.Path, "numTracks", len(pls.Tracks))
|
log.Debug("Found playlist", "name", pls.Name, "lastUpdated", pls.UpdatedAt, "path", pls.Path, "numTracks", len(pls.Tracks))
|
||||||
err = s.updatePlaylist(ctx, pls)
|
err = s.updatePlaylist(ctx, pls)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error(ctx, "Error updating playlist", "playlist", f.Name(), err)
|
log.Error(ctx, "Error updating playlist", "playlist", fname, err)
|
||||||
}
|
}
|
||||||
count++
|
return pls, err
|
||||||
}
|
|
||||||
return count
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *playlistSync) parsePlaylist(ctx context.Context, playlistFile string, baseDir string) (*model.Playlist, error) {
|
func (s *playlists) parsePlaylist(ctx context.Context, playlistFile string, baseDir string) (*model.Playlist, error) {
|
||||||
playlistPath := filepath.Join(baseDir, playlistFile)
|
pls, err := s.newSyncedPlaylist(baseDir, playlistFile)
|
||||||
file, err := os.Open(playlistPath)
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
file, err := os.Open(pls.Path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
defer file.Close()
|
defer file.Close()
|
||||||
|
|
||||||
|
extension := strings.ToLower(filepath.Ext(playlistFile))
|
||||||
|
switch extension {
|
||||||
|
case ".nsp":
|
||||||
|
return s.parseNSP(ctx, pls, file)
|
||||||
|
default:
|
||||||
|
return s.parseM3U(ctx, pls, baseDir, file)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *playlists) newSyncedPlaylist(baseDir string, playlistFile string) (*model.Playlist, error) {
|
||||||
|
playlistPath := filepath.Join(baseDir, playlistFile)
|
||||||
info, err := os.Stat(playlistPath)
|
info, err := os.Stat(playlistPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -79,7 +87,24 @@ func (s *playlistSync) parsePlaylist(ctx context.Context, playlistFile string, b
|
||||||
Sync: true,
|
Sync: true,
|
||||||
UpdatedAt: info.ModTime(),
|
UpdatedAt: info.ModTime(),
|
||||||
}
|
}
|
||||||
|
return pls, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *playlists) parseNSP(ctx context.Context, pls *model.Playlist, file io.Reader) (*model.Playlist, error) {
|
||||||
|
content, err := io.ReadAll(file)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
pls.Rules = &model.SmartPlaylist{}
|
||||||
|
err = json.Unmarshal(content, pls.Rules)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return pls, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *playlists) parseM3U(ctx context.Context, pls *model.Playlist, baseDir string, file io.Reader) (*model.Playlist, error) {
|
||||||
mediaFileRepository := s.ds.MediaFile(ctx)
|
mediaFileRepository := s.ds.MediaFile(ctx)
|
||||||
scanner := bufio.NewScanner(file)
|
scanner := bufio.NewScanner(file)
|
||||||
scanner.Split(scanLines)
|
scanner.Split(scanLines)
|
||||||
|
@ -99,7 +124,7 @@ func (s *playlistSync) parsePlaylist(ctx context.Context, playlistFile string, b
|
||||||
}
|
}
|
||||||
mf, err := mediaFileRepository.FindByPath(path)
|
mf, err := mediaFileRepository.FindByPath(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Warn(ctx, "Path in playlist not found", "playlist", playlistFile, "path", path, err)
|
log.Warn(ctx, "Path in playlist not found", "playlist", pls.Name, "path", path, err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
mfs = append(mfs, *mf)
|
mfs = append(mfs, *mf)
|
||||||
|
@ -110,7 +135,7 @@ func (s *playlistSync) parsePlaylist(ctx context.Context, playlistFile string, b
|
||||||
return pls, scanner.Err()
|
return pls, scanner.Err()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *playlistSync) updatePlaylist(ctx context.Context, newPls *model.Playlist) error {
|
func (s *playlists) updatePlaylist(ctx context.Context, newPls *model.Playlist) error {
|
||||||
owner, _ := request.UsernameFrom(ctx)
|
owner, _ := request.UsernameFrom(ctx)
|
||||||
|
|
||||||
pls, err := s.ds.Playlist(ctx).FindByPath(newPls.Path)
|
pls, err := s.ds.Playlist(ctx).FindByPath(newPls.Path)
|
||||||
|
@ -136,16 +161,6 @@ func (s *playlistSync) updatePlaylist(ctx context.Context, newPls *model.Playlis
|
||||||
return s.ds.Playlist(ctx).Put(newPls)
|
return s.ds.Playlist(ctx).Put(newPls)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *playlistSync) inPlaylistsPath(dir string) bool {
|
|
||||||
rel, _ := filepath.Rel(s.rootFolder, dir)
|
|
||||||
for _, path := range strings.Split(conf.Server.PlaylistsPath, string(filepath.ListSeparator)) {
|
|
||||||
if match, _ := zglob.Match(path, rel); match {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// From https://stackoverflow.com/a/41433698
|
// From https://stackoverflow.com/a/41433698
|
||||||
func scanLines(data []byte, atEOF bool) (advance int, token []byte, err error) {
|
func scanLines(data []byte, atEOF bool) (advance int, token []byte, err error) {
|
||||||
if atEOF && len(data) == 0 {
|
if atEOF && len(data) == 0 {
|
90
core/playlists_test.go
Normal file
90
core/playlists_test.go
Normal file
|
@ -0,0 +1,90 @@
|
||||||
|
package core
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"path/filepath"
|
||||||
|
|
||||||
|
"github.com/navidrome/navidrome/model"
|
||||||
|
"github.com/navidrome/navidrome/tests"
|
||||||
|
|
||||||
|
. "github.com/onsi/ginkgo"
|
||||||
|
. "github.com/onsi/gomega"
|
||||||
|
)
|
||||||
|
|
||||||
|
var _ = Describe("IsPlaylist", func() {
|
||||||
|
It("returns true for a M3U file", func() {
|
||||||
|
Expect(IsPlaylist(filepath.Join("path", "to", "test.m3u"))).To(BeTrue())
|
||||||
|
})
|
||||||
|
|
||||||
|
It("returns true for a M3U8 file", func() {
|
||||||
|
Expect(IsPlaylist(filepath.Join("path", "to", "test.m3u8"))).To(BeTrue())
|
||||||
|
})
|
||||||
|
|
||||||
|
It("returns false for a non-playlist file", func() {
|
||||||
|
Expect(IsPlaylist("testm3u")).To(BeFalse())
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
var _ = Describe("Playlists", func() {
|
||||||
|
var ds model.DataStore
|
||||||
|
var ps Playlists
|
||||||
|
ctx := context.Background()
|
||||||
|
|
||||||
|
BeforeEach(func() {
|
||||||
|
ds = &tests.MockDataStore{
|
||||||
|
MockedMediaFile: &mockedMediaFile{},
|
||||||
|
MockedPlaylist: &mockedPlaylist{},
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
Describe("ImportFile", func() {
|
||||||
|
BeforeEach(func() {
|
||||||
|
ps = NewPlaylists(ds)
|
||||||
|
})
|
||||||
|
|
||||||
|
It("parses well-formed playlists", func() {
|
||||||
|
pls, err := ps.ImportFile(ctx, "tests/fixtures", "playlists/pls1.m3u")
|
||||||
|
Expect(err).To(BeNil())
|
||||||
|
Expect(pls.Tracks).To(HaveLen(3))
|
||||||
|
Expect(pls.Tracks[0].Path).To(Equal("tests/fixtures/test.mp3"))
|
||||||
|
Expect(pls.Tracks[1].Path).To(Equal("tests/fixtures/test.ogg"))
|
||||||
|
Expect(pls.Tracks[2].Path).To(Equal("/tests/fixtures/01 Invisible (RED) Edit Version.mp3"))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("parses playlists using LF ending", func() {
|
||||||
|
pls, err := ps.ImportFile(ctx, "tests/fixtures/playlists", "lf-ended.m3u")
|
||||||
|
Expect(err).To(BeNil())
|
||||||
|
Expect(pls.Tracks).To(HaveLen(2))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("parses playlists using CR ending (old Mac format)", func() {
|
||||||
|
pls, err := ps.ImportFile(ctx, "tests/fixtures/playlists", "cr-ended.m3u")
|
||||||
|
Expect(err).To(BeNil())
|
||||||
|
Expect(pls.Tracks).To(HaveLen(2))
|
||||||
|
})
|
||||||
|
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
type mockedMediaFile struct {
|
||||||
|
model.MediaFileRepository
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *mockedMediaFile) FindByPath(s string) (*model.MediaFile, error) {
|
||||||
|
return &model.MediaFile{
|
||||||
|
ID: "123",
|
||||||
|
Path: s,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type mockedPlaylist struct {
|
||||||
|
model.PlaylistRepository
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *mockedPlaylist) FindByPath(path string) (*model.Playlist, error) {
|
||||||
|
return nil, model.ErrNotFound
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *mockedPlaylist) Put(pls *model.Playlist) error {
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -20,4 +20,5 @@ var Set = wire.NewSet(
|
||||||
transcoder.New,
|
transcoder.New,
|
||||||
scrobbler.GetPlayTracker,
|
scrobbler.GetPlayTracker,
|
||||||
NewShare,
|
NewShare,
|
||||||
|
NewPlaylists,
|
||||||
)
|
)
|
||||||
|
|
63
scanner/playlist_importer.go
Normal file
63
scanner/playlist_importer.go
Normal file
|
@ -0,0 +1,63 @@
|
||||||
|
package scanner
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/mattn/go-zglob"
|
||||||
|
"github.com/navidrome/navidrome/conf"
|
||||||
|
"github.com/navidrome/navidrome/core"
|
||||||
|
"github.com/navidrome/navidrome/log"
|
||||||
|
"github.com/navidrome/navidrome/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
type playlistImporter struct {
|
||||||
|
ds model.DataStore
|
||||||
|
pls core.Playlists
|
||||||
|
rootFolder string
|
||||||
|
}
|
||||||
|
|
||||||
|
func newPlaylistImporter(ds model.DataStore, playlists core.Playlists, rootFolder string) *playlistImporter {
|
||||||
|
return &playlistImporter{ds: ds, pls: playlists, rootFolder: rootFolder}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *playlistImporter) processPlaylists(ctx context.Context, dir string) int64 {
|
||||||
|
if !s.inPlaylistsPath(dir) {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
var count int64
|
||||||
|
files, err := os.ReadDir(dir)
|
||||||
|
if err != nil {
|
||||||
|
log.Error(ctx, "Error reading files", "dir", dir, err)
|
||||||
|
return count
|
||||||
|
}
|
||||||
|
for _, f := range files {
|
||||||
|
if !core.IsPlaylist(f.Name()) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
pls, err := s.pls.ImportFile(ctx, dir, f.Name())
|
||||||
|
if err != nil {
|
||||||
|
log.Error(ctx, "Error parsing playlist", "playlist", f.Name(), err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if pls.IsSmartPlaylist() {
|
||||||
|
log.Debug("Imported smart playlist", "name", pls.Name, "lastUpdated", pls.UpdatedAt, "path", pls.Path, "numTracks", len(pls.Tracks))
|
||||||
|
} else {
|
||||||
|
log.Debug("Imported playlist", "name", pls.Name, "lastUpdated", pls.UpdatedAt, "path", pls.Path, "numTracks", len(pls.Tracks))
|
||||||
|
}
|
||||||
|
count++
|
||||||
|
}
|
||||||
|
return count
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *playlistImporter) inPlaylistsPath(dir string) bool {
|
||||||
|
rel, _ := filepath.Rel(s.rootFolder, dir)
|
||||||
|
for _, path := range strings.Split(conf.Server.PlaylistsPath, string(filepath.ListSeparator)) {
|
||||||
|
if match, _ := zglob.Match(path, rel); match {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
|
@ -3,6 +3,8 @@ package scanner
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
|
||||||
|
"github.com/navidrome/navidrome/core"
|
||||||
|
|
||||||
"github.com/navidrome/navidrome/conf"
|
"github.com/navidrome/navidrome/conf"
|
||||||
"github.com/navidrome/navidrome/consts"
|
"github.com/navidrome/navidrome/consts"
|
||||||
"github.com/navidrome/navidrome/model"
|
"github.com/navidrome/navidrome/model"
|
||||||
|
@ -11,9 +13,10 @@ import (
|
||||||
. "github.com/onsi/gomega"
|
. "github.com/onsi/gomega"
|
||||||
)
|
)
|
||||||
|
|
||||||
var _ = Describe("playlistSync", func() {
|
var _ = Describe("playlistImporter", func() {
|
||||||
var ds model.DataStore
|
var ds model.DataStore
|
||||||
var ps *playlistSync
|
var ps *playlistImporter
|
||||||
|
var pls core.Playlists
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
|
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
|
@ -21,34 +24,7 @@ var _ = Describe("playlistSync", func() {
|
||||||
MockedMediaFile: &mockedMediaFile{},
|
MockedMediaFile: &mockedMediaFile{},
|
||||||
MockedPlaylist: &mockedPlaylist{},
|
MockedPlaylist: &mockedPlaylist{},
|
||||||
}
|
}
|
||||||
})
|
pls = core.NewPlaylists(ds)
|
||||||
|
|
||||||
Describe("parsePlaylist", func() {
|
|
||||||
BeforeEach(func() {
|
|
||||||
ps = newPlaylistSync(ds, "tests/")
|
|
||||||
})
|
|
||||||
|
|
||||||
It("parses well-formed playlists", func() {
|
|
||||||
pls, err := ps.parsePlaylist(ctx, "playlists/pls1.m3u", "tests/fixtures")
|
|
||||||
Expect(err).To(BeNil())
|
|
||||||
Expect(pls.Tracks).To(HaveLen(3))
|
|
||||||
Expect(pls.Tracks[0].Path).To(Equal("tests/fixtures/test.mp3"))
|
|
||||||
Expect(pls.Tracks[1].Path).To(Equal("tests/fixtures/test.ogg"))
|
|
||||||
Expect(pls.Tracks[2].Path).To(Equal("/tests/fixtures/01 Invisible (RED) Edit Version.mp3"))
|
|
||||||
})
|
|
||||||
|
|
||||||
It("parses playlists using LF ending", func() {
|
|
||||||
pls, err := ps.parsePlaylist(ctx, "lf-ended.m3u", "tests/fixtures/playlists")
|
|
||||||
Expect(err).To(BeNil())
|
|
||||||
Expect(pls.Tracks).To(HaveLen(2))
|
|
||||||
})
|
|
||||||
|
|
||||||
It("parses playlists using CR ending (old Mac format)", func() {
|
|
||||||
pls, err := ps.parsePlaylist(ctx, "cr-ended.m3u", "tests/fixtures/playlists")
|
|
||||||
Expect(err).To(BeNil())
|
|
||||||
Expect(pls.Tracks).To(HaveLen(2))
|
|
||||||
})
|
|
||||||
|
|
||||||
})
|
})
|
||||||
|
|
||||||
Describe("processPlaylists", func() {
|
Describe("processPlaylists", func() {
|
||||||
|
@ -57,19 +33,19 @@ var _ = Describe("playlistSync", func() {
|
||||||
conf.Server.PlaylistsPath = consts.DefaultPlaylistsPath
|
conf.Server.PlaylistsPath = consts.DefaultPlaylistsPath
|
||||||
})
|
})
|
||||||
It("finds and import playlists at the top level", func() {
|
It("finds and import playlists at the top level", func() {
|
||||||
ps = newPlaylistSync(ds, "tests/fixtures/playlists/subfolder1")
|
ps = newPlaylistImporter(ds, pls, "tests/fixtures/playlists/subfolder1")
|
||||||
Expect(ps.processPlaylists(ctx, "tests/fixtures/playlists/subfolder1")).To(Equal(int64(1)))
|
Expect(ps.processPlaylists(ctx, "tests/fixtures/playlists/subfolder1")).To(Equal(int64(1)))
|
||||||
})
|
})
|
||||||
|
|
||||||
It("finds and import playlists at any subfolder level", func() {
|
It("finds and import playlists at any subfolder level", func() {
|
||||||
ps = newPlaylistSync(ds, "tests")
|
ps = newPlaylistImporter(ds, pls, "tests")
|
||||||
Expect(ps.processPlaylists(ctx, "tests/fixtures/playlists/subfolder1")).To(Equal(int64(1)))
|
Expect(ps.processPlaylists(ctx, "tests/fixtures/playlists/subfolder1")).To(Equal(int64(1)))
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
It("ignores playlists not in the PlaylistsPath", func() {
|
It("ignores playlists not in the PlaylistsPath", func() {
|
||||||
conf.Server.PlaylistsPath = "subfolder1"
|
conf.Server.PlaylistsPath = "subfolder1"
|
||||||
ps = newPlaylistSync(ds, "tests/fixtures/playlists")
|
ps = newPlaylistImporter(ds, pls, "tests/fixtures/playlists")
|
||||||
|
|
||||||
Expect(ps.processPlaylists(ctx, "tests/fixtures/playlists/subfolder1")).To(Equal(int64(1)))
|
Expect(ps.processPlaylists(ctx, "tests/fixtures/playlists/subfolder1")).To(Equal(int64(1)))
|
||||||
Expect(ps.processPlaylists(ctx, "tests/fixtures/playlists/subfolder2")).To(Equal(int64(0)))
|
Expect(ps.processPlaylists(ctx, "tests/fixtures/playlists/subfolder2")).To(Equal(int64(0)))
|
||||||
|
@ -77,7 +53,7 @@ var _ = Describe("playlistSync", func() {
|
||||||
|
|
||||||
It("only imports playlists from the root of MusicFolder if PlaylistsPath is '.'", func() {
|
It("only imports playlists from the root of MusicFolder if PlaylistsPath is '.'", func() {
|
||||||
conf.Server.PlaylistsPath = "."
|
conf.Server.PlaylistsPath = "."
|
||||||
ps = newPlaylistSync(ds, "tests/fixtures/playlists")
|
ps = newPlaylistImporter(ds, pls, "tests/fixtures/playlists")
|
||||||
|
|
||||||
Expect(ps.processPlaylists(ctx, "tests/fixtures/playlists")).To(Equal(int64(3)))
|
Expect(ps.processPlaylists(ctx, "tests/fixtures/playlists")).To(Equal(int64(3)))
|
||||||
Expect(ps.processPlaylists(ctx, "tests/fixtures/playlists/subfolder1")).To(Equal(int64(0)))
|
Expect(ps.processPlaylists(ctx, "tests/fixtures/playlists/subfolder1")).To(Equal(int64(0)))
|
|
@ -46,6 +46,7 @@ type scanner struct {
|
||||||
status map[string]*scanStatus
|
status map[string]*scanStatus
|
||||||
lock *sync.RWMutex
|
lock *sync.RWMutex
|
||||||
ds model.DataStore
|
ds model.DataStore
|
||||||
|
pls core.Playlists
|
||||||
cacheWarmer core.CacheWarmer
|
cacheWarmer core.CacheWarmer
|
||||||
broker events.Broker
|
broker events.Broker
|
||||||
}
|
}
|
||||||
|
@ -57,9 +58,10 @@ type scanStatus struct {
|
||||||
lastUpdate time.Time
|
lastUpdate time.Time
|
||||||
}
|
}
|
||||||
|
|
||||||
func New(ds model.DataStore, cacheWarmer core.CacheWarmer, broker events.Broker) Scanner {
|
func New(ds model.DataStore, playlists core.Playlists, cacheWarmer core.CacheWarmer, broker events.Broker) Scanner {
|
||||||
s := &scanner{
|
s := &scanner{
|
||||||
ds: ds,
|
ds: ds,
|
||||||
|
pls: playlists,
|
||||||
cacheWarmer: cacheWarmer,
|
cacheWarmer: cacheWarmer,
|
||||||
broker: broker,
|
broker: broker,
|
||||||
folders: map[string]FolderScanner{},
|
folders: map[string]FolderScanner{},
|
||||||
|
@ -250,5 +252,5 @@ func (s *scanner) loadFolders() {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *scanner) newScanner(f model.MediaFolder) FolderScanner {
|
func (s *scanner) newScanner(f model.MediaFolder) FolderScanner {
|
||||||
return NewTagScanner(f.Path, s.ds, s.cacheWarmer)
|
return NewTagScanner(f.Path, s.ds, s.pls, s.cacheWarmer)
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,15 +22,15 @@ type TagScanner struct {
|
||||||
rootFolder string
|
rootFolder string
|
||||||
ds model.DataStore
|
ds model.DataStore
|
||||||
cacheWarmer core.CacheWarmer
|
cacheWarmer core.CacheWarmer
|
||||||
plsSync *playlistSync
|
plsSync *playlistImporter
|
||||||
cnt *counters
|
cnt *counters
|
||||||
mapper *mediaFileMapper
|
mapper *mediaFileMapper
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewTagScanner(rootFolder string, ds model.DataStore, cacheWarmer core.CacheWarmer) *TagScanner {
|
func NewTagScanner(rootFolder string, ds model.DataStore, playlists core.Playlists, cacheWarmer core.CacheWarmer) *TagScanner {
|
||||||
return &TagScanner{
|
return &TagScanner{
|
||||||
rootFolder: rootFolder,
|
rootFolder: rootFolder,
|
||||||
plsSync: newPlaylistSync(ds, rootFolder),
|
plsSync: newPlaylistImporter(ds, playlists, rootFolder),
|
||||||
ds: ds,
|
ds: ds,
|
||||||
cacheWarmer: cacheWarmer,
|
cacheWarmer: cacheWarmer,
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,7 @@ import (
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/navidrome/navidrome/consts"
|
"github.com/navidrome/navidrome/consts"
|
||||||
|
"github.com/navidrome/navidrome/core"
|
||||||
"github.com/navidrome/navidrome/log"
|
"github.com/navidrome/navidrome/log"
|
||||||
"github.com/navidrome/navidrome/utils"
|
"github.com/navidrome/navidrome/utils"
|
||||||
)
|
)
|
||||||
|
@ -95,7 +96,7 @@ func loadDir(ctx context.Context, dirPath string) ([]string, *dirStats, error) {
|
||||||
if utils.IsAudioFile(entry.Name()) {
|
if utils.IsAudioFile(entry.Name()) {
|
||||||
stats.AudioFilesCount++
|
stats.AudioFilesCount++
|
||||||
} else {
|
} else {
|
||||||
stats.HasPlaylist = stats.HasPlaylist || utils.IsPlaylist(entry.Name())
|
stats.HasPlaylist = stats.HasPlaylist || core.IsPlaylist(entry.Name())
|
||||||
stats.HasImages = stats.HasImages || utils.IsImageFile(entry.Name())
|
stats.HasImages = stats.HasImages || utils.IsImageFile(entry.Name())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
// Code generated by Wire. DO NOT EDIT.
|
// Code generated by Wire. DO NOT EDIT.
|
||||||
|
|
||||||
//go:generate go run github.com/google/wire/cmd/wire
|
//go:generate go run github.com/google/wire/cmd/wire
|
||||||
//+build !wireinject
|
//go:build !wireinject
|
||||||
|
// +build !wireinject
|
||||||
|
|
||||||
package subsonic
|
package subsonic
|
||||||
|
|
||||||
|
|
|
@ -21,8 +21,3 @@ func IsImageFile(filePath string) bool {
|
||||||
extension := filepath.Ext(filePath)
|
extension := filepath.Ext(filePath)
|
||||||
return strings.HasPrefix(mime.TypeByExtension(extension), "image/")
|
return strings.HasPrefix(mime.TypeByExtension(extension), "image/")
|
||||||
}
|
}
|
||||||
|
|
||||||
func IsPlaylist(filePath string) bool {
|
|
||||||
extension := strings.ToLower(filepath.Ext(filePath))
|
|
||||||
return extension == ".m3u" || extension == ".m3u8"
|
|
||||||
}
|
|
||||||
|
|
|
@ -43,18 +43,4 @@ var _ = Describe("Files", func() {
|
||||||
Expect(IsImageFile("test.mp3")).To(BeFalse())
|
Expect(IsImageFile("test.mp3")).To(BeFalse())
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
Describe("IsPlaylist", func() {
|
|
||||||
It("returns true for a M3U file", func() {
|
|
||||||
Expect(IsPlaylist(filepath.Join("path", "to", "test.m3u"))).To(BeTrue())
|
|
||||||
})
|
|
||||||
|
|
||||||
It("returns true for a M3U8 file", func() {
|
|
||||||
Expect(IsPlaylist(filepath.Join("path", "to", "test.m3u8"))).To(BeTrue())
|
|
||||||
})
|
|
||||||
|
|
||||||
It("returns false for a non-playlist file", func() {
|
|
||||||
Expect(IsPlaylist("testm3u")).To(BeFalse())
|
|
||||||
})
|
|
||||||
})
|
|
||||||
})
|
})
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue