auf success_url/cancel_url weiter. Gleichzeitig werden diese URLs jetzt in custom_data bei Add‑ons, Package‑Checkout und Gift‑Vouchern gespeichert, damit der Return‑Handler sie kennt. Details (relevant): - app/Http/Controllers/PaddleReturnController.php verarbeitet _ptxn, prüft Status, schützt vor Open‑Redirects. - routes/web.php neue Route paddle.return (öffentlich). - app/Services/Addons/EventAddonCheckoutService.php, app/Services/Paddle/PaddleCheckoutService.php, app/Services/ GiftVouchers/GiftVoucherCheckoutService.php speichern success_url/cancel_url in custom_data. - tests/Feature/PaddleReturnTest.php prüft Success/Cancel‑Redirects. - Tests aktualisiert: tests/Unit/PaddleCheckoutServiceTest.php. Wichtig für die Rückleitung: - Bitte in Paddle (Sandbox + Live) die Checkout‑Success/Cancel URL auf http://fotospiel-app.test/paddle/return setzen. Ohne diese Einstellung schickt Paddle den Nutzer nicht zu unserem Return‑Handler. Nebenwirkung: Add‑on‑Checkout gibt jetzt als checkout_id die Transaktions‑ID (txn_…) zurück (statt chk_…).
68 lines
2.6 KiB
PHP
68 lines
2.6 KiB
PHP
<?php
|
|
|
|
namespace Tests\Unit;
|
|
|
|
use App\Models\Package;
|
|
use App\Models\Tenant;
|
|
use App\Services\Paddle\PaddleCheckoutService;
|
|
use App\Services\Paddle\PaddleClient;
|
|
use App\Services\Paddle\PaddleCustomerService;
|
|
use Illuminate\Foundation\Testing\RefreshDatabase;
|
|
use Mockery;
|
|
use Tests\TestCase;
|
|
|
|
class PaddleCheckoutServiceTest extends TestCase
|
|
{
|
|
use RefreshDatabase;
|
|
|
|
public function test_create_checkout_sends_custom_data_payload(): void
|
|
{
|
|
$tenant = Tenant::factory()->create([
|
|
'contact_email' => 'buyer@example.com',
|
|
]);
|
|
|
|
$package = Package::factory()->create([
|
|
'paddle_price_id' => 'pri_123',
|
|
]);
|
|
|
|
$client = Mockery::mock(PaddleClient::class);
|
|
$customers = Mockery::mock(PaddleCustomerService::class);
|
|
|
|
$customers->shouldReceive('ensureCustomerId')
|
|
->once()
|
|
->with($tenant)
|
|
->andReturn('ctm_123');
|
|
|
|
$client->shouldReceive('post')
|
|
->once()
|
|
->with('/transactions', Mockery::on(function (array $payload) use ($tenant, $package) {
|
|
return $payload['items'][0]['price_id'] === 'pri_123'
|
|
&& $payload['customer_id'] === 'ctm_123'
|
|
&& ($payload['custom_data']['tenant_id'] ?? null) === (string) $tenant->id
|
|
&& ($payload['custom_data']['package_id'] ?? null) === (string) $package->id
|
|
&& ($payload['custom_data']['source'] ?? null) === 'test'
|
|
&& ($payload['custom_data']['success_url'] ?? null) === 'https://example.test/success'
|
|
&& ($payload['custom_data']['cancel_url'] ?? null) === 'https://example.test/cancel'
|
|
&& ! isset($payload['metadata'])
|
|
&& ! isset($payload['success_url'])
|
|
&& ! isset($payload['cancel_url'])
|
|
&& ! isset($payload['customer_email']);
|
|
}))
|
|
->andReturn(['data' => ['checkout' => ['url' => 'https://paddle.test/checkout/123'], 'id' => 'txn_123']]);
|
|
|
|
$this->app->instance(PaddleClient::class, $client);
|
|
$this->app->instance(PaddleCustomerService::class, $customers);
|
|
|
|
$service = $this->app->make(PaddleCheckoutService::class);
|
|
|
|
$checkout = $service->createCheckout($tenant, $package, [
|
|
'success_url' => 'https://example.test/success',
|
|
'return_url' => 'https://example.test/cancel',
|
|
'metadata' => ['source' => 'test'],
|
|
]);
|
|
|
|
$this->assertSame('https://paddle.test/checkout/123', $checkout['checkout_url']);
|
|
$this->assertSame('txn_123', $checkout['id']);
|
|
}
|
|
}
|