2024-04-14 20:21:45 +07:00
|
|
|
package routes
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
2024-04-24 21:57:13 +07:00
|
|
|
"regexp"
|
2024-04-14 20:21:45 +07:00
|
|
|
|
2024-04-24 21:57:13 +07:00
|
|
|
"github.com/tigorlazuardi/redmage/models"
|
|
|
|
"github.com/tigorlazuardi/redmage/pkg/errs"
|
2024-04-14 20:21:45 +07:00
|
|
|
"github.com/tigorlazuardi/redmage/pkg/log"
|
|
|
|
"github.com/tigorlazuardi/redmage/pkg/telemetry"
|
|
|
|
)
|
|
|
|
|
|
|
|
func (routes *Routes) APIDeviceCreate(rw http.ResponseWriter, r *http.Request) {
|
|
|
|
var err error
|
|
|
|
ctx, span := tracer.Start(r.Context(), "*Routes.APIDeviceCreate")
|
|
|
|
defer func() { telemetry.EndWithStatus(span, err) }()
|
|
|
|
|
2024-04-29 21:45:18 +07:00
|
|
|
var body *models.Device
|
2024-04-14 20:21:45 +07:00
|
|
|
|
|
|
|
if err = json.NewDecoder(r.Body).Decode(&body); err != nil {
|
|
|
|
log.New(ctx).Err(err).Error("failed to decode json body")
|
|
|
|
rw.WriteHeader(http.StatusBadRequest)
|
|
|
|
_ = json.NewEncoder(rw).Encode(map[string]string{"error": fmt.Sprintf("cannot decode json body: %s", err)})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-04-24 21:57:13 +07:00
|
|
|
if err = validateCreateDevice(body); err != nil {
|
2024-04-14 20:21:45 +07:00
|
|
|
rw.WriteHeader(http.StatusBadRequest)
|
|
|
|
_ = json.NewEncoder(rw).Encode(map[string]string{"error": err.Error()})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-04-29 21:45:18 +07:00
|
|
|
device, err := routes.API.DevicesCreate(ctx, body)
|
2024-04-14 20:21:45 +07:00
|
|
|
if err != nil {
|
2024-04-25 20:22:05 +07:00
|
|
|
log.New(ctx).Err(err).Error("failed to create device", "body", body)
|
2024-04-24 22:26:04 +07:00
|
|
|
code, message := errs.HTTPMessage(err)
|
|
|
|
rw.WriteHeader(code)
|
|
|
|
_ = json.NewEncoder(rw).Encode(map[string]string{"error": message})
|
2024-04-14 20:21:45 +07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-04-24 21:57:13 +07:00
|
|
|
rw.WriteHeader(http.StatusCreated)
|
2024-04-14 20:21:45 +07:00
|
|
|
if err := json.NewEncoder(rw).Encode(device); err != nil {
|
|
|
|
log.New(ctx).Err(err).Error("failed to marshal json api devices")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-24 21:57:13 +07:00
|
|
|
var slugRegex = regexp.MustCompile(`^[a-z0-9-]+$`)
|
|
|
|
|
2024-04-29 21:45:18 +07:00
|
|
|
func validateCreateDevice(params *models.Device) error {
|
2024-04-14 20:21:45 +07:00
|
|
|
if params.Name == "" {
|
|
|
|
return errors.New("name is required")
|
|
|
|
}
|
|
|
|
if params.Slug == "" {
|
|
|
|
return errors.New("slug is required")
|
|
|
|
}
|
2024-04-24 21:57:13 +07:00
|
|
|
if !slugRegex.MatchString(params.Slug) {
|
|
|
|
return errors.New("slug must be lowercase alphanumeric with dash. eg: my-awesome-laptop")
|
|
|
|
}
|
2024-04-14 20:21:45 +07:00
|
|
|
if params.ResolutionX < 1 {
|
|
|
|
return errors.New("device width resolution is required")
|
|
|
|
}
|
|
|
|
if params.ResolutionY < 1 {
|
|
|
|
return errors.New("device height resolution is required")
|
|
|
|
}
|
2024-04-24 21:57:13 +07:00
|
|
|
if params.AspectRatioTolerance < 0 {
|
|
|
|
return errors.New("aspect ratio tolerance cannot be negative value")
|
|
|
|
}
|
2024-04-14 20:21:45 +07:00
|
|
|
if params.MaxX < 0 {
|
|
|
|
params.MaxX = 0
|
|
|
|
}
|
|
|
|
if params.MaxY < 0 {
|
|
|
|
params.MaxY = 0
|
|
|
|
}
|
|
|
|
if params.MinX < 0 {
|
|
|
|
params.MinX = 0
|
|
|
|
}
|
|
|
|
if params.MinY < 0 {
|
|
|
|
params.MinY = 0
|
|
|
|
}
|
2024-04-24 21:57:13 +07:00
|
|
|
if params.NSFW < 0 {
|
|
|
|
params.NSFW = 0
|
|
|
|
}
|
|
|
|
if params.NSFW > 1 {
|
|
|
|
params.NSFW = 1
|
|
|
|
}
|
|
|
|
if params.WindowsWallpaperMode < 0 {
|
|
|
|
params.WindowsWallpaperMode = 0
|
|
|
|
}
|
|
|
|
if params.WindowsWallpaperMode > 1 {
|
|
|
|
params.WindowsWallpaperMode = 1
|
|
|
|
}
|
2024-04-14 20:21:45 +07:00
|
|
|
return nil
|
|
|
|
}
|