WebhookService.php 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228
  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. private readonly ProviderWithdrawalService $providerWithdrawalService,
  32. ) {
  33. }
  34. public function handlePagarme(array $payload): ?Payment
  35. {
  36. $event = $payload['type'] ?? null;
  37. $data = $payload['data'] ?? [];
  38. $data = is_array($data) ? $data : [];
  39. $webhook = $this->recordWebhook($payload, $event, $data);
  40. Log::channel('pagarme')->info('Pagar.me webhook received', [
  41. 'hook_id' => $payload['id'] ?? null,
  42. 'event' => $event,
  43. ]);
  44. if (! in_array($event, self::PAYMENT_EVENTS, true)) {
  45. if (str_starts_with((string) $event, 'transfer.')) {
  46. $this->providerWithdrawalService->handleTransferWebhook($data);
  47. $webhook->update([
  48. 'status' => 'processed',
  49. 'processed_at' => now(),
  50. ]);
  51. return null;
  52. }
  53. $webhook->update([
  54. 'status' => 'ignored',
  55. 'processed_at' => now(),
  56. ]);
  57. return null;
  58. }
  59. $orderResponse = $this->normalizeOrderResponse($event, $data);
  60. $payment = $this->findPayment($orderResponse);
  61. if (! $payment) {
  62. $webhook->update([
  63. 'status' => 'payment_not_found',
  64. 'processed_at' => now(),
  65. ]);
  66. Log::channel('pagarme')->warning('Payment not found for Pagar.me webhook', [
  67. 'hook_id' => $payload['id'] ?? null,
  68. 'event' => $event,
  69. 'order_id' => $orderResponse['id'] ?? null,
  70. 'charge_id' => $orderResponse['charges'][0]['id'] ?? null,
  71. 'transaction_id' => $orderResponse['charges'][0]['last_transaction']['id'] ?? null,
  72. 'metadata' => $orderResponse['metadata'] ?? [],
  73. ]);
  74. return null;
  75. }
  76. try {
  77. $payment = $this->pagarmePaymentService->applyGatewayResponseToPayment($payment, $orderResponse);
  78. $this->paymentService->syncScheduleStatusAfterPayment($payment->schedule, $payment);
  79. $webhook->update([
  80. 'payment_id' => $payment->id,
  81. 'status' => 'processed',
  82. 'processed_at' => now(),
  83. ]);
  84. return $payment;
  85. } catch (\Throwable $e) {
  86. $webhook->update([
  87. 'payment_id' => $payment->id,
  88. 'status' => 'failed',
  89. 'processed_at' => now(),
  90. 'error_message' => $e->getMessage(),
  91. ]);
  92. throw $e;
  93. }
  94. }
  95. //
  96. private function recordWebhook(array $payload, ?string $event, array $data): Webhook
  97. {
  98. $references = $this->extractReferences($event, $data);
  99. $hookId = $payload['id'] ?? null;
  100. $attributes = [
  101. 'hook_id' => $hookId,
  102. 'provider' => 'pagarme',
  103. 'event' => $event,
  104. 'account_id' => $payload['account']['id'] ?? null,
  105. 'order_id' => $references['order_id'],
  106. 'charge_id' => $references['charge_id'],
  107. 'transaction_id' => $references['transaction_id'],
  108. 'status' => 'received',
  109. 'payload' => $payload,
  110. 'received_at' => now(),
  111. 'processed_at' => null,
  112. 'error_message' => null,
  113. ];
  114. if (empty($hookId)) {
  115. return Webhook::create($attributes);
  116. }
  117. $webhook = Webhook::firstOrNew([
  118. 'provider' => 'pagarme',
  119. 'hook_id' => $hookId,
  120. ]);
  121. $webhook->fill([
  122. ...$attributes,
  123. 'attempts_count' => $webhook->exists ? $webhook->attempts_count + 1 : 1,
  124. ])->save();
  125. return $webhook;
  126. }
  127. private function extractReferences(?string $event, array $data): array
  128. {
  129. if (str_starts_with((string) $event, 'order.')) {
  130. $charge = $data['charges'][0] ?? [];
  131. return [
  132. 'order_id' => $data['id'] ?? null,
  133. 'charge_id' => $charge['id'] ?? null,
  134. 'transaction_id' => $charge['last_transaction']['id'] ?? null,
  135. ];
  136. }
  137. if (str_starts_with((string) $event, 'charge.')) {
  138. return [
  139. 'order_id' => $data['order']['id'] ?? $data['order_id'] ?? null,
  140. 'charge_id' => $data['id'] ?? null,
  141. 'transaction_id' => $data['last_transaction']['id'] ?? null,
  142. ];
  143. }
  144. return [
  145. 'order_id' => null,
  146. 'charge_id' => null,
  147. 'transaction_id' => null,
  148. ];
  149. }
  150. private function normalizeOrderResponse(?string $event, array $data): array
  151. {
  152. if (str_starts_with((string) $event, 'order.')) {
  153. return $data;
  154. }
  155. return [
  156. 'id' => $data['order']['id'] ?? $data['order_id'] ?? null,
  157. 'metadata' => $data['metadata'] ?? $data['order']['metadata'] ?? [],
  158. 'charges' => [$data],
  159. ];
  160. }
  161. private function findPayment(array $orderResponse): ?Payment
  162. {
  163. $charge = $orderResponse['charges'][0] ?? [];
  164. $transaction = $charge['last_transaction'] ?? [];
  165. $metadata = $orderResponse['metadata'] ?? $charge['metadata'] ?? [];
  166. $metadataId = filter_var($metadata['payment_id'] ?? null, FILTER_VALIDATE_INT) ?: null;
  167. $chargeId = $charge['id'] ?? null;
  168. $transactionId = $transaction['id'] ?? null;
  169. $orderId = $orderResponse['id'] ?? null;
  170. $references = array_filter([$metadataId, $chargeId, $transactionId, $orderId]);
  171. if (empty($references)) {
  172. return null;
  173. }
  174. return Payment::query()
  175. ->where('gateway_provider', 'pagarme')
  176. ->where(function ($query) use ($metadataId, $chargeId, $transactionId, $orderId) {
  177. $query
  178. ->when($metadataId, fn ($query) => $query->orWhere('id', $metadataId))
  179. ->when($chargeId, fn ($query) => $query->orWhere('gateway_entity_reference', $chargeId))
  180. ->when($transactionId, fn ($query) => $query->orWhere('gateway_operation_reference', $transactionId))
  181. ->when($orderId, fn ($query) => $query->orWhere(function ($query) use ($orderId) {
  182. $query->where('gateway_entity_label', 'order')
  183. ->where('gateway_entity_reference', $orderId);
  184. }));
  185. })
  186. ->latest('id')
  187. ->first();
  188. }
  189. }