Started implementing getIndex, now with TDD (brought to you by DI)!

This commit is contained in:
Deluan 2016-03-01 12:35:30 -05:00
parent b2cdf8cadb
commit 2bb4c74ba6
7 changed files with 129 additions and 21 deletions

View file

@ -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))
}

View file

@ -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
}

View file

@ -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
View 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"`
}

View file

@ -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)
}

View file

@ -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

View file

@ -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()
}