mirror of
				https://github.com/elyby/chrly.git
				synced 2025-05-31 14:11:51 +05:30 
			
		
		
		
	Implemented worker command
This commit is contained in:
		
							
								
								
									
										89
									
								
								http/uuids_worker.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								http/uuids_worker.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,89 @@
 | 
			
		||||
package http
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"encoding/json"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"net"
 | 
			
		||||
	"net/http"
 | 
			
		||||
	"time"
 | 
			
		||||
 | 
			
		||||
	"github.com/gorilla/mux"
 | 
			
		||||
	"github.com/mono83/slf/wd"
 | 
			
		||||
 | 
			
		||||
	"github.com/elyby/chrly/api/mojang"
 | 
			
		||||
	"github.com/elyby/chrly/mojangtextures"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type UuidsProvider interface {
 | 
			
		||||
	GetUuid(username string) (*mojang.ProfileInfo, error)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type UUIDsWorker struct {
 | 
			
		||||
	ListenSpec string
 | 
			
		||||
 | 
			
		||||
	UUIDsProvider mojangtextures.UUIDsProvider
 | 
			
		||||
	Logger        wd.Watchdog
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (ctx *UUIDsWorker) Run() error {
 | 
			
		||||
	ctx.Logger.Info(fmt.Sprintf("Starting, HTTP on: %s\n", ctx.ListenSpec))
 | 
			
		||||
 | 
			
		||||
	listener, err := net.Listen("tcp", ctx.ListenSpec)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	server := &http.Server{
 | 
			
		||||
		ReadTimeout:    60 * time.Second,
 | 
			
		||||
		WriteTimeout:   60 * time.Second, // TODO: should I adjust this values?
 | 
			
		||||
		MaxHeaderBytes: 1 << 16,
 | 
			
		||||
		Handler:        ctx.CreateHandler(),
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// noinspection GoUnhandledErrorResult
 | 
			
		||||
	go server.Serve(listener)
 | 
			
		||||
 | 
			
		||||
	s := waitForSignal()
 | 
			
		||||
	ctx.Logger.Info(fmt.Sprintf("Got signal: %v, exiting.", s))
 | 
			
		||||
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (ctx *UUIDsWorker) CreateHandler() http.Handler {
 | 
			
		||||
	router := mux.NewRouter().StrictSlash(true)
 | 
			
		||||
	router.NotFoundHandler = http.HandlerFunc(NotFound)
 | 
			
		||||
 | 
			
		||||
	router.Handle("/api/worker/mojang-uuid/{username}", http.HandlerFunc(ctx.GetUUID)).Methods("GET")
 | 
			
		||||
 | 
			
		||||
	return router
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (ctx *UUIDsWorker) GetUUID(response http.ResponseWriter, request *http.Request) {
 | 
			
		||||
	username := parseUsername(mux.Vars(request)["username"])
 | 
			
		||||
	profile, err := ctx.UUIDsProvider.GetUuid(username)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		if _, ok := err.(*mojang.TooManyRequestsError); ok {
 | 
			
		||||
			ctx.Logger.Warning("Got 429 Too Many Requests")
 | 
			
		||||
			response.WriteHeader(http.StatusTooManyRequests)
 | 
			
		||||
			return
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		ctx.Logger.Warning("Got non success response: :err", wd.ErrParam(err))
 | 
			
		||||
		response.Header().Set("Content-Type", "application/json")
 | 
			
		||||
		response.WriteHeader(http.StatusInternalServerError)
 | 
			
		||||
		result, _ := json.Marshal(map[string]interface{}{
 | 
			
		||||
			"provider": err.Error(),
 | 
			
		||||
		})
 | 
			
		||||
		_, _ = response.Write(result)
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if profile == nil {
 | 
			
		||||
		response.WriteHeader(http.StatusNoContent)
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	response.Header().Set("Content-Type", "application/json")
 | 
			
		||||
	responseData, _ := json.Marshal(profile)
 | 
			
		||||
	_, _ = response.Write(responseData)
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										157
									
								
								http/uuids_worker_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										157
									
								
								http/uuids_worker_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,157 @@
 | 
			
		||||
package http
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"errors"
 | 
			
		||||
	"github.com/elyby/chrly/api/mojang"
 | 
			
		||||
	"github.com/elyby/chrly/tests"
 | 
			
		||||
	"github.com/stretchr/testify/mock"
 | 
			
		||||
	"github.com/stretchr/testify/suite"
 | 
			
		||||
	"io/ioutil"
 | 
			
		||||
	"net/http"
 | 
			
		||||
	"net/http/httptest"
 | 
			
		||||
	"testing"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
/***************
 | 
			
		||||
 * Setup mocks *
 | 
			
		||||
 ***************/
 | 
			
		||||
 | 
			
		||||
type uuidsProviderMock struct {
 | 
			
		||||
	mock.Mock
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (m *uuidsProviderMock) GetUuid(username string) (*mojang.ProfileInfo, error) {
 | 
			
		||||
	args := m.Called(username)
 | 
			
		||||
	var result *mojang.ProfileInfo
 | 
			
		||||
	if casted, ok := args.Get(0).(*mojang.ProfileInfo); ok {
 | 
			
		||||
		result = casted
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return result, args.Error(1)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type uuidsWorkerTestSuite struct {
 | 
			
		||||
	suite.Suite
 | 
			
		||||
 | 
			
		||||
	App *UUIDsWorker
 | 
			
		||||
 | 
			
		||||
	UuidsProvider *uuidsProviderMock
 | 
			
		||||
	Logger        *tests.WdMock
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/********************
 | 
			
		||||
 * Setup test suite *
 | 
			
		||||
 ********************/
 | 
			
		||||
 | 
			
		||||
func (suite *uuidsWorkerTestSuite) SetupTest() {
 | 
			
		||||
	suite.UuidsProvider = &uuidsProviderMock{}
 | 
			
		||||
	suite.Logger = &tests.WdMock{}
 | 
			
		||||
 | 
			
		||||
	suite.App = &UUIDsWorker{
 | 
			
		||||
		UUIDsProvider: suite.UuidsProvider,
 | 
			
		||||
		Logger:        suite.Logger,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (suite *uuidsWorkerTestSuite) TearDownTest() {
 | 
			
		||||
	suite.UuidsProvider.AssertExpectations(suite.T())
 | 
			
		||||
	suite.Logger.AssertExpectations(suite.T())
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (suite *uuidsWorkerTestSuite) RunSubTest(name string, subTest func()) {
 | 
			
		||||
	suite.SetupTest()
 | 
			
		||||
	suite.Run(name, subTest)
 | 
			
		||||
	suite.TearDownTest()
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*************
 | 
			
		||||
 * Run tests *
 | 
			
		||||
 *************/
 | 
			
		||||
 | 
			
		||||
func TestUUIDsWorker(t *testing.T) {
 | 
			
		||||
	suite.Run(t, new(uuidsWorkerTestSuite))
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type uuidsWorkerTestCase struct {
 | 
			
		||||
	Name       string
 | 
			
		||||
	BeforeTest func(suite *uuidsWorkerTestSuite)
 | 
			
		||||
	AfterTest  func(suite *uuidsWorkerTestSuite, response *http.Response)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/************************
 | 
			
		||||
 * Get UUID tests cases *
 | 
			
		||||
 ************************/
 | 
			
		||||
 | 
			
		||||
var getUuidTestsCases = []*uuidsWorkerTestCase{
 | 
			
		||||
	{
 | 
			
		||||
		Name: "Success provider response",
 | 
			
		||||
		BeforeTest: func(suite *uuidsWorkerTestSuite) {
 | 
			
		||||
			suite.UuidsProvider.On("GetUuid", "mock_username").Return(&mojang.ProfileInfo{
 | 
			
		||||
				Id:   "0fcc38620f1845f3a54e1b523c1bd1c7",
 | 
			
		||||
				Name: "mock_username",
 | 
			
		||||
			}, nil)
 | 
			
		||||
		},
 | 
			
		||||
		AfterTest: func(suite *uuidsWorkerTestSuite, response *http.Response) {
 | 
			
		||||
			suite.Equal(200, response.StatusCode)
 | 
			
		||||
			suite.Equal("application/json", response.Header.Get("Content-Type"))
 | 
			
		||||
			body, _ := ioutil.ReadAll(response.Body)
 | 
			
		||||
			suite.JSONEq(`{
 | 
			
		||||
				"id": "0fcc38620f1845f3a54e1b523c1bd1c7",
 | 
			
		||||
				"name": "mock_username"
 | 
			
		||||
			}`, string(body))
 | 
			
		||||
		},
 | 
			
		||||
	},
 | 
			
		||||
	{
 | 
			
		||||
		Name: "Receive empty response from UUIDs provider",
 | 
			
		||||
		BeforeTest: func(suite *uuidsWorkerTestSuite) {
 | 
			
		||||
			suite.UuidsProvider.On("GetUuid", "mock_username").Return(nil, nil)
 | 
			
		||||
		},
 | 
			
		||||
		AfterTest: func(suite *uuidsWorkerTestSuite, response *http.Response) {
 | 
			
		||||
			suite.Equal(204, response.StatusCode)
 | 
			
		||||
			body, _ := ioutil.ReadAll(response.Body)
 | 
			
		||||
			suite.Assert().Empty(body)
 | 
			
		||||
		},
 | 
			
		||||
	},
 | 
			
		||||
	{
 | 
			
		||||
		Name: "Receive error from UUIDs provider",
 | 
			
		||||
		BeforeTest: func(suite *uuidsWorkerTestSuite) {
 | 
			
		||||
			suite.UuidsProvider.On("GetUuid", "mock_username").Return(nil, errors.New("this is an error"))
 | 
			
		||||
			suite.Logger.On("Warning", "Got non success response: :err", mock.Anything).Times(1)
 | 
			
		||||
		},
 | 
			
		||||
		AfterTest: func(suite *uuidsWorkerTestSuite, response *http.Response) {
 | 
			
		||||
			suite.Equal(500, response.StatusCode)
 | 
			
		||||
			suite.Equal("application/json", response.Header.Get("Content-Type"))
 | 
			
		||||
			body, _ := ioutil.ReadAll(response.Body)
 | 
			
		||||
			suite.JSONEq(`{
 | 
			
		||||
				"provider": "this is an error"
 | 
			
		||||
			}`, string(body))
 | 
			
		||||
		},
 | 
			
		||||
	},
 | 
			
		||||
	{
 | 
			
		||||
		Name: "Receive Too Many Requests from UUIDs provider",
 | 
			
		||||
		BeforeTest: func(suite *uuidsWorkerTestSuite) {
 | 
			
		||||
			suite.UuidsProvider.On("GetUuid", "mock_username").Return(nil, &mojang.TooManyRequestsError{})
 | 
			
		||||
			suite.Logger.On("Warning", "Got 429 Too Many Requests").Times(1)
 | 
			
		||||
		},
 | 
			
		||||
		AfterTest: func(suite *uuidsWorkerTestSuite, response *http.Response) {
 | 
			
		||||
			suite.Equal(429, response.StatusCode)
 | 
			
		||||
			body, _ := ioutil.ReadAll(response.Body)
 | 
			
		||||
			suite.Empty(body)
 | 
			
		||||
		},
 | 
			
		||||
	},
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (suite *uuidsWorkerTestSuite) TestGetUUID() {
 | 
			
		||||
	for _, testCase := range getUuidTestsCases {
 | 
			
		||||
		suite.RunSubTest(testCase.Name, func() {
 | 
			
		||||
			testCase.BeforeTest(suite)
 | 
			
		||||
 | 
			
		||||
			req := httptest.NewRequest("GET", "http://chrly/api/worker/mojang-uuid/mock_username", nil)
 | 
			
		||||
			w := httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
			suite.App.CreateHandler().ServeHTTP(w, req)
 | 
			
		||||
 | 
			
		||||
			testCase.AfterTest(suite, w.Result())
 | 
			
		||||
		})
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
		Reference in New Issue
	
	Block a user