WebhookService.php 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. <?php
  2. namespace App\Services;
  3. use App\Models\Payment;
  4. use App\Models\Webhook;
  5. use App\Services\Pagarme\PagarmePaymentService;
  6. use Illuminate\Support\Facades\Log;
  7. class WebhookService
  8. {
  9. private const PAYMENT_EVENTS = [
  10. 'order.paid',
  11. 'order.payment_failed',
  12. 'order.canceled',
  13. 'order.created',
  14. 'order.closed',
  15. 'order.updated',
  16. 'charge.created',
  17. 'charge.paid',
  18. 'charge.payment_failed',
  19. 'charge.refunded',
  20. 'charge.pending',
  21. 'charge.processing',
  22. 'charge.underpaid',
  23. 'charge.overpaid',
  24. 'charge.partial_canceled',
  25. 'charge.chargedback',
  26. 'charge.updated',
  27. ];
  28. public function __construct(
  29. private readonly PagarmePaymentService $pagarmePaymentService,
  30. private readonly PaymentService $paymentService,
  31. ) {}
  32. public function handlePagarme(array $payload): ?Payment
  33. {
  34. $event = $payload['type'] ?? null;
  35. $data = $payload['data'] ?? [];
  36. $data = is_array($data) ? $data : [];
  37. $webhook = $this->recordWebhook($payload, $event, $data);
  38. Log::channel('pagarme')->info('Pagar.me webhook received', [
  39. 'hook_id' => $payload['id'] ?? null,
  40. 'event' => $event,
  41. ]);
  42. if (! in_array($event, self::PAYMENT_EVENTS, true)) {
  43. $webhook->update([
  44. 'status' => 'ignored',
  45. 'processed_at' => now(),
  46. ]);
  47. return null;
  48. }
  49. $orderResponse = $this->normalizeOrderResponse($event, $data);
  50. $payment = $this->findPayment($orderResponse);
  51. if (! $payment) {
  52. $webhook->update([
  53. 'status' => 'payment_not_found',
  54. 'processed_at' => now(),
  55. ]);
  56. Log::channel('pagarme')->warning('Payment not found for Pagar.me webhook', [
  57. 'hook_id' => $payload['id'] ?? null,
  58. 'event' => $event,
  59. 'order_id' => $orderResponse['id'] ?? null,
  60. 'charge_id' => $orderResponse['charges'][0]['id'] ?? null,
  61. 'transaction_id' => $orderResponse['charges'][0]['last_transaction']['id'] ?? null,
  62. 'metadata' => $orderResponse['metadata'] ?? [],
  63. ]);
  64. return null;
  65. }
  66. try {
  67. $payment = $this->pagarmePaymentService->applyGatewayResponseToPayment($payment, $orderResponse);
  68. $this->paymentService->syncScheduleStatusAfterPayment($payment->schedule, $payment);
  69. $webhook->update([
  70. 'payment_id' => $payment->id,
  71. 'status' => 'processed',
  72. 'processed_at' => now(),
  73. ]);
  74. return $payment;
  75. } catch (\Throwable $e) {
  76. $webhook->update([
  77. 'payment_id' => $payment->id,
  78. 'status' => 'failed',
  79. 'processed_at' => now(),
  80. 'error_message' => $e->getMessage(),
  81. ]);
  82. throw $e;
  83. }
  84. }
  85. private function recordWebhook(array $payload, ?string $event, array $data): Webhook
  86. {
  87. $references = $this->extractReferences($event, $data);
  88. $hookId = $payload['id'] ?? null;
  89. $attributes = [
  90. 'hook_id' => $hookId,
  91. 'provider' => 'pagarme',
  92. 'event' => $event,
  93. 'account_id' => $payload['account']['id'] ?? null,
  94. 'order_id' => $references['order_id'],
  95. 'charge_id' => $references['charge_id'],
  96. 'transaction_id' => $references['transaction_id'],
  97. 'status' => 'received',
  98. 'payload' => $payload,
  99. 'received_at' => now(),
  100. 'processed_at' => null,
  101. 'error_message' => null,
  102. ];
  103. if (empty($hookId)) {
  104. return Webhook::create($attributes);
  105. }
  106. $webhook = Webhook::firstOrNew([
  107. 'provider' => 'pagarme',
  108. 'hook_id' => $hookId,
  109. ]);
  110. $webhook->fill([
  111. ...$attributes,
  112. 'attempts_count' => $webhook->exists ? $webhook->attempts_count + 1 : 1,
  113. ])->save();
  114. return $webhook;
  115. }
  116. private function extractReferences(?string $event, array $data): array
  117. {
  118. if (str_starts_with((string) $event, 'order.')) {
  119. $charge = $data['charges'][0] ?? [];
  120. return [
  121. 'order_id' => $data['id'] ?? null,
  122. 'charge_id' => $charge['id'] ?? null,
  123. 'transaction_id' => $charge['last_transaction']['id'] ?? null,
  124. ];
  125. }
  126. if (str_starts_with((string) $event, 'charge.')) {
  127. return [
  128. 'order_id' => $data['order']['id'] ?? $data['order_id'] ?? null,
  129. 'charge_id' => $data['id'] ?? null,
  130. 'transaction_id' => $data['last_transaction']['id'] ?? null,
  131. ];
  132. }
  133. return [
  134. 'order_id' => null,
  135. 'charge_id' => null,
  136. 'transaction_id' => null,
  137. ];
  138. }
  139. private function normalizeOrderResponse(?string $event, array $data): array
  140. {
  141. if (str_starts_with((string) $event, 'order.')) {
  142. return $data;
  143. }
  144. return [
  145. 'id' => $data['order']['id'] ?? $data['order_id'] ?? null,
  146. 'metadata' => $data['metadata'] ?? $data['order']['metadata'] ?? [],
  147. 'charges' => [$data],
  148. ];
  149. }
  150. private function findPayment(array $orderResponse): ?Payment
  151. {
  152. $charge = $orderResponse['charges'][0] ?? [];
  153. $transaction = $charge['last_transaction'] ?? [];
  154. $metadata = $orderResponse['metadata'] ?? $charge['metadata'] ?? [];
  155. $metadataId = filter_var($metadata['payment_id'] ?? null, FILTER_VALIDATE_INT) ?: null;
  156. $chargeId = $charge['id'] ?? null;
  157. $transactionId = $transaction['id'] ?? null;
  158. $orderId = $orderResponse['id'] ?? null;
  159. $references = array_filter([$metadataId, $chargeId, $transactionId, $orderId]);
  160. if (empty($references)) {
  161. return null;
  162. }
  163. return Payment::query()
  164. ->where('gateway_provider', 'pagarme')
  165. ->where(function ($query) use ($metadataId, $chargeId, $transactionId, $orderId) {
  166. $query
  167. ->when($metadataId, fn ($query) => $query->orWhere('id', $metadataId))
  168. ->when($chargeId, fn ($query) => $query->orWhere('gateway_entity_reference', $chargeId))
  169. ->when($transactionId, fn ($query) => $query->orWhere('gateway_operation_reference', $transactionId))
  170. ->when($orderId, fn ($query) => $query->orWhere(function ($query) use ($orderId) {
  171. $query->where('gateway_entity_label', 'order')
  172. ->where('gateway_entity_reference', $orderId);
  173. }));
  174. })
  175. ->latest('id')
  176. ->first();
  177. }
  178. }