php-commerce/Models/Payment.php
Snider a774f4e285 refactor: migrate namespace from Core\Commerce to Core\Mod\Commerce
Align commerce module with the monorepo module structure by updating
all namespaces to use the Core\Mod\Commerce convention. This change
supports the recent monorepo separation and ensures consistency with
other modules.

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-27 16:23:12 +00:00

179 lines
4.2 KiB
PHP

<?php
namespace Core\Mod\Commerce\Models;
use Core\Mod\Tenant\Models\Workspace;
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Database\Eloquent\Relations\HasMany;
/**
* Payment model representing money received.
*
* @property int $id
* @property int $workspace_id
* @property int|null $invoice_id
* @property string $gateway
* @property string|null $gateway_payment_id
* @property string|null $gateway_customer_id
* @property string $currency
* @property float $amount
* @property float $fee
* @property float $net_amount
* @property string $status
* @property string|null $failure_reason
* @property string|null $payment_method_type
* @property string|null $payment_method_last4
* @property string|null $payment_method_brand
* @property array|null $gateway_response
* @property float $refunded_amount
*/
class Payment extends Model
{
use HasFactory;
protected static function newFactory(): \Core\Mod\Commerce\Database\Factories\PaymentFactory
{
return \Core\Mod\Commerce\Database\Factories\PaymentFactory::new();
}
protected $fillable = [
'workspace_id',
'invoice_id',
'order_id',
'gateway',
'gateway_payment_id',
'gateway_customer_id',
'currency',
'amount',
'fee',
'net_amount',
'status',
'failure_reason',
'payment_method_type',
'payment_method_last4',
'payment_method_brand',
'gateway_response',
'refunded_amount',
'paid_at',
];
protected $casts = [
'amount' => 'decimal:2',
'fee' => 'decimal:2',
'net_amount' => 'decimal:2',
'refunded_amount' => 'decimal:2',
'gateway_response' => 'array',
'paid_at' => 'datetime',
];
// Relationships
public function workspace(): BelongsTo
{
return $this->belongsTo(Workspace::class);
}
public function invoice(): BelongsTo
{
return $this->belongsTo(Invoice::class);
}
public function refunds(): HasMany
{
return $this->hasMany(Refund::class);
}
// Status helpers
public function isPending(): bool
{
return $this->status === 'pending';
}
public function isProcessing(): bool
{
return $this->status === 'processing';
}
public function isSucceeded(): bool
{
return $this->status === 'succeeded';
}
public function isFailed(): bool
{
return $this->status === 'failed';
}
public function isRefunded(): bool
{
return $this->status === 'refunded';
}
public function isPartiallyRefunded(): bool
{
return $this->status === 'partially_refunded';
}
public function canRefund(): bool
{
return $this->isSucceeded() || $this->isPartiallyRefunded();
}
public function isFullyRefunded(): bool
{
return $this->refunded_amount >= $this->amount;
}
public function getRefundableAmount(): float
{
return $this->amount - $this->refunded_amount;
}
// Actions
public function markAsSucceeded(): void
{
$this->update(['status' => 'succeeded']);
}
public function markAsFailed(?string $reason = null): void
{
$this->update([
'status' => 'failed',
'failure_reason' => $reason,
]);
}
public function recordRefund(float $amount): void
{
$newRefundedAmount = $this->refunded_amount + $amount;
$status = $newRefundedAmount >= $this->amount
? 'refunded'
: 'partially_refunded';
$this->update([
'refunded_amount' => $newRefundedAmount,
'status' => $status,
]);
}
// Scopes
public function scopeSucceeded($query)
{
return $query->where('status', 'succeeded');
}
public function scopeForWorkspace($query, int $workspaceId)
{
return $query->where('workspace_id', $workspaceId);
}
public function scopeForGateway($query, string $gateway)
{
return $query->where('gateway', $gateway);
}
}