feat(packages): implement package-based business model

This commit is contained in:
Codex Agent
2025-09-26 22:13:56 +02:00
parent 6fc36ebaf4
commit 0a643c3e4d
54 changed files with 3301 additions and 282 deletions

View File

@@ -20,9 +20,6 @@ class Tenant extends Model
'features' => 'array',
'settings' => 'array',
'last_activity_at' => 'datetime',
'event_credits_balance' => 'integer',
'subscription_tier' => 'string',
'subscription_expires_at' => 'datetime',
'total_revenue' => 'decimal:2',
'settings_updated_at' => 'datetime',
];
@@ -46,17 +43,38 @@ class Tenant extends Model
public function purchases(): HasMany
{
return $this->hasMany(PurchaseHistory::class);
return $this->hasMany(PackagePurchase::class);
}
public function eventPurchases(): HasMany
public function tenantPackages(): HasMany
{
return $this->hasMany(EventPurchase::class);
return $this->hasMany(TenantPackage::class);
}
public function creditsLedger(): HasMany
public function activeResellerPackage()
{
return $this->hasMany(EventCreditsLedger::class);
return $this->tenantPackages()->where('active', true)->first();
}
public function canCreateEvent(): bool
{
$package = $this->activeResellerPackage();
if (!$package) {
return false;
}
return $package->canCreateEvent();
}
public function incrementUsedEvents(int $amount = 1): bool
{
$package = $this->activeResellerPackage();
if (!$package) {
return false;
}
$package->increment('used_events', $amount);
return true;
}
public function setSettingsAttribute($value): void
@@ -72,88 +90,7 @@ class Tenant extends Model
public function activeSubscription(): Attribute
{
return Attribute::make(
get: fn () => $this->subscription_expires_at && $this->subscription_expires_at->isFuture(),
get: fn () => $this->activeResellerPackage() !== null,
);
}
public function decrementCredits(int $amount, string $reason = 'event_create', ?string $note = null, ?int $relatedPurchaseId = null): bool
{
if ($amount <= 0) {
return true;
}
$operation = function () use ($amount, $reason, $note, $relatedPurchaseId) {
$locked = static::query()
->whereKey($this->getKey())
->lockForUpdate()
->first();
if (! $locked || $locked->event_credits_balance < $amount) {
return false;
}
EventCreditsLedger::create([
'tenant_id' => $this->id,
'delta' => -$amount,
'reason' => $reason,
'related_purchase_id' => $relatedPurchaseId,
'note' => $note,
]);
$locked->event_credits_balance -= $amount;
$locked->save();
$this->event_credits_balance = $locked->event_credits_balance;
return true;
};
return $this->runCreditOperation($operation);
}
public function incrementCredits(int $amount, string $reason = 'manual_adjust', ?string $note = null, ?int $relatedPurchaseId = null): bool
{
if ($amount <= 0) {
return true;
}
$operation = function () use ($amount, $reason, $note, $relatedPurchaseId) {
$locked = static::query()
->whereKey($this->getKey())
->lockForUpdate()
->first();
if (! $locked) {
return false;
}
EventCreditsLedger::create([
'tenant_id' => $this->id,
'delta' => $amount,
'reason' => $reason,
'related_purchase_id' => $relatedPurchaseId,
'note' => $note,
]);
$locked->event_credits_balance += $amount;
$locked->save();
$this->event_credits_balance = $locked->event_credits_balance;
return true;
};
return $this->runCreditOperation($operation);
}
private function runCreditOperation(callable $operation): bool
{
$connection = DB::connection();
if ($connection->transactionLevel() > 0) {
return (bool) $operation();
}
return (bool) $connection->transaction($operation);
}
}