refactor: flatten plans.features JSON to typed columns
The features JSON column required defensive fallback stubs in three places (Plan::resolveForUser, PlanFeatures::__construct, PlanSeeder) and silently swallowed misspelled keys. Typed columns give Eloquent type-safe reads, simplify the Filament form (no more dotted JSON paths), and let resolveForUser fail loud when the free row is missing. PlanFeatures public API is unchanged so consumers (jobs, middleware) need no rewrites — one missed JSON read in SendScheduledWhatsAppJob was caught and converted to a typed where() query. tests/Pest.php seeds PlanSeeder in beforeEach so any feature test that resolves a plan finds the free row, mirroring production where plans always exist. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -14,71 +14,75 @@ class PlanSeeder extends Seeder
|
||||
PlanTier::Free->value => [
|
||||
'stripe_price_id_monthly' => null,
|
||||
'stripe_price_id_annual' => null,
|
||||
'features' => [
|
||||
'fuel_types' => ['max' => 1],
|
||||
'email' => ['enabled' => true, 'frequency' => 'weekly_digest'],
|
||||
'push' => ['enabled' => false, 'frequency' => 'none'],
|
||||
'whatsapp' => ['enabled' => false, 'daily_limit' => 0, 'scheduled_updates' => 0],
|
||||
'sms' => ['enabled' => false, 'daily_limit' => 0],
|
||||
'ai_predictions' => false,
|
||||
'price_threshold' => false,
|
||||
'score_alerts' => false,
|
||||
],
|
||||
'max_fuel_types' => 1,
|
||||
'email_enabled' => true,
|
||||
'email_frequency' => 'weekly_digest',
|
||||
'push_enabled' => false,
|
||||
'push_frequency' => 'none',
|
||||
'whatsapp_enabled' => false,
|
||||
'whatsapp_daily_limit' => 0,
|
||||
'whatsapp_scheduled_updates' => 0,
|
||||
'sms_enabled' => false,
|
||||
'sms_daily_limit' => 0,
|
||||
'ai_predictions' => false,
|
||||
'price_threshold' => false,
|
||||
'score_alerts' => false,
|
||||
],
|
||||
PlanTier::Basic->value => [
|
||||
'stripe_price_id_monthly' => config('services.stripe.prices.basic.monthly'),
|
||||
'stripe_price_id_annual' => config('services.stripe.prices.basic.annual'),
|
||||
'features' => [
|
||||
'fuel_types' => ['max' => 1],
|
||||
'email' => ['enabled' => true, 'frequency' => 'daily'],
|
||||
'push' => ['enabled' => true, 'frequency' => 'daily'],
|
||||
'whatsapp' => ['enabled' => true, 'daily_limit' => 5, 'scheduled_updates' => 2],
|
||||
'sms' => ['enabled' => false, 'daily_limit' => 0],
|
||||
'ai_predictions' => false,
|
||||
'price_threshold' => true,
|
||||
'score_alerts' => true,
|
||||
],
|
||||
'max_fuel_types' => 1,
|
||||
'email_enabled' => true,
|
||||
'email_frequency' => 'daily',
|
||||
'push_enabled' => true,
|
||||
'push_frequency' => 'daily',
|
||||
'whatsapp_enabled' => true,
|
||||
'whatsapp_daily_limit' => 5,
|
||||
'whatsapp_scheduled_updates' => 2,
|
||||
'sms_enabled' => false,
|
||||
'sms_daily_limit' => 0,
|
||||
'ai_predictions' => false,
|
||||
'price_threshold' => true,
|
||||
'score_alerts' => true,
|
||||
],
|
||||
PlanTier::Plus->value => [
|
||||
'stripe_price_id_monthly' => config('services.stripe.prices.plus.monthly'),
|
||||
'stripe_price_id_annual' => config('services.stripe.prices.plus.annual'),
|
||||
'features' => [
|
||||
'fuel_types' => ['max' => 1],
|
||||
'email' => ['enabled' => true, 'frequency' => 'triggered'],
|
||||
'push' => ['enabled' => true, 'frequency' => 'triggered'],
|
||||
'whatsapp' => ['enabled' => true, 'daily_limit' => 5, 'scheduled_updates' => 2],
|
||||
'sms' => ['enabled' => true, 'daily_limit' => 1],
|
||||
'ai_predictions' => true,
|
||||
'price_threshold' => true,
|
||||
'score_alerts' => true,
|
||||
],
|
||||
'max_fuel_types' => 1,
|
||||
'email_enabled' => true,
|
||||
'email_frequency' => 'triggered',
|
||||
'push_enabled' => true,
|
||||
'push_frequency' => 'triggered',
|
||||
'whatsapp_enabled' => true,
|
||||
'whatsapp_daily_limit' => 5,
|
||||
'whatsapp_scheduled_updates' => 2,
|
||||
'sms_enabled' => true,
|
||||
'sms_daily_limit' => 1,
|
||||
'ai_predictions' => true,
|
||||
'price_threshold' => true,
|
||||
'score_alerts' => true,
|
||||
],
|
||||
PlanTier::Pro->value => [
|
||||
'stripe_price_id_monthly' => config('services.stripe.prices.pro.monthly'),
|
||||
'stripe_price_id_annual' => config('services.stripe.prices.pro.annual'),
|
||||
'features' => [
|
||||
'fuel_types' => ['max' => null],
|
||||
'email' => ['enabled' => true, 'frequency' => 'triggered'],
|
||||
'push' => ['enabled' => true, 'frequency' => 'triggered'],
|
||||
'whatsapp' => ['enabled' => true, 'daily_limit' => 5, 'scheduled_updates' => 2],
|
||||
'sms' => ['enabled' => true, 'daily_limit' => 3],
|
||||
'ai_predictions' => true,
|
||||
'price_threshold' => true,
|
||||
'score_alerts' => true,
|
||||
],
|
||||
'max_fuel_types' => null,
|
||||
'email_enabled' => true,
|
||||
'email_frequency' => 'triggered',
|
||||
'push_enabled' => true,
|
||||
'push_frequency' => 'triggered',
|
||||
'whatsapp_enabled' => true,
|
||||
'whatsapp_daily_limit' => 5,
|
||||
'whatsapp_scheduled_updates' => 2,
|
||||
'sms_enabled' => true,
|
||||
'sms_daily_limit' => 3,
|
||||
'ai_predictions' => true,
|
||||
'price_threshold' => true,
|
||||
'score_alerts' => true,
|
||||
],
|
||||
];
|
||||
|
||||
foreach ($plans as $name => $data) {
|
||||
Plan::updateOrCreate(
|
||||
['name' => $name],
|
||||
[
|
||||
'stripe_price_id_monthly' => $data['stripe_price_id_monthly'],
|
||||
'stripe_price_id_annual' => $data['stripe_price_id_annual'],
|
||||
'features' => $data['features'],
|
||||
'active' => true,
|
||||
]
|
||||
);
|
||||
foreach ($plans as $name => $attributes) {
|
||||
Plan::updateOrCreate(['name' => $name], [...$attributes, 'active' => true]);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user