Tarif-Datenmodell (Decision-Update): - plans: Starter/Business/Pro/Agency mit Monats-/Jahrespreis (Jahres = 10 x Monat), PM-Kontingent, Tageslimit, Stripe-IDs; idempotenter Seeder - single_purchases: Einzel-PM, Extra-PM, Boost, PDF-Nachweis mit Status-Lifecycle und Stripe-Checkout-Referenzen - laravel/cashier ^16.5 installiert (freigegeben); User ist Billable, Cashier-Migrationen published + ausgefuehrt; lokale invoices()-Relation ueberschreibt bewusst die Cashier-Methode Hybride Rechnungskreise (Entscheidung 12.06.2026): - invoice_number_sequences + InvoiceNumberGenerator: atomare fortlaufende Nummern pro Kreis (STR- fuer den neuen Stripe-Shop, MAN- fuer den manuellen Legacy-Kreis); Alt-Archiv legacy_invoices bleibt unveraendert - ManualInvoiceService + billing:generate-manual-invoices (Scheduler taeglich 04:30): prueft aktive/grandfathered user_payment_options ohne Stripe-Subscription auf erreichtes Periodenende, friert die Rechnungsadresse als Snapshot ein, stellt die MAN-Rechnung aus (Zahlungsziel billing.manual_due_days) und schaltet die Periode weiter; Konditions-Overrides via legacy_conditions, sonst Netto-Preis + billing.vat_rate; nicht abrechenbare Faelle werden geloggt und beim naechsten Lauf erneut geprueft Submit-Gate: - User::hasActiveBooking() prueft jetzt echt (hinter billing.enforce_booking): Cashier-Abo, bezahlter Einzel-/Extra-PM-Kauf oder laufende Legacy-Vereinbarung (MAN-Kreis) Suite: 468 passed, 4 skipped (17 neue Billing-Tests). Pint clean. Offen fuer 9E: Stripe-Checkout/Webhooks, STR-Spiegelung, Slot-Logik auf Plan-Kontingent, Migration der aktiven Legacy-Zahlungen in user_payment_options. Co-Authored-By: Claude Fable 5 <noreply@anthropic.com>
75 lines
2.3 KiB
PHP
75 lines
2.3 KiB
PHP
<?php
|
|
|
|
use App\Enums\UserPaymentOptionStatus;
|
|
use App\Models\PaymentOption;
|
|
use App\Models\SinglePurchase;
|
|
use App\Models\User;
|
|
use App\Models\UserPaymentOption;
|
|
|
|
beforeEach(function (): void {
|
|
config()->set('billing.enforce_booking', true);
|
|
});
|
|
|
|
test('without the enforce flag everyone counts as booked', function () {
|
|
config()->set('billing.enforce_booking', false);
|
|
|
|
expect(User::factory()->create()->hasActiveBooking())->toBeTrue();
|
|
});
|
|
|
|
test('a user without any booking is rejected when the gate is enforced', function () {
|
|
expect(User::factory()->create()->hasActiveBooking())->toBeFalse();
|
|
});
|
|
|
|
test('an active cashier subscription counts as booking', function () {
|
|
$user = User::factory()->create();
|
|
|
|
$user->subscriptions()->create([
|
|
'type' => 'default',
|
|
'stripe_id' => 'sub_test_123',
|
|
'stripe_status' => 'active',
|
|
'stripe_price' => 'price_test',
|
|
'quantity' => 1,
|
|
]);
|
|
|
|
expect($user->hasActiveBooking())->toBeTrue();
|
|
});
|
|
|
|
test('a paid unconsumed single purchase counts as booking', function () {
|
|
$user = User::factory()->create();
|
|
SinglePurchase::factory()->paid()->create(['user_id' => $user->id]);
|
|
|
|
expect($user->hasActiveBooking())->toBeTrue();
|
|
});
|
|
|
|
test('a consumed single purchase does not count as booking', function () {
|
|
$user = User::factory()->create();
|
|
SinglePurchase::factory()->consumed()->create(['user_id' => $user->id]);
|
|
|
|
expect($user->hasActiveBooking())->toBeFalse();
|
|
});
|
|
|
|
test('an active legacy payment agreement counts as booking (manual circle)', function () {
|
|
$user = User::factory()->create();
|
|
|
|
UserPaymentOption::factory()->create([
|
|
'user_id' => $user->id,
|
|
'payment_option_id' => PaymentOption::factory()->create()->id,
|
|
'status' => UserPaymentOptionStatus::Active->value,
|
|
'stripe_subscription_id' => null,
|
|
]);
|
|
|
|
expect($user->hasActiveBooking())->toBeTrue();
|
|
});
|
|
|
|
test('a cancelled legacy agreement does not count as booking', function () {
|
|
$user = User::factory()->create();
|
|
|
|
UserPaymentOption::factory()->create([
|
|
'user_id' => $user->id,
|
|
'payment_option_id' => PaymentOption::factory()->create()->id,
|
|
'status' => UserPaymentOptionStatus::Cancelled->value,
|
|
'stripe_subscription_id' => null,
|
|
]);
|
|
|
|
expect($user->hasActiveBooking())->toBeFalse();
|
|
});
|