create(['stripe_id' => 'cus_created_1']); Cache::tags(['plans'])->put("plan_for_user_{$user->id}", 'stale', 3600); (new HandleStripeWebhook)->handle(new WebhookReceived([ 'type' => 'customer.subscription.created', 'data' => ['object' => ['customer' => 'cus_created_1']], ])); expect(Cache::tags(['plans'])->get("plan_for_user_{$user->id}"))->toBeNull(); }); it('ignores subscription.created when the user is not found', function (): void { (new HandleStripeWebhook)->handle(new WebhookReceived([ 'type' => 'customer.subscription.created', 'data' => ['object' => ['customer' => 'cus_unknown']], ])); expect(true)->toBeTrue(); }); it('busts the plan cache on customer.subscription.updated', function (): void { $user = User::factory()->create(['stripe_id' => 'cus_updated_1']); Cache::tags(['plans'])->put("plan_for_user_{$user->id}", 'stale', 3600); (new HandleStripeWebhook)->handle(new WebhookReceived([ 'type' => 'customer.subscription.updated', 'data' => ['object' => ['customer' => 'cus_updated_1']], ])); expect(Cache::tags(['plans'])->get("plan_for_user_{$user->id}"))->toBeNull(); }); it('on customer.subscription.deleted disables whatsapp+sms prefs, clears grace, busts cache', function (): void { $user = User::factory()->create([ 'stripe_id' => 'cus_deleted_1', 'grace_period_until' => now()->addDays(5), ]); UserNotificationPreference::query()->insert([ ['user_id' => $user->id, 'channel' => 'whatsapp', 'fuel_type' => 'E10', 'enabled' => true, 'created_at' => now(), 'updated_at' => now()], ['user_id' => $user->id, 'channel' => 'sms', 'fuel_type' => 'E10', 'enabled' => true, 'created_at' => now(), 'updated_at' => now()], ['user_id' => $user->id, 'channel' => 'email', 'fuel_type' => 'E10', 'enabled' => true, 'created_at' => now(), 'updated_at' => now()], ]); Cache::tags(['plans'])->put("plan_for_user_{$user->id}", 'stale', 3600); (new HandleStripeWebhook)->handle(new WebhookReceived([ 'type' => 'customer.subscription.deleted', 'data' => ['object' => ['customer' => 'cus_deleted_1']], ])); expect(UserNotificationPreference::where('user_id', $user->id)->where('channel', 'whatsapp')->value('enabled'))->toBeFalse(); expect(UserNotificationPreference::where('user_id', $user->id)->where('channel', 'sms')->value('enabled'))->toBeFalse(); expect(UserNotificationPreference::where('user_id', $user->id)->where('channel', 'email')->value('enabled'))->toBeTrue(); expect($user->fresh()->grace_period_until)->toBeNull(); expect(Cache::tags(['plans'])->get("plan_for_user_{$user->id}"))->toBeNull(); }); it('on invoice.payment_succeeded clears grace_period_until and busts cache', function (): void { $user = User::factory()->create([ 'stripe_id' => 'cus_paid_1', 'grace_period_until' => now()->addDays(4), ]); Cache::tags(['plans'])->put("plan_for_user_{$user->id}", 'stale', 3600); (new HandleStripeWebhook)->handle(new WebhookReceived([ 'type' => 'invoice.payment_succeeded', 'data' => ['object' => ['customer' => 'cus_paid_1']], ])); expect($user->fresh()->grace_period_until)->toBeNull(); expect(Cache::tags(['plans'])->get("plan_for_user_{$user->id}"))->toBeNull(); }); it('invoice.payment_succeeded is a no-op when grace was never set', function (): void { $user = User::factory()->create(['stripe_id' => 'cus_paid_2', 'grace_period_until' => null]); (new HandleStripeWebhook)->handle(new WebhookReceived([ 'type' => 'invoice.payment_succeeded', 'data' => ['object' => ['customer' => 'cus_paid_2']], ])); expect($user->fresh()->grace_period_until)->toBeNull(); }); it('on invoice.payment_failed sets grace_period_until 5 days out and queues both reminders', function (): void { Queue::fake(); $user = User::factory()->create(['stripe_id' => 'cus_failed_1', 'grace_period_until' => null]); $before = now(); (new HandleStripeWebhook)->handle(new WebhookReceived([ 'type' => 'invoice.payment_failed', 'data' => ['object' => ['customer' => 'cus_failed_1']], ])); $grace = $user->fresh()->grace_period_until; expect($grace)->not->toBeNull(); expect($grace->greaterThanOrEqualTo($before->copy()->addDays(5)->subSecond()))->toBeTrue(); expect($grace->lessThanOrEqualTo($before->copy()->addDays(5)->addSeconds(5)))->toBeTrue(); Queue::assertPushed(SendPaymentFailedReminderJob::class, function ($job) use ($user) { return $job->userId === $user->id && $job->day === 3; }); Queue::assertPushed(SendPaymentFailedReminderJob::class, function ($job) use ($user) { return $job->userId === $user->id && $job->day === 5; }); Queue::assertPushed(SendPaymentFailedReminderJob::class, 2); }); it('repeat invoice.payment_failed within grace does not re-dispatch reminders', function (): void { Queue::fake(); $existingGrace = now()->addDays(3)->startOfSecond(); $user = User::factory()->create(['stripe_id' => 'cus_failed_2', 'grace_period_until' => $existingGrace]); (new HandleStripeWebhook)->handle(new WebhookReceived([ 'type' => 'invoice.payment_failed', 'data' => ['object' => ['customer' => 'cus_failed_2']], ])); // grace_period_until unchanged (same value) expect($user->fresh()->grace_period_until->equalTo($existingGrace))->toBeTrue(); // No new jobs queued Queue::assertNothingPushed(); });