Proof-PDF + Extra-PM-Verkauf ueber die Credit-Wallet (Decision-Update 2.1/2.3)
- ProofPdfService: Veroeffentlichungsnachweis 3 Credits pauschal, einmal pro
PM (Zweitdownload kostenfrei); ProofPdfRenderer erzeugt das PDF on-demand
aus vorhandenen PM-Daten (kein externer Renderer); GET-Download-Endpoint
/admin/me/press-releases/{id}/nachweis hinter downloadProof-Policy + Kauf-Gate
- ExtraPmPurchaseService: tier-gestaffelter Nachkauf (19/15/12/10/8) aus der
Wallet; verbucht als bezahlter SinglePurchase(ExtraPm) und greift damit in
die bestehende Kontingent-/Slot-Mechanik. InsufficientCreditsException
liefert das Mini-Checkout-Signal (required/available/shortfall)
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
parent
c7a4c8bfd4
commit
69411b4c87
8 changed files with 609 additions and 0 deletions
159
tests/Feature/ProofPdfAndExtraPmTest.php
Normal file
159
tests/Feature/ProofPdfAndExtraPmTest.php
Normal file
|
|
@ -0,0 +1,159 @@
|
|||
<?php
|
||||
|
||||
use App\Enums\PressReleaseStatus;
|
||||
use App\Enums\SinglePurchaseStatus;
|
||||
use App\Enums\SinglePurchaseType;
|
||||
use App\Exceptions\InsufficientCreditsException;
|
||||
use App\Exceptions\ProofPdfNotAvailableException;
|
||||
use App\Models\Plan;
|
||||
use App\Models\PressRelease;
|
||||
use App\Models\User;
|
||||
use App\Services\Billing\CreditWalletService;
|
||||
use App\Services\Billing\ExtraPmPurchaseService;
|
||||
use App\Services\Billing\ProofPdfService;
|
||||
use App\Services\PressRelease\ProofPdfRenderer;
|
||||
use Database\Seeders\RolesAndPermissionsSeeder;
|
||||
|
||||
beforeEach(function (): void {
|
||||
$this->seed(RolesAndPermissionsSeeder::class);
|
||||
$this->wallet = app(CreditWalletService::class);
|
||||
$this->proof = app(ProofPdfService::class);
|
||||
$this->extraPm = app(ExtraPmPurchaseService::class);
|
||||
});
|
||||
|
||||
function customerWithRole(): User
|
||||
{
|
||||
$user = User::factory()->create(['is_active' => true, 'email_verified_at' => now()]);
|
||||
$user->assignRole('customer');
|
||||
|
||||
return $user;
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------- Proof-PDF
|
||||
|
||||
test('buying a proof debits three credits and records a paid purchase', function () {
|
||||
$user = customerWithRole();
|
||||
$this->wallet->credit($user, 10);
|
||||
$pr = PressRelease::factory()->published()->create(['user_id' => $user->id]);
|
||||
|
||||
$purchase = $this->proof->purchase($user, $pr);
|
||||
|
||||
expect($purchase->type)->toBe(SinglePurchaseType::ProofPdf);
|
||||
expect($purchase->status)->toBe(SinglePurchaseStatus::Paid);
|
||||
expect($this->wallet->balance($user))->toBe(7);
|
||||
expect($this->proof->hasPurchased($user, $pr))->toBeTrue();
|
||||
});
|
||||
|
||||
test('re-buying the same proof does not charge again', function () {
|
||||
$user = customerWithRole();
|
||||
$this->wallet->credit($user, 10);
|
||||
$pr = PressRelease::factory()->published()->create(['user_id' => $user->id]);
|
||||
|
||||
$first = $this->proof->purchase($user, $pr);
|
||||
$second = $this->proof->purchase($user, $pr);
|
||||
|
||||
expect($second->id)->toBe($first->id);
|
||||
expect($this->wallet->balance($user))->toBe(7);
|
||||
expect($user->singlePurchases()->where('type', SinglePurchaseType::ProofPdf->value)->count())->toBe(1);
|
||||
});
|
||||
|
||||
test('a proof for an unpublished release is rejected', function () {
|
||||
$user = customerWithRole();
|
||||
$this->wallet->credit($user, 10);
|
||||
$draft = PressRelease::factory()->create(['user_id' => $user->id, 'status' => PressReleaseStatus::Draft->value]);
|
||||
|
||||
expect(fn () => $this->proof->purchase($user, $draft))
|
||||
->toThrow(ProofPdfNotAvailableException::class);
|
||||
expect($this->wallet->balance($user))->toBe(10);
|
||||
});
|
||||
|
||||
test('a proof without enough credits throws and records nothing', function () {
|
||||
$user = customerWithRole();
|
||||
$this->wallet->credit($user, 2);
|
||||
$pr = PressRelease::factory()->published()->create(['user_id' => $user->id]);
|
||||
|
||||
expect(fn () => $this->proof->purchase($user, $pr))
|
||||
->toThrow(InsufficientCreditsException::class);
|
||||
expect($user->singlePurchases()->count())->toBe(0);
|
||||
});
|
||||
|
||||
test('the rendered proof is a non-trivial pdf document', function () {
|
||||
$pr = PressRelease::factory()->published()->create(['title' => 'Testmeldung', 'text' => str_repeat('Inhalt. ', 50)]);
|
||||
|
||||
$pdf = app(ProofPdfRenderer::class)->render($pr);
|
||||
|
||||
expect($pdf)->toStartWith('%PDF-1.4');
|
||||
expect($pdf)->toContain('%%EOF');
|
||||
expect(strlen($pdf))->toBeGreaterThan(800);
|
||||
});
|
||||
|
||||
test('the proof download requires a prior purchase', function () {
|
||||
$user = customerWithRole();
|
||||
$pr = PressRelease::factory()->published()->create(['user_id' => $user->id]);
|
||||
|
||||
// Ohne Kauf: 403.
|
||||
$this->actingAs($user)
|
||||
->get(route('me.press-releases.proof', $pr->id))
|
||||
->assertForbidden();
|
||||
|
||||
// Nach Kauf: PDF-Download.
|
||||
$this->wallet->credit($user, 5);
|
||||
$this->proof->purchase($user, $pr);
|
||||
|
||||
$this->actingAs($user)
|
||||
->get(route('me.press-releases.proof', $pr->id))
|
||||
->assertOk()
|
||||
->assertHeader('content-type', 'application/pdf');
|
||||
});
|
||||
|
||||
// ---------------------------------------------------------------- Extra-PM
|
||||
|
||||
test('an Einzel user pays the full extra-pm rate from the wallet', function () {
|
||||
$user = customerWithRole();
|
||||
$this->wallet->credit($user, 25);
|
||||
|
||||
$purchase = $this->extraPm->purchase($user);
|
||||
|
||||
expect($purchase->type)->toBe(SinglePurchaseType::ExtraPm);
|
||||
expect($purchase->status)->toBe(SinglePurchaseStatus::Paid);
|
||||
expect($purchase->price_cents)->toBe(1900);
|
||||
expect($this->wallet->balance($user))->toBe(6);
|
||||
});
|
||||
|
||||
test('an extra-pm purchase extends the press release quota', function () {
|
||||
config()->set('billing.enforce_booking', true);
|
||||
|
||||
$user = customerWithRole();
|
||||
$plan = Plan::factory()->create([
|
||||
'slug' => 'business',
|
||||
'press_release_quota' => 3,
|
||||
'stripe_price_id_monthly' => 'price_biz_'.fake()->unique()->randomNumber(6),
|
||||
]);
|
||||
subscribeUserToPlan($user, $plan);
|
||||
$user->update(['press_release_quota_used_this_month' => 3]); // Kontingent voll
|
||||
$this->wallet->credit($user, 20);
|
||||
|
||||
expect($user->pressReleaseQuotaRemaining())->toBe(0);
|
||||
|
||||
// Business-Tier zahlt 12 Credits für die Extra-PM.
|
||||
$this->extraPm->purchase($user);
|
||||
|
||||
expect($this->wallet->balance($user))->toBe(8);
|
||||
expect($user->fresh()->pressReleaseQuotaRemaining())->toBe(1);
|
||||
});
|
||||
|
||||
test('an extra-pm without enough credits throws the mini-checkout signal', function () {
|
||||
$user = customerWithRole();
|
||||
$this->wallet->credit($user, 8);
|
||||
|
||||
try {
|
||||
$this->extraPm->purchase($user);
|
||||
$this->fail('Expected InsufficientCreditsException.');
|
||||
} catch (InsufficientCreditsException $e) {
|
||||
expect($e->required)->toBe(19);
|
||||
expect($e->available)->toBe(8);
|
||||
expect($e->shortfall())->toBe(11);
|
||||
}
|
||||
|
||||
expect($user->singlePurchases()->count())->toBe(0);
|
||||
});
|
||||
Loading…
Add table
Add a link
Reference in a new issue