Echte Credit-Wallet (1 Credit = 1 EUR) mit append-only Ledger als Basis fuer die Credit-Oekonomie aus dem Decision-Update (Rev. 4): - credit_wallets (denormalisierter Saldo) + credit_transactions (Ledger, vorzeichenbehaftet, balance_after, polymorphe reference) - CreditWalletService: einziger Schreibpfad, atomar mit Row-Lock, InsufficientCreditsException mit shortfall fuer den Mini-Checkout - Tier-Enum (Einzel/Starter/Business/Pro/Agency) + User::currentTier() - CreditPricingService: tier-gestaffelte Ableitung aus config/credits.php (Extra-PM 19/15/12/10/8, Boost 12/20/35, PDF 3, Depublish 25, Pruef-Quota) Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
44 lines
1.1 KiB
PHP
44 lines
1.1 KiB
PHP
<?php
|
|
|
|
namespace App\Models;
|
|
|
|
use App\Services\Billing\CreditWalletService;
|
|
use Database\Factories\CreditWalletFactory;
|
|
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
|
use Illuminate\Database\Eloquent\Model;
|
|
use Illuminate\Database\Eloquent\Relations\BelongsTo;
|
|
use Illuminate\Database\Eloquent\Relations\HasMany;
|
|
|
|
/**
|
|
* Credit-Guthaben eines Users. `balance_credits` ist der denormalisierte
|
|
* Saldo (1 Credit = 1 €); die maßgebliche Wahrheit ist der Ledger in
|
|
* `transactions`. Schreibzugriffe laufen ausschließlich über
|
|
* {@see CreditWalletService} (atomar + gesperrt).
|
|
*/
|
|
class CreditWallet extends Model
|
|
{
|
|
/** @use HasFactory<CreditWalletFactory> */
|
|
use HasFactory;
|
|
|
|
protected $fillable = [
|
|
'user_id',
|
|
'balance_credits',
|
|
];
|
|
|
|
protected function casts(): array
|
|
{
|
|
return [
|
|
'balance_credits' => 'integer',
|
|
];
|
|
}
|
|
|
|
public function user(): BelongsTo
|
|
{
|
|
return $this->belongsTo(User::class);
|
|
}
|
|
|
|
public function transactions(): HasMany
|
|
{
|
|
return $this->hasMany(CreditTransaction::class)->orderByDesc('created_at');
|
|
}
|
|
}
|