mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 21:17:37 +03:00
Implemented ProperyRepository. Now the Scanner stores the LastScan timestamp
This commit is contained in:
parent
1ceefda6ca
commit
841d8f457f
10 changed files with 178 additions and 38 deletions
|
@ -6,15 +6,18 @@ import (
|
||||||
"github.com/deluan/gosonic/utils"
|
"github.com/deluan/gosonic/utils"
|
||||||
"github.com/karlkfi/inject"
|
"github.com/karlkfi/inject"
|
||||||
"github.com/deluan/gosonic/api/responses"
|
"github.com/deluan/gosonic/api/responses"
|
||||||
|
"github.com/deluan/gosonic/consts"
|
||||||
)
|
)
|
||||||
|
|
||||||
type GetIndexesController struct {
|
type GetIndexesController struct {
|
||||||
beego.Controller
|
beego.Controller
|
||||||
repo repositories.ArtistIndex
|
repo repositories.ArtistIndex
|
||||||
|
properties repositories.Property
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GetIndexesController) Prepare() {
|
func (c *GetIndexesController) Prepare() {
|
||||||
inject.ExtractAssignable(utils.Graph, &c.repo)
|
inject.ExtractAssignable(utils.Graph, &c.repo)
|
||||||
|
inject.ExtractAssignable(utils.Graph, &c.properties)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GetIndexesController) Get() {
|
func (c *GetIndexesController) Get() {
|
||||||
|
@ -23,7 +26,15 @@ func (c *GetIndexesController) Get() {
|
||||||
beego.Error("Error retrieving Indexes:", err)
|
beego.Error("Error retrieving Indexes:", err)
|
||||||
c.CustomAbort(200, string(responses.NewError(responses.ERROR_GENERIC, "Internal Error")))
|
c.CustomAbort(200, string(responses.NewError(responses.ERROR_GENERIC, "Internal Error")))
|
||||||
}
|
}
|
||||||
res := &responses.ArtistIndex{IgnoredArticles: beego.AppConfig.String("ignoredArticles")}
|
res := &responses.ArtistIndex{}
|
||||||
|
|
||||||
|
res.LastModified, err = c.properties.Get(consts.LastScan)
|
||||||
|
if err != nil {
|
||||||
|
beego.Error("Error retrieving LastScan property:", err)
|
||||||
|
c.CustomAbort(200, string(responses.NewError(responses.ERROR_GENERIC, "Internal Error")))
|
||||||
|
}
|
||||||
|
|
||||||
|
res.IgnoredArticles = beego.AppConfig.String("ignoredArticles")
|
||||||
res.Index = make([]responses.IdxIndex, len(indexes))
|
res.Index = make([]responses.IdxIndex, len(indexes))
|
||||||
for i, idx := range indexes {
|
for i, idx := range indexes {
|
||||||
res.Index[i].Name = idx.Id
|
res.Index[i].Name = idx.Id
|
||||||
|
|
|
@ -6,25 +6,39 @@ import (
|
||||||
"github.com/deluan/gosonic/utils"
|
"github.com/deluan/gosonic/utils"
|
||||||
. "github.com/smartystreets/goconvey/convey"
|
. "github.com/smartystreets/goconvey/convey"
|
||||||
"github.com/deluan/gosonic/tests"
|
"github.com/deluan/gosonic/tests"
|
||||||
"github.com/deluan/gosonic/models"
|
|
||||||
"github.com/deluan/gosonic/repositories"
|
"github.com/deluan/gosonic/repositories"
|
||||||
"encoding/xml"
|
"encoding/xml"
|
||||||
"encoding/json"
|
|
||||||
"fmt"
|
|
||||||
"errors"
|
|
||||||
"github.com/deluan/gosonic/api/responses"
|
"github.com/deluan/gosonic/api/responses"
|
||||||
|
"github.com/deluan/gosonic/consts"
|
||||||
|
"github.com/deluan/gosonic/tests/mocks"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
emptyResponse = `<indexes lastModified="1" ignoredArticles="The El La Los Las Le Les Os As O A"></indexes>`
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestGetIndexes(t *testing.T) {
|
func TestGetIndexes(t *testing.T) {
|
||||||
tests.Init(t, false)
|
tests.Init(t, false)
|
||||||
mockRepo := &mockArtistIndex{}
|
mockRepo := mocks.CreateMockArtistIndexRepo()
|
||||||
utils.DefineSingleton(new(repositories.ArtistIndex), func() repositories.ArtistIndex {
|
utils.DefineSingleton(new(repositories.ArtistIndex), func() repositories.ArtistIndex {
|
||||||
return mockRepo
|
return mockRepo
|
||||||
})
|
})
|
||||||
|
propRepo := mocks.CreateMockPropertyRepo()
|
||||||
|
utils.DefineSingleton(new(repositories.Property), func() repositories.Property {
|
||||||
|
return propRepo
|
||||||
|
})
|
||||||
|
|
||||||
Convey("Subject: GetIndexes Endpoint", t, func() {
|
Convey("Subject: GetIndexes Endpoint", t, func() {
|
||||||
Convey("Return fail on DB error", func() {
|
Convey("Return fail on Index Table error", func() {
|
||||||
mockRepo.err = true
|
mockRepo.SetError(true)
|
||||||
|
_, w := Get(AddParams("/rest/getIndexes.view"), "TestGetIndexes")
|
||||||
|
|
||||||
|
v := responses.Subsonic{}
|
||||||
|
xml.Unmarshal(w.Body.Bytes(), &v)
|
||||||
|
So(v.Status, ShouldEqual, "fail")
|
||||||
|
})
|
||||||
|
Convey("Return fail on Property Table error", func() {
|
||||||
|
propRepo.SetError(true)
|
||||||
_, w := Get(AddParams("/rest/getIndexes.view"), "TestGetIndexes")
|
_, w := Get(AddParams("/rest/getIndexes.view"), "TestGetIndexes")
|
||||||
|
|
||||||
v := responses.Subsonic{}
|
v := responses.Subsonic{}
|
||||||
|
@ -43,45 +57,26 @@ func TestGetIndexes(t *testing.T) {
|
||||||
So(err, ShouldBeNil)
|
So(err, ShouldBeNil)
|
||||||
})
|
})
|
||||||
Convey("Then it should return an empty collection", func() {
|
Convey("Then it should return an empty collection", func() {
|
||||||
So(w.Body.String(), ShouldContainSubstring, `<indexes ignoredArticles="The El La Los Las Le Les Os As O A"></indexes>`)
|
So(w.Body.String(), ShouldContainSubstring, emptyResponse)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
Convey("When the index is not empty", func() {
|
Convey("When the index is not empty", func() {
|
||||||
mockRepo.data = makeMockData(`[{"Id": "A","Artists": [
|
mockRepo.SetData(`[{"Id": "A","Artists": [
|
||||||
{"ArtistId": "21", "Artist": "Afrolicious"}
|
{"ArtistId": "21", "Artist": "Afrolicious"}
|
||||||
]}]`, 2)
|
]}]`, 2)
|
||||||
_, w := Get(AddParams("/rest/getIndexes.view"), "TestGetIndexes")
|
|
||||||
|
|
||||||
Convey("Then it should return the the items in the response", func() {
|
Convey("Then it should return the the items in the response", func() {
|
||||||
|
_, w := Get(AddParams("/rest/getIndexes.view"), "TestGetIndexes")
|
||||||
|
|
||||||
So(w.Body.String(), ShouldContainSubstring,
|
So(w.Body.String(), ShouldContainSubstring,
|
||||||
`<indexes ignoredArticles="The El La Los Las Le Les Os As O A"><index name="A"><artist id="21" name="Afrolicious"></artist></index></indexes>`)
|
`<indexes lastModified="1" ignoredArticles="The El La Los Las Le Les Os As O A"><index name="A"><artist id="21" name="Afrolicious"></artist></index></indexes>`)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
Reset(func() {
|
Reset(func() {
|
||||||
mockRepo.data = make([]models.ArtistIndex, 0)
|
mockRepo.SetData("[]", 0)
|
||||||
mockRepo.err = false
|
mockRepo.SetError(false)
|
||||||
|
propRepo.Put(consts.LastScan, "1")
|
||||||
|
propRepo.SetError(false)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func makeMockData(j string, length int) []models.ArtistIndex {
|
|
||||||
data := make([]models.ArtistIndex, length)
|
|
||||||
err := json.Unmarshal([]byte(j), &data)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println("ERROR: ", err)
|
|
||||||
}
|
|
||||||
return data
|
|
||||||
}
|
|
||||||
|
|
||||||
type mockArtistIndex struct {
|
|
||||||
repositories.ArtistIndexImpl
|
|
||||||
data []models.ArtistIndex
|
|
||||||
err bool
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *mockArtistIndex) GetAll() ([]models.ArtistIndex, error) {
|
|
||||||
if m.err {
|
|
||||||
return nil, errors.New("Error!")
|
|
||||||
}
|
|
||||||
return m.data, nil
|
|
||||||
}
|
|
|
@ -17,6 +17,7 @@ type IdxIndex struct {
|
||||||
type ArtistIndex struct {
|
type ArtistIndex struct {
|
||||||
XMLName xml.Name `xml:"indexes"`
|
XMLName xml.Name `xml:"indexes"`
|
||||||
Index []IdxIndex `xml:"indexes"`
|
Index []IdxIndex `xml:"indexes"`
|
||||||
|
LastModified string `xml:"lastModified,attr"`
|
||||||
IgnoredArticles string `xml:"ignoredArticles,attr"`
|
IgnoredArticles string `xml:"ignoredArticles,attr"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -7,4 +7,5 @@ import (
|
||||||
|
|
||||||
func init () {
|
func init () {
|
||||||
utils.DefineSingleton(new(repositories.ArtistIndex), repositories.NewArtistIndexRepository)
|
utils.DefineSingleton(new(repositories.ArtistIndex), repositories.NewArtistIndexRepository)
|
||||||
|
utils.DefineSingleton(new(repositories.Property), repositories.NewPropertyRepository)
|
||||||
}
|
}
|
||||||
|
|
5
consts/properties.go
Normal file
5
consts/properties.go
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
package consts
|
||||||
|
|
||||||
|
const (
|
||||||
|
LastScan = "LastScan"
|
||||||
|
)
|
6
models/property.go
Normal file
6
models/property.go
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
package models
|
||||||
|
|
||||||
|
type Property struct {
|
||||||
|
Id string
|
||||||
|
Value string
|
||||||
|
}
|
35
repositories/property_repository.go
Normal file
35
repositories/property_repository.go
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
package repositories
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/deluan/gosonic/models"
|
||||||
|
"errors"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Property interface {
|
||||||
|
Put(id string, value string) error
|
||||||
|
Get(id string) (string, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type PropertyImpl struct {
|
||||||
|
BaseRepository
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewPropertyRepository() *PropertyImpl {
|
||||||
|
r := &PropertyImpl{}
|
||||||
|
r.init("property", &models.Property{})
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *PropertyImpl) Put(id string, value string) error {
|
||||||
|
m := &models.Property{Id: id, Value: value}
|
||||||
|
if m.Id == "" {
|
||||||
|
return errors.New("Id is required")
|
||||||
|
}
|
||||||
|
return r.saveOrUpdate(m.Id, m)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *PropertyImpl) Get(id string) (string, error) {
|
||||||
|
var rec interface{}
|
||||||
|
rec, err := r.readEntity(id)
|
||||||
|
return rec.(*models.Property).Value, err
|
||||||
|
}
|
|
@ -6,6 +6,9 @@ import (
|
||||||
"github.com/deluan/gosonic/models"
|
"github.com/deluan/gosonic/models"
|
||||||
"strings"
|
"strings"
|
||||||
"github.com/deluan/gosonic/utils"
|
"github.com/deluan/gosonic/utils"
|
||||||
|
"github.com/deluan/gosonic/consts"
|
||||||
|
"time"
|
||||||
|
"fmt"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Scanner interface {
|
type Scanner interface {
|
||||||
|
@ -26,7 +29,7 @@ func doImport(mediaFolder string, scanner Scanner) {
|
||||||
beego.Info("Finished importing", len(files), "files")
|
beego.Info("Finished importing", len(files), "files")
|
||||||
}
|
}
|
||||||
|
|
||||||
func importLibrary(files []Track) {
|
func importLibrary(files []Track) (err error){
|
||||||
mfRepo := repositories.NewMediaFileRepository()
|
mfRepo := repositories.NewMediaFileRepository()
|
||||||
albumRepo := repositories.NewAlbumRepository()
|
albumRepo := repositories.NewAlbumRepository()
|
||||||
artistRepo := repositories.NewArtistRepository()
|
artistRepo := repositories.NewArtistRepository()
|
||||||
|
@ -38,7 +41,7 @@ func importLibrary(files []Track) {
|
||||||
collectIndex(artist, artistIndex)
|
collectIndex(artist, artistIndex)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := saveIndex(artistIndex); err != nil {
|
if err = saveIndex(artistIndex); err != nil {
|
||||||
beego.Error(err)
|
beego.Error(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -48,6 +51,15 @@ func importLibrary(files []Track) {
|
||||||
beego.Info("Total Albums in database:", c)
|
beego.Info("Total Albums in database:", c)
|
||||||
c, _ = mfRepo.CountAll()
|
c, _ = mfRepo.CountAll()
|
||||||
beego.Info("Total MediaFiles in database:", c)
|
beego.Info("Total MediaFiles in database:", c)
|
||||||
|
|
||||||
|
if err == nil {
|
||||||
|
propertyRepo := repositories.NewPropertyRepository()
|
||||||
|
millis := time.Now().UnixNano() / 1000000
|
||||||
|
propertyRepo.Put(consts.LastScan, fmt.Sprint(millis))
|
||||||
|
beego.Info("LastScan timestamp:", millis)
|
||||||
|
}
|
||||||
|
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func parseTrack(t *Track) (*models.MediaFile, *models.Album, *models.Artist) {
|
func parseTrack(t *Track) (*models.MediaFile, *models.Album, *models.Artist) {
|
||||||
|
|
38
tests/mocks/mock_index_repo_tests.go
Normal file
38
tests/mocks/mock_index_repo_tests.go
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
package mocks
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/deluan/gosonic/models"
|
||||||
|
"fmt"
|
||||||
|
"encoding/json"
|
||||||
|
"github.com/deluan/gosonic/repositories"
|
||||||
|
"errors"
|
||||||
|
)
|
||||||
|
|
||||||
|
func CreateMockArtistIndexRepo() *MockArtistIndex {
|
||||||
|
return &MockArtistIndex{}
|
||||||
|
}
|
||||||
|
|
||||||
|
type MockArtistIndex struct {
|
||||||
|
repositories.ArtistIndexImpl
|
||||||
|
data []models.ArtistIndex
|
||||||
|
err bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockArtistIndex) SetError(err bool) {
|
||||||
|
m.err = err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockArtistIndex) SetData(j string, length int) {
|
||||||
|
m.data = make([]models.ArtistIndex, length)
|
||||||
|
err := json.Unmarshal([]byte(j), &m.data)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("ERROR: ", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockArtistIndex) GetAll() ([]models.ArtistIndex, error) {
|
||||||
|
if m.err {
|
||||||
|
return nil, errors.New("Error!")
|
||||||
|
}
|
||||||
|
return m.data, nil
|
||||||
|
}
|
36
tests/mocks/mock_property_repo_tests.go
Normal file
36
tests/mocks/mock_property_repo_tests.go
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
package mocks
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/deluan/gosonic/repositories"
|
||||||
|
"errors"
|
||||||
|
)
|
||||||
|
|
||||||
|
func CreateMockPropertyRepo() *MockProperty {
|
||||||
|
return &MockProperty{data: make(map[string]string)}
|
||||||
|
}
|
||||||
|
|
||||||
|
type MockProperty struct {
|
||||||
|
repositories.PropertyImpl
|
||||||
|
data map[string]string
|
||||||
|
err bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockProperty) SetError(err bool) {
|
||||||
|
m.err = err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockProperty) Put(id string, value string) error {
|
||||||
|
if (m.err) {
|
||||||
|
return errors.New("Error!")
|
||||||
|
}
|
||||||
|
m.data[id] = value
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockProperty) Get(id string) (string, error) {
|
||||||
|
if (m.err) {
|
||||||
|
return "", errors.New("Error!")
|
||||||
|
} else {
|
||||||
|
return m.data[id], nil
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Add a link
Reference in a new issue