Files
fotospiel-app/app/Filament/Clusters/DailyOps/Resources/TenantCheckoutHealths/Tables/TenantCheckoutHealthTable.php
Codex Agent 7262617897
Some checks failed
linter / quality (push) Has been cancelled
tests / ci (push) Has been cancelled
tests / ui (push) Has been cancelled
Refactor checkout health resource
2026-02-05 10:54:05 +01:00

268 lines
12 KiB
PHP

<?php
namespace App\Filament\Clusters\DailyOps\Resources\TenantCheckoutHealths\Tables;
use App\Filament\Clusters\DailyOps\Resources\TenantCheckoutHealths\TenantCheckoutHealthResource;
use App\Models\CheckoutSession;
use App\Models\Tenant;
use Filament\Forms\Components\TextInput;
use Filament\Tables\Columns\IconColumn;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Filters\Filter;
use Filament\Tables\Filters\SelectFilter;
use Filament\Tables\Table;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Support\Carbon;
class TenantCheckoutHealthTable
{
public static function configure(Table $table): Table
{
return $table
->columns([
TextColumn::make('name')
->label(__('admin.common.tenant'))
->searchable()
->sortable(),
TextColumn::make('slug')
->label(__('admin.common.slug'))
->searchable()
->toggleable(isToggledHiddenByDefault: true),
TextColumn::make('contact_email')
->label(__('admin.tenants.fields.contact_email'))
->searchable()
->toggleable(isToggledHiddenByDefault: true),
TextColumn::make('subscription_status')
->label('Subscription')
->badge()
->color(fn (?string $state) => match ($state) {
'active' => 'success',
'suspended' => 'warning',
'expired' => 'danger',
'free' => 'gray',
default => 'gray',
}),
TextColumn::make('active_reseller_package')
->label('Active package')
->getStateUsing(fn (Tenant $record) => $record->activeResellerPackage?->package?->name ?? '—')
->badge()
->color(fn (string $state) => $state === '—' ? 'gray' : 'success')
->toggleable(isToggledHiddenByDefault: true),
IconColumn::make('status_mismatch')
->label('Status mismatch')
->boolean()
->getStateUsing(fn (Tenant $record) => self::hasStatusMismatch($record)),
TextColumn::make('last_checkout_transaction_at')
->label('Last transaction')
->badge()
->color(fn (?Carbon $state) => self::transactionAgeColor($state))
->getStateUsing(fn (Tenant $record) => $record->last_checkout_transaction_at
? Carbon::parse($record->last_checkout_transaction_at)
: null)
->formatStateUsing(fn (?Carbon $state) => $state?->diffForHumans() ?? '—')
->toggleable(isToggledHiddenByDefault: true),
TextColumn::make('checkout_transaction_count_window')
->label('Transactions (30d)')
->default('0')
->sortable()
->toggleable(),
TextColumn::make('checkout_transaction_total_window')
->label('Total (30d)')
->default(0)
->money('EUR')
->sortable()
->toggleable(),
TextColumn::make('checkout_refund_count_window')
->label('Refunds (30d)')
->badge()
->color(fn (?int $state) => $state && $state > 0 ? 'danger' : 'gray')
->default('0')
->sortable()
->toggleable(isToggledHiddenByDefault: true),
TextColumn::make('checkout_refund_total_window')
->label('Refund total (30d)')
->default(0)
->money('EUR')
->sortable()
->toggleable(isToggledHiddenByDefault: true),
TextColumn::make('checkout_requires_action_count')
->label('Checkout action required')
->badge()
->color(fn (?int $state) => $state && $state > 0 ? 'warning' : 'gray')
->default('0')
->sortable()
->toggleable(isToggledHiddenByDefault: true),
TextColumn::make('checkout_processing_count')
->label('Checkout processing')
->badge()
->color(fn (?int $state) => $state && $state > 0 ? 'warning' : 'gray')
->default('0')
->sortable()
->toggleable(isToggledHiddenByDefault: true),
TextColumn::make('checkout_expired_count')
->label('Checkout expired')
->badge()
->color(fn (?int $state) => $state && $state > 0 ? 'danger' : 'gray')
->default('0')
->sortable()
->toggleable(isToggledHiddenByDefault: true),
TextColumn::make('checkout_transaction_count')
->label('Transactions (all)')
->default('0')
->sortable()
->toggleable(isToggledHiddenByDefault: true),
TextColumn::make('checkout_transaction_total')
->label('Total (all)')
->default(0)
->money('EUR')
->sortable()
->toggleable(isToggledHiddenByDefault: true),
])
->filters([
Filter::make('status_mismatch')
->label('Status mismatch')
->indicator('Status mismatch')
->query(fn (Builder $query) => self::applyStatusMismatchFilter($query)),
Filter::make('active_package')
->label('Active package')
->indicator('Active package')
->query(fn (Builder $query) => $query->whereHas('activeResellerPackage', function (Builder $query) {
$query->where('active', true)
->where(function (Builder $query) {
$query->whereNull('expires_at')
->orWhere('expires_at', '>=', now());
});
})),
Filter::make('not_suspended_or_deleted')
->label('Not suspended/deleted')
->indicator('Not suspended/deleted')
->query(fn (Builder $query) => $query
->where('is_suspended', false)
->whereNull('pending_deletion_at')
->whereNull('anonymized_at')),
Filter::make('checkout_transaction_stale')
->label('Stale transactions')
->indicator('Stale transactions')
->query(function (Builder $query): Builder {
$cutoff = now()->subDays(TenantCheckoutHealthResource::TRANSACTION_WINDOW_DAYS);
$provider = TenantCheckoutHealthResource::provider();
return $query
->whereHas('purchases', fn (Builder $query) => $query->where('provider', $provider))
->whereDoesntHave('purchases', fn (Builder $query) => $query
->where('provider', $provider)
->where('purchased_at', '>=', $cutoff));
}),
Filter::make('checkout_attention')
->label('Checkout attention')
->indicator('Checkout attention')
->query(fn (Builder $query) => $query->whereHas('checkoutSessions', function (Builder $query) {
$query->where('provider', TenantCheckoutHealthResource::provider())
->where(function (Builder $query) {
$query->whereIn('status', [
CheckoutSession::STATUS_REQUIRES_CUSTOMER_ACTION,
CheckoutSession::STATUS_PROCESSING,
])
->orWhere(function (Builder $query) {
$query->whereNotIn('status', [
CheckoutSession::STATUS_COMPLETED,
CheckoutSession::STATUS_CANCELLED,
])
->whereNotNull('expires_at')
->where('expires_at', '<', now());
});
});
})),
Filter::make('refund_spike')
->label('Refund spike (30d)')
->form([
TextInput::make('min_refunds')
->label('Minimum refunds')
->numeric()
->default(1)
->minValue(1),
])
->indicateUsing(function (array $data): ?string {
$min = (int) ($data['min_refunds'] ?? 0);
return $min > 0 ? "Refunds >= {$min} (30d)" : null;
})
->query(function (Builder $query, array $data): Builder {
$min = (int) ($data['min_refunds'] ?? 0);
if ($min < 1) {
return $query;
}
$cutoff = now()->subDays(TenantCheckoutHealthResource::TRANSACTION_WINDOW_DAYS);
$provider = TenantCheckoutHealthResource::provider();
return $query->whereHas('purchases', fn (Builder $query) => $query
->where('provider', $provider)
->where('refunded', true)
->where('purchased_at', '>=', $cutoff), '>=', $min);
}),
SelectFilter::make('subscription_status')
->label('Subscription')
->options([
'active' => 'Active',
'suspended' => 'Suspended',
'expired' => 'Expired',
'free' => 'Free',
]),
])
->actions([]);
}
private static function hasStatusMismatch(Tenant $record): bool
{
$hasActivePackage = (bool) ($record->has_active_reseller_package ?? $record->activeResellerPackage);
$status = (string) ($record->subscription_status ?? '');
$expiresAt = $record->subscription_expires_at;
if ($status === 'active' && ! $hasActivePackage) {
return true;
}
if ($status !== 'active' && $hasActivePackage) {
return true;
}
if ($status === 'active' && $expiresAt && $expiresAt->isPast()) {
return true;
}
return false;
}
private static function applyStatusMismatchFilter(Builder $query): Builder
{
return $query->where(function (Builder $query) {
$query->where(function (Builder $query) {
$query->where('subscription_status', 'active')
->whereDoesntHave('activeResellerPackage');
})->orWhere(function (Builder $query) {
$query->where('subscription_status', '!=', 'active')
->whereHas('activeResellerPackage');
})->orWhere(function (Builder $query) {
$query->where('subscription_status', 'active')
->whereNotNull('subscription_expires_at')
->where('subscription_expires_at', '<', now());
});
});
}
private static function transactionAgeColor(?Carbon $state): string
{
if (! $state) {
return 'gray';
}
if ($state->lt(now()->subDays(TenantCheckoutHealthResource::TRANSACTION_WINDOW_DAYS))) {
return 'danger';
}
return 'success';
}
}