feat(addons): finalize event addon catalog and ai styling upgrade flow
This commit is contained in:
@@ -6,6 +6,7 @@ use App\Models\Event;
|
||||
use App\Models\EventPackage;
|
||||
use App\Models\EventPackageAddon;
|
||||
use App\Models\Package;
|
||||
use App\Models\PackageAddon;
|
||||
use App\Models\Tenant;
|
||||
use Tests\Feature\Tenant\TenantTestCase;
|
||||
|
||||
@@ -225,4 +226,49 @@ class BillingAddonHistoryTest extends TenantTestCase
|
||||
$response->assertStatus(404);
|
||||
$response->assertJsonPath('message', 'Event scope not found.');
|
||||
}
|
||||
|
||||
public function test_tenant_addon_history_uses_catalog_label_when_metadata_label_missing(): void
|
||||
{
|
||||
$package = Package::factory()->endcustomer()->create();
|
||||
|
||||
$event = Event::factory()->for($this->tenant)->create([
|
||||
'slug' => 'catalog-label-event',
|
||||
]);
|
||||
|
||||
$eventPackage = EventPackage::create([
|
||||
'event_id' => $event->id,
|
||||
'package_id' => $package->id,
|
||||
'purchased_price' => $package->price,
|
||||
'purchased_at' => now()->subWeek(),
|
||||
'used_photos' => 0,
|
||||
'used_guests' => 0,
|
||||
'gallery_expires_at' => now()->addDays(20),
|
||||
]);
|
||||
|
||||
PackageAddon::create([
|
||||
'key' => 'catalog_label_test',
|
||||
'label' => 'Catalog label fallback',
|
||||
'active' => true,
|
||||
'sort' => 5,
|
||||
'metadata' => ['price_eur' => 4],
|
||||
]);
|
||||
|
||||
EventPackageAddon::create([
|
||||
'event_package_id' => $eventPackage->id,
|
||||
'event_id' => $event->id,
|
||||
'tenant_id' => $this->tenant->id,
|
||||
'addon_key' => 'catalog_label_test',
|
||||
'quantity' => 1,
|
||||
'status' => 'completed',
|
||||
'amount' => 4.00,
|
||||
'currency' => 'EUR',
|
||||
'metadata' => [],
|
||||
'purchased_at' => now(),
|
||||
]);
|
||||
|
||||
$response = $this->authenticatedRequest('GET', '/api/v1/tenant/billing/addons');
|
||||
|
||||
$response->assertOk();
|
||||
$response->assertJsonPath('data.0.label', 'Catalog label fallback');
|
||||
}
|
||||
}
|
||||
|
||||
@@ -6,6 +6,7 @@ use App\Models\Event;
|
||||
use App\Models\EventPackage;
|
||||
use App\Models\EventPackageAddon;
|
||||
use App\Models\Package;
|
||||
use App\Models\PackageAddon;
|
||||
use Tests\Feature\Tenant\TenantTestCase;
|
||||
|
||||
class EventAddonsSummaryTest extends TenantTestCase
|
||||
@@ -87,4 +88,49 @@ class EventAddonsSummaryTest extends TenantTestCase
|
||||
$response->assertJsonPath('data.capabilities.ai_styling', true);
|
||||
$response->assertJsonPath('data.capabilities.ai_styling_granted_by', 'addon');
|
||||
}
|
||||
|
||||
public function test_event_resource_uses_catalog_label_when_addon_metadata_label_missing(): void
|
||||
{
|
||||
$package = Package::factory()->endcustomer()->create([
|
||||
'features' => ['basic_uploads'],
|
||||
]);
|
||||
|
||||
$event = Event::factory()->for($this->tenant)->create([
|
||||
'status' => 'published',
|
||||
]);
|
||||
|
||||
$eventPackage = EventPackage::create([
|
||||
'event_id' => $event->id,
|
||||
'package_id' => $package->id,
|
||||
'purchased_price' => $package->price,
|
||||
'purchased_at' => now(),
|
||||
'used_photos' => 0,
|
||||
'used_guests' => 0,
|
||||
'gallery_expires_at' => now()->addDays(30),
|
||||
]);
|
||||
|
||||
PackageAddon::create([
|
||||
'key' => 'catalog_label_summary_test',
|
||||
'label' => 'Catalog Summary Label',
|
||||
'active' => true,
|
||||
'sort' => 3,
|
||||
'metadata' => ['price_eur' => 3],
|
||||
]);
|
||||
|
||||
EventPackageAddon::create([
|
||||
'event_package_id' => $eventPackage->id,
|
||||
'event_id' => $event->id,
|
||||
'tenant_id' => $this->tenant->id,
|
||||
'addon_key' => 'catalog_label_summary_test',
|
||||
'quantity' => 1,
|
||||
'status' => 'completed',
|
||||
'purchased_at' => now(),
|
||||
'metadata' => [],
|
||||
]);
|
||||
|
||||
$response = $this->authenticatedRequest('GET', "/api/v1/tenant/events/{$event->slug}");
|
||||
|
||||
$response->assertOk();
|
||||
$response->assertJsonPath('data.addons.0.label', 'Catalog Summary Label');
|
||||
}
|
||||
}
|
||||
|
||||
79
tests/Feature/Tenant/EventAddonCatalogControllerTest.php
Normal file
79
tests/Feature/Tenant/EventAddonCatalogControllerTest.php
Normal file
@@ -0,0 +1,79 @@
|
||||
<?php
|
||||
|
||||
namespace Tests\Feature\Tenant;
|
||||
|
||||
use App\Models\PackageAddon;
|
||||
use Illuminate\Support\Facades\Config;
|
||||
|
||||
class EventAddonCatalogControllerTest extends TenantTestCase
|
||||
{
|
||||
public function test_paypal_catalog_only_returns_addons_with_paypal_price(): void
|
||||
{
|
||||
Config::set('package-addons.provider', 'paypal');
|
||||
|
||||
PackageAddon::create([
|
||||
'key' => 'extend_gallery_30d',
|
||||
'label' => 'Galerie +30 Tage',
|
||||
'active' => true,
|
||||
'sort' => 10,
|
||||
'extra_gallery_days' => 30,
|
||||
'metadata' => ['price_eur' => 4],
|
||||
]);
|
||||
|
||||
PackageAddon::create([
|
||||
'key' => 'extend_gallery_90d',
|
||||
'label' => 'Galerie +90 Tage',
|
||||
'active' => true,
|
||||
'sort' => 20,
|
||||
'extra_gallery_days' => 90,
|
||||
'variant_id' => 'variant_only',
|
||||
'metadata' => [],
|
||||
]);
|
||||
|
||||
$response = $this->authenticatedRequest('GET', '/api/v1/tenant/addons/catalog');
|
||||
|
||||
$response->assertOk();
|
||||
$response->assertJsonFragment([
|
||||
'key' => 'extend_gallery_30d',
|
||||
'price_id' => 'paypal',
|
||||
]);
|
||||
$response->assertJsonMissing([
|
||||
'key' => 'extend_gallery_90d',
|
||||
]);
|
||||
}
|
||||
|
||||
public function test_lemonsqueezy_catalog_only_returns_addons_with_variant(): void
|
||||
{
|
||||
Config::set('package-addons.provider', 'lemonsqueezy');
|
||||
|
||||
PackageAddon::create([
|
||||
'key' => 'extend_gallery_30d',
|
||||
'label' => 'Galerie +30 Tage',
|
||||
'active' => true,
|
||||
'sort' => 10,
|
||||
'extra_gallery_days' => 30,
|
||||
'variant_id' => 'var_30d',
|
||||
'metadata' => ['price_eur' => 4],
|
||||
]);
|
||||
|
||||
PackageAddon::create([
|
||||
'key' => 'extend_gallery_90d',
|
||||
'label' => 'Galerie +90 Tage',
|
||||
'active' => true,
|
||||
'sort' => 20,
|
||||
'extra_gallery_days' => 90,
|
||||
'metadata' => ['price_eur' => 10],
|
||||
]);
|
||||
|
||||
$response = $this->authenticatedRequest('GET', '/api/v1/tenant/addons/catalog');
|
||||
|
||||
$response->assertOk();
|
||||
$response->assertJsonFragment([
|
||||
'key' => 'extend_gallery_30d',
|
||||
'price_id' => 'var_30d',
|
||||
]);
|
||||
$response->assertJsonMissing([
|
||||
'key' => 'extend_gallery_90d',
|
||||
]);
|
||||
}
|
||||
}
|
||||
@@ -84,6 +84,9 @@ class EventAddonCheckoutTest extends TenantTestCase
|
||||
|
||||
$addon = EventPackageAddon::where('event_package_id', $eventPackage->id)->latest()->first();
|
||||
$this->assertSame(1000, $addon->extra_photos); // increments * quantity
|
||||
$this->assertSame('Extra photos (500)', $addon->metadata['label'] ?? null);
|
||||
$this->assertSame(500, $addon->metadata['increments']['extra_photos'] ?? null);
|
||||
$this->assertNull($addon->metadata['price_eur'] ?? null);
|
||||
}
|
||||
|
||||
public function test_paypal_checkout_creates_pending_addon_record(): void
|
||||
@@ -149,5 +152,114 @@ class EventAddonCheckoutTest extends TenantTestCase
|
||||
|
||||
$addon = EventPackageAddon::where('event_package_id', $eventPackage->id)->latest()->first();
|
||||
$this->assertSame(1000, $addon->extra_photos);
|
||||
$this->assertSame('Extra photos (500)', $addon->metadata['label'] ?? null);
|
||||
$this->assertSame(12.5, $addon->metadata['price_eur'] ?? null);
|
||||
}
|
||||
|
||||
public function test_ai_styling_checkout_persists_feature_entitlement_metadata(): void
|
||||
{
|
||||
Config::set('package-addons.provider', CheckoutSession::PROVIDER_PAYPAL);
|
||||
Config::set('checkout.currency', 'EUR');
|
||||
Config::set('package-addons.ai_styling_unlock', [
|
||||
'label' => 'AI Styling Add-on',
|
||||
'price' => 9.00,
|
||||
'increments' => [],
|
||||
'metadata' => [
|
||||
'scope' => 'feature',
|
||||
],
|
||||
]);
|
||||
|
||||
$package = Package::factory()->endcustomer()->create([
|
||||
'max_photos' => 100,
|
||||
'max_guests' => 50,
|
||||
'gallery_days' => 7,
|
||||
]);
|
||||
|
||||
$event = Event::factory()->for($this->tenant)->create([
|
||||
'status' => 'published',
|
||||
]);
|
||||
|
||||
$eventPackage = EventPackage::create([
|
||||
'event_id' => $event->id,
|
||||
'package_id' => $package->id,
|
||||
'purchased_price' => $package->price,
|
||||
'purchased_at' => now(),
|
||||
'used_photos' => 0,
|
||||
'used_guests' => 0,
|
||||
'gallery_expires_at' => now()->addDays(7),
|
||||
]);
|
||||
|
||||
$orders = Mockery::mock(PayPalOrderService::class);
|
||||
$orders->shouldReceive('createSimpleOrder')
|
||||
->once()
|
||||
->andReturn([
|
||||
'id' => 'ORDER-AI-1',
|
||||
'links' => [
|
||||
['rel' => 'approve', 'href' => 'https://paypal.test/approve-ai'],
|
||||
],
|
||||
]);
|
||||
$orders->shouldReceive('resolveApproveUrl')
|
||||
->once()
|
||||
->andReturn('https://paypal.test/approve-ai');
|
||||
$this->app->instance(PayPalOrderService::class, $orders);
|
||||
|
||||
$response = $this->authenticatedRequest('POST', "/api/v1/tenant/events/{$event->slug}/addons/checkout", [
|
||||
'addon_key' => 'ai_styling_unlock',
|
||||
'quantity' => 1,
|
||||
'accepted_terms' => true,
|
||||
'accepted_waiver' => true,
|
||||
]);
|
||||
|
||||
$response->assertOk();
|
||||
$response->assertJsonPath('checkout_id', 'ORDER-AI-1');
|
||||
|
||||
$this->assertDatabaseHas('event_package_addons', [
|
||||
'event_package_id' => $eventPackage->id,
|
||||
'addon_key' => 'ai_styling_unlock',
|
||||
'status' => 'pending',
|
||||
'amount' => 9.00,
|
||||
'currency' => 'EUR',
|
||||
]);
|
||||
|
||||
$addon = EventPackageAddon::where('event_package_id', $eventPackage->id)->latest()->first();
|
||||
$this->assertSame('AI Styling Add-on', $addon->metadata['label'] ?? null);
|
||||
$this->assertSame('feature', $addon->metadata['scope'] ?? null);
|
||||
$this->assertSame(['ai_styling'], $addon->metadata['entitlements']['features'] ?? null);
|
||||
$this->assertEquals(9.0, $addon->metadata['price_eur'] ?? null);
|
||||
$this->assertSame(0, $addon->extra_photos);
|
||||
$this->assertSame(0, $addon->extra_guests);
|
||||
$this->assertSame(0, $addon->extra_gallery_days);
|
||||
}
|
||||
|
||||
public function test_checkout_requires_both_legal_consents(): void
|
||||
{
|
||||
$package = Package::factory()->endcustomer()->create([
|
||||
'max_photos' => 100,
|
||||
'max_guests' => 50,
|
||||
'gallery_days' => 7,
|
||||
]);
|
||||
|
||||
$event = Event::factory()->for($this->tenant)->create([
|
||||
'status' => 'published',
|
||||
]);
|
||||
|
||||
EventPackage::create([
|
||||
'event_id' => $event->id,
|
||||
'package_id' => $package->id,
|
||||
'purchased_price' => $package->price,
|
||||
'purchased_at' => now(),
|
||||
'used_photos' => 0,
|
||||
'used_guests' => 0,
|
||||
'gallery_expires_at' => now()->addDays(7),
|
||||
]);
|
||||
|
||||
$response = $this->authenticatedRequest('POST', "/api/v1/tenant/events/{$event->slug}/addons/checkout", [
|
||||
'addon_key' => 'extra_photos_small',
|
||||
'quantity' => 1,
|
||||
'accepted_terms' => true,
|
||||
]);
|
||||
|
||||
$response->assertStatus(422);
|
||||
$response->assertJsonValidationErrors(['accepted_waiver']);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -5,11 +5,10 @@ namespace Tests\Feature\Tenant;
|
||||
use App\Models\Event;
|
||||
use App\Models\EventPackage;
|
||||
use App\Models\Package;
|
||||
use Carbon\Carbon;
|
||||
|
||||
class EventAddonControllerTest extends TenantTestCase
|
||||
{
|
||||
public function test_tenant_admin_can_apply_addons(): void
|
||||
public function test_apply_endpoint_is_not_available_for_tenant_admins(): void
|
||||
{
|
||||
$package = Package::factory()->endcustomer()->create([
|
||||
'max_photos' => 100,
|
||||
@@ -28,11 +27,9 @@ class EventAddonControllerTest extends TenantTestCase
|
||||
'purchased_at' => now()->subDay(),
|
||||
'used_photos' => 10,
|
||||
'used_guests' => 5,
|
||||
'gallery_expires_at' => Carbon::now()->addDays(7),
|
||||
'gallery_expires_at' => now()->addDays(7),
|
||||
]);
|
||||
|
||||
$originalExpiry = $eventPackage->gallery_expires_at->copy();
|
||||
|
||||
$response = $this->authenticatedRequest('POST', "/api/v1/tenant/events/{$event->slug}/addons/apply", [
|
||||
'extra_photos' => 50,
|
||||
'extra_guests' => 25,
|
||||
@@ -40,39 +37,12 @@ class EventAddonControllerTest extends TenantTestCase
|
||||
'reason' => 'Manual boost for event',
|
||||
]);
|
||||
|
||||
$response->assertOk();
|
||||
$response->assertJsonPath('data.limits.photos.limit', 150);
|
||||
$response->assertJsonPath('data.limits.guests.limit', 75);
|
||||
$response->assertNotFound();
|
||||
|
||||
$eventPackage->refresh();
|
||||
|
||||
$this->assertSame(50, $eventPackage->extra_photos);
|
||||
$this->assertSame(25, $eventPackage->extra_guests);
|
||||
$this->assertSame(3, $eventPackage->extra_gallery_days);
|
||||
$this->assertTrue($eventPackage->gallery_expires_at->isSameDay($originalExpiry->addDays(3)));
|
||||
}
|
||||
|
||||
public function test_validation_fails_when_no_addons_provided(): void
|
||||
{
|
||||
$package = Package::factory()->endcustomer()->create();
|
||||
|
||||
$event = Event::factory()->for($this->tenant)->create([
|
||||
'status' => 'published',
|
||||
]);
|
||||
|
||||
EventPackage::create([
|
||||
'event_id' => $event->id,
|
||||
'package_id' => $package->id,
|
||||
'purchased_price' => $package->price,
|
||||
'purchased_at' => now(),
|
||||
'used_photos' => 0,
|
||||
'used_guests' => 0,
|
||||
'gallery_expires_at' => now()->addDays(7),
|
||||
]);
|
||||
|
||||
$response = $this->authenticatedRequest('POST', "/api/v1/tenant/events/{$event->slug}/addons/apply", []);
|
||||
|
||||
$response->assertStatus(422);
|
||||
$response->assertJsonValidationErrors('addons');
|
||||
$this->assertSame(0, (int) $eventPackage->extra_photos);
|
||||
$this->assertSame(0, (int) $eventPackage->extra_guests);
|
||||
$this->assertSame(0, (int) $eventPackage->extra_gallery_days);
|
||||
}
|
||||
}
|
||||
|
||||
50
tests/Feature/Tenant/PackageCatalogAvailabilityTest.php
Normal file
50
tests/Feature/Tenant/PackageCatalogAvailabilityTest.php
Normal file
@@ -0,0 +1,50 @@
|
||||
<?php
|
||||
|
||||
namespace Tests\Feature\Tenant;
|
||||
|
||||
use App\Models\Package;
|
||||
use Illuminate\Support\Facades\Config;
|
||||
|
||||
class PackageCatalogAvailabilityTest extends TenantTestCase
|
||||
{
|
||||
public function test_paypal_catalog_marks_paid_reseller_packages_as_checkout_ready(): void
|
||||
{
|
||||
Config::set('checkout.default_provider', 'paypal');
|
||||
|
||||
Package::factory()->create([
|
||||
'type' => 'reseller',
|
||||
'price' => 99,
|
||||
'lemonsqueezy_variant_id' => null,
|
||||
]);
|
||||
|
||||
$response = $this->authenticatedRequest('GET', '/api/v1/tenant/packages?type=reseller');
|
||||
|
||||
$response->assertOk();
|
||||
$response->assertJsonPath('data.0.checkout_provider', 'paypal');
|
||||
$response->assertJsonPath('data.0.can_checkout', true);
|
||||
}
|
||||
|
||||
public function test_lemonsqueezy_catalog_requires_variant_for_checkout(): void
|
||||
{
|
||||
Config::set('checkout.default_provider', 'lemonsqueezy');
|
||||
|
||||
Package::factory()->create([
|
||||
'type' => 'reseller',
|
||||
'price' => 99,
|
||||
'lemonsqueezy_variant_id' => null,
|
||||
]);
|
||||
|
||||
Package::factory()->create([
|
||||
'type' => 'reseller',
|
||||
'price' => 199,
|
||||
'lemonsqueezy_variant_id' => 'pri_reseller_2',
|
||||
]);
|
||||
|
||||
$response = $this->authenticatedRequest('GET', '/api/v1/tenant/packages?type=reseller');
|
||||
|
||||
$response->assertOk();
|
||||
$response->assertJsonPath('data.0.checkout_provider', 'lemonsqueezy');
|
||||
$response->assertJsonPath('data.0.can_checkout', false);
|
||||
$response->assertJsonPath('data.1.can_checkout', true);
|
||||
}
|
||||
}
|
||||
@@ -31,6 +31,10 @@ class EventAddonCatalogTest extends TestCase
|
||||
'sort' => 1,
|
||||
'metadata' => [
|
||||
'price_eur' => 12,
|
||||
'scope' => 'feature',
|
||||
'entitlements' => [
|
||||
'features' => ['ai_styling'],
|
||||
],
|
||||
],
|
||||
]);
|
||||
|
||||
@@ -44,5 +48,7 @@ class EventAddonCatalogTest extends TestCase
|
||||
$this->assertSame(200, $addon['increments']['extra_photos']);
|
||||
$this->assertSame(12.0, $addon['price']);
|
||||
$this->assertSame('EUR', $addon['currency']);
|
||||
$this->assertSame('feature', $addon['metadata']['scope'] ?? null);
|
||||
$this->assertSame(['ai_styling'], $addon['metadata']['entitlements']['features'] ?? null);
|
||||
}
|
||||
}
|
||||
|
||||
108
tests/Unit/Services/EventFeatureEntitlementServiceTest.php
Normal file
108
tests/Unit/Services/EventFeatureEntitlementServiceTest.php
Normal file
@@ -0,0 +1,108 @@
|
||||
<?php
|
||||
|
||||
namespace Tests\Unit\Services;
|
||||
|
||||
use App\Models\Event;
|
||||
use App\Models\EventPackage;
|
||||
use App\Models\EventPackageAddon;
|
||||
use App\Models\Package;
|
||||
use App\Services\Addons\EventFeatureEntitlementService;
|
||||
use Illuminate\Foundation\Testing\RefreshDatabase;
|
||||
use Tests\TestCase;
|
||||
|
||||
class EventFeatureEntitlementServiceTest extends TestCase
|
||||
{
|
||||
use RefreshDatabase;
|
||||
|
||||
public function test_it_grants_feature_via_package(): void
|
||||
{
|
||||
$service = app(EventFeatureEntitlementService::class);
|
||||
$event = Event::factory()->create();
|
||||
$package = Package::factory()->endcustomer()->create([
|
||||
'features' => ['basic_uploads', 'ai_styling'],
|
||||
]);
|
||||
|
||||
EventPackage::query()->create([
|
||||
'event_id' => $event->id,
|
||||
'package_id' => $package->id,
|
||||
'purchased_price' => $package->price,
|
||||
'purchased_at' => now(),
|
||||
'gallery_expires_at' => now()->addDays(30),
|
||||
]);
|
||||
|
||||
$result = $service->resolveForEvent($event->fresh(), 'ai_styling', ['ai_styling_unlock']);
|
||||
|
||||
$this->assertTrue($result['allowed']);
|
||||
$this->assertSame('package', $result['granted_by']);
|
||||
}
|
||||
|
||||
public function test_it_grants_feature_via_completed_addon_key(): void
|
||||
{
|
||||
$service = app(EventFeatureEntitlementService::class);
|
||||
$event = Event::factory()->create();
|
||||
$package = Package::factory()->endcustomer()->create([
|
||||
'features' => ['basic_uploads'],
|
||||
]);
|
||||
|
||||
$eventPackage = EventPackage::query()->create([
|
||||
'event_id' => $event->id,
|
||||
'package_id' => $package->id,
|
||||
'purchased_price' => $package->price,
|
||||
'purchased_at' => now(),
|
||||
'gallery_expires_at' => now()->addDays(30),
|
||||
]);
|
||||
|
||||
EventPackageAddon::query()->create([
|
||||
'event_package_id' => $eventPackage->id,
|
||||
'event_id' => $event->id,
|
||||
'tenant_id' => $event->tenant_id,
|
||||
'addon_key' => 'custom_entitlement',
|
||||
'quantity' => 1,
|
||||
'status' => 'completed',
|
||||
'purchased_at' => now(),
|
||||
]);
|
||||
|
||||
$result = $service->resolveForEvent($event->fresh(), 'ai_styling', ['custom_entitlement']);
|
||||
|
||||
$this->assertTrue($result['allowed']);
|
||||
$this->assertSame('addon', $result['granted_by']);
|
||||
}
|
||||
|
||||
public function test_it_denies_feature_when_addon_is_expired(): void
|
||||
{
|
||||
$service = app(EventFeatureEntitlementService::class);
|
||||
$event = Event::factory()->create();
|
||||
$package = Package::factory()->endcustomer()->create([
|
||||
'features' => ['basic_uploads'],
|
||||
]);
|
||||
|
||||
$eventPackage = EventPackage::query()->create([
|
||||
'event_id' => $event->id,
|
||||
'package_id' => $package->id,
|
||||
'purchased_price' => $package->price,
|
||||
'purchased_at' => now(),
|
||||
'gallery_expires_at' => now()->addDays(30),
|
||||
]);
|
||||
|
||||
EventPackageAddon::query()->create([
|
||||
'event_package_id' => $eventPackage->id,
|
||||
'event_id' => $event->id,
|
||||
'tenant_id' => $event->tenant_id,
|
||||
'addon_key' => 'custom_entitlement',
|
||||
'quantity' => 1,
|
||||
'status' => 'completed',
|
||||
'purchased_at' => now()->subDays(2),
|
||||
'metadata' => [
|
||||
'entitlements' => [
|
||||
'expires_at' => now()->subDay()->toIso8601String(),
|
||||
'features' => ['ai_styling'],
|
||||
],
|
||||
],
|
||||
]);
|
||||
|
||||
$result = $service->resolveForEvent($event->fresh(), 'ai_styling', ['custom_entitlement']);
|
||||
|
||||
$this->assertFalse($result['allowed']);
|
||||
$this->assertNull($result['granted_by']);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user