- Buchungs-Seite zeigt das echte 4-Tier-Raster aus plans (Monat/Jahr-
Toggle, Jahrespreis als "2 Monate gratis") mit Checkout-Buttons,
Einzel-PM als separaten No-Abo-Block und Enterprise-Hinweis;
Credit-Konzept-Mock entfernt (Credits folgen mit 9I bzw. Phase 2)
- Aktueller-Tarif-Panel real: Abo (Preis, Kontingent, Kündigungsstatus),
Bestandstarif (unbegrenzt, nächste MAN-Rechnung), offene Einzelkäufe;
Kontingent-Kachel zeigt "Unbegrenzt" bei Bestandsschutz
- "Abo verwalten" über das Stripe Billing Portal
(me.checkout.billing-portal; Zahlungsmethode, Rechnungen, Kündigung)
- Aktive Buchungen + Verlauf aus echten Daten (Abo, Legacy-Vereinbarung,
offene/eingelöste Einzelkäufe mit PM-Verknüpfung)
- Tests: BookingsPageTest (9 Tests), PanelConsolidationTest angepasst;
Suite 519 passed / 4 skipped
- Doku: PHASE-9-Plan 9F ✅, Billing-Doku (Routen, Stripe Tax aktiviert),
STATUS-ABGLEICH, Checkliste, PROGRESS
Co-Authored-By: Claude Fable 5 <noreply@anthropic.com>
60 lines
2.1 KiB
PHP
60 lines
2.1 KiB
PHP
<?php
|
|
|
|
namespace App\Services\Billing;
|
|
|
|
use App\Models\Plan;
|
|
use App\Models\SinglePurchase;
|
|
use App\Models\User;
|
|
use Laravel\Cashier\Checkout;
|
|
|
|
/**
|
|
* Dünner Wrapper um die Cashier-Checkout-Erzeugung (Phase 9E).
|
|
*
|
|
* Hier passiert ausschließlich der Stripe-Aufruf — alle Guards (Tarif
|
|
* synchronisiert, kein Doppel-Abo, Preis konfiguriert) liegen im
|
|
* CheckoutController. So bleibt der Controller ohne Stripe-Anbindung
|
|
* testbar, indem dieser Service im Container gemockt wird.
|
|
*/
|
|
class StripeCheckoutService
|
|
{
|
|
/**
|
|
* Stripe-Checkout für ein Tarif-Abo (monatlich/jährlich). Die Steuer
|
|
* ergänzt Stripe Tax automatisch (Cashier::calculateTaxes, Netto-Preise).
|
|
*/
|
|
public function forSubscription(User $user, Plan $plan, string $interval): Checkout
|
|
{
|
|
$priceId = $interval === 'yearly'
|
|
? $plan->stripe_price_id_yearly
|
|
: $plan->stripe_price_id_monthly;
|
|
|
|
return $user
|
|
->newSubscription('default', $priceId)
|
|
->checkout([
|
|
'success_url' => route('me.bookings.index', ['checkout' => 'erfolg']),
|
|
'cancel_url' => route('me.bookings.index', ['checkout' => 'abbruch']),
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* URL zum Stripe Billing Portal (Zahlungsmethode, Rechnungen, Kündigung).
|
|
* Rücksprung auf die Buchungs-Seite.
|
|
*/
|
|
public function billingPortalUrl(User $user): string
|
|
{
|
|
return $user->billingPortalUrl(route('me.bookings.index'));
|
|
}
|
|
|
|
/**
|
|
* Stripe-Checkout für eine Einzel-PM. Die `single_purchase_id` in den
|
|
* Session-Metadaten schließt den Kreis: `checkout.session.completed`
|
|
* markiert den Kauf über ProcessStripeWebhook als bezahlt.
|
|
*/
|
|
public function forSinglePurchase(User $user, SinglePurchase $purchase): Checkout
|
|
{
|
|
return $user->checkout([config('billing.single_pm_stripe_price_id') => 1], [
|
|
'success_url' => route('me.bookings.index', ['checkout' => 'erfolg']),
|
|
'cancel_url' => route('me.bookings.index', ['checkout' => 'abbruch']),
|
|
'metadata' => ['single_purchase_id' => (string) $purchase->id],
|
|
]);
|
|
}
|
|
}
|