Migrate billing from Paddle to Lemon Squeezy
This commit is contained in:
@@ -3,7 +3,7 @@
|
||||
namespace App\Jobs;
|
||||
|
||||
use App\Models\Package;
|
||||
use App\Services\Paddle\PaddleCatalogService;
|
||||
use App\Services\LemonSqueezy\LemonSqueezyCatalogService;
|
||||
use Illuminate\Bus\Queueable;
|
||||
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||
use Illuminate\Foundation\Bus\Dispatchable;
|
||||
@@ -13,7 +13,7 @@ use Illuminate\Support\Arr;
|
||||
use Illuminate\Support\Facades\Log;
|
||||
use Throwable;
|
||||
|
||||
class PullPackageFromPaddle implements ShouldQueue
|
||||
class PullPackageFromLemonSqueezy implements ShouldQueue
|
||||
{
|
||||
use Dispatchable;
|
||||
use InteractsWithQueue;
|
||||
@@ -22,7 +22,7 @@ class PullPackageFromPaddle implements ShouldQueue
|
||||
|
||||
public function __construct(private readonly int $packageId) {}
|
||||
|
||||
public function handle(PaddleCatalogService $catalog): void
|
||||
public function handle(LemonSqueezyCatalogService $catalog): void
|
||||
{
|
||||
$package = Package::query()->find($this->packageId);
|
||||
|
||||
@@ -30,8 +30,8 @@ class PullPackageFromPaddle implements ShouldQueue
|
||||
return;
|
||||
}
|
||||
|
||||
if (! $package->paddle_product_id && ! $package->paddle_price_id) {
|
||||
Log::channel('paddle-sync')->warning('Paddle pull skipped for package without linkage', [
|
||||
if (! $package->lemonsqueezy_product_id && ! $package->lemonsqueezy_variant_id) {
|
||||
Log::channel('lemonsqueezy-sync')->warning('Lemon Squeezy pull skipped for package without linkage', [
|
||||
'package_id' => $package->id,
|
||||
]);
|
||||
|
||||
@@ -39,41 +39,41 @@ class PullPackageFromPaddle implements ShouldQueue
|
||||
}
|
||||
|
||||
try {
|
||||
$product = $package->paddle_product_id ? $catalog->fetchProduct($package->paddle_product_id) : null;
|
||||
$price = $package->paddle_price_id ? $catalog->fetchPrice($package->paddle_price_id) : null;
|
||||
$product = $package->lemonsqueezy_product_id ? $catalog->fetchProduct($package->lemonsqueezy_product_id) : null;
|
||||
$price = $package->lemonsqueezy_variant_id ? $catalog->fetchPrice($package->lemonsqueezy_variant_id) : null;
|
||||
|
||||
$snapshot = $package->paddle_snapshot ?? [];
|
||||
$snapshot = $package->lemonsqueezy_snapshot ?? [];
|
||||
$snapshot['remote'] = array_filter([
|
||||
'product' => $product,
|
||||
'price' => $price,
|
||||
], static fn ($value) => $value !== null);
|
||||
|
||||
$package->forceFill([
|
||||
'paddle_sync_status' => 'pulled',
|
||||
'paddle_synced_at' => now(),
|
||||
'paddle_snapshot' => $snapshot,
|
||||
'lemonsqueezy_sync_status' => 'pulled',
|
||||
'lemonsqueezy_synced_at' => now(),
|
||||
'lemonsqueezy_snapshot' => $snapshot,
|
||||
])->save();
|
||||
|
||||
Log::channel('paddle-sync')->info('Paddle package pull completed', [
|
||||
Log::channel('lemonsqueezy-sync')->info('Lemon Squeezy package pull completed', [
|
||||
'package_id' => $package->id,
|
||||
]);
|
||||
} catch (Throwable $exception) {
|
||||
Log::channel('paddle-sync')->error('Paddle package pull failed', [
|
||||
Log::channel('lemonsqueezy-sync')->error('Lemon Squeezy package pull failed', [
|
||||
'package_id' => $package->id,
|
||||
'message' => $exception->getMessage(),
|
||||
'exception' => $exception,
|
||||
]);
|
||||
|
||||
$snapshot = $package->paddle_snapshot ?? [];
|
||||
$snapshot = $package->lemonsqueezy_snapshot ?? [];
|
||||
$snapshot['error'] = array_merge(Arr::get($snapshot, 'error', []), [
|
||||
'message' => $exception->getMessage(),
|
||||
'class' => $exception::class,
|
||||
]);
|
||||
|
||||
$package->forceFill([
|
||||
'paddle_sync_status' => 'pull-failed',
|
||||
'paddle_synced_at' => now(),
|
||||
'paddle_snapshot' => $snapshot,
|
||||
'lemonsqueezy_sync_status' => 'pull-failed',
|
||||
'lemonsqueezy_synced_at' => now(),
|
||||
'lemonsqueezy_snapshot' => $snapshot,
|
||||
])->save();
|
||||
|
||||
throw $exception;
|
||||
@@ -3,8 +3,8 @@
|
||||
namespace App\Jobs;
|
||||
|
||||
use App\Models\Coupon;
|
||||
use App\Services\Paddle\Exceptions\PaddleException;
|
||||
use App\Services\Paddle\PaddleDiscountService;
|
||||
use App\Services\LemonSqueezy\Exceptions\LemonSqueezyException;
|
||||
use App\Services\LemonSqueezy\LemonSqueezyDiscountService;
|
||||
use Illuminate\Bus\Queueable;
|
||||
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||
use Illuminate\Foundation\Bus\Dispatchable;
|
||||
@@ -12,7 +12,7 @@ use Illuminate\Queue\InteractsWithQueue;
|
||||
use Illuminate\Queue\SerializesModels;
|
||||
use Illuminate\Support\Facades\Log;
|
||||
|
||||
class SyncCouponToPaddle implements ShouldQueue
|
||||
class SyncCouponToLemonSqueezy implements ShouldQueue
|
||||
{
|
||||
use Dispatchable;
|
||||
use InteractsWithQueue;
|
||||
@@ -24,16 +24,16 @@ class SyncCouponToPaddle implements ShouldQueue
|
||||
public bool $archive = false,
|
||||
) {}
|
||||
|
||||
public function handle(PaddleDiscountService $discounts): void
|
||||
public function handle(LemonSqueezyDiscountService $discounts): void
|
||||
{
|
||||
try {
|
||||
if ($this->archive) {
|
||||
$discounts->archiveDiscount($this->coupon);
|
||||
|
||||
$this->coupon->forceFill([
|
||||
'paddle_discount_id' => null,
|
||||
'paddle_snapshot' => null,
|
||||
'paddle_last_synced_at' => now(),
|
||||
'lemonsqueezy_discount_id' => null,
|
||||
'lemonsqueezy_snapshot' => null,
|
||||
'lemonsqueezy_last_synced_at' => now(),
|
||||
])->save();
|
||||
|
||||
return;
|
||||
@@ -42,12 +42,12 @@ class SyncCouponToPaddle implements ShouldQueue
|
||||
$data = $discounts->updateDiscount($this->coupon);
|
||||
|
||||
$this->coupon->forceFill([
|
||||
'paddle_discount_id' => $data['id'] ?? $this->coupon->paddle_discount_id,
|
||||
'paddle_snapshot' => $data,
|
||||
'paddle_last_synced_at' => now(),
|
||||
'lemonsqueezy_discount_id' => $data['id'] ?? $this->coupon->lemonsqueezy_discount_id,
|
||||
'lemonsqueezy_snapshot' => $data,
|
||||
'lemonsqueezy_last_synced_at' => now(),
|
||||
])->save();
|
||||
} catch (PaddleException $exception) {
|
||||
Log::channel('paddle-sync')->error('Failed syncing coupon to Paddle', [
|
||||
} catch (LemonSqueezyException $exception) {
|
||||
Log::channel('lemonsqueezy-sync')->error('Failed syncing coupon to Lemon Squeezy', [
|
||||
'coupon_id' => $this->coupon->id,
|
||||
'message' => $exception->getMessage(),
|
||||
'status' => $exception->status(),
|
||||
@@ -55,7 +55,7 @@ class SyncCouponToPaddle implements ShouldQueue
|
||||
]);
|
||||
|
||||
$this->coupon->forceFill([
|
||||
'paddle_snapshot' => [
|
||||
'lemonsqueezy_snapshot' => [
|
||||
'error' => $exception->getMessage(),
|
||||
'status' => $exception->status(),
|
||||
'context' => $exception->context(),
|
||||
@@ -3,8 +3,8 @@
|
||||
namespace App\Jobs;
|
||||
|
||||
use App\Models\PackageAddon;
|
||||
use App\Services\Paddle\Exceptions\PaddleException;
|
||||
use App\Services\Paddle\PaddleAddonCatalogService;
|
||||
use App\Services\LemonSqueezy\Exceptions\LemonSqueezyException;
|
||||
use App\Services\LemonSqueezy\LemonSqueezyAddonCatalogService;
|
||||
use Illuminate\Bus\Queueable;
|
||||
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||
use Illuminate\Foundation\Bus\Dispatchable;
|
||||
@@ -14,7 +14,7 @@ use Illuminate\Support\Arr;
|
||||
use Illuminate\Support\Facades\Log;
|
||||
use Throwable;
|
||||
|
||||
class SyncPackageAddonToPaddle implements ShouldQueue
|
||||
class SyncPackageAddonToLemonSqueezy implements ShouldQueue
|
||||
{
|
||||
use Dispatchable;
|
||||
use InteractsWithQueue;
|
||||
@@ -26,7 +26,7 @@ class SyncPackageAddonToPaddle implements ShouldQueue
|
||||
*/
|
||||
public function __construct(private readonly int $addonId, private readonly array $options = []) {}
|
||||
|
||||
public function handle(PaddleAddonCatalogService $catalog): void
|
||||
public function handle(LemonSqueezyAddonCatalogService $catalog): void
|
||||
{
|
||||
$addon = PackageAddon::query()->find($this->addonId);
|
||||
|
||||
@@ -39,7 +39,7 @@ class SyncPackageAddonToPaddle implements ShouldQueue
|
||||
$priceOverrides = Arr::get($this->options, 'price', []);
|
||||
|
||||
if ($dryRun) {
|
||||
$this->storeDryRunSnapshot($catalog, $addon, $productOverrides, $priceOverrides);
|
||||
$this->storeDryRunSnapshot($addon, $productOverrides, $priceOverrides);
|
||||
|
||||
return;
|
||||
}
|
||||
@@ -47,41 +47,41 @@ class SyncPackageAddonToPaddle implements ShouldQueue
|
||||
// Mark syncing (metadata)
|
||||
$addon->forceFill([
|
||||
'metadata' => array_merge($addon->metadata ?? [], [
|
||||
'paddle_sync_status' => 'syncing',
|
||||
'paddle_synced_at' => now()->toIso8601String(),
|
||||
'lemonsqueezy_sync_status' => 'syncing',
|
||||
'lemonsqueezy_synced_at' => now()->toIso8601String(),
|
||||
]),
|
||||
])->save();
|
||||
|
||||
try {
|
||||
$payloadOverrides = $this->buildPayloadOverrides($addon, $productOverrides, $priceOverrides);
|
||||
|
||||
$productResponse = $addon->metadata['paddle_product_id'] ?? null
|
||||
? $catalog->updateProduct($addon->metadata['paddle_product_id'], $addon, $payloadOverrides['product'])
|
||||
$productResponse = $addon->metadata['lemonsqueezy_product_id'] ?? null
|
||||
? $catalog->updateProduct($addon->metadata['lemonsqueezy_product_id'], $addon, $payloadOverrides['product'])
|
||||
: $catalog->createProduct($addon, $payloadOverrides['product']);
|
||||
|
||||
$productId = (string) ($productResponse['id'] ?? $addon->metadata['paddle_product_id'] ?? null);
|
||||
$productId = (string) ($productResponse['id'] ?? $addon->metadata['lemonsqueezy_product_id'] ?? null);
|
||||
|
||||
if (! $productId) {
|
||||
throw new PaddleException('Paddle product ID missing after addon sync.');
|
||||
throw new LemonSqueezyException('Lemon Squeezy product ID missing after addon sync.');
|
||||
}
|
||||
|
||||
$priceResponse = $addon->price_id
|
||||
? $catalog->updatePrice($addon->price_id, $addon, array_merge($payloadOverrides['price'], ['product_id' => $productId]))
|
||||
$priceResponse = $addon->variant_id
|
||||
? $catalog->updatePrice($addon->variant_id, $addon, array_merge($payloadOverrides['price'], ['product_id' => $productId]))
|
||||
: $catalog->createPrice($addon, $productId, $payloadOverrides['price']);
|
||||
|
||||
$priceId = (string) ($priceResponse['id'] ?? $addon->price_id);
|
||||
$priceId = (string) ($priceResponse['id'] ?? $addon->variant_id);
|
||||
|
||||
if (! $priceId) {
|
||||
throw new PaddleException('Paddle price ID missing after addon sync.');
|
||||
throw new LemonSqueezyException('Lemon Squeezy variant ID missing after addon sync.');
|
||||
}
|
||||
|
||||
$addon->forceFill([
|
||||
'price_id' => $priceId,
|
||||
'variant_id' => $priceId,
|
||||
'metadata' => array_merge($addon->metadata ?? [], [
|
||||
'paddle_sync_status' => 'synced',
|
||||
'paddle_synced_at' => now()->toIso8601String(),
|
||||
'paddle_product_id' => $productId,
|
||||
'paddle_snapshot' => [
|
||||
'lemonsqueezy_sync_status' => 'synced',
|
||||
'lemonsqueezy_synced_at' => now()->toIso8601String(),
|
||||
'lemonsqueezy_product_id' => $productId,
|
||||
'lemonsqueezy_snapshot' => [
|
||||
'product' => $productResponse,
|
||||
'price' => $priceResponse,
|
||||
'payload' => $payloadOverrides,
|
||||
@@ -89,7 +89,7 @@ class SyncPackageAddonToPaddle implements ShouldQueue
|
||||
]),
|
||||
])->save();
|
||||
} catch (Throwable $exception) {
|
||||
Log::channel('paddle-sync')->error('Paddle addon sync failed', [
|
||||
Log::channel('lemonsqueezy-sync')->error('Lemon Squeezy addon sync failed', [
|
||||
'addon_id' => $addon->id,
|
||||
'message' => $exception->getMessage(),
|
||||
'exception' => $exception,
|
||||
@@ -97,9 +97,9 @@ class SyncPackageAddonToPaddle implements ShouldQueue
|
||||
|
||||
$addon->forceFill([
|
||||
'metadata' => array_merge($addon->metadata ?? [], [
|
||||
'paddle_sync_status' => 'failed',
|
||||
'paddle_synced_at' => now()->toIso8601String(),
|
||||
'paddle_error' => [
|
||||
'lemonsqueezy_sync_status' => 'failed',
|
||||
'lemonsqueezy_synced_at' => now()->toIso8601String(),
|
||||
'lemonsqueezy_error' => [
|
||||
'message' => $exception->getMessage(),
|
||||
'class' => $exception::class,
|
||||
],
|
||||
@@ -145,22 +145,22 @@ class SyncPackageAddonToPaddle implements ShouldQueue
|
||||
* @param array<string, mixed> $productOverrides
|
||||
* @param array<string, mixed> $priceOverrides
|
||||
*/
|
||||
protected function storeDryRunSnapshot(PaddleCatalogService $catalog, PackageAddon $addon, array $productOverrides, array $priceOverrides): void
|
||||
protected function storeDryRunSnapshot(PackageAddon $addon, array $productOverrides, array $priceOverrides): void
|
||||
{
|
||||
$payloadOverrides = $this->buildPayloadOverrides($addon, $productOverrides, $priceOverrides);
|
||||
|
||||
$addon->forceFill([
|
||||
'metadata' => array_merge($addon->metadata ?? [], [
|
||||
'paddle_sync_status' => 'dry-run',
|
||||
'paddle_synced_at' => now()->toIso8601String(),
|
||||
'paddle_snapshot' => [
|
||||
'lemonsqueezy_sync_status' => 'dry-run',
|
||||
'lemonsqueezy_synced_at' => now()->toIso8601String(),
|
||||
'lemonsqueezy_snapshot' => [
|
||||
'dry_run' => true,
|
||||
'payload' => $payloadOverrides,
|
||||
],
|
||||
]),
|
||||
])->save();
|
||||
|
||||
Log::channel('paddle-sync')->info('Paddle addon dry-run snapshot generated', [
|
||||
Log::channel('lemonsqueezy-sync')->info('Lemon Squeezy addon dry-run snapshot generated', [
|
||||
'addon_id' => $addon->id,
|
||||
]);
|
||||
}
|
||||
@@ -3,8 +3,8 @@
|
||||
namespace App\Jobs;
|
||||
|
||||
use App\Models\Package;
|
||||
use App\Services\Paddle\Exceptions\PaddleException;
|
||||
use App\Services\Paddle\PaddleCatalogService;
|
||||
use App\Services\LemonSqueezy\Exceptions\LemonSqueezyException;
|
||||
use App\Services\LemonSqueezy\LemonSqueezyCatalogService;
|
||||
use Illuminate\Bus\Queueable;
|
||||
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||
use Illuminate\Foundation\Bus\Dispatchable;
|
||||
@@ -14,7 +14,7 @@ use Illuminate\Support\Arr;
|
||||
use Illuminate\Support\Facades\Log;
|
||||
use Throwable;
|
||||
|
||||
class SyncPackageToPaddle implements ShouldQueue
|
||||
class SyncPackageToLemonSqueezy implements ShouldQueue
|
||||
{
|
||||
use Dispatchable;
|
||||
use InteractsWithQueue;
|
||||
@@ -26,7 +26,7 @@ class SyncPackageToPaddle implements ShouldQueue
|
||||
*/
|
||||
public function __construct(private readonly int $packageId, private readonly array $options = []) {}
|
||||
|
||||
public function handle(PaddleCatalogService $catalog): void
|
||||
public function handle(LemonSqueezyCatalogService $catalog): void
|
||||
{
|
||||
$package = Package::query()->find($this->packageId);
|
||||
|
||||
@@ -45,37 +45,37 @@ class SyncPackageToPaddle implements ShouldQueue
|
||||
}
|
||||
|
||||
$package->forceFill([
|
||||
'paddle_sync_status' => 'syncing',
|
||||
'lemonsqueezy_sync_status' => 'syncing',
|
||||
])->save();
|
||||
|
||||
try {
|
||||
$productResponse = $package->paddle_product_id
|
||||
? $catalog->updateProduct($package->paddle_product_id, $package, $productOverrides)
|
||||
$productResponse = $package->lemonsqueezy_product_id
|
||||
? $catalog->updateProduct($package->lemonsqueezy_product_id, $package, $productOverrides)
|
||||
: $catalog->createProduct($package, $productOverrides);
|
||||
|
||||
$productId = (string) ($productResponse['id'] ?? $package->paddle_product_id);
|
||||
$productId = (string) ($productResponse['id'] ?? $package->lemonsqueezy_product_id);
|
||||
|
||||
if (! $productId) {
|
||||
throw new PaddleException('Paddle product ID missing after sync.');
|
||||
throw new LemonSqueezyException('Lemon Squeezy product ID missing after sync.');
|
||||
}
|
||||
|
||||
$package->paddle_product_id = $productId;
|
||||
$package->lemonsqueezy_product_id = $productId;
|
||||
|
||||
$priceResponse = $package->paddle_price_id
|
||||
? $catalog->updatePrice($package->paddle_price_id, $package, array_merge($priceOverrides, ['product_id' => $productId]))
|
||||
$priceResponse = $package->lemonsqueezy_variant_id
|
||||
? $catalog->updatePrice($package->lemonsqueezy_variant_id, $package, array_merge($priceOverrides, ['product_id' => $productId]))
|
||||
: $catalog->createPrice($package, $productId, $priceOverrides);
|
||||
|
||||
$priceId = (string) ($priceResponse['id'] ?? $package->paddle_price_id);
|
||||
$priceId = (string) ($priceResponse['id'] ?? $package->lemonsqueezy_variant_id);
|
||||
|
||||
if (! $priceId) {
|
||||
throw new PaddleException('Paddle price ID missing after sync.');
|
||||
throw new LemonSqueezyException('Lemon Squeezy variant ID missing after sync.');
|
||||
}
|
||||
|
||||
$package->forceFill([
|
||||
'paddle_price_id' => $priceId,
|
||||
'paddle_sync_status' => 'synced',
|
||||
'paddle_synced_at' => now(),
|
||||
'paddle_snapshot' => [
|
||||
'lemonsqueezy_variant_id' => $priceId,
|
||||
'lemonsqueezy_sync_status' => 'synced',
|
||||
'lemonsqueezy_synced_at' => now(),
|
||||
'lemonsqueezy_snapshot' => [
|
||||
'product' => $productResponse,
|
||||
'price' => $priceResponse,
|
||||
'payload' => [
|
||||
@@ -85,16 +85,16 @@ class SyncPackageToPaddle implements ShouldQueue
|
||||
],
|
||||
])->save();
|
||||
} catch (Throwable $exception) {
|
||||
Log::channel('paddle-sync')->error('Paddle package sync failed', [
|
||||
Log::channel('lemonsqueezy-sync')->error('Lemon Squeezy package sync failed', [
|
||||
'package_id' => $package->id,
|
||||
'message' => $exception->getMessage(),
|
||||
'exception' => $exception,
|
||||
]);
|
||||
|
||||
$package->forceFill([
|
||||
'paddle_sync_status' => 'failed',
|
||||
'paddle_synced_at' => now(),
|
||||
'paddle_snapshot' => array_merge($package->paddle_snapshot ?? [], [
|
||||
'lemonsqueezy_sync_status' => 'failed',
|
||||
'lemonsqueezy_synced_at' => now(),
|
||||
'lemonsqueezy_snapshot' => array_merge($package->lemonsqueezy_snapshot ?? [], [
|
||||
'error' => [
|
||||
'message' => $exception->getMessage(),
|
||||
'class' => $exception::class,
|
||||
@@ -110,19 +110,19 @@ class SyncPackageToPaddle implements ShouldQueue
|
||||
* @param array<string, mixed> $productOverrides
|
||||
* @param array<string, mixed> $priceOverrides
|
||||
*/
|
||||
protected function storeDryRunSnapshot(PaddleCatalogService $catalog, Package $package, array $productOverrides, array $priceOverrides): void
|
||||
protected function storeDryRunSnapshot(LemonSqueezyCatalogService $catalog, Package $package, array $productOverrides, array $priceOverrides): void
|
||||
{
|
||||
$productPayload = $catalog->buildProductPayload($package, $productOverrides);
|
||||
$pricePayload = $catalog->buildPricePayload(
|
||||
$package,
|
||||
$package->paddle_product_id ?: ($priceOverrides['product_id'] ?? 'pending'),
|
||||
$package->lemonsqueezy_product_id ?: ($priceOverrides['product_id'] ?? 'pending'),
|
||||
$priceOverrides
|
||||
);
|
||||
|
||||
$package->forceFill([
|
||||
'paddle_sync_status' => 'dry-run',
|
||||
'paddle_synced_at' => now(),
|
||||
'paddle_snapshot' => [
|
||||
'lemonsqueezy_sync_status' => 'dry-run',
|
||||
'lemonsqueezy_synced_at' => now(),
|
||||
'lemonsqueezy_snapshot' => [
|
||||
'dry_run' => true,
|
||||
'payload' => [
|
||||
'product' => $productPayload,
|
||||
@@ -131,7 +131,7 @@ class SyncPackageToPaddle implements ShouldQueue
|
||||
],
|
||||
])->save();
|
||||
|
||||
Log::channel('paddle-sync')->info('Paddle package dry-run snapshot generated', [
|
||||
Log::channel('lemonsqueezy-sync')->info('Lemon Squeezy package dry-run snapshot generated', [
|
||||
'package_id' => $package->id,
|
||||
]);
|
||||
}
|
||||
Reference in New Issue
Block a user