2019-04-18 02:56:20 +03:00
|
|
|
package queue
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/rand"
|
|
|
|
"encoding/base64"
|
2019-04-30 00:36:51 +03:00
|
|
|
"errors"
|
2019-04-21 03:04:03 +03:00
|
|
|
"net"
|
2019-05-05 23:06:29 +03:00
|
|
|
"net/url"
|
2019-04-20 22:22:02 +03:00
|
|
|
"strings"
|
2019-04-21 03:04:03 +03:00
|
|
|
"syscall"
|
2019-04-20 22:22:02 +03:00
|
|
|
"time"
|
2019-04-25 00:45:04 +03:00
|
|
|
|
|
|
|
"github.com/elyby/chrly/api/mojang"
|
|
|
|
|
2019-05-05 23:06:29 +03:00
|
|
|
"testing"
|
|
|
|
|
2019-04-25 00:45:04 +03:00
|
|
|
"github.com/stretchr/testify/mock"
|
|
|
|
"github.com/stretchr/testify/suite"
|
2019-05-05 23:06:29 +03:00
|
|
|
|
|
|
|
mocks "github.com/elyby/chrly/tests"
|
2019-04-18 02:56:20 +03:00
|
|
|
)
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
type mojangApiMocks struct {
|
2019-04-19 01:41:52 +03:00
|
|
|
mock.Mock
|
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (o *mojangApiMocks) UsernamesToUuids(usernames []string) ([]*mojang.ProfileInfo, error) {
|
2019-04-19 01:41:52 +03:00
|
|
|
args := o.Called(usernames)
|
|
|
|
var result []*mojang.ProfileInfo
|
|
|
|
if casted, ok := args.Get(0).([]*mojang.ProfileInfo); ok {
|
|
|
|
result = casted
|
|
|
|
}
|
|
|
|
|
|
|
|
return result, args.Error(1)
|
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (o *mojangApiMocks) UuidToTextures(uuid string, signed bool) (*mojang.SignedTexturesResponse, error) {
|
2019-04-19 01:41:52 +03:00
|
|
|
args := o.Called(uuid, signed)
|
|
|
|
var result *mojang.SignedTexturesResponse
|
|
|
|
if casted, ok := args.Get(0).(*mojang.SignedTexturesResponse); ok {
|
|
|
|
result = casted
|
|
|
|
}
|
|
|
|
|
|
|
|
return result, args.Error(1)
|
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
type mockStorage struct {
|
2019-04-20 19:35:37 +03:00
|
|
|
mock.Mock
|
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (m *mockStorage) GetUuid(username string) (string, error) {
|
2019-04-20 19:35:37 +03:00
|
|
|
args := m.Called(username)
|
2019-04-20 22:22:02 +03:00
|
|
|
return args.String(0), args.Error(1)
|
|
|
|
}
|
|
|
|
|
2019-05-06 17:12:37 +03:00
|
|
|
func (m *mockStorage) StoreUuid(username string, uuid string) error {
|
|
|
|
args := m.Called(username, uuid)
|
|
|
|
return args.Error(0)
|
2019-04-20 22:22:02 +03:00
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (m *mockStorage) GetTextures(uuid string) (*mojang.SignedTexturesResponse, error) {
|
2019-04-20 22:22:02 +03:00
|
|
|
args := m.Called(uuid)
|
2019-04-20 19:35:37 +03:00
|
|
|
var result *mojang.SignedTexturesResponse
|
|
|
|
if casted, ok := args.Get(0).(*mojang.SignedTexturesResponse); ok {
|
|
|
|
result = casted
|
|
|
|
}
|
|
|
|
|
2019-04-20 22:22:02 +03:00
|
|
|
return result, args.Error(1)
|
2019-04-20 19:35:37 +03:00
|
|
|
}
|
|
|
|
|
2019-05-05 23:06:29 +03:00
|
|
|
func (m *mockStorage) StoreTextures(uuid string, textures *mojang.SignedTexturesResponse) {
|
|
|
|
m.Called(uuid, textures)
|
2019-04-20 19:35:37 +03:00
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
type queueTestSuite struct {
|
2019-04-19 01:41:52 +03:00
|
|
|
suite.Suite
|
|
|
|
Queue *JobsQueue
|
2019-04-21 03:04:03 +03:00
|
|
|
Storage *mockStorage
|
|
|
|
MojangApi *mojangApiMocks
|
2019-04-25 00:45:04 +03:00
|
|
|
Logger *mocks.WdMock
|
2019-04-19 01:41:52 +03:00
|
|
|
Iterate func()
|
|
|
|
|
|
|
|
iterateChan chan bool
|
|
|
|
done func()
|
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (suite *queueTestSuite) SetupSuite() {
|
2019-05-05 23:06:29 +03:00
|
|
|
uuidsQueueIterationDelay = 0
|
2019-04-18 02:56:20 +03:00
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (suite *queueTestSuite) SetupTest() {
|
|
|
|
suite.Storage = &mockStorage{}
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Logger = &mocks.WdMock{}
|
2019-04-20 19:35:37 +03:00
|
|
|
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Queue = &JobsQueue{Storage: suite.Storage, Logger: suite.Logger}
|
2019-04-19 01:41:52 +03:00
|
|
|
|
|
|
|
suite.iterateChan = make(chan bool)
|
|
|
|
forever = func() bool {
|
|
|
|
return <-suite.iterateChan
|
|
|
|
}
|
|
|
|
|
|
|
|
suite.Iterate = func() {
|
|
|
|
suite.iterateChan <- true
|
|
|
|
}
|
|
|
|
|
|
|
|
suite.done = func() {
|
|
|
|
suite.iterateChan <- false
|
2019-04-18 02:56:20 +03:00
|
|
|
}
|
2019-04-19 01:41:52 +03:00
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
suite.MojangApi = new(mojangApiMocks)
|
|
|
|
usernamesToUuids = suite.MojangApi.UsernamesToUuids
|
2019-04-19 01:41:52 +03:00
|
|
|
uuidToTextures = suite.MojangApi.UuidToTextures
|
2019-04-18 02:56:20 +03:00
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (suite *queueTestSuite) TearDownTest() {
|
2019-04-19 01:41:52 +03:00
|
|
|
suite.done()
|
2019-04-28 20:24:08 +03:00
|
|
|
time.Sleep(10 * time.Millisecond) // Add delay to let finish all goroutines before assert mocks calls
|
2019-04-19 01:41:52 +03:00
|
|
|
suite.MojangApi.AssertExpectations(suite.T())
|
2019-04-20 19:35:37 +03:00
|
|
|
suite.Storage.AssertExpectations(suite.T())
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Logger.AssertExpectations(suite.T())
|
2019-04-18 02:56:20 +03:00
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (suite *queueTestSuite) TestReceiveTexturesForOneUsernameWithoutAnyCache() {
|
2019-04-20 19:35:37 +03:00
|
|
|
expectedResult := &mojang.SignedTexturesResponse{Id: "0d252b7218b648bfb86c2ae476954d32", Name: "maksimkurb"}
|
|
|
|
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.request", int64(1)).Once()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.queued", int64(1)).Once()
|
|
|
|
suite.Logger.On("UpdateGauge", "mojang_textures.usernames.iteration_size", int64(1)).Once()
|
|
|
|
suite.Logger.On("UpdateGauge", "mojang_textures.usernames.queue_size", int64(0)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.usernames.round_time", mock.Anything)
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.textures.request", int64(1)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.textures.request_time", mock.Anything).Once()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.uuid_hit", int64(1)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.result_time", mock.Anything).Once()
|
|
|
|
|
2019-04-20 22:22:02 +03:00
|
|
|
suite.Storage.On("GetUuid", "maksimkurb").Once().Return("", &ValueNotFound{})
|
2019-05-06 17:12:37 +03:00
|
|
|
suite.Storage.On("StoreUuid", "maksimkurb", "0d252b7218b648bfb86c2ae476954d32").Once().Return(nil)
|
2019-04-20 22:22:02 +03:00
|
|
|
suite.Storage.On("GetTextures", "0d252b7218b648bfb86c2ae476954d32").Once().Return(nil, &ValueNotFound{})
|
2019-05-05 23:06:29 +03:00
|
|
|
suite.Storage.On("StoreTextures", "0d252b7218b648bfb86c2ae476954d32", expectedResult).Once()
|
2019-04-25 00:45:04 +03:00
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
suite.MojangApi.On("UsernamesToUuids", []string{"maksimkurb"}).Once().Return([]*mojang.ProfileInfo{
|
2019-04-19 01:41:52 +03:00
|
|
|
{Id: "0d252b7218b648bfb86c2ae476954d32", Name: "maksimkurb"},
|
|
|
|
}, nil)
|
2019-04-20 19:35:37 +03:00
|
|
|
suite.MojangApi.On("UuidToTextures", "0d252b7218b648bfb86c2ae476954d32", true).Once().Return(expectedResult, nil)
|
2019-04-19 01:41:52 +03:00
|
|
|
|
|
|
|
resultChan := suite.Queue.GetTexturesForUsername("maksimkurb")
|
|
|
|
|
|
|
|
suite.Iterate()
|
|
|
|
|
|
|
|
result := <-resultChan
|
2019-04-20 19:35:37 +03:00
|
|
|
suite.Assert().Equal(expectedResult, result)
|
2019-04-19 01:41:52 +03:00
|
|
|
}
|
2019-04-18 02:56:20 +03:00
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (suite *queueTestSuite) TestReceiveTexturesForFewUsernamesWithoutAnyCache() {
|
2019-04-20 19:35:37 +03:00
|
|
|
expectedResult1 := &mojang.SignedTexturesResponse{Id: "0d252b7218b648bfb86c2ae476954d32", Name: "maksimkurb"}
|
|
|
|
expectedResult2 := &mojang.SignedTexturesResponse{Id: "4566e69fc90748ee8d71d7ba5aa00d20", Name: "Thinkofdeath"}
|
|
|
|
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.request", int64(1)).Twice()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.queued", int64(1)).Twice()
|
|
|
|
suite.Logger.On("UpdateGauge", "mojang_textures.usernames.iteration_size", int64(2)).Once()
|
|
|
|
suite.Logger.On("UpdateGauge", "mojang_textures.usernames.queue_size", int64(0)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.usernames.round_time", mock.Anything)
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.textures.request", int64(1)).Twice()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.textures.request_time", mock.Anything).Twice()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.uuid_hit", int64(1)).Twice()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.result_time", mock.Anything).Twice()
|
|
|
|
|
2019-04-20 22:22:02 +03:00
|
|
|
suite.Storage.On("GetUuid", "maksimkurb").Once().Return("", &ValueNotFound{})
|
|
|
|
suite.Storage.On("GetUuid", "Thinkofdeath").Once().Return("", &ValueNotFound{})
|
2019-05-06 17:12:37 +03:00
|
|
|
suite.Storage.On("StoreUuid", "maksimkurb", "0d252b7218b648bfb86c2ae476954d32").Once().Return(nil)
|
|
|
|
suite.Storage.On("StoreUuid", "Thinkofdeath", "4566e69fc90748ee8d71d7ba5aa00d20").Once().Return(nil)
|
2019-04-20 22:22:02 +03:00
|
|
|
suite.Storage.On("GetTextures", "0d252b7218b648bfb86c2ae476954d32").Once().Return(nil, &ValueNotFound{})
|
|
|
|
suite.Storage.On("GetTextures", "4566e69fc90748ee8d71d7ba5aa00d20").Once().Return(nil, &ValueNotFound{})
|
2019-05-05 23:06:29 +03:00
|
|
|
suite.Storage.On("StoreTextures", "0d252b7218b648bfb86c2ae476954d32", expectedResult1).Once()
|
|
|
|
suite.Storage.On("StoreTextures", "4566e69fc90748ee8d71d7ba5aa00d20", expectedResult2).Once()
|
2019-04-25 00:45:04 +03:00
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
suite.MojangApi.On("UsernamesToUuids", []string{"maksimkurb", "Thinkofdeath"}).Once().Return([]*mojang.ProfileInfo{
|
2019-04-19 01:41:52 +03:00
|
|
|
{Id: "0d252b7218b648bfb86c2ae476954d32", Name: "maksimkurb"},
|
|
|
|
{Id: "4566e69fc90748ee8d71d7ba5aa00d20", Name: "Thinkofdeath"},
|
|
|
|
}, nil)
|
2019-04-20 19:35:37 +03:00
|
|
|
suite.MojangApi.On("UuidToTextures", "0d252b7218b648bfb86c2ae476954d32", true).Once().Return(expectedResult1, nil)
|
|
|
|
suite.MojangApi.On("UuidToTextures", "4566e69fc90748ee8d71d7ba5aa00d20", true).Once().Return(expectedResult2, nil)
|
2019-04-19 01:41:52 +03:00
|
|
|
|
|
|
|
resultChan1 := suite.Queue.GetTexturesForUsername("maksimkurb")
|
|
|
|
resultChan2 := suite.Queue.GetTexturesForUsername("Thinkofdeath")
|
|
|
|
|
|
|
|
suite.Iterate()
|
|
|
|
|
2019-04-20 19:35:37 +03:00
|
|
|
suite.Assert().Equal(expectedResult1, <-resultChan1)
|
|
|
|
suite.Assert().Equal(expectedResult2, <-resultChan2)
|
2019-04-18 02:56:20 +03:00
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (suite *queueTestSuite) TestReceiveTexturesForUsernameWithCachedUuid() {
|
2019-04-20 22:22:02 +03:00
|
|
|
expectedResult := &mojang.SignedTexturesResponse{Id: "0d252b7218b648bfb86c2ae476954d32", Name: "maksimkurb"}
|
|
|
|
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.request", int64(1)).Once()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.cache_hit", int64(1)).Once()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.textures.request", int64(1)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.textures.request_time", mock.Anything).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.result_time", mock.Anything).Once()
|
|
|
|
|
2019-04-20 22:22:02 +03:00
|
|
|
suite.Storage.On("GetUuid", "maksimkurb").Once().Return("0d252b7218b648bfb86c2ae476954d32", nil)
|
|
|
|
// Storage.StoreUuid shouldn't be called
|
|
|
|
suite.Storage.On("GetTextures", "0d252b7218b648bfb86c2ae476954d32").Once().Return(nil, &ValueNotFound{})
|
2019-05-05 23:06:29 +03:00
|
|
|
suite.Storage.On("StoreTextures", "0d252b7218b648bfb86c2ae476954d32", expectedResult).Once()
|
2019-04-25 00:45:04 +03:00
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
// MojangApi.UsernamesToUuids shouldn't be called
|
2019-04-20 22:22:02 +03:00
|
|
|
suite.MojangApi.On("UuidToTextures", "0d252b7218b648bfb86c2ae476954d32", true).Once().Return(expectedResult, nil)
|
|
|
|
|
|
|
|
resultChan := suite.Queue.GetTexturesForUsername("maksimkurb")
|
|
|
|
|
|
|
|
// Note that there is no iteration
|
|
|
|
|
|
|
|
result := <-resultChan
|
|
|
|
suite.Assert().Equal(expectedResult, result)
|
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (suite *queueTestSuite) TestReceiveTexturesForUsernameWithFullyCachedResult() {
|
2019-04-20 22:22:02 +03:00
|
|
|
expectedResult := &mojang.SignedTexturesResponse{Id: "0d252b7218b648bfb86c2ae476954d32", Name: "maksimkurb"}
|
|
|
|
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.request", int64(1)).Once()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.cache_hit", int64(1)).Once()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.textures.cache_hit", int64(1)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.result_time", mock.Anything).Once()
|
|
|
|
|
2019-04-20 22:22:02 +03:00
|
|
|
suite.Storage.On("GetUuid", "maksimkurb").Once().Return("0d252b7218b648bfb86c2ae476954d32", nil)
|
|
|
|
// Storage.StoreUuid shouldn't be called
|
|
|
|
suite.Storage.On("GetTextures", "0d252b7218b648bfb86c2ae476954d32").Once().Return(expectedResult, nil)
|
|
|
|
// Storage.StoreTextures shouldn't be called
|
2019-04-25 00:45:04 +03:00
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
// MojangApi.UsernamesToUuids shouldn't be called
|
2019-04-20 22:22:02 +03:00
|
|
|
// MojangApi.UuidToTextures shouldn't be called
|
|
|
|
|
|
|
|
resultChan := suite.Queue.GetTexturesForUsername("maksimkurb")
|
|
|
|
|
|
|
|
// Note that there is no iteration
|
|
|
|
|
|
|
|
result := <-resultChan
|
|
|
|
suite.Assert().Equal(expectedResult, result)
|
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (suite *queueTestSuite) TestReceiveTexturesForUsernameWithCachedUnknownUuid() {
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.request", int64(1)).Once()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.cache_hit_nil", int64(1)).Once()
|
|
|
|
|
2019-04-20 22:22:02 +03:00
|
|
|
suite.Storage.On("GetUuid", "maksimkurb").Once().Return("", nil)
|
|
|
|
// Storage.StoreUuid shouldn't be called
|
|
|
|
// Storage.GetTextures shouldn't be called
|
|
|
|
// Storage.StoreTextures shouldn't be called
|
2019-04-25 00:45:04 +03:00
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
// MojangApi.UsernamesToUuids shouldn't be called
|
2019-04-20 22:22:02 +03:00
|
|
|
// MojangApi.UuidToTextures shouldn't be called
|
|
|
|
|
|
|
|
resultChan := suite.Queue.GetTexturesForUsername("maksimkurb")
|
|
|
|
|
|
|
|
// Note that there is no iteration
|
|
|
|
|
|
|
|
suite.Assert().Nil(<-resultChan)
|
|
|
|
}
|
|
|
|
|
2019-10-03 01:00:27 +03:00
|
|
|
func (suite *queueTestSuite) TestReceiveTexturesForMoreThan10Usernames() {
|
|
|
|
usernames := make([]string, 12)
|
|
|
|
for i := 0; i < cap(usernames); i++ {
|
2019-04-19 01:41:52 +03:00
|
|
|
usernames[i] = randStr(8)
|
|
|
|
}
|
|
|
|
|
2019-10-03 01:00:27 +03:00
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.request", int64(1)).Times(12)
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.queued", int64(1)).Times(12)
|
|
|
|
suite.Logger.On("UpdateGauge", "mojang_textures.usernames.iteration_size", int64(10)).Once()
|
|
|
|
suite.Logger.On("UpdateGauge", "mojang_textures.usernames.iteration_size", int64(2)).Once()
|
|
|
|
suite.Logger.On("UpdateGauge", "mojang_textures.usernames.queue_size", int64(2)).Once()
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Logger.On("UpdateGauge", "mojang_textures.usernames.queue_size", int64(0)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.usernames.round_time", mock.Anything).Twice()
|
2019-10-03 01:00:27 +03:00
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.uuid_miss", int64(1)).Times(12)
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.result_time", mock.Anything).Times(12)
|
2019-04-25 00:45:04 +03:00
|
|
|
|
2019-10-03 01:00:27 +03:00
|
|
|
suite.Storage.On("GetUuid", mock.Anything).Times(12).Return("", &ValueNotFound{})
|
|
|
|
suite.Storage.On("StoreUuid", mock.Anything, "").Times(12).Return(nil) // should be called with "" if username is not compared to uuid
|
2019-04-20 22:22:02 +03:00
|
|
|
// Storage.GetTextures and Storage.SetTextures shouldn't be called
|
2019-04-25 00:45:04 +03:00
|
|
|
|
2019-10-03 01:00:27 +03:00
|
|
|
suite.MojangApi.On("UsernamesToUuids", usernames[0:10]).Once().Return([]*mojang.ProfileInfo{}, nil)
|
|
|
|
suite.MojangApi.On("UsernamesToUuids", usernames[10:12]).Once().Return([]*mojang.ProfileInfo{}, nil)
|
2019-04-19 01:41:52 +03:00
|
|
|
|
2019-10-03 01:00:27 +03:00
|
|
|
channels := make([]chan *mojang.SignedTexturesResponse, 12)
|
2019-04-25 00:45:04 +03:00
|
|
|
for i, username := range usernames {
|
|
|
|
channels[i] = suite.Queue.GetTexturesForUsername(username)
|
2019-04-18 02:56:20 +03:00
|
|
|
}
|
2019-04-19 01:41:52 +03:00
|
|
|
|
|
|
|
suite.Iterate()
|
|
|
|
suite.Iterate()
|
2019-04-25 00:45:04 +03:00
|
|
|
|
|
|
|
for _, channel := range channels {
|
|
|
|
<-channel
|
|
|
|
}
|
2019-04-19 01:41:52 +03:00
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (suite *queueTestSuite) TestReceiveTexturesForTheSameUsernames() {
|
2019-04-20 19:35:37 +03:00
|
|
|
expectedResult := &mojang.SignedTexturesResponse{Id: "0d252b7218b648bfb86c2ae476954d32", Name: "maksimkurb"}
|
|
|
|
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.request", int64(1)).Twice()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.queued", int64(1)).Once()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.already_in_queue", int64(1)).Once()
|
|
|
|
suite.Logger.On("UpdateGauge", "mojang_textures.usernames.iteration_size", int64(1)).Once()
|
|
|
|
suite.Logger.On("UpdateGauge", "mojang_textures.usernames.queue_size", int64(0)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.usernames.round_time", mock.Anything)
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.textures.request", int64(1)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.textures.request_time", mock.Anything).Once()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.uuid_hit", int64(1)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.result_time", mock.Anything).Once()
|
|
|
|
|
2019-04-20 22:22:02 +03:00
|
|
|
suite.Storage.On("GetUuid", "maksimkurb").Twice().Return("", &ValueNotFound{})
|
2019-05-06 17:12:37 +03:00
|
|
|
suite.Storage.On("StoreUuid", "maksimkurb", "0d252b7218b648bfb86c2ae476954d32").Once().Return(nil)
|
2019-04-20 22:22:02 +03:00
|
|
|
suite.Storage.On("GetTextures", "0d252b7218b648bfb86c2ae476954d32").Once().Return(nil, &ValueNotFound{})
|
2019-05-05 23:06:29 +03:00
|
|
|
suite.Storage.On("StoreTextures", "0d252b7218b648bfb86c2ae476954d32", expectedResult).Once()
|
2019-04-21 03:04:03 +03:00
|
|
|
suite.MojangApi.On("UsernamesToUuids", []string{"maksimkurb"}).Once().Return([]*mojang.ProfileInfo{
|
2019-04-20 03:23:49 +03:00
|
|
|
{Id: "0d252b7218b648bfb86c2ae476954d32", Name: "maksimkurb"},
|
|
|
|
}, nil)
|
2019-04-20 19:35:37 +03:00
|
|
|
suite.MojangApi.On("UuidToTextures", "0d252b7218b648bfb86c2ae476954d32", true).Once().Return(expectedResult, nil)
|
2019-04-20 03:23:49 +03:00
|
|
|
|
|
|
|
resultChan1 := suite.Queue.GetTexturesForUsername("maksimkurb")
|
|
|
|
resultChan2 := suite.Queue.GetTexturesForUsername("maksimkurb")
|
|
|
|
|
|
|
|
suite.Iterate()
|
|
|
|
|
2019-04-20 19:35:37 +03:00
|
|
|
suite.Assert().Equal(expectedResult, <-resultChan1)
|
|
|
|
suite.Assert().Equal(expectedResult, <-resultChan2)
|
2019-04-20 03:23:49 +03:00
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (suite *queueTestSuite) TestReceiveTexturesForUsernameThatAlreadyProcessing() {
|
2019-04-20 19:35:37 +03:00
|
|
|
expectedResult := &mojang.SignedTexturesResponse{Id: "0d252b7218b648bfb86c2ae476954d32", Name: "maksimkurb"}
|
|
|
|
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.request", int64(1)).Twice()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.queued", int64(1)).Once()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.already_in_queue", int64(1)).Once()
|
|
|
|
suite.Logger.On("UpdateGauge", "mojang_textures.usernames.iteration_size", int64(1)).Once()
|
|
|
|
suite.Logger.On("UpdateGauge", "mojang_textures.usernames.queue_size", int64(0)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.usernames.round_time", mock.Anything)
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.textures.request", int64(1)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.textures.request_time", mock.Anything).Once()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.uuid_hit", int64(1)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.result_time", mock.Anything).Once()
|
|
|
|
|
2019-04-20 22:22:02 +03:00
|
|
|
suite.Storage.On("GetUuid", "maksimkurb").Twice().Return("", &ValueNotFound{})
|
2019-05-06 17:12:37 +03:00
|
|
|
suite.Storage.On("StoreUuid", "maksimkurb", "0d252b7218b648bfb86c2ae476954d32").Once().Return(nil)
|
2019-04-20 22:22:02 +03:00
|
|
|
suite.Storage.On("GetTextures", "0d252b7218b648bfb86c2ae476954d32").Once().Return(nil, &ValueNotFound{})
|
2019-05-05 23:06:29 +03:00
|
|
|
suite.Storage.On("StoreTextures", "0d252b7218b648bfb86c2ae476954d32", expectedResult).Once()
|
2019-04-25 00:45:04 +03:00
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
suite.MojangApi.On("UsernamesToUuids", []string{"maksimkurb"}).Once().Return([]*mojang.ProfileInfo{
|
2019-04-20 03:23:49 +03:00
|
|
|
{Id: "0d252b7218b648bfb86c2ae476954d32", Name: "maksimkurb"},
|
|
|
|
}, nil)
|
|
|
|
suite.MojangApi.On("UuidToTextures", "0d252b7218b648bfb86c2ae476954d32", true).
|
|
|
|
Once().
|
|
|
|
After(10*time.Millisecond). // Simulate long round trip
|
2019-04-20 19:35:37 +03:00
|
|
|
Return(expectedResult, nil)
|
2019-04-20 03:23:49 +03:00
|
|
|
|
|
|
|
resultChan1 := suite.Queue.GetTexturesForUsername("maksimkurb")
|
|
|
|
|
|
|
|
// Note that for entire test there is only one iteration
|
|
|
|
suite.Iterate()
|
|
|
|
|
|
|
|
// Let it meet delayed UuidToTextures request
|
|
|
|
time.Sleep(5 * time.Millisecond)
|
|
|
|
|
|
|
|
resultChan2 := suite.Queue.GetTexturesForUsername("maksimkurb")
|
|
|
|
|
2019-04-20 19:35:37 +03:00
|
|
|
suite.Assert().Equal(expectedResult, <-resultChan1)
|
|
|
|
suite.Assert().Equal(expectedResult, <-resultChan2)
|
2019-04-20 03:23:49 +03:00
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (suite *queueTestSuite) TestDoNothingWhenNoTasks() {
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.request", int64(1)).Once()
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.queued", int64(1)).Once()
|
|
|
|
suite.Logger.On("UpdateGauge", "mojang_textures.usernames.iteration_size", int64(1)).Once()
|
|
|
|
suite.Logger.On("UpdateGauge", "mojang_textures.usernames.queue_size", int64(0)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.usernames.round_time", mock.Anything)
|
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.usernames.uuid_miss", int64(1)).Once()
|
|
|
|
suite.Logger.On("RecordTimer", "mojang_textures.result_time", mock.Anything).Once()
|
|
|
|
|
2019-04-20 22:22:02 +03:00
|
|
|
suite.Storage.On("GetUuid", "maksimkurb").Once().Return("", &ValueNotFound{})
|
2019-05-06 17:12:37 +03:00
|
|
|
suite.Storage.On("StoreUuid", "maksimkurb", "").Once().Return(nil)
|
2019-04-20 22:22:02 +03:00
|
|
|
// Storage.GetTextures and Storage.StoreTextures shouldn't be called
|
2019-04-25 00:45:04 +03:00
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
suite.MojangApi.On("UsernamesToUuids", []string{"maksimkurb"}).Once().Return([]*mojang.ProfileInfo{}, nil)
|
2019-04-19 01:41:52 +03:00
|
|
|
|
|
|
|
// Perform first iteration and await it finish
|
|
|
|
resultChan := suite.Queue.GetTexturesForUsername("maksimkurb")
|
|
|
|
|
|
|
|
suite.Iterate()
|
|
|
|
|
|
|
|
suite.Assert().Nil(<-resultChan)
|
|
|
|
|
|
|
|
// Let it to perform a few more iterations to ensure, that there is no calls to external APIs
|
|
|
|
suite.Iterate()
|
|
|
|
suite.Iterate()
|
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
type timeoutError struct {
|
2019-04-19 01:41:52 +03:00
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (*timeoutError) Error() string { return "timeout error" }
|
|
|
|
func (*timeoutError) Timeout() bool { return true }
|
|
|
|
func (*timeoutError) Temporary() bool { return false }
|
|
|
|
|
|
|
|
var expectedErrors = []error{
|
|
|
|
&mojang.BadRequestError{},
|
2019-11-08 01:32:26 +03:00
|
|
|
&mojang.ForbiddenError{},
|
2019-04-21 03:04:03 +03:00
|
|
|
&mojang.TooManyRequestsError{},
|
|
|
|
&mojang.ServerError{},
|
|
|
|
&timeoutError{},
|
2019-05-05 23:06:29 +03:00
|
|
|
&url.Error{Op: "GET", URL: "http://localhost"},
|
2019-04-21 03:04:03 +03:00
|
|
|
&net.OpError{Op: "read"},
|
|
|
|
&net.OpError{Op: "dial"},
|
|
|
|
syscall.ECONNREFUSED,
|
2019-04-20 22:39:17 +03:00
|
|
|
}
|
|
|
|
|
2019-04-30 00:36:51 +03:00
|
|
|
func (suite *queueTestSuite) TestShouldNotLogErrorWhenExpectedErrorReturnedFromUsernameToUuidRequest() {
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Logger.On("IncCounter", mock.Anything, mock.Anything)
|
|
|
|
suite.Logger.On("UpdateGauge", mock.Anything, mock.Anything)
|
|
|
|
suite.Logger.On("RecordTimer", mock.Anything, mock.Anything)
|
2019-05-05 23:06:29 +03:00
|
|
|
suite.Logger.On("Debug", ":name: Got response error :err", mock.Anything, mock.Anything).Times(len(expectedErrors))
|
2019-10-03 01:00:27 +03:00
|
|
|
suite.Logger.On("Warning", ":name: Got 400 Bad Request :err", mock.Anything, mock.Anything).Once()
|
2019-11-08 01:32:26 +03:00
|
|
|
suite.Logger.On("Warning", ":name: Got 403 Forbidden :err", mock.Anything, mock.Anything).Once()
|
2019-05-05 23:06:29 +03:00
|
|
|
suite.Logger.On("Warning", ":name: Got 429 Too Many Requests :err", mock.Anything, mock.Anything).Once()
|
2019-04-25 00:45:04 +03:00
|
|
|
|
|
|
|
suite.Storage.On("GetUuid", "maksimkurb").Return("", &ValueNotFound{})
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
for _, err := range expectedErrors {
|
|
|
|
suite.MojangApi.On("UsernamesToUuids", []string{"maksimkurb"}).Once().Return(nil, err)
|
2019-04-25 00:45:04 +03:00
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
resultChan := suite.Queue.GetTexturesForUsername("maksimkurb")
|
|
|
|
suite.Iterate()
|
|
|
|
suite.Assert().Nil(<-resultChan)
|
|
|
|
suite.MojangApi.AssertExpectations(suite.T())
|
|
|
|
suite.MojangApi.ExpectedCalls = nil // https://github.com/stretchr/testify/issues/558#issuecomment-372112364
|
|
|
|
}
|
2019-04-19 01:41:52 +03:00
|
|
|
}
|
|
|
|
|
2019-04-30 00:36:51 +03:00
|
|
|
func (suite *queueTestSuite) TestShouldLogEmergencyOnUnexpectedErrorReturnedFromUsernameToUuidRequest() {
|
|
|
|
suite.Logger.On("IncCounter", mock.Anything, mock.Anything)
|
|
|
|
suite.Logger.On("UpdateGauge", mock.Anything, mock.Anything)
|
|
|
|
suite.Logger.On("RecordTimer", mock.Anything, mock.Anything)
|
2019-05-05 23:06:29 +03:00
|
|
|
suite.Logger.On("Debug", ":name: Got response error :err", mock.Anything, mock.Anything).Once()
|
|
|
|
suite.Logger.On("Emergency", ":name: Unknown Mojang response error: :err", mock.Anything, mock.Anything).Once()
|
2019-04-30 00:36:51 +03:00
|
|
|
|
|
|
|
suite.Storage.On("GetUuid", "maksimkurb").Return("", &ValueNotFound{})
|
|
|
|
|
|
|
|
suite.MojangApi.On("UsernamesToUuids", []string{"maksimkurb"}).Once().Return(nil, errors.New("unexpected error"))
|
|
|
|
|
|
|
|
resultChan := suite.Queue.GetTexturesForUsername("maksimkurb")
|
|
|
|
suite.Iterate()
|
|
|
|
suite.Assert().Nil(<-resultChan)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (suite *queueTestSuite) TestShouldNotLogErrorWhenExpectedErrorReturnedFromUuidToTexturesRequest() {
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Logger.On("IncCounter", mock.Anything, mock.Anything)
|
|
|
|
suite.Logger.On("UpdateGauge", mock.Anything, mock.Anything)
|
|
|
|
suite.Logger.On("RecordTimer", mock.Anything, mock.Anything)
|
2019-05-05 23:06:29 +03:00
|
|
|
suite.Logger.On("Debug", ":name: Got response error :err", mock.Anything, mock.Anything).Times(len(expectedErrors))
|
2019-10-03 01:00:27 +03:00
|
|
|
suite.Logger.On("Warning", ":name: Got 400 Bad Request :err", mock.Anything, mock.Anything).Once()
|
2019-11-08 01:32:26 +03:00
|
|
|
suite.Logger.On("Warning", ":name: Got 403 Forbidden :err", mock.Anything, mock.Anything).Once()
|
2019-05-05 23:06:29 +03:00
|
|
|
suite.Logger.On("Warning", ":name: Got 429 Too Many Requests :err", mock.Anything, mock.Anything).Once()
|
2019-04-25 00:45:04 +03:00
|
|
|
|
|
|
|
suite.Storage.On("GetUuid", "maksimkurb").Return("", &ValueNotFound{})
|
2019-05-06 17:12:37 +03:00
|
|
|
suite.Storage.On("StoreUuid", "maksimkurb", "0d252b7218b648bfb86c2ae476954d32").Return(nil)
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Storage.On("GetTextures", "0d252b7218b648bfb86c2ae476954d32").Return(nil, &ValueNotFound{})
|
2019-05-05 23:06:29 +03:00
|
|
|
suite.Storage.On("StoreTextures", "0d252b7218b648bfb86c2ae476954d32", (*mojang.SignedTexturesResponse)(nil))
|
2019-04-25 00:45:04 +03:00
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
for _, err := range expectedErrors {
|
|
|
|
suite.MojangApi.On("UsernamesToUuids", []string{"maksimkurb"}).Once().Return([]*mojang.ProfileInfo{
|
|
|
|
{Id: "0d252b7218b648bfb86c2ae476954d32", Name: "maksimkurb"},
|
|
|
|
}, nil)
|
|
|
|
suite.MojangApi.On("UuidToTextures", "0d252b7218b648bfb86c2ae476954d32", true).Once().Return(nil, err)
|
2019-04-25 00:45:04 +03:00
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
resultChan := suite.Queue.GetTexturesForUsername("maksimkurb")
|
|
|
|
suite.Iterate()
|
|
|
|
suite.Assert().Nil(<-resultChan)
|
|
|
|
suite.MojangApi.AssertExpectations(suite.T())
|
|
|
|
suite.MojangApi.ExpectedCalls = nil // https://github.com/stretchr/testify/issues/558#issuecomment-372112364
|
|
|
|
}
|
2019-04-20 23:04:29 +03:00
|
|
|
}
|
|
|
|
|
2019-04-30 00:36:51 +03:00
|
|
|
func (suite *queueTestSuite) TestShouldLogEmergencyOnUnexpectedErrorReturnedFromUuidToTexturesRequest() {
|
|
|
|
suite.Logger.On("IncCounter", mock.Anything, mock.Anything)
|
|
|
|
suite.Logger.On("UpdateGauge", mock.Anything, mock.Anything)
|
|
|
|
suite.Logger.On("RecordTimer", mock.Anything, mock.Anything)
|
2019-05-05 23:06:29 +03:00
|
|
|
suite.Logger.On("Debug", ":name: Got response error :err", mock.Anything, mock.Anything).Once()
|
|
|
|
suite.Logger.On("Emergency", ":name: Unknown Mojang response error: :err", mock.Anything, mock.Anything).Once()
|
2019-04-30 00:36:51 +03:00
|
|
|
|
|
|
|
suite.Storage.On("GetUuid", "maksimkurb").Return("", &ValueNotFound{})
|
2019-05-06 17:12:37 +03:00
|
|
|
suite.Storage.On("StoreUuid", "maksimkurb", "0d252b7218b648bfb86c2ae476954d32").Return(nil)
|
2019-04-30 00:36:51 +03:00
|
|
|
suite.Storage.On("GetTextures", "0d252b7218b648bfb86c2ae476954d32").Return(nil, &ValueNotFound{})
|
2019-05-05 23:06:29 +03:00
|
|
|
suite.Storage.On("StoreTextures", "0d252b7218b648bfb86c2ae476954d32", (*mojang.SignedTexturesResponse)(nil))
|
2019-04-30 00:36:51 +03:00
|
|
|
|
|
|
|
suite.MojangApi.On("UsernamesToUuids", []string{"maksimkurb"}).Once().Return([]*mojang.ProfileInfo{
|
|
|
|
{Id: "0d252b7218b648bfb86c2ae476954d32", Name: "maksimkurb"},
|
|
|
|
}, nil)
|
|
|
|
suite.MojangApi.On("UuidToTextures", "0d252b7218b648bfb86c2ae476954d32", true).Once().Return(nil, errors.New("unexpected error"))
|
|
|
|
|
|
|
|
resultChan := suite.Queue.GetTexturesForUsername("maksimkurb")
|
|
|
|
suite.Iterate()
|
|
|
|
suite.Assert().Nil(<-resultChan)
|
|
|
|
}
|
|
|
|
|
2019-04-21 03:04:03 +03:00
|
|
|
func (suite *queueTestSuite) TestReceiveTexturesForNotAllowedMojangUsername() {
|
2019-04-25 00:45:04 +03:00
|
|
|
suite.Logger.On("IncCounter", "mojang_textures.invalid_username", int64(1)).Once()
|
|
|
|
|
2019-04-20 19:51:55 +03:00
|
|
|
resultChan := suite.Queue.GetTexturesForUsername("Not allowed")
|
|
|
|
suite.Assert().Nil(<-resultChan)
|
|
|
|
}
|
|
|
|
|
2019-04-19 01:41:52 +03:00
|
|
|
func TestJobsQueueSuite(t *testing.T) {
|
2019-04-21 03:04:03 +03:00
|
|
|
suite.Run(t, new(queueTestSuite))
|
2019-04-18 02:56:20 +03:00
|
|
|
}
|
|
|
|
|
2019-04-20 20:04:57 +03:00
|
|
|
var replacer = strings.NewReplacer("-", "_", "=", "")
|
|
|
|
|
2019-04-18 02:56:20 +03:00
|
|
|
// https://stackoverflow.com/a/50581165
|
|
|
|
func randStr(len int) string {
|
|
|
|
buff := make([]byte, len)
|
2019-04-19 01:41:52 +03:00
|
|
|
_, _ = rand.Read(buff)
|
2019-04-20 20:04:57 +03:00
|
|
|
str := replacer.Replace(base64.URLEncoding.EncodeToString(buff))
|
2019-04-18 02:56:20 +03:00
|
|
|
|
|
|
|
// Base 64 can be longer than len
|
|
|
|
return str[:len]
|
|
|
|
}
|