Migrate billing from Paddle to Lemon Squeezy
This commit is contained in:
@@ -4,10 +4,9 @@ namespace App\Http\Controllers\Api;
|
||||
|
||||
use App\Http\Controllers\Controller;
|
||||
use App\Models\EventPackageAddon;
|
||||
use App\Services\Paddle\Exceptions\PaddleException;
|
||||
use App\Services\Paddle\PaddleCustomerPortalService;
|
||||
use App\Services\Paddle\PaddleCustomerService;
|
||||
use App\Services\Paddle\PaddleTransactionService;
|
||||
use App\Services\LemonSqueezy\Exceptions\LemonSqueezyException;
|
||||
use App\Services\LemonSqueezy\LemonSqueezyOrderService;
|
||||
use App\Services\LemonSqueezy\LemonSqueezySubscriptionService;
|
||||
use Illuminate\Http\JsonResponse;
|
||||
use Illuminate\Http\Request;
|
||||
use Illuminate\Support\Arr;
|
||||
@@ -16,9 +15,8 @@ use Illuminate\Support\Facades\Log;
|
||||
class TenantBillingController extends Controller
|
||||
{
|
||||
public function __construct(
|
||||
private readonly PaddleTransactionService $paddleTransactions,
|
||||
private readonly PaddleCustomerService $paddleCustomers,
|
||||
private readonly PaddleCustomerPortalService $portalSessions,
|
||||
private readonly LemonSqueezyOrderService $orders,
|
||||
private readonly LemonSqueezySubscriptionService $subscriptions,
|
||||
) {}
|
||||
|
||||
public function transactions(Request $request): JsonResponse
|
||||
@@ -32,20 +30,15 @@ class TenantBillingController extends Controller
|
||||
], 404);
|
||||
}
|
||||
|
||||
if (! $tenant->paddle_customer_id) {
|
||||
try {
|
||||
$this->paddleCustomers->ensureCustomerId($tenant);
|
||||
} catch (\Throwable $exception) {
|
||||
Log::warning('Failed to resolve Paddle customer for tenant', [
|
||||
'tenant_id' => $tenant->id,
|
||||
'error' => $exception->getMessage(),
|
||||
]);
|
||||
|
||||
return response()->json([
|
||||
'data' => [],
|
||||
'message' => 'Failed to resolve Paddle customer.',
|
||||
], 502);
|
||||
}
|
||||
if (! $tenant->lemonsqueezy_customer_id) {
|
||||
return response()->json([
|
||||
'data' => [],
|
||||
'meta' => [
|
||||
'next' => null,
|
||||
'previous' => null,
|
||||
'has_more' => false,
|
||||
],
|
||||
]);
|
||||
}
|
||||
|
||||
$cursor = $request->query('cursor');
|
||||
@@ -60,16 +53,16 @@ class TenantBillingController extends Controller
|
||||
}
|
||||
|
||||
try {
|
||||
$result = $this->paddleTransactions->listForCustomer($tenant->paddle_customer_id, $query);
|
||||
$result = $this->orders->listForCustomer($tenant->lemonsqueezy_customer_id, $query);
|
||||
} catch (\Throwable $exception) {
|
||||
Log::warning('Failed to load Paddle transactions', [
|
||||
Log::warning('Failed to load Lemon Squeezy transactions', [
|
||||
'tenant_id' => $tenant->id,
|
||||
'error' => $exception->getMessage(),
|
||||
]);
|
||||
|
||||
return response()->json([
|
||||
'data' => [],
|
||||
'message' => 'Failed to load Paddle transactions.',
|
||||
'message' => 'Failed to load Lemon Squeezy transactions.',
|
||||
], 502);
|
||||
}
|
||||
|
||||
@@ -143,68 +136,64 @@ class TenantBillingController extends Controller
|
||||
], 404);
|
||||
}
|
||||
|
||||
$customerId = null;
|
||||
$subscriptionId = null;
|
||||
|
||||
try {
|
||||
$customerId = $this->paddleCustomers->ensureCustomerId($tenant);
|
||||
$subscriptionId = $tenant->getActiveResellerPackage()?->lemonsqueezy_subscription_id;
|
||||
if (! $subscriptionId) {
|
||||
return response()->json([
|
||||
'message' => 'No active subscription found.',
|
||||
], 404);
|
||||
}
|
||||
|
||||
Log::debug('Creating Paddle customer portal session', [
|
||||
Log::debug('Fetching Lemon Squeezy subscription portal URL', [
|
||||
'tenant_id' => $tenant->id,
|
||||
'paddle_customer_id' => $customerId,
|
||||
'paddle_environment' => config('paddle.environment'),
|
||||
'paddle_base_url' => config('paddle.base_url'),
|
||||
'lemonsqueezy_subscription_id' => $subscriptionId,
|
||||
]);
|
||||
|
||||
$session = $this->portalSessions->createSession($customerId);
|
||||
$subscription = $this->subscriptions->retrieve($subscriptionId);
|
||||
} catch (\Throwable $exception) {
|
||||
$context = [
|
||||
'tenant_id' => $tenant->id,
|
||||
'paddle_customer_id' => $customerId ?? $tenant->paddle_customer_id,
|
||||
'lemonsqueezy_customer_id' => $tenant->lemonsqueezy_customer_id,
|
||||
'lemonsqueezy_subscription_id' => $subscriptionId ?? null,
|
||||
'error' => $exception->getMessage(),
|
||||
'paddle_environment' => config('paddle.environment'),
|
||||
'paddle_base_url' => config('paddle.base_url'),
|
||||
];
|
||||
|
||||
if ($exception instanceof PaddleException) {
|
||||
$context['paddle_status'] = $exception->status();
|
||||
$context['paddle_error_code'] = Arr::get($exception->context(), 'error.code');
|
||||
$context['paddle_error_message'] = Arr::get($exception->context(), 'error.message');
|
||||
$context['paddle_error_detail'] = Arr::get($exception->context(), 'error.detail');
|
||||
$context['paddle_error_doc_url'] = Arr::get($exception->context(), 'error.documentation_url');
|
||||
$context['paddle_request_id'] = Arr::get($exception->context(), 'meta.request_id');
|
||||
$context['paddle_errors'] = Arr::get($exception->context(), 'error.errors');
|
||||
if ($exception instanceof LemonSqueezyException) {
|
||||
$context['lemonsqueezy_status'] = $exception->status();
|
||||
$context['lemonsqueezy_error'] = Arr::get($exception->context(), 'errors.0');
|
||||
$context['lemonsqueezy_errors'] = Arr::get($exception->context(), 'errors');
|
||||
$context['lemonsqueezy_request_id'] = Arr::get($exception->context(), 'meta.request_id');
|
||||
}
|
||||
|
||||
Log::warning('Failed to create Paddle customer portal session', [
|
||||
Log::warning('Failed to fetch Lemon Squeezy subscription portal URL', [
|
||||
...$context,
|
||||
]);
|
||||
|
||||
return response()->json([
|
||||
'message' => 'Failed to create Paddle customer portal session.',
|
||||
'message' => 'Failed to fetch Lemon Squeezy subscription portal URL.',
|
||||
], 502);
|
||||
}
|
||||
|
||||
$url = Arr::get($session, 'data.urls.general.overview')
|
||||
?? Arr::get($session, 'data.urls.general')
|
||||
?? Arr::get($session, 'urls.general.overview')
|
||||
?? Arr::get($session, 'urls.general');
|
||||
$url = $this->subscriptions->portalUrl($subscription)
|
||||
?? $this->subscriptions->updatePaymentMethodUrl($subscription);
|
||||
|
||||
if (! $url) {
|
||||
$sessionData = Arr::get($session, 'data');
|
||||
$sessionUrls = Arr::get($session, 'data.urls') ?? Arr::get($session, 'urls');
|
||||
$sessionData = Arr::get($subscription, 'data');
|
||||
$sessionUrls = Arr::get($subscription, 'attributes.urls');
|
||||
|
||||
Log::warning('Paddle customer portal session missing URL', [
|
||||
Log::warning('Lemon Squeezy subscription missing portal URL', [
|
||||
'tenant_id' => $tenant->id,
|
||||
'paddle_customer_id' => $customerId ?? $tenant->paddle_customer_id,
|
||||
'paddle_environment' => config('paddle.environment'),
|
||||
'paddle_base_url' => config('paddle.base_url'),
|
||||
'session_keys' => array_keys($session),
|
||||
'lemonsqueezy_customer_id' => $tenant->lemonsqueezy_customer_id,
|
||||
'lemonsqueezy_subscription_id' => $subscriptionId ?? null,
|
||||
'subscription_keys' => array_keys($subscription),
|
||||
'session_data_keys' => is_array($sessionData) ? array_keys($sessionData) : null,
|
||||
'session_url_keys' => is_array($sessionUrls) ? array_keys($sessionUrls) : null,
|
||||
]);
|
||||
|
||||
return response()->json([
|
||||
'message' => 'Paddle customer portal session missing URL.',
|
||||
'message' => 'Lemon Squeezy subscription missing portal URL.',
|
||||
], 502);
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user