presseportale/app/Models/CreditWallet.php
Kevin Adametz b63cd26326 Credit-Wallet + Ledger + Tier-Preisableitung (Fundament)
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>
2026-06-17 14:16:43 +00:00

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');
}
}