161 lines
5.7 KiB
PHP
161 lines
5.7 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Tests\Feature\Academy;
|
|
|
|
use App\Http\Middleware\ConditionalValidateCsrfToken;
|
|
use App\Models\AcademyLesson;
|
|
use App\Models\AcademyPromptTemplate;
|
|
use App\Models\User;
|
|
use Illuminate\Foundation\Testing\RefreshDatabase;
|
|
use Inertia\Testing\AssertableInertia;
|
|
use Laravel\Cashier\Subscription;
|
|
use Tests\TestCase;
|
|
|
|
final class AcademyBillingAccessTest extends TestCase
|
|
{
|
|
use RefreshDatabase;
|
|
|
|
protected function setUp(): void
|
|
{
|
|
parent::setUp();
|
|
|
|
$this->withoutMiddleware(ConditionalValidateCsrfToken::class);
|
|
$this->configureBilling();
|
|
}
|
|
|
|
public function test_success_page_does_not_grant_access_by_itself(): void
|
|
{
|
|
$prompt = AcademyPromptTemplate::query()->create([
|
|
'title' => 'Creator Prompt',
|
|
'slug' => 'billing-success-does-not-unlock',
|
|
'excerpt' => 'Locked creator prompt.',
|
|
'prompt' => 'SECRET CREATOR PROMPT',
|
|
'difficulty' => 'beginner',
|
|
'access_level' => 'creator',
|
|
'active' => true,
|
|
'published_at' => now()->subMinute(),
|
|
]);
|
|
|
|
$user = User::factory()->create(['email_verified_at' => now()]);
|
|
|
|
$this->actingAs($user)
|
|
->get(route('academy.billing.success', ['session_id' => 'cs_test_only']))
|
|
->assertOk()
|
|
->assertInertia(fn (AssertableInertia $page) => $page
|
|
->where('currentTier', 'free')
|
|
->where('isSubscribed', false));
|
|
|
|
$this->actingAs($user)
|
|
->get(route('academy.prompts.show', ['slug' => $prompt->slug]))
|
|
->assertOk()
|
|
->assertDontSee('SECRET CREATOR PROMPT')
|
|
->assertInertia(fn (AssertableInertia $page) => $page
|
|
->where('item.locked', true)
|
|
->where('item.prompt', null));
|
|
}
|
|
|
|
public function test_canceled_subscription_on_grace_period_still_has_access(): void
|
|
{
|
|
$lesson = AcademyLesson::query()->create([
|
|
'title' => 'Creator Grace Lesson',
|
|
'slug' => 'creator-grace-lesson',
|
|
'excerpt' => 'Should remain available in grace period.',
|
|
'content' => 'VISIBLE DURING GRACE PERIOD',
|
|
'difficulty' => 'beginner',
|
|
'access_level' => 'creator',
|
|
'lesson_type' => 'article',
|
|
'active' => true,
|
|
'published_at' => now()->subMinute(),
|
|
]);
|
|
|
|
$user = User::factory()->create(['email_verified_at' => now()]);
|
|
$this->attachSubscription($user, 'sub_grace', 'price_creator_month', now()->addDay());
|
|
|
|
$this->actingAs($user)
|
|
->get(route('academy.lessons.show', ['slug' => $lesson->slug]))
|
|
->assertOk()
|
|
->assertSee('VISIBLE DURING GRACE PERIOD')
|
|
->assertInertia(fn (AssertableInertia $page) => $page
|
|
->where('item.locked', false)
|
|
->where('item.content', 'VISIBLE DURING GRACE PERIOD'));
|
|
}
|
|
|
|
public function test_ended_subscription_loses_paid_access(): void
|
|
{
|
|
$lesson = AcademyLesson::query()->create([
|
|
'title' => 'Creator Ended Lesson',
|
|
'slug' => 'creator-ended-lesson',
|
|
'excerpt' => 'Should lock after grace period.',
|
|
'content' => 'NO LONGER VISIBLE',
|
|
'difficulty' => 'beginner',
|
|
'access_level' => 'creator',
|
|
'lesson_type' => 'article',
|
|
'active' => true,
|
|
'published_at' => now()->subMinute(),
|
|
]);
|
|
|
|
$user = User::factory()->create(['email_verified_at' => now()]);
|
|
$this->attachSubscription($user, 'sub_ended', 'price_creator_month', now()->subMinute());
|
|
|
|
$this->actingAs($user)
|
|
->get(route('academy.lessons.show', ['slug' => $lesson->slug]))
|
|
->assertOk()
|
|
->assertDontSee('NO LONGER VISIBLE')
|
|
->assertInertia(fn (AssertableInertia $page) => $page
|
|
->where('item.locked', true)
|
|
->where('item.content', null));
|
|
}
|
|
|
|
public function test_billing_portal_route_requires_authentication(): void
|
|
{
|
|
$this->get(route('academy.billing.portal'))
|
|
->assertRedirect(route('login'));
|
|
}
|
|
|
|
private function attachSubscription(User $user, string $subscriptionId, string $priceId, ?\Illuminate\Support\Carbon $endsAt = null): Subscription
|
|
{
|
|
$subscription = $user->subscriptions()->create([
|
|
'type' => 'academy',
|
|
'stripe_id' => $subscriptionId,
|
|
'stripe_status' => 'active',
|
|
'stripe_price' => $priceId,
|
|
'quantity' => 1,
|
|
'ends_at' => $endsAt,
|
|
]);
|
|
|
|
$subscription->items()->create([
|
|
'stripe_id' => 'si_'.$subscriptionId,
|
|
'stripe_product' => 'prod_'.($priceId === 'price_pro_month' ? 'pro' : 'creator'),
|
|
'stripe_price' => $priceId,
|
|
'quantity' => 1,
|
|
]);
|
|
|
|
return $subscription;
|
|
}
|
|
|
|
private function configureBilling(): void
|
|
{
|
|
config()->set('academy.enabled', true);
|
|
config()->set('academy.payments_enabled', true);
|
|
config()->set('academy_billing.enabled', true);
|
|
config()->set('academy_billing.subscription_name', 'academy');
|
|
config()->set('academy_billing.plans', [
|
|
'creator_monthly' => [
|
|
'label' => 'Creator Monthly',
|
|
'tier' => 'creator',
|
|
'interval' => 'monthly',
|
|
'stripe_price_id' => 'price_creator_month',
|
|
'featured' => false,
|
|
],
|
|
'pro_monthly' => [
|
|
'label' => 'Pro Monthly',
|
|
'tier' => 'pro',
|
|
'interval' => 'monthly',
|
|
'stripe_price_id' => 'price_pro_month',
|
|
'featured' => true,
|
|
],
|
|
]);
|
|
}
|
|
} |