mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-03 20:47:35 +03:00
* wip: API endpoint for creating playlists from m3u files * wip: get user id from context * temporarily disable failing test * custom logic for playlist route to accomodate m3u content type * incorporate playlist parsing into existing logic in core * re-enable test * fix locally failing test * Address requested changes. * Improve ImportFile tests. * Remove ownerID as a parameter of ImportM3U. * Write tests for ImportM3U. * Separate ImportM3U test into two. * Test OwnerID and playlist Name. --------- Co-authored-by: Sam Watson <SwatsonCodes@users.noreply.github.com> Co-authored-by: caiocotts <caio@cotts.com.br>
116 lines
3 KiB
Go
116 lines
3 KiB
Go
package core
|
|
|
|
import (
|
|
"context"
|
|
"os"
|
|
"time"
|
|
|
|
"github.com/navidrome/navidrome/model/request"
|
|
|
|
"github.com/navidrome/navidrome/model"
|
|
"github.com/navidrome/navidrome/tests"
|
|
. "github.com/onsi/ginkgo/v2"
|
|
. "github.com/onsi/gomega"
|
|
)
|
|
|
|
var _ = Describe("Playlists", func() {
|
|
var ds model.DataStore
|
|
var ps Playlists
|
|
var mp mockedPlaylist
|
|
ctx := context.Background()
|
|
|
|
BeforeEach(func() {
|
|
mp = mockedPlaylist{}
|
|
ds = &tests.MockDataStore{
|
|
MockedMediaFile: &mockedMediaFile{},
|
|
MockedPlaylist: &mp,
|
|
}
|
|
ctx = request.WithUser(ctx, model.User{ID: "123"})
|
|
})
|
|
|
|
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.OwnerID).To(Equal("123"))
|
|
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"))
|
|
Expect(mp.last).To(Equal(pls))
|
|
})
|
|
|
|
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))
|
|
})
|
|
|
|
})
|
|
|
|
Describe("ImportM3U", func() {
|
|
BeforeEach(func() {
|
|
ps = NewPlaylists(ds)
|
|
ctx = request.WithUser(ctx, model.User{ID: "123"})
|
|
})
|
|
|
|
It("parses well-formed playlists", func() {
|
|
f, _ := os.Open("tests/fixtures/playlists/pls-post-with-name.m3u")
|
|
defer f.Close()
|
|
pls, err := ps.ImportM3U(ctx, f)
|
|
Expect(pls.OwnerID).To(Equal("123"))
|
|
Expect(pls.Name).To(Equal("playlist 1"))
|
|
Expect(err).To(BeNil())
|
|
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"))
|
|
Expect(mp.last).To(Equal(pls))
|
|
f.Close()
|
|
|
|
})
|
|
|
|
It("sets the playlist name as a timestamp if the #PLAYLIST directive is not present", func() {
|
|
f, _ := os.Open("tests/fixtures/playlists/pls-post.m3u")
|
|
defer f.Close()
|
|
pls, err := ps.ImportM3U(ctx, f)
|
|
Expect(err).To(BeNil())
|
|
_, err = time.Parse(time.RFC3339, pls.Name)
|
|
Expect(err).To(BeNil())
|
|
})
|
|
})
|
|
})
|
|
|
|
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 {
|
|
last *model.Playlist
|
|
model.PlaylistRepository
|
|
}
|
|
|
|
func (r *mockedPlaylist) FindByPath(string) (*model.Playlist, error) {
|
|
return nil, model.ErrNotFound
|
|
}
|
|
|
|
func (r *mockedPlaylist) Put(pls *model.Playlist) error {
|
|
r.last = pls
|
|
return nil
|
|
}
|