Migrate billing from Paddle to Lemon Squeezy

This commit is contained in:
Codex Agent
2026-02-03 10:59:54 +01:00
parent 2f4ebfefd4
commit a0ef90e13a
228 changed files with 4369 additions and 4067 deletions

View File

@@ -21,7 +21,7 @@ class CouponSeeder extends Seeder
/** @var Collection<string, int> $packageIds */
$packageIds = Package::query()
->whereNotNull('paddle_price_id')
->whereNotNull('lemonsqueezy_variant_id')
->pluck('id', 'slug');
$coupons = [

View File

@@ -2,75 +2,32 @@
namespace Database\Seeders;
use App\Models\Package;
use App\Services\Paddle\PaddleGiftVoucherCatalogService;
use App\Services\LemonSqueezy\LemonSqueezyGiftVoucherCatalogService;
use Illuminate\Database\Seeder;
use Illuminate\Support\Facades\Schema;
class GiftVoucherTierSeeder extends Seeder
{
public function __construct(private readonly PaddleGiftVoucherCatalogService $catalog) {}
public function __construct(private readonly LemonSqueezyGiftVoucherCatalogService $catalog) {}
public function run(): void
{
if (! config('paddle.api_key')) {
$this->command?->warn('Skipping gift voucher Paddle sync: paddle.api_key not configured.');
if (! config('lemonsqueezy.api_key')) {
$this->command?->warn('Skipping gift voucher Lemon Squeezy sync: lemonsqueezy.api_key not configured.');
return;
}
$tiers = $this->buildTiers();
$tiers = config('gift-vouchers.tiers', []);
foreach ($tiers as $tier) {
$result = $this->catalog->ensureTier($tier);
$this->command?->info(sprintf(
'%s → product %s, price %s',
'%s → product %s, variant %s',
$tier['key'],
$result['product_id'],
$result['price_id']
$result['variant_id']
));
}
}
/**
* @return array<int, array{key:string,label:string,amount:float,currency?:string,paddle_product_id?:string|null,paddle_price_id?:string|null}>
*/
protected function buildTiers(): array
{
$columns = ['slug', 'name', 'price'];
if (Schema::hasColumn('packages', 'currency')) {
$columns[] = 'currency';
}
$packages = Package::query()
->where('type', 'endcustomer')
->whereNotNull('price')
->get($columns)
->unique(fn (Package $package) => $package->price.'|'.($package->currency ?? 'EUR'));
return $packages->map(function (Package $package): array {
$amount = (float) $package->price;
$currency = $package->currency ?? 'EUR';
return [
'key' => 'gift-'.$package->slug,
'label' => 'Gutschein '.$package->name,
'amount' => $amount,
'currency' => $currency,
'paddle_price_id' => $this->lookupPaddlePriceId($package->slug),
];
})->values()->all();
}
protected function lookupPaddlePriceId(string $slug): ?string
{
return match ($slug) {
'starter' => 'pri_01kbwccfe1mpwh7hh60eygemx6',
'standard' => 'pri_01kbwccfvzrf4z2f1r62vns7gh',
'pro' => 'pri_01kbwccg8vjc5cwz0kftfvf9wm',
'premium' => 'pri_01kbwccgnjzwrjy5xg1yp981p6',
default => null,
};
}
}

View File

@@ -32,8 +32,8 @@ class PackageSeeder extends Seeder
'watermark_allowed' => false,
'branding_allowed' => false,
'features' => ['basic_uploads', 'limited_sharing', 'custom_tasks', 'live_slideshow'],
'paddle_product_id' => 'pro_01k8jcxx2g1vj9snqbga4283ej',
'paddle_price_id' => 'pri_01k8jcxx8qktxvqzzv0nkjjj27',
'lemonsqueezy_product_id' => 'pro_01k8jcxx2g1vj9snqbga4283ej',
'lemonsqueezy_variant_id' => 'pri_01k8jcxx8qktxvqzzv0nkjjj27',
'description' => <<<'TEXT'
Ideal für Geburtstage, Gartenpartys oder Polterabende! {{max_guests}} Gäste teilen ihre besten Schnappschüsse, lösen {{max_tasks}} Fotoaufgaben und haben {{gallery_duration}} Zugriff auf die Online-Galerie. {{max_photos}} Bilder sind inklusive genug Platz für jede Menge Lieblingsmomente.
TEXT,
@@ -65,8 +65,8 @@ TEXT,
'watermark_allowed' => true,
'branding_allowed' => true,
'features' => ['basic_uploads', 'unlimited_sharing', 'custom_branding', 'custom_tasks', 'live_slideshow', 'no_watermark'],
'paddle_product_id' => 'pro_01k8jcxwjv4ne8vf9pvd9bye3j',
'paddle_price_id' => 'pri_01k8jcxws51pze5xc3vj2ea0yc',
'lemonsqueezy_product_id' => 'pro_01k8jcxwjv4ne8vf9pvd9bye3j',
'lemonsqueezy_variant_id' => 'pri_01k8jcxws51pze5xc3vj2ea0yc',
'description' => <<<'TEXT'
Das Rundum-Sorglos-Paket für Hochzeiten, Firmenfeiern oder Jubiläen. {{max_photos}} Bilder, {{max_guests}} Gäste und {{max_tasks}} Fotoaufgaben dazu eine Galerie, die {{gallery_duration}} online bleibt. Eigenes Logo oder Wasserzeichen inklusive.
TEXT,
@@ -98,8 +98,8 @@ TEXT,
'watermark_allowed' => true,
'branding_allowed' => true,
'features' => ['basic_uploads', 'unlimited_sharing', 'custom_branding', 'custom_tasks', 'live_slideshow', 'advanced_analytics', 'priority_support'],
'paddle_product_id' => 'pro_01k8jcxvwp38gay6jj2akjg76s',
'paddle_price_id' => 'pri_01k8jcxw5sap4r306wcvc0ephy',
'lemonsqueezy_product_id' => 'pro_01k8jcxvwp38gay6jj2akjg76s',
'lemonsqueezy_variant_id' => 'pri_01k8jcxw5sap4r306wcvc0ephy',
'description' => <<<'TEXT'
Das volle Erlebnis für alle, die keine Kompromisse machen wollen. {{max_photos}} Bilder, unbegrenzt viele Gäste, {{gallery_duration}} Galerie-Zugang und {{max_tasks}} Aufgaben dazu eigenes Wasserzeichen, Live-Slideshow und Premium-Support.
TEXT,
@@ -134,8 +134,8 @@ TEXT,
'max_events_per_year' => 5,
'expires_after' => null,
'features' => ['reseller_dashboard', 'custom_branding', 'priority_support'],
'paddle_product_id' => 'pro_01k8jcxvax48mhmwsfydw8ha9y',
'paddle_price_id' => 'pri_01k8jcxvhe0bfasg9gg1rw70sy',
'lemonsqueezy_product_id' => 'pro_01k8jcxvax48mhmwsfydw8ha9y',
'lemonsqueezy_variant_id' => 'pri_01k8jcxvhe0bfasg9gg1rw70sy',
'description' => <<<'TEXT'
Event-Kontingent für Partner / Agenturen: {{max_events_per_year}} Events auf StarterNiveau. Empfohlen innerhalb von 24 Monaten zu nutzen.
TEXT,
@@ -168,8 +168,8 @@ TEXT,
'max_events_per_year' => 15,
'expires_after' => null,
'features' => ['reseller_dashboard', 'custom_branding', 'priority_support', 'advanced_reporting'],
'paddle_product_id' => 'pro_01k8jcxtrxw7jsew52jnax901q',
'paddle_price_id' => 'pri_01k8jcxv06nsgy8ym8mnfrfm5v',
'lemonsqueezy_product_id' => 'pro_01k8jcxtrxw7jsew52jnax901q',
'lemonsqueezy_variant_id' => 'pri_01k8jcxv06nsgy8ym8mnfrfm5v',
'description' => <<<'TEXT'
Event-Kontingent für Partner / Agenturen: {{max_events_per_year}} Events auf ClassicNiveau. Empfohlen innerhalb von 24 Monaten zu nutzen.
TEXT,
@@ -202,8 +202,8 @@ TEXT,
'max_events_per_year' => 35,
'expires_after' => null,
'features' => ['reseller_dashboard', 'custom_branding', 'priority_support', 'advanced_reporting', 'live_slideshow'],
'paddle_product_id' => 'pro_01k8jcxt7gc6g6ddavmq65txzz',
'paddle_price_id' => 'pri_01k8jcxtfa07gvq43kpvpe0t8z',
'lemonsqueezy_product_id' => 'pro_01k8jcxt7gc6g6ddavmq65txzz',
'lemonsqueezy_variant_id' => 'pri_01k8jcxtfa07gvq43kpvpe0t8z',
'description' => <<<'TEXT'
Event-Kontingent für Partner / Agenturen: {{max_events_per_year}} Events auf PremiumNiveau. Empfohlen innerhalb von 24 Monaten zu nutzen.
TEXT,
@@ -236,8 +236,8 @@ TEXT,
'max_events_per_year' => 5,
'expires_after' => null,
'features' => ['reseller_dashboard', 'custom_branding', 'priority_support'],
'paddle_product_id' => 'pro_01kf16ttp0fph79j59x0z1cdqc',
'paddle_price_id' => 'pri_01kf16v0v2z4hse5cxq5wnah4b',
'lemonsqueezy_product_id' => 'pro_01kf16ttp0fph79j59x0z1cdqc',
'lemonsqueezy_variant_id' => 'pri_01kf16v0v2z4hse5cxq5wnah4b',
'description' => <<<'TEXT'
Premium Event-Kontingent für Partner / Agenturen: {{max_events_per_year}} Events auf PremiumNiveau. Empfohlen innerhalb von 24 Monaten zu nutzen.
TEXT,
@@ -270,8 +270,8 @@ TEXT,
'max_events_per_year' => 24,
'expires_after' => null,
'features' => ['reseller_dashboard', 'custom_branding', 'priority_support', 'advanced_reporting'],
'paddle_product_id' => 'pro_01k8jct3gz9ks5mg6z61q6nrxb',
'paddle_price_id' => 'pri_01k8jcxsa8axwpjnybhjbcrb06',
'lemonsqueezy_product_id' => 'pro_01k8jct3gz9ks5mg6z61q6nrxb',
'lemonsqueezy_variant_id' => 'pri_01k8jcxsa8axwpjnybhjbcrb06',
'description' => <<<'TEXT'
Event-Kontingent für Partner / Agenturen: {{max_events_per_year}} Events auf ClassicNiveau. Empfohlen innerhalb von 24 Monaten zu nutzen.
TEXT,