WS-4: Bestandsschutz-Abrechnung – gekoppelte Kündigung + Admin-Sichtbarkeit
M1 (portalübergreifend) und M5 (04:30-Worker) verifiziert; M2 + M3 umgesetzt. - M2 (Entscheidung: getrennte Rechnungen, gekoppelte Kündigung): neuer LegacySubscriptionService::cancelBundleFor() kündigt das gesamte Bestandsschutz-Bündel eines Kunden gemeinsam – keine Einzelkündigung. Kein Rechnungs-Schema-Umbau (jede Kondition behält ihre MAN-Rechnung). - M3: Admin-Payments-Ansicht zeigt eine "Bestandsschutz-Posten"-Tabelle (Badge, Portal, Netto, nächste Fälligkeit, "Bündel (N)") + Bündel-Kündigung mit Bestätigungsmodal. - Tests: tests/Feature/Billing/LegacyBundleTest.php (M1 cross-portal, M2 Bündel- Kündigung + Gate, M3 Admin-Liste + Kündigung). Detailplan WS-4 dokumentiert. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
parent
6668f492ad
commit
be7d1799a5
4 changed files with 372 additions and 0 deletions
118
tests/Feature/Billing/LegacyBundleTest.php
Normal file
118
tests/Feature/Billing/LegacyBundleTest.php
Normal file
|
|
@ -0,0 +1,118 @@
|
|||
<?php
|
||||
|
||||
use App\Enums\Portal;
|
||||
use App\Enums\UserPaymentOptionStatus;
|
||||
use App\Models\User;
|
||||
use App\Models\UserPaymentOption;
|
||||
use App\Services\Billing\LegacySubscriptionService;
|
||||
use App\Services\CurrentPortalContext;
|
||||
use Database\Seeders\RolesAndPermissionsSeeder;
|
||||
use Livewire\Volt\Volt;
|
||||
use Tests\TestCase;
|
||||
|
||||
afterEach(function () {
|
||||
// Statischer Portal-Kontext darf nicht in andere Tests lecken.
|
||||
CurrentPortalContext::clear();
|
||||
});
|
||||
|
||||
function grandfatheredPost(User $user, string $portal, int $netCents = 100000): UserPaymentOption
|
||||
{
|
||||
return UserPaymentOption::factory()->create([
|
||||
'user_id' => $user->id,
|
||||
'status' => UserPaymentOptionStatus::Grandfathered->value,
|
||||
'stripe_subscription_id' => null,
|
||||
'cancelled_at' => null,
|
||||
'legacy_conditions' => [
|
||||
'legacy_portal' => $portal,
|
||||
'net_cents' => $netCents,
|
||||
'name' => 'Legacy '.$portal,
|
||||
'interval' => 'yearly',
|
||||
],
|
||||
]);
|
||||
}
|
||||
|
||||
// ===== M1: portalübergreifend =====
|
||||
|
||||
test('grandfathered bestandsschutz counts as booking regardless of active portal', function () {
|
||||
/** @var TestCase $this */
|
||||
config()->set('billing.enforce_booking', true);
|
||||
|
||||
$user = User::factory()->create();
|
||||
grandfatheredPost($user, Portal::Presseecho->value);
|
||||
|
||||
// Aktiver Portal-Kontext businessportal24 darf den presseecho-Posten nicht ausschließen.
|
||||
CurrentPortalContext::set(Portal::Businessportal24);
|
||||
expect($user->hasActiveBooking())->toBeTrue();
|
||||
|
||||
CurrentPortalContext::clear();
|
||||
expect($user->fresh()->hasActiveBooking())->toBeTrue();
|
||||
});
|
||||
|
||||
// ===== M2: gekoppelte Kündigung =====
|
||||
|
||||
test('cancelling a bundle cancels all grandfathered posts of the customer together', function () {
|
||||
/** @var TestCase $this */
|
||||
$user = User::factory()->create();
|
||||
$a = grandfatheredPost($user, Portal::Presseecho->value);
|
||||
$b = grandfatheredPost($user, Portal::Businessportal24->value);
|
||||
|
||||
$count = app(LegacySubscriptionService::class)->cancelBundleFor($user);
|
||||
|
||||
expect($count)->toBe(2);
|
||||
expect($a->fresh()->status)->toBe(UserPaymentOptionStatus::Cancelled);
|
||||
expect($b->fresh()->status)->toBe(UserPaymentOptionStatus::Cancelled);
|
||||
expect($a->fresh()->cancelled_at)->not->toBeNull();
|
||||
});
|
||||
|
||||
test('cancelling a bundle leaves other customers untouched', function () {
|
||||
/** @var TestCase $this */
|
||||
$user = User::factory()->create();
|
||||
$other = User::factory()->create();
|
||||
grandfatheredPost($user, Portal::Presseecho->value);
|
||||
$otherPost = grandfatheredPost($other, Portal::Presseecho->value);
|
||||
|
||||
app(LegacySubscriptionService::class)->cancelBundleFor($user);
|
||||
|
||||
expect($otherPost->fresh()->status)->toBe(UserPaymentOptionStatus::Grandfathered);
|
||||
});
|
||||
|
||||
test('a cancelled bundle no longer counts as an active booking', function () {
|
||||
/** @var TestCase $this */
|
||||
config()->set('billing.enforce_booking', true);
|
||||
|
||||
$user = User::factory()->create();
|
||||
grandfatheredPost($user, Portal::Presseecho->value);
|
||||
|
||||
app(LegacySubscriptionService::class)->cancelBundleFor($user);
|
||||
|
||||
expect($user->fresh()->hasActiveBooking())->toBeFalse();
|
||||
});
|
||||
|
||||
// ===== M3: Admin-Sichtbarkeit + Bündel-Kündigung über die UI =====
|
||||
|
||||
test('the payments page lists bestandsschutz posts and cancels a whole bundle', function () {
|
||||
/** @var TestCase $this */
|
||||
$this->seed(RolesAndPermissionsSeeder::class);
|
||||
|
||||
$admin = User::factory()->create(['is_active' => true]);
|
||||
$admin->assignRole('admin');
|
||||
|
||||
$customer = User::factory()->create(['name' => 'Bestandskunde Test', 'is_active' => true]);
|
||||
grandfatheredPost($customer, Portal::Presseecho->value);
|
||||
grandfatheredPost($customer, Portal::Businessportal24->value);
|
||||
|
||||
Volt::actingAs($admin)
|
||||
->test('admin.payments.index')
|
||||
->assertOk()
|
||||
->assertSee('Bestandskunde Test')
|
||||
->assertSee('Bestandsschutz')
|
||||
->call('cancelBundle', $customer->id)
|
||||
->assertHasNoErrors();
|
||||
|
||||
$stillOpen = UserPaymentOption::query()
|
||||
->where('user_id', $customer->id)
|
||||
->where('status', UserPaymentOptionStatus::Grandfathered->value)
|
||||
->count();
|
||||
|
||||
expect($stillOpen)->toBe(0);
|
||||
});
|
||||
Loading…
Add table
Add a link
Reference in a new issue