feat(payment): implement database-driven payment provider system with encrypted configuration support
- Add PaymentProviderSeeder with initial provider data (Stripe, Lemon Squeezy, Polar, OxaPay, Crypto, Activation Key) - Create migration to disable JSON constraints and change configuration column from JSON to TEXT - Update PaymentProvider model cast from 'array' to 'encrypted:array' for secure configuration storage
This commit is contained in:
229
database/seeders/PaymentProviderSeeder.php
Normal file
229
database/seeders/PaymentProviderSeeder.php
Normal file
@@ -0,0 +1,229 @@
|
||||
<?php
|
||||
|
||||
namespace Database\Seeders;
|
||||
|
||||
use App\Models\PaymentProvider;
|
||||
use Illuminate\Database\Seeder;
|
||||
|
||||
class PaymentProviderSeeder extends Seeder
|
||||
{
|
||||
/**
|
||||
* Run the database seeds.
|
||||
*/
|
||||
public function run(): void
|
||||
{
|
||||
$this->command->info('🌱 Seeding payment providers...');
|
||||
|
||||
// Skip encryption test for now to focus on basic seeding
|
||||
$this->command->info('📝 Skipping encryption test to focus on basic data seeding');
|
||||
|
||||
$providers = [
|
||||
[
|
||||
'name' => 'stripe',
|
||||
'display_name' => 'Stripe',
|
||||
'description' => 'Accept payments via Stripe - Credit cards, Apple Pay, Google Pay and more',
|
||||
'is_active' => false,
|
||||
'configuration' => [
|
||||
'class' => 'App\\Services\\Payments\\Providers\\StripeProvider',
|
||||
'secret_key' => env('STRIPE_SECRET') ?: 'sk_test_placeholder',
|
||||
'publishable_key' => env('STRIPE_PUBLISHABLE_KEY') ?: 'pk_test_placeholder',
|
||||
'webhook_secret' => env('STRIPE_WEBHOOK_SECRET') ?: 'whsec_placeholder',
|
||||
'webhook_url' => env('APP_URL', 'https://example.com') . '/webhook/stripe',
|
||||
'success_url' => env('APP_URL', 'https://example.com') . '/payment/success',
|
||||
'cancel_url' => env('APP_URL', 'https://example.com') . '/payment/cancel',
|
||||
'currency' => env('CASHIER_CURRENCY', 'USD'),
|
||||
],
|
||||
'supports_recurring' => true,
|
||||
'supports_one_time' => true,
|
||||
'supported_currencies' => [
|
||||
'USD' => 'US Dollar'
|
||||
],
|
||||
'fee_structure' => [
|
||||
'fixed_fee' => '0.50',
|
||||
'percentage_fee' => '3.9',
|
||||
],
|
||||
'priority' => 60,
|
||||
'is_fallback' => false,
|
||||
],
|
||||
|
||||
[
|
||||
'name' => 'lemon_squeezy',
|
||||
'display_name' => 'Lemon Squeezy',
|
||||
'description' => 'Modern payment platform for digital products and subscriptions',
|
||||
'is_active' => false,
|
||||
'configuration' => [
|
||||
'class' => 'App\\Services\\Payments\\Providers\\LemonSqueezyProvider',
|
||||
'api_key' => env('LEMON_SQUEEZY_API_KEY', 'lsk_...'),
|
||||
'store_id' => env('LEMON_SQUEEZY_STORE_ID', '...'),
|
||||
'webhook_secret' => env('LEMON_SQUEEZY_WEBHOOK_SECRET', 'whsec_...'),
|
||||
'webhook_url' => env('APP_URL') . '/webhook/lemon-squeezy',
|
||||
'success_url' => env('APP_URL') . '/payment/success',
|
||||
'cancel_url' => env('APP_URL') . '/payment/cancel',
|
||||
],
|
||||
'supports_recurring' => true,
|
||||
'supports_one_time' => true,
|
||||
'supported_currencies' => [
|
||||
'USD' => 'US Dollar'
|
||||
],
|
||||
'fee_structure' => [
|
||||
'fixed_fee' => '0.50',
|
||||
'percentage_fee' => '5.0',
|
||||
],
|
||||
'priority' => 50,
|
||||
'is_fallback' => false,
|
||||
],
|
||||
|
||||
[
|
||||
'name' => 'polar',
|
||||
'display_name' => 'Polar.sh',
|
||||
'description' => 'Modern crowdfunding and payment platform for creators',
|
||||
'is_active' => false,
|
||||
'configuration' => [
|
||||
'class' => 'App\\Services\\Payments\\Providers\\PolarProvider',
|
||||
'api_key' => env('POLAR_API_KEY', 'pol_...'),
|
||||
'webhook_secret' => env('POLAR_WEBHOOK_SECRET', 'whsec_...'),
|
||||
'sandbox' => env('POLAR_SANDBOX', false),
|
||||
'sandbox_api_key' => env('POLAR_SANDBOX_API_KEY', 'pol_test_...'),
|
||||
'sandbox_webhook_secret' => env('POLAR_SANDBOX_WEBHOOK_SECRET', 'whsec_test_...'),
|
||||
'access_token' => env('POLAR_ACCESS_TOKEN', 'polar_...'),
|
||||
'webhook_url' => env('APP_URL') . '/webhook/polar',
|
||||
'success_url' => env('APP_URL') . '/payment/success',
|
||||
'cancel_url' => env('APP_URL') . '/payment/cancel',
|
||||
],
|
||||
'supports_recurring' => true,
|
||||
'supports_one_time' => true,
|
||||
'supported_currencies' => [
|
||||
'USD' => 'US Dollar',
|
||||
'EUR' => 'Euro',
|
||||
],
|
||||
'fee_structure' => [
|
||||
'fixed_fee' => '0.50',
|
||||
'percentage_fee' => '5.0',
|
||||
],
|
||||
'priority' => 40,
|
||||
'is_fallback' => false,
|
||||
],
|
||||
|
||||
[
|
||||
'name' => 'oxapay',
|
||||
'display_name' => 'OxaPay',
|
||||
'description' => 'Cryptocurrency payment gateway supporting multiple digital assets',
|
||||
'is_active' => false,
|
||||
'configuration' => [
|
||||
'class' => 'App\\Services\\Payments\\Providers\\OxapayProvider',
|
||||
'merchant_api_key' => env('OXAPAY_MERCHANT_API_KEY', 'merchant_...'),
|
||||
'payout_api_key' => env('OXAPAY_PAYOUT_API_KEY', 'payout_...'),
|
||||
'webhook_url' => env('OXAPAY_WEBHOOK_URL', env('APP_URL') . '/webhook/oxapay'),
|
||||
'success_url' => env('APP_URL') . '/payment/success',
|
||||
'cancel_url' => env('APP_URL') . '/payment/cancel',
|
||||
'sandbox' => env('OXAPAY_SANDBOX', true),
|
||||
],
|
||||
'supports_recurring' => false,
|
||||
'supports_one_time' => true,
|
||||
'supported_currencies' => [
|
||||
'BTC' => 'Bitcoin',
|
||||
'ETH' => 'Ethereum',
|
||||
'USDT' => 'Tether USD',
|
||||
'USDC' => 'USD Coin',
|
||||
'LTC' => 'Litecoin',
|
||||
'BCH' => 'Bitcoin Cash',
|
||||
'BNB' => 'Binance Coin',
|
||||
],
|
||||
'fee_structure' => [
|
||||
'fixed_fee' => '0.00',
|
||||
'percentage_fee' => '1.0',
|
||||
],
|
||||
'priority' => 30,
|
||||
'is_fallback' => false,
|
||||
],
|
||||
|
||||
[
|
||||
'name' => 'crypto',
|
||||
'display_name' => 'Native Crypto',
|
||||
'description' => 'Direct cryptocurrency payments with blockchain confirmations',
|
||||
'is_active' => false,
|
||||
'configuration' => [
|
||||
'class' => 'App\\Services\\Payments\\Providers\\CryptoProvider',
|
||||
'webhook_secret' => env('CRYPTO_WEBHOOK_SECRET', 'crypto_whsec_...'),
|
||||
'confirmation_timeout_minutes' => env('CRYPTO_CONFIRMATION_TIMEOUT', 30),
|
||||
'exchange_rate_provider' => env('CRYPTO_EXCHANGE_RATE_PROVIDER', 'coingecko'),
|
||||
'coingecko_api_key' => env('COINGECKO_API_KEY'),
|
||||
'blockchair_api_key' => env('BLOCKCHAIR_API_KEY'),
|
||||
'webhook_url' => env('APP_URL') . '/webhook/crypto',
|
||||
'success_url' => env('APP_URL') . '/payment/success',
|
||||
'cancel_url' => env('APP_URL') . '/payment/cancel',
|
||||
'supported_wallets' => [
|
||||
'btc' => ['bitcoin', 'lightning'],
|
||||
'eth' => ['ethereum', 'erc20'],
|
||||
'ltc' => ['litecoin'],
|
||||
],
|
||||
],
|
||||
'supports_recurring' => false,
|
||||
'supports_one_time' => true,
|
||||
'supported_currencies' => [
|
||||
'BTC' => 'Bitcoin',
|
||||
'ETH' => 'Ethereum',
|
||||
'LTC' => 'Litecoin',
|
||||
'USDT' => 'Tether USD',
|
||||
'USDC' => 'USD Coin',
|
||||
],
|
||||
'fee_structure' => [
|
||||
'fixed_fee' => '0.00',
|
||||
'percentage_fee' => '0.0',
|
||||
'note' => 'Only blockchain network fees apply',
|
||||
],
|
||||
'priority' => 20,
|
||||
'is_fallback' => false,
|
||||
],
|
||||
|
||||
[
|
||||
'name' => 'activation_key',
|
||||
'display_name' => 'Activation Key',
|
||||
'description' => 'Manual activation using pre-generated activation keys',
|
||||
'is_active' => true,
|
||||
'configuration' => [
|
||||
'class' => 'App\\Services\\Payments\\Providers\\ActivationKeyProvider',
|
||||
'key_prefix' => env('ACTIVATION_KEY_PREFIX', 'AK-'),
|
||||
'key_length' => env('ACTIVATION_KEY_LENGTH', 32),
|
||||
'expiration_days' => env('ACTIVATION_KEY_EXPIRATION_DAYS'),
|
||||
'require_email_verification' => env('ACTIVATION_KEY_REQUIRE_EMAIL', true),
|
||||
'max_keys_per_user' => env('ACTIVATION_KEY_MAX_PER_USER', 5),
|
||||
'allowed_characters' => 'ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789',
|
||||
],
|
||||
'supports_recurring' => false,
|
||||
'supports_one_time' => true,
|
||||
'supported_currencies' => [], // No currency needed for activation keys
|
||||
'fee_structure' => [
|
||||
'fixed_fee' => '0.00',
|
||||
'percentage_fee' => '0.0',
|
||||
'note' => 'No fees for activation key system',
|
||||
],
|
||||
'priority' => 10, // Lowest priority
|
||||
'is_fallback' => true,
|
||||
],
|
||||
];
|
||||
|
||||
foreach ($providers as $providerData) {
|
||||
try {
|
||||
// First try to find existing provider
|
||||
$existingProvider = PaymentProvider::where('name', $providerData['name'])->first();
|
||||
|
||||
if ($existingProvider) {
|
||||
// Update existing provider
|
||||
$existingProvider->update($providerData);
|
||||
$this->command->info("✅ Updated payment provider: {$providerData['display_name']} ({$providerData['name']})");
|
||||
} else {
|
||||
// Create new provider
|
||||
$provider = PaymentProvider::create($providerData);
|
||||
$this->command->info("✅ Created payment provider: {$providerData['display_name']} ({$providerData['name']})");
|
||||
}
|
||||
} catch (\Exception $e) {
|
||||
$this->command->error("❌ Error seeding provider {$providerData['name']}: {$e->getMessage()}");
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
$this->command->info('🎉 Payment providers seeding completed!');
|
||||
$this->command->info('💡 Configuration data is managed by Laravel\'s encrypted:array cast.');
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user