mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 13:07:36 +03:00
Introduced engine.Scrobbler
Also refactored mocks into their original packages, to avoid cyclic references. Is there a better way to have mocks in GoLang tests?
This commit is contained in:
parent
4aa02e68e5
commit
b660a70688
16 changed files with 158 additions and 47 deletions
|
@ -5,8 +5,8 @@ import (
|
||||||
|
|
||||||
"github.com/deluan/gosonic/api/responses"
|
"github.com/deluan/gosonic/api/responses"
|
||||||
"github.com/deluan/gosonic/domain"
|
"github.com/deluan/gosonic/domain"
|
||||||
|
"github.com/deluan/gosonic/persistence"
|
||||||
. "github.com/deluan/gosonic/tests"
|
. "github.com/deluan/gosonic/tests"
|
||||||
"github.com/deluan/gosonic/tests/mocks"
|
|
||||||
"github.com/deluan/gosonic/utils"
|
"github.com/deluan/gosonic/utils"
|
||||||
. "github.com/smartystreets/goconvey/convey"
|
. "github.com/smartystreets/goconvey/convey"
|
||||||
)
|
)
|
||||||
|
@ -14,7 +14,7 @@ import (
|
||||||
func TestGetAlbumList(t *testing.T) {
|
func TestGetAlbumList(t *testing.T) {
|
||||||
Init(t, false)
|
Init(t, false)
|
||||||
|
|
||||||
mockAlbumRepo := mocks.CreateMockAlbumRepo()
|
mockAlbumRepo := persistence.CreateMockAlbumRepo()
|
||||||
utils.DefineSingleton(new(domain.AlbumRepository), func() domain.AlbumRepository {
|
utils.DefineSingleton(new(domain.AlbumRepository), func() domain.AlbumRepository {
|
||||||
return mockAlbumRepo
|
return mockAlbumRepo
|
||||||
})
|
})
|
||||||
|
|
|
@ -46,6 +46,9 @@ func (c *BaseAPIController) ParamInt(param string, def int) int {
|
||||||
|
|
||||||
func (c *BaseAPIController) ParamBool(param string, def bool) bool {
|
func (c *BaseAPIController) ParamBool(param string, def bool) bool {
|
||||||
value := def
|
value := def
|
||||||
|
if c.Input().Get(param) == "" {
|
||||||
|
return def
|
||||||
|
}
|
||||||
c.Ctx.Input.Bind(&value, param)
|
c.Ctx.Input.Bind(&value, param)
|
||||||
return value
|
return value
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,8 +7,8 @@ import (
|
||||||
"github.com/deluan/gosonic/consts"
|
"github.com/deluan/gosonic/consts"
|
||||||
"github.com/deluan/gosonic/domain"
|
"github.com/deluan/gosonic/domain"
|
||||||
"github.com/deluan/gosonic/engine"
|
"github.com/deluan/gosonic/engine"
|
||||||
|
"github.com/deluan/gosonic/persistence"
|
||||||
. "github.com/deluan/gosonic/tests"
|
. "github.com/deluan/gosonic/tests"
|
||||||
"github.com/deluan/gosonic/tests/mocks"
|
|
||||||
"github.com/deluan/gosonic/utils"
|
"github.com/deluan/gosonic/utils"
|
||||||
. "github.com/smartystreets/goconvey/convey"
|
. "github.com/smartystreets/goconvey/convey"
|
||||||
)
|
)
|
||||||
|
@ -35,11 +35,11 @@ const (
|
||||||
func TestGetIndexes(t *testing.T) {
|
func TestGetIndexes(t *testing.T) {
|
||||||
Init(t, false)
|
Init(t, false)
|
||||||
|
|
||||||
mockRepo := mocks.CreateMockArtistIndexRepo()
|
mockRepo := persistence.CreateMockArtistIndexRepo()
|
||||||
utils.DefineSingleton(new(domain.ArtistIndexRepository), func() domain.ArtistIndexRepository {
|
utils.DefineSingleton(new(domain.ArtistIndexRepository), func() domain.ArtistIndexRepository {
|
||||||
return mockRepo
|
return mockRepo
|
||||||
})
|
})
|
||||||
propRepo := mocks.CreateMockPropertyRepo()
|
propRepo := engine.CreateMockPropertyRepo()
|
||||||
utils.DefineSingleton(new(engine.PropertyRepository), func() engine.PropertyRepository {
|
utils.DefineSingleton(new(engine.PropertyRepository), func() engine.PropertyRepository {
|
||||||
return propRepo
|
return propRepo
|
||||||
})
|
})
|
||||||
|
@ -116,15 +116,15 @@ func TestGetIndexes(t *testing.T) {
|
||||||
func TestGetMusicDirectory(t *testing.T) {
|
func TestGetMusicDirectory(t *testing.T) {
|
||||||
Init(t, false)
|
Init(t, false)
|
||||||
|
|
||||||
mockArtistRepo := mocks.CreateMockArtistRepo()
|
mockArtistRepo := persistence.CreateMockArtistRepo()
|
||||||
utils.DefineSingleton(new(domain.ArtistRepository), func() domain.ArtistRepository {
|
utils.DefineSingleton(new(domain.ArtistRepository), func() domain.ArtistRepository {
|
||||||
return mockArtistRepo
|
return mockArtistRepo
|
||||||
})
|
})
|
||||||
mockAlbumRepo := mocks.CreateMockAlbumRepo()
|
mockAlbumRepo := persistence.CreateMockAlbumRepo()
|
||||||
utils.DefineSingleton(new(domain.AlbumRepository), func() domain.AlbumRepository {
|
utils.DefineSingleton(new(domain.AlbumRepository), func() domain.AlbumRepository {
|
||||||
return mockAlbumRepo
|
return mockAlbumRepo
|
||||||
})
|
})
|
||||||
mockMediaFileRepo := mocks.CreateMockMediaFileRepo()
|
mockMediaFileRepo := persistence.CreateMockMediaFileRepo()
|
||||||
utils.DefineSingleton(new(domain.MediaFileRepository), func() domain.MediaFileRepository {
|
utils.DefineSingleton(new(domain.MediaFileRepository), func() domain.MediaFileRepository {
|
||||||
return mockMediaFileRepo
|
return mockMediaFileRepo
|
||||||
})
|
})
|
||||||
|
|
|
@ -10,8 +10,8 @@ import (
|
||||||
"github.com/astaxie/beego"
|
"github.com/astaxie/beego"
|
||||||
"github.com/deluan/gosonic/api/responses"
|
"github.com/deluan/gosonic/api/responses"
|
||||||
"github.com/deluan/gosonic/domain"
|
"github.com/deluan/gosonic/domain"
|
||||||
|
"github.com/deluan/gosonic/persistence"
|
||||||
. "github.com/deluan/gosonic/tests"
|
. "github.com/deluan/gosonic/tests"
|
||||||
"github.com/deluan/gosonic/tests/mocks"
|
|
||||||
"github.com/deluan/gosonic/utils"
|
"github.com/deluan/gosonic/utils"
|
||||||
. "github.com/smartystreets/goconvey/convey"
|
. "github.com/smartystreets/goconvey/convey"
|
||||||
)
|
)
|
||||||
|
@ -28,7 +28,7 @@ func getCoverArt(params ...string) (*http.Request, *httptest.ResponseRecorder) {
|
||||||
func TestGetCoverArt(t *testing.T) {
|
func TestGetCoverArt(t *testing.T) {
|
||||||
Init(t, false)
|
Init(t, false)
|
||||||
|
|
||||||
mockMediaFileRepo := mocks.CreateMockMediaFileRepo()
|
mockMediaFileRepo := persistence.CreateMockMediaFileRepo()
|
||||||
utils.DefineSingleton(new(domain.MediaFileRepository), func() domain.MediaFileRepository {
|
utils.DefineSingleton(new(domain.MediaFileRepository), func() domain.MediaFileRepository {
|
||||||
return mockMediaFileRepo
|
return mockMediaFileRepo
|
||||||
})
|
})
|
||||||
|
|
|
@ -6,38 +6,31 @@ import (
|
||||||
|
|
||||||
"github.com/astaxie/beego"
|
"github.com/astaxie/beego"
|
||||||
"github.com/deluan/gosonic/api/responses"
|
"github.com/deluan/gosonic/api/responses"
|
||||||
"github.com/deluan/gosonic/domain"
|
"github.com/deluan/gosonic/engine"
|
||||||
"github.com/deluan/gosonic/itunesbridge"
|
|
||||||
"github.com/deluan/gosonic/utils"
|
"github.com/deluan/gosonic/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
type MediaAnnotationController struct {
|
type MediaAnnotationController struct {
|
||||||
BaseAPIController
|
BaseAPIController
|
||||||
itunes itunesbridge.ItunesControl
|
scrobbler engine.Scrobbler
|
||||||
mfRepo domain.MediaFileRepository
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *MediaAnnotationController) Prepare() {
|
func (c *MediaAnnotationController) Prepare() {
|
||||||
utils.ResolveDependencies(&c.itunes, &c.mfRepo)
|
utils.ResolveDependencies(&c.scrobbler)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *MediaAnnotationController) Scrobble() {
|
func (c *MediaAnnotationController) Scrobble() {
|
||||||
id := c.RequiredParamString("id", "Required id parameter is missing")
|
id := c.RequiredParamString("id", "Required id parameter is missing")
|
||||||
time := c.ParamTime("time", time.Now())
|
time := c.ParamTime("time", time.Now())
|
||||||
submission := c.ParamBool("submission", true)
|
submission := c.ParamBool("submission", false)
|
||||||
|
println(submission)
|
||||||
if submission {
|
if submission {
|
||||||
mf, err := c.mfRepo.Get(id)
|
mf, err := c.scrobbler.Register(id, time, true)
|
||||||
if err != nil || mf == nil {
|
if err != nil {
|
||||||
beego.Error("Id", id, "not found!")
|
|
||||||
c.SendError(responses.ERROR_DATA_NOT_FOUND, "Id not found")
|
|
||||||
}
|
|
||||||
|
|
||||||
beego.Info(fmt.Sprintf(`Scrobbling (%s) "%s" at %v`, id, mf.Title, time))
|
|
||||||
if err := c.itunes.Scrobble(id, time); err != nil {
|
|
||||||
beego.Error("Error scrobbling:", err)
|
beego.Error("Error scrobbling:", err)
|
||||||
c.SendError(responses.ERROR_GENERIC, "Internal error")
|
c.SendError(responses.ERROR_GENERIC, "Internal error")
|
||||||
}
|
}
|
||||||
|
beego.Info(fmt.Sprintf(`Scrobbled (%s) "%s" at %v`, id, mf.Title, time))
|
||||||
}
|
}
|
||||||
|
|
||||||
response := c.NewEmpty()
|
response := c.NewEmpty()
|
||||||
|
|
|
@ -3,16 +3,17 @@ package api_test
|
||||||
import (
|
import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/deluan/gosonic/api/responses"
|
"fmt"
|
||||||
"github.com/deluan/gosonic/domain"
|
|
||||||
. "github.com/deluan/gosonic/tests"
|
|
||||||
"github.com/deluan/gosonic/tests/mocks"
|
|
||||||
"github.com/deluan/gosonic/utils"
|
|
||||||
. "github.com/smartystreets/goconvey/convey"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/http/httptest"
|
"net/http/httptest"
|
||||||
|
|
||||||
"github.com/astaxie/beego"
|
"github.com/astaxie/beego"
|
||||||
"fmt"
|
"github.com/deluan/gosonic/api/responses"
|
||||||
|
"github.com/deluan/gosonic/domain"
|
||||||
|
"github.com/deluan/gosonic/persistence"
|
||||||
|
. "github.com/deluan/gosonic/tests"
|
||||||
|
"github.com/deluan/gosonic/utils"
|
||||||
|
. "github.com/smartystreets/goconvey/convey"
|
||||||
)
|
)
|
||||||
|
|
||||||
func stream(params ...string) (*http.Request, *httptest.ResponseRecorder) {
|
func stream(params ...string) (*http.Request, *httptest.ResponseRecorder) {
|
||||||
|
@ -27,7 +28,7 @@ func stream(params ...string) (*http.Request, *httptest.ResponseRecorder) {
|
||||||
func TestStream(t *testing.T) {
|
func TestStream(t *testing.T) {
|
||||||
Init(t, false)
|
Init(t, false)
|
||||||
|
|
||||||
mockMediaFileRepo := mocks.CreateMockMediaFileRepo()
|
mockMediaFileRepo := persistence.CreateMockMediaFileRepo()
|
||||||
utils.DefineSingleton(new(domain.MediaFileRepository), func() domain.MediaFileRepository {
|
utils.DefineSingleton(new(domain.MediaFileRepository), func() domain.MediaFileRepository {
|
||||||
return mockMediaFileRepo
|
return mockMediaFileRepo
|
||||||
})
|
})
|
||||||
|
|
|
@ -27,6 +27,7 @@ func init() {
|
||||||
utils.DefineSingleton(new(engine.Cover), engine.NewCover)
|
utils.DefineSingleton(new(engine.Cover), engine.NewCover)
|
||||||
utils.DefineSingleton(new(engine.Playlists), engine.NewPlaylists)
|
utils.DefineSingleton(new(engine.Playlists), engine.NewPlaylists)
|
||||||
utils.DefineSingleton(new(engine.Search), engine.NewSearch)
|
utils.DefineSingleton(new(engine.Search), engine.NewSearch)
|
||||||
|
utils.DefineSingleton(new(engine.Scrobbler), engine.NewScrobbler)
|
||||||
|
|
||||||
utils.DefineSingleton(new(scanner.CheckSumRepository), persistence.NewCheckSumRepository)
|
utils.DefineSingleton(new(scanner.CheckSumRepository), persistence.NewCheckSumRepository)
|
||||||
utils.DefineSingleton(new(scanner.Scanner), scanner.NewItunesScanner)
|
utils.DefineSingleton(new(scanner.Scanner), scanner.NewItunesScanner)
|
||||||
|
|
|
@ -7,15 +7,15 @@ import (
|
||||||
"image"
|
"image"
|
||||||
|
|
||||||
"github.com/deluan/gosonic/engine"
|
"github.com/deluan/gosonic/engine"
|
||||||
|
"github.com/deluan/gosonic/persistence"
|
||||||
. "github.com/deluan/gosonic/tests"
|
. "github.com/deluan/gosonic/tests"
|
||||||
"github.com/deluan/gosonic/tests/mocks"
|
|
||||||
. "github.com/smartystreets/goconvey/convey"
|
. "github.com/smartystreets/goconvey/convey"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestCover(t *testing.T) {
|
func TestCover(t *testing.T) {
|
||||||
Init(t, false)
|
Init(t, false)
|
||||||
|
|
||||||
mockMediaFileRepo := mocks.CreateMockMediaFileRepo()
|
mockMediaFileRepo := persistence.CreateMockMediaFileRepo()
|
||||||
|
|
||||||
cover := engine.NewCover(mockMediaFileRepo)
|
cover := engine.NewCover(mockMediaFileRepo)
|
||||||
out := new(bytes.Buffer)
|
out := new(bytes.Buffer)
|
||||||
|
|
|
@ -1,9 +1,7 @@
|
||||||
package mocks
|
package engine
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
|
|
||||||
"github.com/deluan/gosonic/engine"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func CreateMockPropertyRepo() *MockProperty {
|
func CreateMockPropertyRepo() *MockProperty {
|
||||||
|
@ -11,7 +9,7 @@ func CreateMockPropertyRepo() *MockProperty {
|
||||||
}
|
}
|
||||||
|
|
||||||
type MockProperty struct {
|
type MockProperty struct {
|
||||||
engine.PropertyRepository
|
PropertyRepository
|
||||||
data map[string]string
|
data map[string]string
|
||||||
err bool
|
err bool
|
||||||
}
|
}
|
41
engine/scrobbler.go
Normal file
41
engine/scrobbler.go
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
package engine
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/deluan/gosonic/domain"
|
||||||
|
"github.com/deluan/gosonic/itunesbridge"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Scrobbler interface {
|
||||||
|
Register(id string, playDate time.Time, submit bool) (*domain.MediaFile, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewScrobbler(itunes itunesbridge.ItunesControl, mr domain.MediaFileRepository) Scrobbler {
|
||||||
|
return scrobbler{itunes, mr}
|
||||||
|
}
|
||||||
|
|
||||||
|
type scrobbler struct {
|
||||||
|
itunes itunesbridge.ItunesControl
|
||||||
|
mfRepo domain.MediaFileRepository
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s scrobbler) Register(id string, playDate time.Time, submit bool) (*domain.MediaFile, error) {
|
||||||
|
mf, err := s.mfRepo.Get(id)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if mf == nil {
|
||||||
|
return nil, errors.New(fmt.Sprintf(`Id "%s" not found`, id))
|
||||||
|
}
|
||||||
|
|
||||||
|
if submit {
|
||||||
|
if err := s.itunes.MarkAsPlayed(id, playDate); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return mf, nil
|
||||||
|
}
|
74
engine/scrobbler_test.go
Normal file
74
engine/scrobbler_test.go
Normal file
|
@ -0,0 +1,74 @@
|
||||||
|
package engine_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/deluan/gosonic/engine"
|
||||||
|
"github.com/deluan/gosonic/itunesbridge"
|
||||||
|
"github.com/deluan/gosonic/persistence"
|
||||||
|
. "github.com/deluan/gosonic/tests"
|
||||||
|
. "github.com/smartystreets/goconvey/convey"
|
||||||
|
"github.com/syndtr/goleveldb/leveldb/errors"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestScrobbler(t *testing.T) {
|
||||||
|
|
||||||
|
Init(t, false)
|
||||||
|
|
||||||
|
mfRepo := persistence.CreateMockMediaFileRepo()
|
||||||
|
itCtrl := &mockItunesControl{}
|
||||||
|
|
||||||
|
scrobbler := engine.NewScrobbler(itCtrl, mfRepo)
|
||||||
|
|
||||||
|
Convey("Given a DB with one song", t, func() {
|
||||||
|
mfRepo.SetData(`[{"Id":"2","Title":"Hands Of Time"}]`, 1)
|
||||||
|
|
||||||
|
Convey("When I scrobble an existing song", func() {
|
||||||
|
now := time.Now()
|
||||||
|
mf, err := scrobbler.Register("2", now, true)
|
||||||
|
|
||||||
|
Convey("Then I get the scrobbled song back", func() {
|
||||||
|
So(err, ShouldBeNil)
|
||||||
|
So(mf.Title, ShouldEqual, "Hands Of Time")
|
||||||
|
})
|
||||||
|
|
||||||
|
Convey("And iTunes is notified", func() {
|
||||||
|
So(itCtrl.played, ShouldContainKey, "2")
|
||||||
|
So(itCtrl.played["2"].Equal(now), ShouldBeTrue)
|
||||||
|
})
|
||||||
|
|
||||||
|
})
|
||||||
|
|
||||||
|
Convey("When the ID is not in the DB", func() {
|
||||||
|
_, err := scrobbler.Register("3", time.Now(), true)
|
||||||
|
|
||||||
|
Convey("Then I receive an error", func() {
|
||||||
|
So(err, ShouldNotBeNil)
|
||||||
|
})
|
||||||
|
|
||||||
|
Convey("And iTunes is not notified", func() {
|
||||||
|
So(itCtrl.played, ShouldNotContainKey, "3")
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
})
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
type mockItunesControl struct {
|
||||||
|
itunesbridge.ItunesControl
|
||||||
|
played map[string]time.Time
|
||||||
|
error bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *mockItunesControl) MarkAsPlayed(id string, playDate time.Time) error {
|
||||||
|
if m.error {
|
||||||
|
return errors.New("ID not found")
|
||||||
|
}
|
||||||
|
if m.played == nil {
|
||||||
|
m.played = make(map[string]time.Time)
|
||||||
|
}
|
||||||
|
m.played[id] = playDate
|
||||||
|
return nil
|
||||||
|
}
|
|
@ -6,16 +6,16 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
type ItunesControl interface {
|
type ItunesControl interface {
|
||||||
Scrobble(id string, playDate time.Time) error
|
MarkAsPlayed(id string, playDate time.Time) error
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewItunesControl() ItunesControl {
|
func NewItunesControl() ItunesControl {
|
||||||
return itunesControl{}
|
return &itunesControl{}
|
||||||
}
|
}
|
||||||
|
|
||||||
type itunesControl struct{}
|
type itunesControl struct{}
|
||||||
|
|
||||||
func (c itunesControl) Scrobble(id string, playDate time.Time) error {
|
func (c *itunesControl) MarkAsPlayed(id string, playDate time.Time) error {
|
||||||
script := Script{fmt.Sprintf(
|
script := Script{fmt.Sprintf(
|
||||||
`set theTrack to the first item of (every track whose database ID is equal to "%s")`, id),
|
`set theTrack to the first item of (every track whose database ID is equal to "%s")`, id),
|
||||||
`set c to (get played count of theTrack)`,
|
`set c to (get played count of theTrack)`,
|
||||||
|
@ -26,6 +26,6 @@ func (c itunesControl) Scrobble(id string, playDate time.Time) error {
|
||||||
return script.Run()
|
return script.Run()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c itunesControl) formatDateTime(d time.Time) string {
|
func (c *itunesControl) formatDateTime(d time.Time) string {
|
||||||
return d.Format("Jan _2, 2006 3:04PM")
|
return d.Format("Jan _2, 2006 3:04PM")
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
package mocks
|
package persistence
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
|
@ -1,4 +1,4 @@
|
||||||
package mocks
|
package persistence
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
|
@ -1,4 +1,4 @@
|
||||||
package mocks
|
package persistence
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
|
@ -1,4 +1,4 @@
|
||||||
package mocks
|
package persistence
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
Loading…
Add table
Add a link
Reference in a new issue