// monorepo/cloud/backend/internal/maplefile/interface/http/file/update.go package file import ( "bytes" "context" "encoding/json" "errors" "io" "net/http" "go.uber.org/zap" "github.com/gocql/gocql" "codeberg.org/mapleopentech/monorepo/cloud/maplefile-backend/config" "codeberg.org/mapleopentech/monorepo/cloud/maplefile-backend/internal/interface/http/middleware" svc_file "codeberg.org/mapleopentech/monorepo/cloud/maplefile-backend/internal/service/file" "codeberg.org/mapleopentech/monorepo/cloud/maplefile-backend/pkg/httperror" ) type UpdateFileHTTPHandler struct { config *config.Configuration logger *zap.Logger service svc_file.UpdateFileService middleware middleware.Middleware } func NewUpdateFileHTTPHandler( config *config.Configuration, logger *zap.Logger, service svc_file.UpdateFileService, middleware middleware.Middleware, ) *UpdateFileHTTPHandler { logger = logger.Named("UpdateFileHTTPHandler") return &UpdateFileHTTPHandler{ config: config, logger: logger, service: service, middleware: middleware, } } func (*UpdateFileHTTPHandler) Pattern() string { return "PUT /api/v1/file/{id}" } func (h *UpdateFileHTTPHandler) ServeHTTP(w http.ResponseWriter, req *http.Request) { // Apply middleware before handling the request h.middleware.Attach(h.Execute)(w, req) } func (h *UpdateFileHTTPHandler) unmarshalRequest( ctx context.Context, r *http.Request, fileID gocql.UUID, ) (*svc_file.UpdateFileRequestDTO, error) { // Initialize our structure which will store the parsed request data var requestData svc_file.UpdateFileRequestDTO defer r.Body.Close() var rawJSON bytes.Buffer teeReader := io.TeeReader(r.Body, &rawJSON) // TeeReader allows you to read the JSON and capture it // Read the JSON string and convert it into our golang struct err := json.NewDecoder(teeReader).Decode(&requestData) if err != nil { h.logger.Error("decoding error", zap.Any("err", err)) // Log raw JSON at debug level only to avoid PII exposure in production logs h.logger.Debug("raw request body for debugging", zap.String("json", rawJSON.String())) return nil, httperror.NewForSingleField(http.StatusBadRequest, "non_field_error", "payload structure is wrong") } // Set the file ID from the URL parameter requestData.ID = fileID return &requestData, nil } func (h *UpdateFileHTTPHandler) Execute(w http.ResponseWriter, r *http.Request) { // Set response content type w.Header().Set("Content-Type", "application/json") ctx := r.Context() // Extract file ID from the URL path parameter fileIDStr := r.PathValue("id") if fileIDStr == "" { h.logger.Warn("file_id not found in path parameters or is empty", zap.String("path", r.URL.Path), zap.String("method", r.Method), ) httperror.RespondWithError(w, r, httperror.NewForBadRequestWithSingleField("file_id", "File ID is required")) return } // Convert string ID to ObjectID fileID, err := gocql.ParseUUID(fileIDStr) if err != nil { h.logger.Error("invalid file ID format", zap.String("file_id", fileIDStr), zap.Error(err)) httperror.RespondWithError(w, r, httperror.NewForBadRequestWithSingleField("file_id", "Invalid file ID format")) return } req, err := h.unmarshalRequest(ctx, r, fileID) if err != nil { httperror.RespondWithError(w, r, err) return } resp, err := h.service.Execute(ctx, req) if err != nil { httperror.RespondWithError(w, r, err) return } // Encode response if resp != nil { if err := json.NewEncoder(w).Encode(resp); err != nil { h.logger.Error("failed to encode response", zap.Any("error", err)) httperror.RespondWithError(w, r, err) return } } else { err := errors.New("transaction completed with no result") h.logger.Error("transaction completed with no result", zap.Any("request_payload", req)) httperror.RespondWithError(w, r, err) return } }