mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 21:17:37 +03:00
Started implementing getIndex, now with TDD (brought to you by DI)!
This commit is contained in:
parent
b2cdf8cadb
commit
2bb4c74ba6
7 changed files with 129 additions and 21 deletions
|
@ -5,6 +5,7 @@ import (
|
||||||
"github.com/deluan/gosonic/repositories"
|
"github.com/deluan/gosonic/repositories"
|
||||||
"github.com/deluan/gosonic/utils"
|
"github.com/deluan/gosonic/utils"
|
||||||
"github.com/karlkfi/inject"
|
"github.com/karlkfi/inject"
|
||||||
|
"github.com/deluan/gosonic/api/responses"
|
||||||
)
|
)
|
||||||
|
|
||||||
type GetIndexesController struct {
|
type GetIndexesController struct {
|
||||||
|
@ -17,7 +18,16 @@ func (c *GetIndexesController) Prepare() {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GetIndexesController) Get() {
|
func (c *GetIndexesController) Get() {
|
||||||
if c.repo == nil {
|
indexes, err := c.repo.GetAll()
|
||||||
c.CustomAbort(500, "INJECTION NOT WORKING")
|
if err != nil {
|
||||||
|
beego.Error("Error retrieving Indexes:", err)
|
||||||
|
c.CustomAbort(200, string(responses.NewError(responses.ERROR_GENERIC, "Internal Error")))
|
||||||
}
|
}
|
||||||
|
res := &responses.ArtistIndex{}
|
||||||
|
res.Index = make([]responses.Index, len(indexes))
|
||||||
|
for i, idx := range indexes {
|
||||||
|
res.Index[i].Name = idx.Id
|
||||||
|
}
|
||||||
|
|
||||||
|
c.Ctx.Output.Body(responses.NewXML(res))
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,18 +3,82 @@ package api_test
|
||||||
import (
|
import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"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"
|
||||||
|
"encoding/xml"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"errors"
|
||||||
|
"github.com/deluan/gosonic/api/responses"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestGetIndexes(t *testing.T) {
|
func TestGetIndexes(t *testing.T) {
|
||||||
tests.Init(t, false)
|
tests.Init(t, false)
|
||||||
|
mockRepo := &mockArtistIndex{}
|
||||||
_, w := Get(AddParams("/rest/getIndexes.view"), "TestGetIndexes")
|
utils.DefineSingleton(new(repositories.ArtistIndex), func() repositories.ArtistIndex {
|
||||||
|
return mockRepo
|
||||||
|
})
|
||||||
|
|
||||||
Convey("Subject: GetIndexes Endpoint", t, func() {
|
Convey("Subject: GetIndexes Endpoint", t, func() {
|
||||||
Convey("Status code should be 200", func() {
|
Convey("Return fail on DB error", func() {
|
||||||
So(w.Code, ShouldEqual, 200)
|
mockRepo.err = true
|
||||||
|
_, w := Get(AddParams("/rest/getIndexes.view"), "TestGetIndexes")
|
||||||
|
|
||||||
|
v := responses.Subsonic{}
|
||||||
|
xml.Unmarshal(w.Body.Bytes(), &v)
|
||||||
|
So(v.Status, ShouldEqual, "fail")
|
||||||
|
})
|
||||||
|
Convey("When the index is empty", func() {
|
||||||
|
_, w := Get(AddParams("/rest/getIndexes.view"), "TestGetIndexes")
|
||||||
|
|
||||||
|
Convey("Status code should be 200", func() {
|
||||||
|
So(w.Code, ShouldEqual, 200)
|
||||||
|
})
|
||||||
|
Convey("It should return valid XML", func() {
|
||||||
|
v := new(string)
|
||||||
|
err := xml.Unmarshal(w.Body.Bytes(), &v)
|
||||||
|
So(err, ShouldBeNil)
|
||||||
|
})
|
||||||
|
Convey("Then it should return an empty collection", func() {
|
||||||
|
So(w.Body.String(), ShouldContainSubstring, "<indexes></indexes>")
|
||||||
|
})
|
||||||
|
})
|
||||||
|
Convey("When the index is not empty", func() {
|
||||||
|
mockRepo.data = makeMockData(`[{"Id": "A","Artists": []}]`, 2)
|
||||||
|
_, w := Get(AddParams("/rest/getIndexes.view"), "TestGetIndexes")
|
||||||
|
|
||||||
|
Convey("Then it should return the the items in the response", func() {
|
||||||
|
So(w.Body.String(), ShouldContainSubstring, `<index name="A"></index>`)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
Reset(func() {
|
||||||
|
mockRepo.data = make([]models.ArtistIndex, 0)
|
||||||
|
mockRepo.err = 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
|
||||||
|
}
|
|
@ -2,6 +2,7 @@ package responses
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/xml"
|
"encoding/xml"
|
||||||
|
"fmt"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -37,13 +38,19 @@ type error struct {
|
||||||
Message string `xml:"message,attr"`
|
Message string `xml:"message,attr"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewError(errorCode int) []byte {
|
func NewError(errorCode int, message ...interface{}) []byte {
|
||||||
response := NewEmpty()
|
response := NewEmpty()
|
||||||
response.Status = "fail"
|
response.Status = "fail"
|
||||||
if errors[errorCode] == "" {
|
if errors[errorCode] == "" {
|
||||||
errorCode = ERROR_GENERIC
|
errorCode = ERROR_GENERIC
|
||||||
}
|
}
|
||||||
xmlBody, _ := xml.Marshal(&error{Code: errorCode, Message: errors[errorCode]})
|
var msg string
|
||||||
|
if (len(message) == 0) {
|
||||||
|
msg = errors[errorCode]
|
||||||
|
} else {
|
||||||
|
msg = fmt.Sprintf(message[0].(string), message[1:len(message)]...)
|
||||||
|
}
|
||||||
|
xmlBody, _ := xml.Marshal(&error{Code: errorCode, Message: msg})
|
||||||
response.Body = xmlBody
|
response.Body = xmlBody
|
||||||
xmlResponse, _ := xml.Marshal(response)
|
xmlResponse, _ := xml.Marshal(response)
|
||||||
return []byte(xml.Header + string(xmlResponse))
|
return []byte(xml.Header + string(xmlResponse))
|
||||||
|
|
14
api/responses/indexes.go
Normal file
14
api/responses/indexes.go
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
package responses
|
||||||
|
|
||||||
|
import "encoding/xml"
|
||||||
|
|
||||||
|
type Index struct {
|
||||||
|
XMLName xml.Name `xml:"index"`
|
||||||
|
Name string `xml:"name,attr"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ArtistIndex struct {
|
||||||
|
XMLName xml.Name `xml:"indexes"`
|
||||||
|
Index []Index `xml:"indexes"`
|
||||||
|
}
|
||||||
|
|
|
@ -1,15 +1,10 @@
|
||||||
package conf
|
package conf
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/deluan/gosonic/repositories"
|
|
||||||
"github.com/deluan/gosonic/utils"
|
"github.com/deluan/gosonic/utils"
|
||||||
"github.com/karlkfi/inject"
|
"github.com/deluan/gosonic/repositories"
|
||||||
)
|
)
|
||||||
|
|
||||||
func define(ptr interface{}, constructor interface{}, argPtrs ...interface{}) {
|
|
||||||
utils.Graph.Define(ptr, inject.NewProvider(constructor, argPtrs...))
|
|
||||||
}
|
|
||||||
|
|
||||||
func init () {
|
func init () {
|
||||||
define(new(repositories.ArtistIndex), repositories.NewArtistIndexRepository)
|
utils.DefineSingleton(new(repositories.ArtistIndex), repositories.NewArtistIndexRepository)
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,30 +11,30 @@ type ArtistIndex interface {
|
||||||
GetAll() ([]models.ArtistIndex, error)
|
GetAll() ([]models.ArtistIndex, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
type artistIndex struct {
|
type ArtistIndexImpl struct {
|
||||||
BaseRepository
|
BaseRepository
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewArtistIndexRepository() ArtistIndex {
|
func NewArtistIndexRepository() ArtistIndex {
|
||||||
r := &artistIndex{}
|
r := &ArtistIndexImpl{}
|
||||||
r.init("index", &models.ArtistIndex{})
|
r.init("index", &models.ArtistIndex{})
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *artistIndex) Put(m *models.ArtistIndex) error {
|
func (r *ArtistIndexImpl) Put(m *models.ArtistIndex) error {
|
||||||
if m.Id == "" {
|
if m.Id == "" {
|
||||||
return errors.New("Id is not set")
|
return errors.New("Id is not set")
|
||||||
}
|
}
|
||||||
return r.saveOrUpdate(m.Id, m)
|
return r.saveOrUpdate(m.Id, m)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *artistIndex) Get(id string) (*models.ArtistIndex, error) {
|
func (r *ArtistIndexImpl) Get(id string) (*models.ArtistIndex, error) {
|
||||||
var rec interface{}
|
var rec interface{}
|
||||||
rec, err := r.readEntity(id)
|
rec, err := r.readEntity(id)
|
||||||
return rec.(*models.ArtistIndex), err
|
return rec.(*models.ArtistIndex), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *artistIndex) GetAll() ([]models.ArtistIndex, error) {
|
func (r *ArtistIndexImpl) GetAll() ([]models.ArtistIndex, error) {
|
||||||
var indices = make([]models.ArtistIndex, 0)
|
var indices = make([]models.ArtistIndex, 0)
|
||||||
err := r.loadAll(&indices)
|
err := r.loadAll(&indices)
|
||||||
return indices, err
|
return indices, err
|
||||||
|
|
|
@ -1,9 +1,27 @@
|
||||||
package utils
|
package utils
|
||||||
|
|
||||||
import "github.com/karlkfi/inject"
|
import (
|
||||||
|
"github.com/karlkfi/inject"
|
||||||
|
"reflect"
|
||||||
|
)
|
||||||
|
|
||||||
var Graph inject.Graph
|
var Graph inject.Graph
|
||||||
|
|
||||||
|
var (
|
||||||
|
definitions map[reflect.Type]interface{}
|
||||||
|
)
|
||||||
|
|
||||||
|
func DefineSingleton(ptr interface{}, constructor interface{}, args ...interface{}) {
|
||||||
|
typ := reflect.TypeOf(ptr)
|
||||||
|
if definitions[typ] == nil {
|
||||||
|
Graph.Define(ptr, inject.NewProvider(constructor, args...))
|
||||||
|
} else {
|
||||||
|
Graph.Define(definitions[typ], inject.NewProvider(constructor, args...))
|
||||||
|
}
|
||||||
|
definitions[typ] = ptr
|
||||||
|
}
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
|
definitions = make(map[reflect.Type]interface{})
|
||||||
Graph = inject.NewGraph()
|
Graph = inject.NewGraph()
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue