mirror of
https://github.com/elyby/chrly.git
synced 2024-11-26 16:51:59 +05:30
3f81a0c18a
Implemented graceful server shutdown Extract records manipulating API into separate handlers group
443 lines
13 KiB
Go
443 lines
13 KiB
Go
package http
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/base64"
|
|
"errors"
|
|
"io"
|
|
"io/ioutil"
|
|
"mime/multipart"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"net/url"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/mock"
|
|
"github.com/stretchr/testify/suite"
|
|
|
|
"github.com/elyby/chrly/model"
|
|
)
|
|
|
|
/***************
|
|
* Setup mocks *
|
|
***************/
|
|
|
|
type apiTestSuite struct {
|
|
suite.Suite
|
|
|
|
App *Api
|
|
|
|
SkinsRepository *skinsRepositoryMock
|
|
Emitter *emitterMock
|
|
}
|
|
|
|
/********************
|
|
* Setup test suite *
|
|
********************/
|
|
|
|
func (suite *apiTestSuite) SetupTest() {
|
|
suite.SkinsRepository = &skinsRepositoryMock{}
|
|
suite.Emitter = &emitterMock{}
|
|
|
|
suite.App = &Api{
|
|
SkinsRepo: suite.SkinsRepository,
|
|
Emitter: suite.Emitter,
|
|
}
|
|
}
|
|
|
|
func (suite *apiTestSuite) TearDownTest() {
|
|
suite.SkinsRepository.AssertExpectations(suite.T())
|
|
suite.Emitter.AssertExpectations(suite.T())
|
|
}
|
|
|
|
func (suite *apiTestSuite) RunSubTest(name string, subTest func()) {
|
|
suite.SetupTest()
|
|
suite.Run(name, subTest)
|
|
suite.TearDownTest()
|
|
}
|
|
|
|
/*************
|
|
* Run tests *
|
|
*************/
|
|
|
|
func TestApi(t *testing.T) {
|
|
suite.Run(t, new(apiTestSuite))
|
|
}
|
|
|
|
/*************************
|
|
* Post skin tests cases *
|
|
*************************/
|
|
|
|
type postSkinTestCase struct {
|
|
Name string
|
|
Form io.Reader
|
|
BeforeTest func(suite *apiTestSuite)
|
|
AfterTest func(suite *apiTestSuite, response *http.Response)
|
|
}
|
|
|
|
var postSkinTestsCases = []*postSkinTestCase{
|
|
{
|
|
Name: "Upload new identity with textures data",
|
|
Form: bytes.NewBufferString(url.Values{
|
|
"identityId": {"1"},
|
|
"username": {"mock_username"},
|
|
"uuid": {"0f657aa8-bfbe-415d-b700-5750090d3af3"},
|
|
"skinId": {"5"},
|
|
"is1_8": {"0"},
|
|
"isSlim": {"0"},
|
|
"url": {"http://example.com/skin.png"},
|
|
}.Encode()),
|
|
BeforeTest: func(suite *apiTestSuite) {
|
|
suite.SkinsRepository.On("FindByUserId", 1).Return(nil, &SkinNotFoundError{Who: "unknown"})
|
|
suite.SkinsRepository.On("FindByUsername", "mock_username").Return(nil, &SkinNotFoundError{Who: "mock_username"})
|
|
suite.SkinsRepository.On("Save", mock.MatchedBy(func(model *model.Skin) bool {
|
|
suite.Equal(1, model.UserId)
|
|
suite.Equal("mock_username", model.Username)
|
|
suite.Equal("0f657aa8-bfbe-415d-b700-5750090d3af3", model.Uuid)
|
|
suite.Equal(5, model.SkinId)
|
|
suite.False(model.Is1_8)
|
|
suite.False(model.IsSlim)
|
|
suite.Equal("http://example.com/skin.png", model.Url)
|
|
|
|
return true
|
|
})).Times(1).Return(nil)
|
|
},
|
|
AfterTest: func(suite *apiTestSuite, response *http.Response) {
|
|
suite.Equal(201, response.StatusCode)
|
|
body, _ := ioutil.ReadAll(response.Body)
|
|
suite.Empty(body)
|
|
},
|
|
},
|
|
{
|
|
Name: "Update exists identity by changing only textures data",
|
|
Form: bytes.NewBufferString(url.Values{
|
|
"identityId": {"1"},
|
|
"username": {"mock_username"},
|
|
"uuid": {"0f657aa8-bfbe-415d-b700-5750090d3af3"},
|
|
"skinId": {"5"},
|
|
"is1_8": {"1"},
|
|
"isSlim": {"1"},
|
|
"url": {"http://textures-server.com/skin.png"},
|
|
}.Encode()),
|
|
BeforeTest: func(suite *apiTestSuite) {
|
|
suite.SkinsRepository.On("FindByUserId", 1).Return(createSkinModel("mock_username", false), nil)
|
|
suite.SkinsRepository.On("Save", mock.MatchedBy(func(model *model.Skin) bool {
|
|
suite.Equal(1, model.UserId)
|
|
suite.Equal("mock_username", model.Username)
|
|
suite.Equal("0f657aa8-bfbe-415d-b700-5750090d3af3", model.Uuid)
|
|
suite.Equal(5, model.SkinId)
|
|
suite.True(model.Is1_8)
|
|
suite.True(model.IsSlim)
|
|
suite.Equal("http://textures-server.com/skin.png", model.Url)
|
|
|
|
return true
|
|
})).Times(1).Return(nil)
|
|
},
|
|
AfterTest: func(suite *apiTestSuite, response *http.Response) {
|
|
suite.Equal(201, response.StatusCode)
|
|
body, _ := ioutil.ReadAll(response.Body)
|
|
suite.Empty(body)
|
|
},
|
|
},
|
|
{
|
|
Name: "Update exists identity by changing its identityId",
|
|
Form: bytes.NewBufferString(url.Values{
|
|
"identityId": {"2"},
|
|
"username": {"mock_username"},
|
|
"uuid": {"0f657aa8-bfbe-415d-b700-5750090d3af3"},
|
|
"skinId": {"5"},
|
|
"is1_8": {"0"},
|
|
"isSlim": {"0"},
|
|
"url": {"http://example.com/skin.png"},
|
|
}.Encode()),
|
|
BeforeTest: func(suite *apiTestSuite) {
|
|
suite.SkinsRepository.On("FindByUserId", 2).Return(nil, &SkinNotFoundError{Who: "unknown"})
|
|
suite.SkinsRepository.On("FindByUsername", "mock_username").Return(createSkinModel("mock_username", false), nil)
|
|
suite.SkinsRepository.On("RemoveByUsername", "mock_username").Times(1).Return(nil)
|
|
suite.SkinsRepository.On("Save", mock.MatchedBy(func(model *model.Skin) bool {
|
|
suite.Equal(2, model.UserId)
|
|
suite.Equal("mock_username", model.Username)
|
|
suite.Equal("0f657aa8-bfbe-415d-b700-5750090d3af3", model.Uuid)
|
|
|
|
return true
|
|
})).Times(1).Return(nil)
|
|
},
|
|
AfterTest: func(suite *apiTestSuite, response *http.Response) {
|
|
suite.Equal(201, response.StatusCode)
|
|
body, _ := ioutil.ReadAll(response.Body)
|
|
suite.Empty(body)
|
|
},
|
|
},
|
|
{
|
|
Name: "Update exists identity by changing its username",
|
|
Form: bytes.NewBufferString(url.Values{
|
|
"identityId": {"1"},
|
|
"username": {"changed_username"},
|
|
"uuid": {"0f657aa8-bfbe-415d-b700-5750090d3af3"},
|
|
"skinId": {"5"},
|
|
"is1_8": {"0"},
|
|
"isSlim": {"0"},
|
|
"url": {"http://example.com/skin.png"},
|
|
}.Encode()),
|
|
BeforeTest: func(suite *apiTestSuite) {
|
|
suite.SkinsRepository.On("FindByUserId", 1).Return(createSkinModel("mock_username", false), nil)
|
|
suite.SkinsRepository.On("RemoveByUserId", 1).Times(1).Return(nil)
|
|
suite.SkinsRepository.On("Save", mock.MatchedBy(func(model *model.Skin) bool {
|
|
suite.Equal(1, model.UserId)
|
|
suite.Equal("changed_username", model.Username)
|
|
suite.Equal("0f657aa8-bfbe-415d-b700-5750090d3af3", model.Uuid)
|
|
|
|
return true
|
|
})).Times(1).Return(nil)
|
|
},
|
|
AfterTest: func(suite *apiTestSuite, response *http.Response) {
|
|
suite.Equal(201, response.StatusCode)
|
|
body, _ := ioutil.ReadAll(response.Body)
|
|
suite.Empty(body)
|
|
},
|
|
},
|
|
{
|
|
Name: "Handle an error when loading the data from the repository",
|
|
Form: bytes.NewBufferString(url.Values{
|
|
"identityId": {"1"},
|
|
"username": {"mock_username"},
|
|
"uuid": {"0f657aa8-bfbe-415d-b700-5750090d3af3"},
|
|
"skinId": {"5"},
|
|
"is1_8": {"1"},
|
|
"isSlim": {"1"},
|
|
"url": {"http://textures-server.com/skin.png"},
|
|
}.Encode()),
|
|
BeforeTest: func(suite *apiTestSuite) {
|
|
suite.SkinsRepository.On("FindByUserId", 1).Return(createSkinModel("mock_username", false), nil)
|
|
err := errors.New("mock error")
|
|
suite.SkinsRepository.On("Save", mock.Anything).Return(err)
|
|
suite.Emitter.On("Emit", "skinsystem:error", mock.MatchedBy(func(cErr error) bool {
|
|
return cErr.Error() == "unable to save record to the repository: mock error" &&
|
|
errors.Is(cErr, err)
|
|
})).Once()
|
|
},
|
|
AfterTest: func(suite *apiTestSuite, response *http.Response) {
|
|
suite.Equal(500, response.StatusCode)
|
|
body, _ := ioutil.ReadAll(response.Body)
|
|
suite.Empty(body)
|
|
},
|
|
},
|
|
{
|
|
Name: "Handle an error when saving the data into the repository",
|
|
Form: bytes.NewBufferString(url.Values{
|
|
"identityId": {"1"},
|
|
"username": {"changed_username"},
|
|
"uuid": {"0f657aa8-bfbe-415d-b700-5750090d3af3"},
|
|
"skinId": {"5"},
|
|
"is1_8": {"0"},
|
|
"isSlim": {"0"},
|
|
"url": {"http://example.com/skin.png"},
|
|
}.Encode()),
|
|
BeforeTest: func(suite *apiTestSuite) {
|
|
err := errors.New("mock error")
|
|
suite.SkinsRepository.On("FindByUserId", 1).Return(nil, err)
|
|
suite.Emitter.On("Emit", "skinsystem:error", mock.MatchedBy(func(cErr error) bool {
|
|
return cErr.Error() == "error on requesting a skin from the repository: mock error" &&
|
|
errors.Is(cErr, err)
|
|
})).Once()
|
|
},
|
|
AfterTest: func(suite *apiTestSuite, response *http.Response) {
|
|
suite.Equal(500, response.StatusCode)
|
|
body, _ := ioutil.ReadAll(response.Body)
|
|
suite.Empty(body)
|
|
},
|
|
},
|
|
}
|
|
|
|
func (suite *apiTestSuite) TestPostSkin() {
|
|
for _, testCase := range postSkinTestsCases {
|
|
suite.RunSubTest(testCase.Name, func() {
|
|
testCase.BeforeTest(suite)
|
|
|
|
req := httptest.NewRequest("POST", "http://chrly/skins", testCase.Form)
|
|
req.Header.Add("Content-Type", "application/x-www-form-urlencoded")
|
|
w := httptest.NewRecorder()
|
|
|
|
suite.App.Handler().ServeHTTP(w, req)
|
|
|
|
testCase.AfterTest(suite, w.Result())
|
|
})
|
|
}
|
|
|
|
suite.RunSubTest("Get errors about required fields", func() {
|
|
req := httptest.NewRequest("POST", "http://chrly/skins", bytes.NewBufferString(url.Values{
|
|
"mojangTextures": {"someBase64EncodedString"},
|
|
}.Encode()))
|
|
req.Header.Add("Content-Type", "application/x-www-form-urlencoded")
|
|
w := httptest.NewRecorder()
|
|
|
|
suite.App.Handler().ServeHTTP(w, req)
|
|
|
|
resp := w.Result()
|
|
defer resp.Body.Close()
|
|
suite.Equal(400, resp.StatusCode)
|
|
body, _ := ioutil.ReadAll(resp.Body)
|
|
suite.JSONEq(`{
|
|
"errors": {
|
|
"identityId": [
|
|
"The identityId field is required",
|
|
"The identityId field must be numeric",
|
|
"The identityId field must be minimum 1 char"
|
|
],
|
|
"skinId": [
|
|
"The skinId field is required",
|
|
"The skinId field must be numeric",
|
|
"The skinId field must be minimum 1 char"
|
|
],
|
|
"username": [
|
|
"The username field is required"
|
|
],
|
|
"uuid": [
|
|
"The uuid field is required",
|
|
"The uuid field must contain valid UUID"
|
|
],
|
|
"url": [
|
|
"One of url or skin should be provided, but not both"
|
|
],
|
|
"skin": [
|
|
"One of url or skin should be provided, but not both"
|
|
],
|
|
"mojangSignature": [
|
|
"The mojangSignature field is required"
|
|
]
|
|
}
|
|
}`, string(body))
|
|
})
|
|
|
|
suite.RunSubTest("Upload textures with skin as file", func() {
|
|
inputBody := &bytes.Buffer{}
|
|
writer := multipart.NewWriter(inputBody)
|
|
|
|
part, _ := writer.CreateFormFile("skin", "char.png")
|
|
_, _ = part.Write(loadSkinFile())
|
|
|
|
_ = writer.WriteField("identityId", "1")
|
|
_ = writer.WriteField("username", "mock_user")
|
|
_ = writer.WriteField("uuid", "0f657aa8-bfbe-415d-b700-5750090d3af3")
|
|
_ = writer.WriteField("skinId", "5")
|
|
|
|
err := writer.Close()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
req := httptest.NewRequest("POST", "http://chrly/skins", inputBody)
|
|
req.Header.Add("Content-Type", writer.FormDataContentType())
|
|
w := httptest.NewRecorder()
|
|
|
|
suite.App.Handler().ServeHTTP(w, req)
|
|
|
|
resp := w.Result()
|
|
defer resp.Body.Close()
|
|
suite.Equal(400, resp.StatusCode)
|
|
responseBody, _ := ioutil.ReadAll(resp.Body)
|
|
suite.JSONEq(`{
|
|
"errors": {
|
|
"skin": [
|
|
"Skin uploading is temporary unavailable"
|
|
]
|
|
}
|
|
}`, string(responseBody))
|
|
})
|
|
}
|
|
|
|
/**************************************
|
|
* Delete skin by user id tests cases *
|
|
**************************************/
|
|
|
|
func (suite *apiTestSuite) TestDeleteByUserId() {
|
|
suite.RunSubTest("Delete skin by its identity id", func() {
|
|
suite.SkinsRepository.On("FindByUserId", 1).Return(createSkinModel("mock_username", false), nil)
|
|
suite.SkinsRepository.On("RemoveByUserId", 1).Once().Return(nil)
|
|
|
|
req := httptest.NewRequest("DELETE", "http://chrly/skins/id:1", nil)
|
|
w := httptest.NewRecorder()
|
|
|
|
suite.App.Handler().ServeHTTP(w, req)
|
|
|
|
resp := w.Result()
|
|
defer resp.Body.Close()
|
|
suite.Equal(204, resp.StatusCode)
|
|
body, _ := ioutil.ReadAll(resp.Body)
|
|
suite.Empty(body)
|
|
})
|
|
|
|
suite.RunSubTest("Try to remove not exists identity id", func() {
|
|
suite.SkinsRepository.On("FindByUserId", 1).Return(nil, &SkinNotFoundError{Who: "unknown"})
|
|
|
|
req := httptest.NewRequest("DELETE", "http://chrly/skins/id:1", nil)
|
|
w := httptest.NewRecorder()
|
|
|
|
suite.App.Handler().ServeHTTP(w, req)
|
|
|
|
resp := w.Result()
|
|
defer resp.Body.Close()
|
|
suite.Equal(404, resp.StatusCode)
|
|
body, _ := ioutil.ReadAll(resp.Body)
|
|
suite.JSONEq(`[
|
|
"Cannot find record for the requested identifier"
|
|
]`, string(body))
|
|
})
|
|
}
|
|
|
|
/***************************************
|
|
* Delete skin by username tests cases *
|
|
***************************************/
|
|
|
|
func (suite *apiTestSuite) TestDeleteByUsername() {
|
|
suite.RunSubTest("Delete skin by its identity username", func() {
|
|
suite.SkinsRepository.On("FindByUsername", "mock_username").Return(createSkinModel("mock_username", false), nil)
|
|
suite.SkinsRepository.On("RemoveByUserId", 1).Once().Return(nil)
|
|
|
|
req := httptest.NewRequest("DELETE", "http://chrly/skins/mock_username", nil)
|
|
w := httptest.NewRecorder()
|
|
|
|
suite.App.Handler().ServeHTTP(w, req)
|
|
|
|
resp := w.Result()
|
|
defer resp.Body.Close()
|
|
suite.Equal(204, resp.StatusCode)
|
|
body, _ := ioutil.ReadAll(resp.Body)
|
|
suite.Empty(body)
|
|
})
|
|
|
|
suite.RunSubTest("Try to remove not exists identity username", func() {
|
|
suite.SkinsRepository.On("FindByUsername", "mock_username").Return(nil, &SkinNotFoundError{Who: "mock_username"})
|
|
|
|
req := httptest.NewRequest("DELETE", "http://chrly/skins/mock_username", nil)
|
|
w := httptest.NewRecorder()
|
|
|
|
suite.App.Handler().ServeHTTP(w, req)
|
|
|
|
resp := w.Result()
|
|
defer resp.Body.Close()
|
|
suite.Equal(404, resp.StatusCode)
|
|
body, _ := ioutil.ReadAll(resp.Body)
|
|
suite.JSONEq(`[
|
|
"Cannot find record for the requested identifier"
|
|
]`, string(body))
|
|
})
|
|
}
|
|
|
|
/*************
|
|
* Utilities *
|
|
*************/
|
|
|
|
// base64 https://github.com/mathiasbynens/small/blob/0ca3c51/png-transparent.png
|
|
var OnePxPng = []byte("iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAACklEQVR4nGMAAQAABQABDQottAAAAABJRU5ErkJggg==")
|
|
|
|
func loadSkinFile() []byte {
|
|
result := make([]byte, 92)
|
|
_, err := base64.StdEncoding.Decode(result, OnePxPng)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
return result
|
|
}
|