lthn.io/app/Core/Bouncer/Gate/Models/ActionRequest.php
Claude 41a90cbff8
feat: lthn.io API serving live chain data
Fixed: basePath self→static binding, namespace detection, event wiring,
SQLite cache, file cache driver. All Mod Boot classes converted to
$listens pattern for lifecycle event discovery.

Working endpoints:
- /v1/explorer/info — live chain height, difficulty, aliases
- /v1/explorer/stats — formatted chain statistics
- /v1/names/directory — alias directory grouped by type
- /v1/names/available/{name} — name availability check
- /v1/names/lookup/{name} — name details

Co-Authored-By: Charon <charon@lethean.io>
2026-04-03 17:17:42 +01:00

181 lines
4.5 KiB
PHP

<?php
/*
* Core PHP Framework
*
* Licensed under the European Union Public Licence (EUPL) v1.2.
* See LICENSE file for details.
*/
declare(strict_types=1);
namespace Core\Bouncer\Gate\Models;
use Carbon\Carbon;
use Illuminate\Database\Eloquent\Collection;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
/**
* Action request audit log entry.
*
* Records all action permission checks for auditing and training purposes.
*
* @property int $id
* @property string $method HTTP method (GET, POST, etc.)
* @property string $route Request path
* @property string $action Action identifier
* @property string|null $scope Resource scope
* @property string $guard Guard name
* @property string|null $role User's role at time of request
* @property int|null $user_id User ID if authenticated
* @property string|null $ip_address Client IP
* @property string $status Result: 'allowed', 'denied', 'pending'
* @property bool $was_trained Whether this request triggered training
* @property Carbon $created_at
* @property Carbon $updated_at
*/
class ActionRequest extends Model
{
protected $table = 'core_action_requests';
protected $fillable = [
'method',
'route',
'action',
'scope',
'guard',
'role',
'user_id',
'ip_address',
'status',
'was_trained',
];
protected $casts = [
'was_trained' => 'boolean',
];
/**
* Status constants.
*/
public const STATUS_ALLOWED = 'allowed';
public const STATUS_DENIED = 'denied';
public const STATUS_PENDING = 'pending';
/**
* User who made the request.
*/
public function user(): BelongsTo
{
return $this->belongsTo(config('auth.providers.users.model'), 'user_id');
}
/**
* Log an action request.
*/
public static function log(
string $method,
string $route,
string $action,
string $guard,
string $status,
?string $scope = null,
?string $role = null,
?int $userId = null,
?string $ipAddress = null,
bool $wasTrained = false
): self {
return static::create([
'method' => $method,
'route' => $route,
'action' => $action,
'scope' => $scope,
'guard' => $guard,
'role' => $role,
'user_id' => $userId,
'ip_address' => $ipAddress,
'status' => $status,
'was_trained' => $wasTrained,
]);
}
/**
* Get pending requests (for training review).
*
* @return Collection<int, self>
*/
public static function pending(): Collection
{
return static::where('status', self::STATUS_PENDING)
->orderBy('created_at', 'desc')
->get();
}
/**
* Get denied requests for an action.
*
* @return Collection<int, self>
*/
public static function deniedFor(string $action): Collection
{
return static::where('action', $action)
->where('status', self::STATUS_DENIED)
->orderBy('created_at', 'desc')
->get();
}
/**
* Get requests by user.
*
* @return Collection<int, self>
*/
public static function forUser(int $userId): Collection
{
return static::where('user_id', $userId)
->orderBy('created_at', 'desc')
->get();
}
/**
* Get unique actions that were denied (candidates for training).
*
* @return array<string, array{action: string, count: int, last_at: string}>
*/
public static function deniedActionsSummary(): array
{
return static::where('status', self::STATUS_DENIED)
->selectRaw('action, COUNT(*) as count, MAX(created_at) as last_at')
->groupBy('action')
->orderByDesc('count')
->get()
->keyBy('action')
->map(fn ($row) => [
'action' => $row->action,
'count' => (int) $row->count,
'last_at' => $row->last_at,
])
->toArray();
}
/**
* Prune old request logs.
*/
public static function prune(int $days = 30): int
{
return static::where('created_at', '<', now()->subDays($days))
->delete();
}
/**
* Mark this request as having triggered training.
*/
public function markTrained(): self
{
$this->update(['was_trained' => true]);
return $this;
}
}