2020-04-19 02:31:09 +03:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"regexp"
|
|
|
|
"strconv"
|
|
|
|
|
|
|
|
"github.com/gorilla/mux"
|
|
|
|
"github.com/thedevsaddam/govalidator"
|
|
|
|
|
|
|
|
"github.com/elyby/chrly/model"
|
|
|
|
)
|
|
|
|
|
2023-12-12 01:35:08 +01:00
|
|
|
// noinspection GoSnakeCaseUsage
|
2020-04-19 02:31:09 +03:00
|
|
|
const UUID_ANY = "^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$"
|
|
|
|
|
|
|
|
var regexUuidAny = regexp.MustCompile(UUID_ANY)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
// Add ability to validate any possible uuid form
|
|
|
|
govalidator.AddCustomRule("uuid_any", func(field string, rule string, message string, value interface{}) error {
|
|
|
|
str := value.(string)
|
|
|
|
if !regexUuidAny.MatchString(str) {
|
|
|
|
if message == "" {
|
|
|
|
message = fmt.Sprintf("The %s field must contain valid UUID", field)
|
|
|
|
}
|
|
|
|
|
|
|
|
return errors.New(message)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
type Api struct {
|
|
|
|
SkinsRepo SkinsRepository
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *Api) Handler() *mux.Router {
|
|
|
|
router := mux.NewRouter().StrictSlash(true)
|
|
|
|
router.HandleFunc("/skins", ctx.postSkinHandler).Methods(http.MethodPost)
|
|
|
|
router.HandleFunc("/skins/id:{id:[0-9]+}", ctx.deleteSkinByUserIdHandler).Methods(http.MethodDelete)
|
|
|
|
router.HandleFunc("/skins/{username}", ctx.deleteSkinByUsernameHandler).Methods(http.MethodDelete)
|
|
|
|
|
|
|
|
return router
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *Api) postSkinHandler(resp http.ResponseWriter, req *http.Request) {
|
|
|
|
validationErrors := validatePostSkinRequest(req)
|
|
|
|
if validationErrors != nil {
|
|
|
|
apiBadRequest(resp, validationErrors)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
identityId, _ := strconv.Atoi(req.Form.Get("identityId"))
|
|
|
|
username := req.Form.Get("username")
|
|
|
|
|
|
|
|
record, err := ctx.findIdentityOrCleanup(identityId, username)
|
|
|
|
if err != nil {
|
2021-02-26 02:45:45 +01:00
|
|
|
panic(err)
|
2020-04-19 02:31:09 +03:00
|
|
|
}
|
|
|
|
|
2020-04-20 15:16:15 +03:00
|
|
|
if record == nil {
|
|
|
|
record = &model.Skin{
|
|
|
|
UserId: identityId,
|
|
|
|
Username: username,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-19 02:31:09 +03:00
|
|
|
skinId, _ := strconv.Atoi(req.Form.Get("skinId"))
|
|
|
|
is18, _ := strconv.ParseBool(req.Form.Get("is1_8"))
|
|
|
|
isSlim, _ := strconv.ParseBool(req.Form.Get("isSlim"))
|
|
|
|
|
|
|
|
record.Uuid = req.Form.Get("uuid")
|
|
|
|
record.SkinId = skinId
|
|
|
|
record.Is1_8 = is18
|
|
|
|
record.IsSlim = isSlim
|
|
|
|
record.Url = req.Form.Get("url")
|
|
|
|
record.MojangTextures = req.Form.Get("mojangTextures")
|
|
|
|
record.MojangSignature = req.Form.Get("mojangSignature")
|
|
|
|
|
2020-04-20 22:18:27 +03:00
|
|
|
err = ctx.SkinsRepo.SaveSkin(record)
|
2020-04-19 02:31:09 +03:00
|
|
|
if err != nil {
|
2021-02-26 02:45:45 +01:00
|
|
|
panic(err)
|
2020-04-19 02:31:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
resp.WriteHeader(http.StatusCreated)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *Api) deleteSkinByUserIdHandler(resp http.ResponseWriter, req *http.Request) {
|
|
|
|
id, _ := strconv.Atoi(mux.Vars(req)["id"])
|
2020-04-20 22:18:27 +03:00
|
|
|
skin, err := ctx.SkinsRepo.FindSkinByUserId(id)
|
2020-04-19 02:31:09 +03:00
|
|
|
ctx.deleteSkin(skin, err, resp)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *Api) deleteSkinByUsernameHandler(resp http.ResponseWriter, req *http.Request) {
|
|
|
|
username := mux.Vars(req)["username"]
|
2020-04-20 22:18:27 +03:00
|
|
|
skin, err := ctx.SkinsRepo.FindSkinByUsername(username)
|
2020-04-19 02:31:09 +03:00
|
|
|
ctx.deleteSkin(skin, err, resp)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *Api) deleteSkin(skin *model.Skin, err error, resp http.ResponseWriter) {
|
|
|
|
if err != nil {
|
2021-02-26 02:45:45 +01:00
|
|
|
panic(err)
|
2020-04-20 15:16:15 +03:00
|
|
|
}
|
2020-04-19 02:31:09 +03:00
|
|
|
|
2020-04-20 15:16:15 +03:00
|
|
|
if skin == nil {
|
|
|
|
apiNotFound(resp, "Cannot find record for the requested identifier")
|
2020-04-19 02:31:09 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-04-20 22:18:27 +03:00
|
|
|
err = ctx.SkinsRepo.RemoveSkinByUserId(skin.UserId)
|
2020-04-19 02:31:09 +03:00
|
|
|
if err != nil {
|
2021-02-26 02:45:45 +01:00
|
|
|
panic(err)
|
2020-04-19 02:31:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
resp.WriteHeader(http.StatusNoContent)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *Api) findIdentityOrCleanup(identityId int, username string) (*model.Skin, error) {
|
2020-04-20 22:18:27 +03:00
|
|
|
record, err := ctx.SkinsRepo.FindSkinByUserId(identityId)
|
2020-04-19 02:31:09 +03:00
|
|
|
if err != nil {
|
2020-04-20 15:16:15 +03:00
|
|
|
return nil, err
|
|
|
|
}
|
2020-04-19 02:31:09 +03:00
|
|
|
|
2020-04-20 15:16:15 +03:00
|
|
|
if record != nil {
|
|
|
|
// The username may have changed in the external database,
|
|
|
|
// so we need to remove the old association
|
|
|
|
if record.Username != username {
|
2020-04-20 22:18:27 +03:00
|
|
|
_ = ctx.SkinsRepo.RemoveSkinByUserId(identityId)
|
2020-04-20 15:16:15 +03:00
|
|
|
record.Username = username
|
2020-04-19 02:31:09 +03:00
|
|
|
}
|
2020-04-20 15:16:15 +03:00
|
|
|
|
|
|
|
return record, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the requested id was not found, then username was reassigned to another user
|
|
|
|
// who has not uploaded his data to Chrly yet
|
2020-04-20 22:18:27 +03:00
|
|
|
record, err = ctx.SkinsRepo.FindSkinByUsername(username)
|
2020-04-20 15:16:15 +03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the target username does exist, clear it as it will be reassigned to the new user
|
|
|
|
if record != nil {
|
2020-04-20 22:18:27 +03:00
|
|
|
_ = ctx.SkinsRepo.RemoveSkinByUsername(username)
|
2020-04-20 15:16:15 +03:00
|
|
|
record.UserId = identityId
|
|
|
|
|
|
|
|
return record, nil
|
2020-04-19 02:31:09 +03:00
|
|
|
}
|
|
|
|
|
2020-04-20 15:16:15 +03:00
|
|
|
return nil, nil
|
2020-04-19 02:31:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func validatePostSkinRequest(request *http.Request) map[string][]string {
|
2023-12-22 01:56:02 +01:00
|
|
|
_ = request.ParseForm()
|
2020-04-19 02:31:09 +03:00
|
|
|
|
|
|
|
validationRules := govalidator.MapData{
|
2023-12-22 01:56:02 +01:00
|
|
|
"identityId": {"required", "numeric", "min:1"},
|
|
|
|
"username": {"required"},
|
|
|
|
"uuid": {"required", "uuid_any"},
|
|
|
|
"skinId": {"required", "numeric"},
|
|
|
|
"url": {},
|
|
|
|
"is1_8": {"bool"},
|
|
|
|
"isSlim": {"bool"},
|
|
|
|
"mojangTextures": {},
|
|
|
|
"mojangSignature": {},
|
2020-04-19 02:31:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
url := request.Form.Get("url")
|
2023-12-22 01:56:02 +01:00
|
|
|
if url == "" {
|
|
|
|
validationRules["skinId"] = append(validationRules["skinId"], "numeric_between:0,0")
|
|
|
|
} else {
|
|
|
|
validationRules["url"] = append(validationRules["url"], "url")
|
|
|
|
validationRules["skinId"] = append(validationRules["skinId"], "numeric_between:1,")
|
2020-04-19 02:31:09 +03:00
|
|
|
validationRules["is1_8"] = append(validationRules["is1_8"], "required")
|
|
|
|
validationRules["isSlim"] = append(validationRules["isSlim"], "required")
|
|
|
|
}
|
|
|
|
|
|
|
|
mojangTextures := request.Form.Get("mojangTextures")
|
|
|
|
if mojangTextures != "" {
|
2023-12-22 01:56:02 +01:00
|
|
|
validationRules["mojangSignature"] = append(validationRules["mojangSignature"], "required")
|
2020-04-19 02:31:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
validator := govalidator.New(govalidator.Options{
|
|
|
|
Request: request,
|
|
|
|
Rules: validationRules,
|
|
|
|
RequiredDefault: false,
|
|
|
|
})
|
|
|
|
validationResults := validator.Validate()
|
|
|
|
|
|
|
|
if len(validationResults) != 0 {
|
|
|
|
return validationResults
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|