package controllers import ( "log" "abdanhafidz.com/go-boilerplate/models/dto" http_error "abdanhafidz.com/go-boilerplate/models/error" "abdanhafidz.com/go-boilerplate/services" "abdanhafidz.com/go-boilerplate/utils" "github.com/gin-gonic/gin" ) type PaymentCallbackController interface { HandleCallback(ctx *gin.Context) } type paymentCallbackController struct { paymentService services.PaymentService } func NewPaymentCallbackController( paymentService services.PaymentService, ) PaymentCallbackController { return &paymentCallbackController{ paymentService: paymentService, } } // Handle Payment Callback godoc // @Summary Handle Xendit Payment Callback // @Description Receive and process payment status updates from Xendit // @Tags Payment // @Accept json // @Produce json // @Param request body map[string]interface{} true "Xendit Callback Payload" // @Success 200 {object} dto.SuccessResponse[any] // @Failure 400 {object} dto.ErrorResponse // @Router /api/v1/payment/callback [post] func (c *paymentCallbackController) HandleCallback(ctx *gin.Context) { // Xendit sends JSON payload // Basic structure for Invoice Callback: // { "id": "...", "external_id": "...", "status": "PAID", ... } var callbackData map[string]interface{} if err := ctx.ShouldBindJSON(&callbackData); err != nil { utils.ResponseFAILED(ctx, gin.H(nil), http_error.BAD_REQUEST_ERROR) return } log.Printf("Payment Callback Received: %+v", callbackData) status, ok := callbackData["status"].(string) if !ok { // Not a status update or unknown format var _ dto.SuccessResponse[any] ResponseJSON(ctx, gin.H(nil), "Ignored: No status", nil) return } invoiceId, _ := callbackData["id"].(string) if status == "PAID" || status == "SETTLED" { // Handle Event Payment // We need a method in Service to handle "ConfirmPayment" by InvoiceID // But existing services don't have it. // Let's add it to PaymentService? Yes. err := c.paymentService.ConfirmPayment(ctx.Request.Context(), invoiceId) if err != nil { log.Printf("Payment Confirmation Failed: %v", err) // Don't return error to Xendit if logic failed, but maybe we should? // Xendit expects 200 OK. } } else if status == "EXPIRED" { c.paymentService.ExpirePayment(ctx.Request.Context(), invoiceId) } // Always return 200 to Xendit ResponseJSON(ctx, gin.H(nil), gin.H{"callback": "Callback Received"}, nil) }