|
| 1 | +package handlers |
| 2 | + |
| 3 | +import ( |
| 4 | + "fmt" |
| 5 | + |
| 6 | + "github.com/NdoleStudio/http-sms-manager/pkg/requests" |
| 7 | + "github.com/NdoleStudio/http-sms-manager/pkg/validators" |
| 8 | + "github.com/davecgh/go-spew/spew" |
| 9 | + |
| 10 | + "github.com/NdoleStudio/http-sms-manager/pkg/services" |
| 11 | + "github.com/NdoleStudio/http-sms-manager/pkg/telemetry" |
| 12 | + "github.com/gofiber/fiber/v2" |
| 13 | + "github.com/palantir/stacktrace" |
| 14 | +) |
| 15 | + |
| 16 | +// PhoneHandler handles phone http requests. |
| 17 | +type PhoneHandler struct { |
| 18 | + handler |
| 19 | + logger telemetry.Logger |
| 20 | + tracer telemetry.Tracer |
| 21 | + service *services.PhoneService |
| 22 | + validator *validators.PhoneHandlerValidator |
| 23 | +} |
| 24 | + |
| 25 | +// NewPhoneHandler creates a new PhoneHandler |
| 26 | +func NewPhoneHandler( |
| 27 | + logger telemetry.Logger, |
| 28 | + tracer telemetry.Tracer, |
| 29 | + service *services.PhoneService, |
| 30 | + validator *validators.PhoneHandlerValidator, |
| 31 | +) (h *PhoneHandler) { |
| 32 | + return &PhoneHandler{ |
| 33 | + logger: logger.WithService(fmt.Sprintf("%T", h)), |
| 34 | + tracer: tracer, |
| 35 | + validator: validator, |
| 36 | + service: service, |
| 37 | + } |
| 38 | +} |
| 39 | + |
| 40 | +// RegisterRoutes registers the routes for the PhoneHandler |
| 41 | +func (h *PhoneHandler) RegisterRoutes(router fiber.Router) { |
| 42 | + router.Get("/phones", h.Index) |
| 43 | + router.Put("/phones/:phoneID", h.Upsert) |
| 44 | +} |
| 45 | + |
| 46 | +// Index returns the phones of a user |
| 47 | +// @Summary Get phones of a user |
| 48 | +// @Description Get list of phones which a user has registered on the http sms application |
| 49 | +// @Security ApiKeyAuth |
| 50 | +// @Tags Phones |
| 51 | +// @Accept json |
| 52 | +// @Produce json |
| 53 | +// @Param skip query int false "number of heartbeats to skip" minimum(0) |
| 54 | +// @Param query query string false "filter phones containing query" |
| 55 | +// @Param limit query int false "number of phones to return" minimum(1) maximum(20) |
| 56 | +// @Success 200 {object} responses.PhonesResponse |
| 57 | +// @Failure 400 {object} responses.BadRequest |
| 58 | +// @Failure 403 {object} responses.Unauthorized |
| 59 | +// @Failure 422 {object} responses.UnprocessableEntity |
| 60 | +// @Failure 500 {object} responses.InternalServerError |
| 61 | +// @Router /phones [get] |
| 62 | +func (h *PhoneHandler) Index(c *fiber.Ctx) error { |
| 63 | + ctx, span := h.tracer.StartFromFiberCtx(c) |
| 64 | + defer span.End() |
| 65 | + |
| 66 | + ctxLogger := h.tracer.CtxLogger(h.logger, span) |
| 67 | + |
| 68 | + var request requests.PhoneIndex |
| 69 | + if err := c.QueryParser(&request); err != nil { |
| 70 | + msg := fmt.Sprintf("cannot marshall params [%s] into %T", c.OriginalURL(), request) |
| 71 | + ctxLogger.Warn(stacktrace.Propagate(err, msg)) |
| 72 | + return h.responseBadRequest(c, err) |
| 73 | + } |
| 74 | + |
| 75 | + if errors := h.validator.ValidateIndex(ctx, request.Sanitize()); len(errors) != 0 { |
| 76 | + msg := fmt.Sprintf("validation errors [%s], while fetching phones [%+#v]", spew.Sdump(errors), request) |
| 77 | + ctxLogger.Warn(stacktrace.NewError(msg)) |
| 78 | + return h.responseUnprocessableEntity(c, errors, "validation errors while fetching phones") |
| 79 | + } |
| 80 | + |
| 81 | + phones, err := h.service.Index(ctx, h.userFromContext(c), request.ToIndexParams()) |
| 82 | + if err != nil { |
| 83 | + msg := fmt.Sprintf("cannot index phones with params [%+#v]", request) |
| 84 | + ctxLogger.Error(stacktrace.Propagate(err, msg)) |
| 85 | + return h.responseInternalServerError(c) |
| 86 | + } |
| 87 | + |
| 88 | + return h.responseOK(c, fmt.Sprintf("fetched %d %s", len(*phones), h.pluralize("phone", len(*phones))), phones) |
| 89 | +} |
| 90 | + |
| 91 | +// Upsert a phone |
| 92 | +// @Summary Upsert Phone |
| 93 | +// @Description Updates properties of a user's phone. If the phone with this number does not exist, a new one will be created. Think of this method like an 'upsert' |
| 94 | +// @Security ApiKeyAuth |
| 95 | +// @Tags Phones |
| 96 | +// @Accept json |
| 97 | +// @Produce json |
| 98 | +// @Param payload body requests.PhoneUpsert true "Payload of new phone number." |
| 99 | +// @Success 200 {object} responses.PhoneResponse |
| 100 | +// @Failure 400 {object} responses.BadRequest |
| 101 | +// @Failure 403 {object} responses.Unauthorized |
| 102 | +// @Failure 422 {object} responses.UnprocessableEntity |
| 103 | +// @Failure 500 {object} responses.InternalServerError |
| 104 | +// @Router /phones [put] |
| 105 | +func (h *PhoneHandler) Upsert(c *fiber.Ctx) error { |
| 106 | + ctx, span := h.tracer.StartFromFiberCtx(c) |
| 107 | + defer span.End() |
| 108 | + |
| 109 | + ctxLogger := h.tracer.CtxLogger(h.logger, span) |
| 110 | + |
| 111 | + var request requests.PhoneUpsert |
| 112 | + if err := c.QueryParser(&request); err != nil { |
| 113 | + msg := fmt.Sprintf("cannot marshall params [%s] into %T", c.OriginalURL(), request) |
| 114 | + ctxLogger.Warn(stacktrace.Propagate(err, msg)) |
| 115 | + return h.responseBadRequest(c, err) |
| 116 | + } |
| 117 | + |
| 118 | + if errors := h.validator.ValidateUpsert(ctx, request.Sanitize()); len(errors) != 0 { |
| 119 | + msg := fmt.Sprintf("validation errors [%s], while fetching phones [%+#v]", spew.Sdump(errors), request) |
| 120 | + ctxLogger.Warn(stacktrace.NewError(msg)) |
| 121 | + return h.responseUnprocessableEntity(c, errors, "validation errors while fetching phones") |
| 122 | + } |
| 123 | + |
| 124 | + phone, err := h.service.Upsert(ctx, request.ToUpsertParams(h.userFromContext(c))) |
| 125 | + if err != nil { |
| 126 | + msg := fmt.Sprintf("cannot update phones with params [%+#v]", request) |
| 127 | + ctxLogger.Error(stacktrace.Propagate(err, msg)) |
| 128 | + return h.responseInternalServerError(c) |
| 129 | + } |
| 130 | + |
| 131 | + return h.responseOK(c, "phone updated successfully", phone) |
| 132 | +} |
0 commit comments