2025-11-21 10:49:41 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
return [
|
2026-05-15 08:27:59 +00:00
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Whether the package should auto-run its migrations.
|
|
|
|
|
*
|
|
|
|
|
* Default: true — fresh installs work plug-and-play (composer require +
|
|
|
|
|
* php artisan migrate). The package's own migrations live in vendor/ and
|
|
|
|
|
* are auto-loaded.
|
|
|
|
|
*
|
|
|
|
|
* Set to false if you have already published migrations to your project's
|
|
|
|
|
* database/migrations directory and want to manage the schema yourself.
|
|
|
|
|
* If you publish *and* leave this true, Laravel's migrator will see the
|
|
|
|
|
* same filenames in both locations and run each migration once — but
|
|
|
|
|
* that requires the published filename to match the source filename. If
|
|
|
|
|
* you've published with a different timestamp prefix, disable this flag
|
|
|
|
|
* to avoid re-runs.
|
|
|
|
|
*/
|
|
|
|
|
'run_migrations' => true,
|
|
|
|
|
|
2025-11-21 10:49:41 +00:00
|
|
|
// Table names (customizable for multi-tenancy)
|
|
|
|
|
'tables' => [
|
2025-11-29 19:09:19 +00:00
|
|
|
'cart_items' => 'cart_items',
|
|
|
|
|
'carts' => 'carts',
|
2025-12-29 08:59:02 +00:00
|
|
|
'orders' => 'orders',
|
|
|
|
|
'order_notes' => 'order_notes',
|
2025-11-29 19:09:19 +00:00
|
|
|
'payment_methods' => 'payment_methods',
|
|
|
|
|
'payment_provider_identities' => 'payment_provider_identities',
|
|
|
|
|
'product_action_runs' => 'product_action_runs',
|
2025-11-21 10:49:41 +00:00
|
|
|
'product_attributes' => 'product_attributes',
|
2025-11-29 19:09:19 +00:00
|
|
|
'product_categories' => 'product_categories',
|
2025-12-05 08:21:07 +00:00
|
|
|
'product_relations' => 'product_relations',
|
2025-11-29 19:09:19 +00:00
|
|
|
'product_prices' => 'product_prices',
|
2025-11-21 10:49:41 +00:00
|
|
|
'product_purchases' => 'product_purchases',
|
2025-12-05 08:21:07 +00:00
|
|
|
'product_actions' => 'product_actions',
|
2025-11-21 10:49:41 +00:00
|
|
|
'product_stocks' => 'product_stocks',
|
2026-05-15 08:27:59 +00:00
|
|
|
'product_price_tiers' => 'product_price_tiers',
|
2025-11-29 19:09:19 +00:00
|
|
|
'products' => 'products',
|
2025-12-05 08:21:07 +00:00
|
|
|
'cart_discounts' => 'cart_discounts',
|
2026-06-02 09:40:15 +00:00
|
|
|
'subscriptions' => 'subscriptions',
|
|
|
|
|
'subscription_items' => 'subscription_items',
|
2025-11-21 10:49:41 +00:00
|
|
|
],
|
|
|
|
|
|
|
|
|
|
// Model classes (allow overriding in main instance)
|
|
|
|
|
'models' => [
|
|
|
|
|
'product' => \Blax\Shop\Models\Product::class,
|
|
|
|
|
'product_price' => \Blax\Shop\Models\ProductPrice::class,
|
|
|
|
|
'product_category' => \Blax\Shop\Models\ProductCategory::class,
|
|
|
|
|
'product_stock' => \Blax\Shop\Models\ProductStock::class,
|
2026-05-15 08:27:59 +00:00
|
|
|
'product_price_tier' => \Blax\Shop\Models\ProductPriceTier::class,
|
2025-11-21 10:49:41 +00:00
|
|
|
'product_attribute' => \Blax\Shop\Models\ProductAttribute::class,
|
2025-11-21 14:22:40 +00:00
|
|
|
'product_purchase' => \Blax\Shop\Models\ProductPurchase::class,
|
2025-11-21 10:49:41 +00:00
|
|
|
'cart' => \Blax\Shop\Models\Cart::class,
|
|
|
|
|
'cart_item' => \Blax\Shop\Models\CartItem::class,
|
2025-12-29 08:59:02 +00:00
|
|
|
'order' => \Blax\Shop\Models\Order::class,
|
|
|
|
|
'order_note' => \Blax\Shop\Models\OrderNote::class,
|
2025-11-26 10:09:52 +00:00
|
|
|
'payment_provider_identity' => \Blax\Shop\Models\PaymentProviderIdentity::class,
|
|
|
|
|
'payment_method' => \Blax\Shop\Models\PaymentMethod::class,
|
2026-06-02 09:40:15 +00:00
|
|
|
'subscription' => \Blax\Shop\Models\Subscription::class,
|
|
|
|
|
'subscription_item' => \Blax\Shop\Models\SubscriptionItem::class,
|
|
|
|
|
],
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Subscriptions are Cashier-backed. The package binds its own
|
|
|
|
|
* Cashier-extending Subscription / SubscriptionItem models (above) so it
|
|
|
|
|
* can link a subscription to a product and run product actions on the
|
|
|
|
|
* billing lifecycle. Set `subscriptions.register_cashier_models` to false
|
|
|
|
|
* if the host app wants to point Cashier at its own models instead.
|
|
|
|
|
*/
|
|
|
|
|
'subscriptions' => [
|
|
|
|
|
'register_cashier_models' => env('SHOP_REGISTER_CASHIER_MODELS', true),
|
|
|
|
|
// Stripe interval => the product-action event fired on a NEW subscription.
|
|
|
|
|
'started_event' => 'subscription.started',
|
|
|
|
|
'renewed_event' => 'subscription.renewed',
|
|
|
|
|
'canceled_event' => 'subscription.canceled',
|
2025-11-21 10:49:41 +00:00
|
|
|
],
|
|
|
|
|
|
|
|
|
|
// API Routes configuration
|
|
|
|
|
'routes' => [
|
|
|
|
|
'enabled' => true,
|
|
|
|
|
'prefix' => 'api/shop',
|
|
|
|
|
'middleware' => ['api'],
|
|
|
|
|
'name_prefix' => 'shop.',
|
|
|
|
|
],
|
|
|
|
|
|
|
|
|
|
// Stock management
|
|
|
|
|
'stock' => [
|
|
|
|
|
'track_inventory' => true,
|
|
|
|
|
'allow_backorders' => false,
|
|
|
|
|
'low_stock_threshold' => 5,
|
|
|
|
|
'log_changes' => true,
|
|
|
|
|
'auto_release_expired' => true,
|
|
|
|
|
],
|
|
|
|
|
|
|
|
|
|
// Product actions (extensible by main instance)
|
|
|
|
|
'actions' => [
|
|
|
|
|
'path' => app_path('Jobs/ProductAction'),
|
|
|
|
|
'namespace' => 'App\\Jobs\\ProductAction',
|
|
|
|
|
'auto_discover' => true,
|
|
|
|
|
],
|
|
|
|
|
|
|
|
|
|
// Stripe integration (optional)
|
|
|
|
|
'stripe' => [
|
|
|
|
|
'enabled' => env('SHOP_STRIPE_ENABLED', false),
|
|
|
|
|
'sync_prices' => true,
|
2025-12-15 13:10:59 +00:00
|
|
|
'webhook_secret' => env('STRIPE_WEBHOOK_SECRET'),
|
2025-12-29 08:59:02 +00:00
|
|
|
|
|
|
|
|
// Webhook events that the shop package listens for
|
|
|
|
|
// You can customize this list to add/remove events as needed
|
|
|
|
|
'webhook_events' => [
|
|
|
|
|
// Checkout Session Events
|
|
|
|
|
'checkout.session.completed',
|
|
|
|
|
'checkout.session.async_payment_succeeded',
|
|
|
|
|
'checkout.session.async_payment_failed',
|
|
|
|
|
'checkout.session.expired',
|
|
|
|
|
|
|
|
|
|
// Charge Events
|
|
|
|
|
'charge.succeeded',
|
|
|
|
|
'charge.failed',
|
|
|
|
|
'charge.refunded',
|
|
|
|
|
'charge.dispute.created',
|
|
|
|
|
'charge.dispute.closed',
|
|
|
|
|
|
|
|
|
|
// Payment Intent Events
|
|
|
|
|
'payment_intent.succeeded',
|
|
|
|
|
'payment_intent.payment_failed',
|
|
|
|
|
'payment_intent.canceled',
|
|
|
|
|
|
|
|
|
|
// Refund Events
|
|
|
|
|
'refund.created',
|
|
|
|
|
'refund.updated',
|
|
|
|
|
|
|
|
|
|
// Invoice Events (for subscriptions)
|
|
|
|
|
'invoice.payment_succeeded',
|
|
|
|
|
'invoice.payment_failed',
|
|
|
|
|
],
|
2025-11-21 10:49:41 +00:00
|
|
|
],
|
|
|
|
|
|
2025-12-17 17:33:34 +00:00
|
|
|
// Currency configuration
|
|
|
|
|
'currency' => env('SHOP_CURRENCY', 'usd'),
|
|
|
|
|
|
2025-11-21 10:49:41 +00:00
|
|
|
// Cache configuration
|
|
|
|
|
'cache' => [
|
|
|
|
|
'enabled' => env('SHOP_CACHE_ENABLED', true),
|
|
|
|
|
'ttl' => 3600,
|
|
|
|
|
'prefix' => 'shop:',
|
|
|
|
|
],
|
|
|
|
|
|
|
|
|
|
// Cart configuration
|
|
|
|
|
'cart' => [
|
|
|
|
|
'expire_after_days' => 30,
|
|
|
|
|
'auto_cleanup' => true,
|
|
|
|
|
'merge_on_login' => true,
|
2025-12-29 09:26:51 +00:00
|
|
|
|
|
|
|
|
// Cart expiration: mark carts as expired after this many minutes of inactivity
|
|
|
|
|
'expiration_minutes' => env('SHOP_CART_EXPIRATION_MINUTES', 60),
|
|
|
|
|
|
|
|
|
|
// Cart deletion: delete unused carts after this many hours of inactivity
|
|
|
|
|
'deletion_hours' => env('SHOP_CART_DELETION_HOURS', 24),
|
2025-11-21 10:49:41 +00:00
|
|
|
],
|
|
|
|
|
|
2025-12-29 08:59:02 +00:00
|
|
|
// Order configuration
|
|
|
|
|
'orders' => [
|
|
|
|
|
'number_prefix' => env('SHOP_ORDER_PREFIX', 'ORD-'),
|
|
|
|
|
'auto_complete_virtual' => true, // Auto-complete orders with only virtual products
|
|
|
|
|
'auto_complete_paid' => false, // Auto-complete orders when fully paid
|
|
|
|
|
],
|
|
|
|
|
|
2025-11-21 10:49:41 +00:00
|
|
|
// API Response format
|
|
|
|
|
'api' => [
|
|
|
|
|
'include_meta' => true,
|
|
|
|
|
'wrap_response' => true,
|
|
|
|
|
'response_key' => 'data',
|
|
|
|
|
],
|
2025-11-25 23:05:46 +00:00
|
|
|
|
2026-05-15 08:27:59 +00:00
|
|
|
/*
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
| Loan / rental defaults
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
| Consumed by ProductPurchase::extend() and ProductPurchase::canExtend()
|
|
|
|
|
| when no overrides are passed. Lets a host app (library, equipment-rental
|
|
|
|
|
| etc.) tune the lifecycle without subclassing the model.
|
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
'loan' => [
|
|
|
|
|
// Loan policy knobs (used by ProductPurchase lifecycle helpers).
|
|
|
|
|
// Pricing lives on the ProductPrice itself (billing_scheme=tiered +
|
|
|
|
|
// associated product_price_tiers rows), not here.
|
|
|
|
|
'default_duration_weeks' => env('SHOP_LOAN_DURATION_WEEKS', 2),
|
|
|
|
|
'extension_weeks' => env('SHOP_LOAN_EXTENSION_WEEKS', 1),
|
|
|
|
|
'max_extensions' => env('SHOP_LOAN_MAX_EXTENSIONS', 2),
|
|
|
|
|
],
|
|
|
|
|
|
2026-05-18 11:05:38 +00:00
|
|
|
/*
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
| API pagination
|
|
|
|
|
|--------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
| Consumed by the public API controllers (Http/Controllers/Api/*).
|
|
|
|
|
| `per_page` — default page size when the request doesn't specify one.
|
|
|
|
|
| `max_per_page` — upper bound the controller will honour regardless of
|
|
|
|
|
| what the client asks for, so a malicious or careless
|
|
|
|
|
| caller can't request all rows in one request.
|
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
'pagination' => [
|
|
|
|
|
'per_page' => env('SHOP_API_PER_PAGE', 24),
|
|
|
|
|
'max_per_page' => env('SHOP_API_MAX_PER_PAGE', 100),
|
|
|
|
|
],
|
|
|
|
|
|
2025-11-21 10:49:41 +00:00
|
|
|
];
|