OrderResponseData.php 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. <?php
  2. namespace App\Data\Pagarme\Response\OrderResponseData;
  3. use App\Data\Pagarme\Response\CustomerResponseData\CustomerResponseData;
  4. use App\Data\Pagarme\Response\OrderResponseData\OrderChargeResponseData\OrderChargeResponseData;
  5. use App\Data\Pagarme\Response\OrderResponseData\OrderChargeResponseData\OrderTransactionResponseData;
  6. use App\Data\Pagarme\PagarmeResponseData;
  7. use App\Enums\PaymentStatusEnum;
  8. /**
  9. * @param OrderItemResponseData[] $items
  10. * @param OrderChargeResponseData[] $charges
  11. * @param OrderCheckoutResponseData[] $checkouts
  12. */
  13. final readonly class OrderResponseData extends PagarmeResponseData
  14. {
  15. public function __construct(
  16. public ?string $id,
  17. public ?string $code,
  18. public ?int $amount,
  19. public ?string $currency,
  20. public ?bool $closed,
  21. public ?string $status,
  22. public array $items,
  23. public ?CustomerResponseData $customer,
  24. public array $charges,
  25. public array $checkouts,
  26. public array $metadata,
  27. public ?string $createdAt = null,
  28. public ?string $updatedAt = null,
  29. public ?string $closedAt = null,
  30. ) {}
  31. public function authorizedAt(): ?string
  32. {
  33. $transaction = $this->lastTransaction();
  34. if (! $transaction) {
  35. return null;
  36. }
  37. if (in_array($transaction->status, ['authorized_pending_capture', 'captured', 'partial_capture'], true)) {
  38. return $transaction->createdAt;
  39. }
  40. return null;
  41. }
  42. public function failureCode(): ?string
  43. {
  44. $transaction = $this->lastTransaction();
  45. if (! $transaction) {
  46. return null;
  47. }
  48. $code = $this->filledValue($transaction->gatewayResponse['code'] ?? null);
  49. if ($code === null || $this->isMisleadingFailureCode($code)) {
  50. return $this->filledValue($transaction->status);
  51. }
  52. return $code;
  53. }
  54. public function failureMessage(): ?string
  55. {
  56. $transaction = $this->lastTransaction();
  57. if (! $transaction) {
  58. return null;
  59. }
  60. $acquirerMessage = $this->filledValue($transaction->acquirerMessage);
  61. if ($acquirerMessage && ! $this->isMisleadingAcquirerMessage($acquirerMessage)) {
  62. return $acquirerMessage;
  63. }
  64. $gatewayErrors = $transaction->gatewayResponse['errors'] ?? [];
  65. if (! is_array($gatewayErrors) || empty($gatewayErrors)) {
  66. return $this->failureMessageFromStatus($transaction->status);
  67. }
  68. $message = collect($gatewayErrors)
  69. ->pluck('message')
  70. ->filter()
  71. ->implode('; ') ?: null;
  72. if ($message && str_contains($message, 'Sem ambiente configurado')) {
  73. return 'Pix não esta habilitado ou configurado neste ambiente do Pagar.me.';
  74. }
  75. return $message;
  76. }
  77. public function firstCharge(): ?OrderChargeResponseData
  78. {
  79. return $this->charges[0] ?? null;
  80. }
  81. public function gatewayEntityLabel(): string
  82. {
  83. $charge = $this->firstCharge();
  84. return $charge?->id ? 'charge' : 'order';
  85. }
  86. public function gatewayEntityReference(): ?string
  87. {
  88. $charge = $this->firstCharge();
  89. return $charge?->id ?? $this->id;
  90. }
  91. public function gatewayOperationLabel(): string
  92. {
  93. $charge = $this->firstCharge();
  94. $transaction = $this->lastTransaction();
  95. return $transaction?->id ? 'transaction' : ($charge?->id ? 'charge' : 'order');
  96. }
  97. public function gatewayOperationReference(): ?string
  98. {
  99. $charge = $this->firstCharge();
  100. $transaction = $this->lastTransaction();
  101. return $transaction?->id ?? $charge?->id ?? $this->id;
  102. }
  103. public function lastTransaction(): ?OrderTransactionResponseData
  104. {
  105. return $this->firstCharge()?->transaction();
  106. }
  107. public function paymentStatus(): PaymentStatusEnum
  108. {
  109. $charge = $this->firstCharge();
  110. $transaction = $this->lastTransaction();
  111. $status = strtolower((string) ($transaction?->status ?: $charge?->status));
  112. return match ($status) {
  113. 'captured', 'paid', 'overpaid' => PaymentStatusEnum::PAID,
  114. 'authorized_pending_capture', 'waiting_capture' => PaymentStatusEnum::AUTHORIZED,
  115. 'pending', 'waiting_payment' => PaymentStatusEnum::PENDING,
  116. 'processing' => PaymentStatusEnum::PROCESSING,
  117. 'not_authorized', 'with_error', 'failed',
  118. 'underpaid', 'chargedback' => PaymentStatusEnum::FAILED,
  119. 'voided', 'partial_void', 'canceled',
  120. 'cancelled', 'refunded', 'partial_refunded',
  121. 'partial_canceled' => PaymentStatusEnum::CANCELLED,
  122. default => PaymentStatusEnum::PENDING,
  123. };
  124. }
  125. public function paidAt(): ?string
  126. {
  127. return $this->firstCharge()?->paidAt;
  128. }
  129. public function requireId(): string
  130. {
  131. if (! $this->id) {
  132. throw new \RuntimeException('Pagar.me order creation returned an empty id.');
  133. }
  134. return $this->id;
  135. }
  136. //
  137. public static function fromArray(array $payload): static
  138. {
  139. $customer = static::arrArray($payload, 'customer');
  140. return new self(
  141. id: static::arrString($payload, 'id'),
  142. code: static::arrString($payload, 'code'),
  143. amount: static::arrInt($payload, 'amount'),
  144. currency: static::arrString($payload, 'currency'),
  145. closed: static::arrBool($payload, 'closed'),
  146. status: static::arrString($payload, 'status'),
  147. items: static::arrMap($payload, 'items',
  148. static fn (array $item) => OrderItemResponseData::fromArray($item),
  149. ),
  150. customer: ! empty($customer)
  151. ? CustomerResponseData::fromArray($customer)
  152. : null,
  153. charges: static::arrMap($payload, 'charges',
  154. static fn (array $charge) => OrderChargeResponseData::fromArray($charge),
  155. ),
  156. checkouts: static::arrMap($payload, 'checkouts',
  157. static fn (array $checkout) => OrderCheckoutResponseData::fromArray($checkout),
  158. ),
  159. metadata: static::arrArray($payload, 'metadata'),
  160. createdAt: static::arrString($payload, 'created_at'),
  161. updatedAt: static::arrString($payload, 'updated_at'),
  162. closedAt: static::arrString($payload, 'closed_at'),
  163. );
  164. }
  165. public function toArray(): array
  166. {
  167. return [
  168. 'id' => $this->id,
  169. 'code' => $this->code,
  170. 'amount' => $this->amount,
  171. 'currency' => $this->currency,
  172. 'closed' => $this->closed,
  173. 'items' => array_map(
  174. static fn (OrderItemResponseData $item) => $item->toArray(),
  175. $this->items,
  176. ),
  177. 'customer' => $this->customer?->toArray(),
  178. 'status' => $this->status,
  179. 'created_at' => $this->createdAt,
  180. 'updated_at' => $this->updatedAt,
  181. 'closed_at' => $this->closedAt,
  182. 'charges' => array_map(
  183. static fn (OrderChargeResponseData $charge) => $charge->toArray(),
  184. $this->charges,
  185. ),
  186. 'checkouts' => array_map(
  187. static fn (OrderCheckoutResponseData $checkout) => $checkout->toArray(),
  188. $this->checkouts,
  189. ),
  190. 'metadata' => $this->metadata,
  191. ];
  192. }
  193. //
  194. private function isMisleadingFailureCode(string $code): bool
  195. {
  196. // filtra para nao incluir codigos http que nao sao de erro
  197. if (preg_match('/^[1-5]\d{2}$/', $code)) {
  198. return true;
  199. }
  200. $lower = mb_strtolower($code);
  201. $successCodes = ['00', '0', 'approved', 'success'];
  202. return in_array($lower, $successCodes, true);
  203. }
  204. private function isMisleadingAcquirerMessage(string $message): bool
  205. {
  206. $lower = mb_strtolower($message);
  207. $successPatterns = [
  208. 'aprovada',
  209. 'aprovado',
  210. 'autorizada',
  211. 'autorizado',
  212. 'authorized',
  213. 'sucesso',
  214. ];
  215. foreach ($successPatterns as $pattern) {
  216. if (str_contains($lower, $pattern)) {
  217. return true;
  218. }
  219. }
  220. return false;
  221. }
  222. //
  223. private function failureMessageFromStatus(?string $status): ?string
  224. {
  225. return match (strtolower((string) $status)) {
  226. 'not_authorized' => 'Transação não autorizada pela operadora do cartão.',
  227. 'with_error' => 'Erro ao processar a transação.',
  228. 'failed' => 'Transação falhou.',
  229. 'underpaid' => 'Valor pago inferior ao esperado.',
  230. 'chargedback' => 'Transação sofreu chargeback.',
  231. default => null,
  232. };
  233. }
  234. private function filledValue(mixed $value): ?string
  235. {
  236. if ($value === null || $value === '' || $value === []) {
  237. return null;
  238. }
  239. return (string) $value;
  240. }
  241. }