ProviderWithdrawalService.php 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. <?php
  2. namespace App\Services;
  3. use App\Data\Pagarme\Response\PagarmeTransferResponseData;
  4. use App\Enums\PaymentStatusEnum;
  5. use App\Enums\ProviderWithdrawalStatusEnum;
  6. use App\Models\PaymentSplit;
  7. use App\Models\Provider;
  8. use App\Models\ProviderWithdrawal;
  9. use App\Services\Pagarme\PagarmeTransferService;
  10. use Carbon\Carbon;
  11. use Illuminate\Database\Eloquent\Collection;
  12. use Illuminate\Support\Facades\DB;
  13. use Illuminate\Support\Facades\Log;
  14. class ProviderWithdrawalService
  15. {
  16. public function __construct(
  17. private readonly PagarmeTransferService $pagarmeTransfer,
  18. ) {}
  19. public function getAvailableBalance(Provider $provider): float
  20. {
  21. $earnings = (float) PaymentSplit::query()
  22. ->where('provider_id', $provider->id)
  23. ->whereHas('payment', fn ($q) => $q->where('status', PaymentStatusEnum::PAID))
  24. ->whereHas('payment.schedule', fn ($q) => $this->availableScheduleQuery($q))
  25. ->sum('gross_amount');
  26. $withdrawn = (float) ProviderWithdrawal::query()
  27. ->where('provider_id', $provider->id)
  28. ->whereIn('status', [
  29. ProviderWithdrawalStatusEnum::PENDING_TRANSFER,
  30. ProviderWithdrawalStatusEnum::PROCESSING,
  31. ProviderWithdrawalStatusEnum::TRANSFERRED,
  32. ])
  33. ->sum('gross_amount');
  34. return max(0, $earnings - $withdrawn);
  35. }
  36. public function getPendingBalance(Provider $provider): float
  37. {
  38. return (float) PaymentSplit::query()
  39. ->where('provider_id', $provider->id)
  40. ->whereHas('payment', fn ($q) => $q->where('status', PaymentStatusEnum::PAID))
  41. ->whereHas('payment.schedule', fn ($q) => $this->pendingScheduleQuery($q))
  42. ->sum('gross_amount');
  43. }
  44. public function getPaymentSplits(Provider $provider): Collection
  45. {
  46. return PaymentSplit::query()
  47. ->where('provider_id', $provider->id)
  48. ->with(['payment.schedule.client.user'])
  49. ->orderBy('created_at', 'desc')
  50. ->get();
  51. }
  52. public function requestWithdrawal(Provider $provider): ProviderWithdrawal
  53. {
  54. if (empty($provider->recipient_id)) {
  55. throw new \RuntimeException('Prestador nao possui recipient_id no Pagar.me.');
  56. }
  57. return DB::transaction(function () use ($provider) {
  58. Provider::query()
  59. ->where('id', $provider->id)
  60. ->lockForUpdate()
  61. ->first();
  62. $pending = ProviderWithdrawal::query()
  63. ->where('provider_id', $provider->id)
  64. ->whereIn('status', [
  65. ProviderWithdrawalStatusEnum::PENDING_TRANSFER,
  66. ProviderWithdrawalStatusEnum::PROCESSING,
  67. ])
  68. ->exists();
  69. if ($pending) {
  70. throw new \RuntimeException('Voce ja possui um saque em andamento.');
  71. }
  72. $available = $this->getAvailableBalance($provider);
  73. if ($available <= 0) {
  74. throw new \RuntimeException('Saldo disponivel insuficiente para saque.');
  75. }
  76. $amountInCents = (int) round($available * 100);
  77. $idempotencyKey = sprintf('withdrawal-%d-%s', $provider->id, now()->timestamp);
  78. $bankAccount = $provider->recipient_default_bank_account;
  79. $withdrawal = ProviderWithdrawal::create([
  80. 'provider_id' => $provider->id,
  81. 'recipient_id' => $provider->recipient_id,
  82. 'idempotency_key' => $idempotencyKey,
  83. 'gross_amount' => $available,
  84. 'net_amount' => $available,
  85. 'status' => ProviderWithdrawalStatusEnum::PENDING_TRANSFER,
  86. 'bank_account' => $bankAccount,
  87. 'metadata' => [
  88. 'provider_id' => $provider->id,
  89. 'amount' => $available,
  90. ],
  91. ]);
  92. try {
  93. $response = $this->pagarmeTransfer->createTransfer(
  94. $amountInCents,
  95. $provider->recipient_id,
  96. $idempotencyKey,
  97. );
  98. $transfer = PagarmeTransferResponseData::fromArray($response);
  99. $withdrawal->forceFill([
  100. 'transfer_id' => $transfer->id(),
  101. 'status' => $transfer->status() ?? ProviderWithdrawalStatusEnum::PROCESSING->value,
  102. 'type' => $transfer->type,
  103. 'gateway_fee_amount' => $transfer->fee ?? 0,
  104. 'net_amount' => max(0, $available - ($transfer->fee ?? 0) / 100),
  105. 'gateway_payload' => $response,
  106. ])->save();
  107. PaymentSplit::query()
  108. ->where('provider_id', $provider->id)
  109. ->whereNull('provider_withdrawal_id')
  110. ->whereHas('payment', fn ($q) => $q->where('status', PaymentStatusEnum::PAID))
  111. ->whereHas('payment.schedule', fn ($q) => $this->availableScheduleQuery($q))
  112. ->update(['provider_withdrawal_id' => $withdrawal->id]);
  113. return $withdrawal->fresh();
  114. } catch (\Throwable $e) {
  115. $withdrawal->forceFill([
  116. 'status' => ProviderWithdrawalStatusEnum::FAILED,
  117. 'failed_at' => now(),
  118. 'bank_response' => $e->getMessage(),
  119. ])->save();
  120. throw $e;
  121. }
  122. });
  123. }
  124. public function getWithdrawals(Provider $provider): Collection
  125. {
  126. return ProviderWithdrawal::query()
  127. ->where('provider_id', $provider->id)
  128. ->orderBy('created_at', 'desc')
  129. ->get();
  130. }
  131. public function getWithdrawal(int $id, Provider $provider): ProviderWithdrawal
  132. {
  133. return ProviderWithdrawal::query()
  134. ->where('id', $id)
  135. ->where('provider_id', $provider->id)
  136. ->firstOrFail();
  137. }
  138. public function handleTransferWebhook(array $data): void
  139. {
  140. $transferId = $data['id'] ?? null;
  141. if (empty($transferId)) {
  142. return;
  143. }
  144. $withdrawal = ProviderWithdrawal::query()
  145. ->where('transfer_id', $transferId)
  146. ->first();
  147. if (! $withdrawal) {
  148. Log::channel('pagarme')->warning('ProviderWithdrawal not found for transfer webhook', [
  149. 'transfer_id' => $transferId,
  150. ]);
  151. return;
  152. }
  153. $status = $data['status'] ?? null;
  154. if ($status === ProviderWithdrawalStatusEnum::TRANSFERRED->value) {
  155. $withdrawal->forceFill([
  156. 'status' => ProviderWithdrawalStatusEnum::TRANSFERRED,
  157. 'completed_at' => now(),
  158. ])->save();
  159. } elseif (in_array($status, [ProviderWithdrawalStatusEnum::FAILED->value, ProviderWithdrawalStatusEnum::CANCELED->value], true)) {
  160. $withdrawal->paymentSplits()
  161. ->update(['provider_withdrawal_id' => null]);
  162. $withdrawal->forceFill([
  163. 'status' => ProviderWithdrawalStatusEnum::fromString($status),
  164. 'failed_at' => $status === ProviderWithdrawalStatusEnum::FAILED->value ? now() : null,
  165. 'bank_response' => $status === ProviderWithdrawalStatusEnum::FAILED->value ? ($data['bank_response'] ?? null) : null,
  166. ])->save();
  167. } elseif ($status === ProviderWithdrawalStatusEnum::PROCESSING->value) {
  168. $withdrawal->forceFill([
  169. 'status' => ProviderWithdrawalStatusEnum::PROCESSING,
  170. ])->save();
  171. }
  172. }
  173. private function availableScheduleQuery($query)
  174. {
  175. return $query
  176. ->where('code_verified', true)
  177. ->when(
  178. ! app()->environment('local', 'development'),
  179. fn ($query) => $query->whereRaw(
  180. $this->scheduleEndedAtExpression() . ' <= ?',
  181. [$this->withdrawalReleaseCutoff()]
  182. )
  183. );
  184. }
  185. private function pendingScheduleQuery($query)
  186. {
  187. return $query
  188. ->whereNotIn('status', ['cancelled', 'rejected'])
  189. ->where(function ($q) {
  190. $q->where('code_verified', false)
  191. ->orWhereRaw(
  192. $this->scheduleEndedAtExpression().' > ?',
  193. [$this->withdrawalReleaseCutoff()]
  194. );
  195. });
  196. }
  197. private function scheduleEndedAtExpression(): string
  198. {
  199. return match (DB::connection()->getDriverName()) {
  200. 'pgsql' => '(date + end_time)',
  201. 'sqlite' => "datetime(date || ' ' || end_time)",
  202. default => 'TIMESTAMP(date, end_time)',
  203. };
  204. }
  205. private function withdrawalReleaseCutoff(): string
  206. {
  207. return Carbon::now()->subDays(5)->format('Y-m-d H:i:s');
  208. }
  209. }