mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-03 20:47:35 +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/utils"
|
||||
"github.com/karlkfi/inject"
|
||||
"github.com/deluan/gosonic/api/responses"
|
||||
)
|
||||
|
||||
type GetIndexesController struct {
|
||||
|
@ -17,7 +18,16 @@ func (c *GetIndexesController) Prepare() {
|
|||
}
|
||||
|
||||
func (c *GetIndexesController) Get() {
|
||||
if c.repo == nil {
|
||||
c.CustomAbort(500, "INJECTION NOT WORKING")
|
||||
indexes, err := c.repo.GetAll()
|
||||
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 (
|
||||
"testing"
|
||||
|
||||
"github.com/deluan/gosonic/utils"
|
||||
. "github.com/smartystreets/goconvey/convey"
|
||||
"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) {
|
||||
tests.Init(t, false)
|
||||
|
||||
_, w := Get(AddParams("/rest/getIndexes.view"), "TestGetIndexes")
|
||||
mockRepo := &mockArtistIndex{}
|
||||
utils.DefineSingleton(new(repositories.ArtistIndex), func() repositories.ArtistIndex {
|
||||
return mockRepo
|
||||
})
|
||||
|
||||
Convey("Subject: GetIndexes Endpoint", t, func() {
|
||||
Convey("Status code should be 200", func() {
|
||||
So(w.Code, ShouldEqual, 200)
|
||||
Convey("Return fail on DB error", func() {
|
||||
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 (
|
||||
"encoding/xml"
|
||||
"fmt"
|
||||
)
|
||||
|
||||
const (
|
||||
|
@ -37,13 +38,19 @@ type error struct {
|
|||
Message string `xml:"message,attr"`
|
||||
}
|
||||
|
||||
func NewError(errorCode int) []byte {
|
||||
func NewError(errorCode int, message ...interface{}) []byte {
|
||||
response := NewEmpty()
|
||||
response.Status = "fail"
|
||||
if errors[errorCode] == "" {
|
||||
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
|
||||
xmlResponse, _ := xml.Marshal(response)
|
||||
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
|
||||
|
||||
import (
|
||||
"github.com/deluan/gosonic/repositories"
|
||||
"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 () {
|
||||
define(new(repositories.ArtistIndex), repositories.NewArtistIndexRepository)
|
||||
utils.DefineSingleton(new(repositories.ArtistIndex), repositories.NewArtistIndexRepository)
|
||||
}
|
||||
|
|
|
@ -11,30 +11,30 @@ type ArtistIndex interface {
|
|||
GetAll() ([]models.ArtistIndex, error)
|
||||
}
|
||||
|
||||
type artistIndex struct {
|
||||
type ArtistIndexImpl struct {
|
||||
BaseRepository
|
||||
}
|
||||
|
||||
func NewArtistIndexRepository() ArtistIndex {
|
||||
r := &artistIndex{}
|
||||
r := &ArtistIndexImpl{}
|
||||
r.init("index", &models.ArtistIndex{})
|
||||
return r
|
||||
}
|
||||
|
||||
func (r *artistIndex) Put(m *models.ArtistIndex) error {
|
||||
func (r *ArtistIndexImpl) Put(m *models.ArtistIndex) error {
|
||||
if m.Id == "" {
|
||||
return errors.New("Id is not set")
|
||||
}
|
||||
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{}
|
||||
rec, err := r.readEntity(id)
|
||||
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)
|
||||
err := r.loadAll(&indices)
|
||||
return indices, err
|
||||
|
|
|
@ -1,9 +1,27 @@
|
|||
package utils
|
||||
|
||||
import "github.com/karlkfi/inject"
|
||||
import (
|
||||
"github.com/karlkfi/inject"
|
||||
"reflect"
|
||||
)
|
||||
|
||||
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() {
|
||||
definitions = make(map[reflect.Type]interface{})
|
||||
Graph = inject.NewGraph()
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue