chore: add payment routes for our UPS

This commit is contained in:
idevakk
2025-11-28 06:03:43 -08:00
parent d4de074161
commit cc34c9aca3
4 changed files with 229 additions and 2 deletions

View File

@@ -254,6 +254,105 @@ class PaymentController extends Controller
}
}
/**
* Handle enhanced checkout with provider selection
*/
public function enhancedCheckout(Request $request, int $plan, string $provider)
{
try {
$user = $request->user();
$planModel = Plan::with(['planProviders', 'trialConfiguration'])->findOrFail($plan);
// Validate provider support
if (! $planModel->supportsProvider($provider)) {
session()->flash('error', "Provider '{$provider}' is not supported for this plan.");
return redirect()->route('dashboard');
}
// Create checkout session via orchestrator
$result = $this->orchestrator->createCheckoutSession($user, $planModel, $provider, [
'success_url' => route('checkout.success'),
'cancel_url' => route('checkout.cancel'),
'is_trial' => false,
]);
// Redirect to provider's checkout page
if (isset($result['redirect_url'])) {
return redirect($result['redirect_url']);
}
// Fallback to session-based checkout
if (isset($result['session_url'])) {
return redirect($result['session_url']);
}
session()->flash('error', 'Unable to create checkout session. Please try again.');
return redirect()->route('dashboard');
} catch (\Exception $e) {
session()->flash('error', 'Checkout error: '.$e->getMessage());
return redirect()->route('dashboard');
}
}
/**
* Handle trial-specific checkout flow
*/
public function trialCheckout(Request $request, int $plan, string $provider)
{
try {
$user = $request->user();
$planModel = Plan::with(['trialConfiguration', 'planProviders'])->findOrFail($plan);
// Validate trial availability
if (! $planModel->hasTrial()) {
session()->flash('error', 'This plan does not offer trials.');
return redirect()->route('dashboard');
}
// Validate provider support
if (! $planModel->supportsProvider($provider)) {
session()->flash('error', "Provider '{$provider}' is not supported for this plan.");
return redirect()->route('dashboard');
}
$trialConfig = $planModel->getTrialConfig();
// Create trial checkout session
$result = $this->orchestrator->createCheckoutSession($user, $planModel, $provider, [
'success_url' => route('checkout.success'),
'cancel_url' => route('checkout.cancel'),
'is_trial' => true,
'trial_duration_days' => $trialConfig?->trial_duration_days ?? 14,
'trial_requires_payment_method' => $trialConfig?->trial_requires_payment_method ?? true,
]);
// Redirect to provider's checkout page
if (isset($result['redirect_url'])) {
return redirect($result['redirect_url']);
}
// Fallback to session-based checkout
if (isset($result['session_url'])) {
return redirect($result['session_url']);
}
session()->flash('error', 'Unable to create trial checkout session. Please try again.');
return redirect()->route('dashboard');
} catch (\Exception $e) {
session()->flash('error', 'Trial checkout error: '.$e->getMessage());
return redirect()->route('dashboard');
}
}
/**
* Handle successful payment redirect
*/

View File

