Fixing stripe controller webhook

This commit is contained in:
Matt Young 2026-01-29 23:28:53 -06:00
parent 9d6c12ac5f
commit 9cf128a887
1 changed files with 57 additions and 44 deletions

View File

@ -2,13 +2,12 @@
namespace App\Http\Controllers;
use App\Enums\PaymentStatus;
use App\Enums\PaymentMethod;
use App\Enums\PaymentStatus;
use App\Models\Invoice;
use App\Models\Payment;
use Illuminate\Http\Request;
use Stripe\Stripe;
use Stripe\Webhook;
class StripeController extends Controller
{
@ -48,8 +47,6 @@ class StripeController extends Controller
public function webhook(Request $request)
{
\Log::info('Stripe webhook received');
$payload = $request->getContent();
$signature = $request->header('Stripe-Signature');
@ -60,26 +57,28 @@ class StripeController extends Controller
config('services.stripe.webhook_secret')
);
} catch (\Exception $e) {
\Log::error('Stripe webhook signature verification failed', ['error' => $e->getMessage()]);
return response('Invalid signature', 400);
}
\Log::info('Stripe webhook event', ['type' => $event->type]);
match ($event->type) {
'checkout.session.completed' => $this->handleCheckoutSessionCompleted($event->data->object),
'charge.updated' => $this->handleChargeUpdated($event->data->object),
default => null,
};
if ($event->type === 'checkout.session.completed') {
$session = $event->data->object;
\Log::info('Processing checkout.session.completed', [
'invoice_id' => $session->metadata->invoice_id ?? 'not set',
'payment_intent' => $session->payment_intent,
]);
return response('OK', 200);
}
protected function handleCheckoutSessionCompleted($session): void
{
$invoice = Invoice::find($session->metadata->invoice_id);
if ($invoice) {
if (! $invoice) {
return;
}
Stripe::setApiKey(config('services.stripe.secret'));
// Retrieve PaymentIntent with expanded charge and balance_transaction
$paymentIntent = \Stripe\PaymentIntent::retrieve([
'id' => $session->payment_intent,
'expand' => ['latest_charge.balance_transaction'],
@ -87,7 +86,7 @@ class StripeController extends Controller
$feeAmount = $paymentIntent->latest_charge?->balance_transaction?->fee ?? 0;
$payment = Payment::create([
Payment::create([
'invoice_id' => $invoice->id,
'payment_date' => now(),
'status' => PaymentStatus::COMPLETED,
@ -97,14 +96,28 @@ class StripeController extends Controller
'amount' => $session->amount_total / 100,
'fee_amount' => $feeAmount / 100,
]);
\Log::info('Payment created', ['payment_id' => $payment->id, 'fee_amount' => $feeAmount]);
} else {
\Log::warning('Invoice not found for Stripe webhook', ['invoice_id' => $session->metadata->invoice_id]);
}
}
return response('OK', 200);
protected function handleChargeUpdated($charge): void
{
if (! $charge->payment_intent || ! $charge->balance_transaction) {
return;
}
$payment = Payment::where('stripe_payment_intent_id', $charge->payment_intent)->first();
if (! $payment) {
return;
}
Stripe::setApiKey(config('services.stripe.secret'));
$balanceTransaction = \Stripe\BalanceTransaction::retrieve($charge->balance_transaction);
// Always update the fee - handles both initial capture and any corrections
$payment->updateQuietly([
'fee_amount' => $balanceTransaction->fee / 100,
]);
}
public function checkoutTutorial()