Files
galaxy-game/backend/internal/server/handlers_user_account.go
T
2026-05-06 10:14:55 +03:00

155 lines
5.1 KiB
Go

package server
import (
"errors"
"net/http"
"galaxy/backend/internal/server/handlers"
"galaxy/backend/internal/server/httperr"
"galaxy/backend/internal/server/middleware/userid"
"galaxy/backend/internal/telemetry"
"galaxy/backend/internal/user"
"github.com/gin-gonic/gin"
"go.uber.org/zap"
)
// UserAccountHandlers groups the handlers under `/api/v1/user/account/*`.
// The current implementation ships real implementations backed by `*user.Service`; tests
// that supply a nil service fall back to the Stage-3 placeholder body
// so the contract test continues to validate the OpenAPI envelope
// without booting a database.
type UserAccountHandlers struct {
svc *user.Service
logger *zap.Logger
}
// NewUserAccountHandlers constructs the handler set. svc may be nil —
// in that case every handler returns 501 not_implemented, matching the
// pre-Stage-5.2 placeholder. logger may also be nil; zap.NewNop is
// used in that case.
func NewUserAccountHandlers(svc *user.Service, logger *zap.Logger) *UserAccountHandlers {
if logger == nil {
logger = zap.NewNop()
}
return &UserAccountHandlers{svc: svc, logger: logger.Named("http.user.account")}
}
// Get handles GET /api/v1/user/account.
func (h *UserAccountHandlers) Get() gin.HandlerFunc {
if h.svc == nil {
return handlers.NotImplemented("userAccountGet")
}
return func(c *gin.Context) {
userID, ok := userid.FromContext(c.Request.Context())
if !ok {
httperr.Abort(c, http.StatusBadRequest, httperr.CodeInvalidRequest, "X-User-ID header is required")
return
}
ctx := c.Request.Context()
account, err := h.svc.GetAccount(ctx, userID)
if err != nil {
respondAccountError(c, h.logger, "user account get", ctx, err)
return
}
c.JSON(http.StatusOK, accountResponseToWire(account))
}
}
// UpdateProfile handles PATCH /api/v1/user/account/profile.
func (h *UserAccountHandlers) UpdateProfile() gin.HandlerFunc {
if h.svc == nil {
return handlers.NotImplemented("userAccountUpdateProfile")
}
return func(c *gin.Context) {
userID, ok := userid.FromContext(c.Request.Context())
if !ok {
httperr.Abort(c, http.StatusBadRequest, httperr.CodeInvalidRequest, "X-User-ID header is required")
return
}
var req updateProfileRequestWire
if err := c.ShouldBindJSON(&req); err != nil {
httperr.Abort(c, http.StatusBadRequest, httperr.CodeInvalidRequest, "request body must be valid JSON")
return
}
ctx := c.Request.Context()
account, err := h.svc.UpdateProfile(ctx, userID, user.UpdateProfileInput{
DisplayName: req.DisplayName,
})
if err != nil {
respondAccountError(c, h.logger, "user account update profile", ctx, err)
return
}
c.JSON(http.StatusOK, accountResponseToWire(account))
}
}
// UpdateSettings handles PATCH /api/v1/user/account/settings.
func (h *UserAccountHandlers) UpdateSettings() gin.HandlerFunc {
if h.svc == nil {
return handlers.NotImplemented("userAccountUpdateSettings")
}
return func(c *gin.Context) {
userID, ok := userid.FromContext(c.Request.Context())
if !ok {
httperr.Abort(c, http.StatusBadRequest, httperr.CodeInvalidRequest, "X-User-ID header is required")
return
}
var req updateSettingsRequestWire
if err := c.ShouldBindJSON(&req); err != nil {
httperr.Abort(c, http.StatusBadRequest, httperr.CodeInvalidRequest, "request body must be valid JSON")
return
}
ctx := c.Request.Context()
account, err := h.svc.UpdateSettings(ctx, userID, user.UpdateSettingsInput{
PreferredLanguage: req.PreferredLanguage,
TimeZone: req.TimeZone,
})
if err != nil {
respondAccountError(c, h.logger, "user account update settings", ctx, err)
return
}
c.JSON(http.StatusOK, accountResponseToWire(account))
}
}
// Delete handles POST /api/v1/user/account/delete.
func (h *UserAccountHandlers) Delete() gin.HandlerFunc {
if h.svc == nil {
return handlers.NotImplemented("userAccountDelete")
}
return func(c *gin.Context) {
userID, ok := userid.FromContext(c.Request.Context())
if !ok {
httperr.Abort(c, http.StatusBadRequest, httperr.CodeInvalidRequest, "X-User-ID header is required")
return
}
ctx := c.Request.Context()
actor := user.ActorRef{Type: "user", ID: userID.String()}
if err := h.svc.SoftDelete(ctx, userID, actor); err != nil {
if errors.Is(err, user.ErrAccountNotFound) {
httperr.Abort(c, http.StatusNotFound, httperr.CodeNotFound, "account not found")
return
}
h.logger.Warn("user account soft-delete returned cascade errors",
append(telemetry.TraceFieldsFromContext(ctx), zap.Error(err))...,
)
// Cascade errors do not change the canonical state — the
// account is soft-deleted in Postgres. Surface 204 so the
// caller's UI proceeds to a logged-out state.
}
c.Status(http.StatusNoContent)
}
}
// updateProfileRequestWire mirrors `UpdateProfileRequest` from openapi.yaml.
type updateProfileRequestWire struct {
DisplayName *string `json:"display_name,omitempty"`
}
// updateSettingsRequestWire mirrors `UpdateSettingsRequest` from openapi.yaml.
type updateSettingsRequestWire struct {
PreferredLanguage *string `json:"preferred_language,omitempty"`
TimeZone *string `json:"time_zone,omitempty"`
}