@@ -15,13 +15,19 @@ Route::prefix('payment')->name('payment.')->group(function () {
Route::get('/success', [PaymentController::class, 'success'])->name('success');
Route::get('/cancel', [PaymentController::class, 'cancel'])->name('cancel');
// Payment processing endpoints
// UNIFIED: Payment processing endpoints (new unified payment system)
Route::post('/checkout', [PaymentController::class, 'createCheckout'])->name('checkout');
Route::post('/subscribe', [PaymentController::class, 'createSubscription'])->name('subscribe');
Route::get('/methods', [PaymentController::class, 'getPaymentMethods'])->name('methods');
Route::get('/history', [PaymentController::class, 'getHistory'])->name('history');
});
// UNIFIED: Enhanced checkout routes with provider selection
Route::middleware(['auth', 'verified'])->prefix('checkout')->name('checkout.')->group(function () {
Route::get('/enhanced/{plan}/{provider}', [PaymentController::class, 'enhancedCheckout'])->name('enhanced');
Route::get('/trial/{plan}/{provider}', [PaymentController::class, 'trialCheckout'])->name('trial');
});
Route::prefix('webhook')->name('webhook.')->group(function () {
// Unified webhook handler
Route::post('/{provider}', [WebhookController::class, 'handle'])->name('unified');

View File

@@ -69,7 +69,7 @@ Route::middleware(['auth', 'verified', CheckUserBanned::class])->group(function
Route::get('dashboard/compose-email', Dashboard::class)->name('dashboard.compose');
Route::get('dashboard/support', Support::class)->name('dashboard.support');
// Checkout Routes
// LEGACY: Old Stripe Cashier checkout route (deprecated - use unified payment system)
Route::get('checkout/{plan}', function ($pricing_id) {
$plans = config('app.plans');
$pricingData = [];
@@ -91,6 +91,7 @@ Route::middleware(['auth', 'verified', CheckUserBanned::class])->group(function
abort(404);
})->name('checkout');
// LEGACY: Payment status routes (used by both legacy and unified systems)
Route::get('dashboard/success', [Dashboard::class, 'paymentStatus'])->name('checkout.success')->defaults('status', 'success');
Route::get('dashboard/cancel', [Dashboard::class, 'paymentStatus'])->name('checkout.cancel')->defaults('status', 'cancel');

View File

@@ -0,0 +1,121 @@
<?php
namespace Tests\Feature;
use App\Models\Plan;
use App\Models\User;
use Livewire\Livewire;
use Tests\TestCase;
class PricingCheckoutTest extends TestCase
{
/** @test */
public function it_renders_pricing_component_with_working_checkout_buttons()
{
$user = User::factory()->create();
$plan = Plan::factory()->create([
'name' => 'Test Plan',
'price' => 99.99,
'monthly_billing' => true,
]);
// Enable polar provider for this plan
$plan->planProviders()->create([
'provider' => 'polar',
'is_enabled' => true,
]);
Livewire::actingAs($user)
->test(\App\Livewire\Dashboard\Pricing::class)
->assertSet('plans')
->assertSet('planTiers')
->assertSee($plan->name);
}
/** @test */
public function it_can_redirect_to_enhanced_checkout_for_polar_provider()
{
$user = User::factory()->create();
$plan = Plan::factory()->create([
'name' => 'Test Plan',
'price' => 99.99,
'monthly_billing' => true,
]);
// Enable polar provider for this plan
$plan->planProviders()->create([
'provider' => 'polar',
'is_enabled' => true,
]);
Livewire::actingAs($user)
->test(\App\Livewire\Dashboard\Pricing::class)
->call('choosePlan', $plan->id, 'polar')
->assertRedirect(route('checkout.enhanced', [
'plan' => $plan->id,
'provider' => 'polar',
]));
}
/** @test */
public function it_can_redirect_to_trial_checkout_when_plan_supports_trials()
{
$user = User::factory()->create();
$plan = Plan::factory()->create([
'name' => 'Test Plan with Trial',
'price' => 99.99,
'monthly_billing' => true,
]);
// Enable polar provider for this plan
$plan->planProviders()->create([
'provider' => 'polar',
'is_enabled' => true,
]);
// Create trial configuration
$plan->trialConfiguration()->create([
'trial_duration_days' => 14,
'trial_requires_payment_method' => true,
'trial_auto_converts' => true,
]);
Livewire::actingAs($user)
->test(\App\Livewire\Dashboard\Pricing::class)
->call('startTrial', $plan->id, 'polar')
->assertRedirect(route('checkout.trial', [
'plan' => $plan->id,
'provider' => 'polar',
]));
}
/** @test */
public function it_shows_error_when_provider_is_not_supported()
{
$user = User::factory()->create();
$plan = Plan::factory()->create();
Livewire::actingAs($user)
->test(\App\Livewire\Dashboard\Pricing::class)
->call('choosePlan', $plan->id, 'unsupported_provider')
->assertSessionHas('error', "This plan doesn't support unsupported_provider payments.");
}
/** @test */
public function it_shows_error_when_trying_trial_on_plan_without_trial()
{
$user = User::factory()->create();
$plan = Plan::factory()->create();
// Enable provider but no trial configuration
$plan->planProviders()->create([
'provider' => 'polar',
'is_enabled' => true,
]);
Livewire::actingAs($user)
->test(\App\Livewire\Dashboard\Pricing::class)
->call('startTrial', $plan->id, 'polar')
->assertSessionHas('error', "This plan doesn't offer trials.");
}
}