2024-09-21 12:27:41 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
|
|
namespace App\Models;
|
|
|
|
|
|
2025-04-04 09:30:45 +02:00
|
|
|
use App\Enums\RolePermissionModels;
|
|
|
|
|
use App\Enums\RolePermissionPrefixes;
|
2026-01-27 11:36:07 +01:00
|
|
|
use App\Enums\TablerIcon;
|
|
|
|
|
use BackedEnum;
|
2025-09-24 13:34:19 +02:00
|
|
|
use Illuminate\Database\Eloquent\Collection;
|
2025-04-28 10:20:33 -04:00
|
|
|
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
2025-05-05 12:58:55 +02:00
|
|
|
use Illuminate\Database\Eloquent\Relations\BelongsToMany;
|
2026-03-17 09:09:01 +01:00
|
|
|
use Illuminate\Support\Carbon;
|
2025-09-24 13:34:19 +02:00
|
|
|
use Spatie\Permission\Models\Permission;
|
2024-09-21 12:27:41 +02:00
|
|
|
use Spatie\Permission\Models\Role as BaseRole;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @property int $id
|
|
|
|
|
* @property string $name
|
|
|
|
|
* @property string $guard_name
|
2026-03-17 09:09:01 +01:00
|
|
|
* @property Carbon|null $created_at
|
|
|
|
|
* @property Carbon|null $updated_at
|
|
|
|
|
* @property-read NodeRole|null $pivot
|
|
|
|
|
* @property-read Collection<int, Node> $nodes
|
|
|
|
|
* @property-read int|null $nodes_count
|
|
|
|
|
* @property-read Collection<int, Permission> $permissions
|
|
|
|
|
* @property-read int|null $permissions_count
|
|
|
|
|
* @property-read Collection<int, User> $users
|
|
|
|
|
* @property-read int|null $users_count
|
|
|
|
|
*
|
|
|
|
|
* @method static \Database\Factories\RoleFactory factory($count = null, $state = [])
|
|
|
|
|
* @method static \Illuminate\Database\Eloquent\Builder<static>|Role newModelQuery()
|
|
|
|
|
* @method static \Illuminate\Database\Eloquent\Builder<static>|Role newQuery()
|
|
|
|
|
* @method static \Illuminate\Database\Eloquent\Builder<static>|Role permission($permissions, $without = false)
|
|
|
|
|
* @method static \Illuminate\Database\Eloquent\Builder<static>|Role query()
|
|
|
|
|
* @method static \Illuminate\Database\Eloquent\Builder<static>|Role whereCreatedAt($value)
|
|
|
|
|
* @method static \Illuminate\Database\Eloquent\Builder<static>|Role whereGuardName($value)
|
|
|
|
|
* @method static \Illuminate\Database\Eloquent\Builder<static>|Role whereId($value)
|
|
|
|
|
* @method static \Illuminate\Database\Eloquent\Builder<static>|Role whereName($value)
|
|
|
|
|
* @method static \Illuminate\Database\Eloquent\Builder<static>|Role whereUpdatedAt($value)
|
|
|
|
|
* @method static \Illuminate\Database\Eloquent\Builder<static>|Role withoutPermission($permissions)
|
2024-09-21 12:27:41 +02:00
|
|
|
*/
|
|
|
|
|
class Role extends BaseRole
|
|
|
|
|
{
|
2025-04-28 10:20:33 -04:00
|
|
|
use HasFactory;
|
|
|
|
|
|
2024-09-21 12:27:41 +02:00
|
|
|
public const RESOURCE_NAME = 'role';
|
|
|
|
|
|
|
|
|
|
public const ROOT_ADMIN = 'Root Admin';
|
|
|
|
|
|
2024-10-29 18:29:25 +01:00
|
|
|
public const DEFAULT_GUARD_NAME = 'web';
|
|
|
|
|
|
2024-09-21 12:27:41 +02:00
|
|
|
public const MODEL_SPECIFIC_PERMISSIONS = [
|
|
|
|
|
'egg' => [
|
|
|
|
|
'import',
|
|
|
|
|
'export',
|
|
|
|
|
],
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
public const SPECIAL_PERMISSIONS = [
|
|
|
|
|
'settings' => [
|
|
|
|
|
'view',
|
|
|
|
|
'update',
|
|
|
|
|
],
|
2025-01-06 15:43:29 +01:00
|
|
|
'health' => [
|
|
|
|
|
'view',
|
|
|
|
|
],
|
2025-08-31 12:59:48 +02:00
|
|
|
'activityLog' => [
|
feat: Client UI translate to Filament (from React) (#416)
* Add new panel
* Add some basic resource pages
* Wip
* Wip terminal
* Wip
* Add new panel
* Add some basic resource pages
* Wip
* [Sub-Users] Add Invite
TODO: The logic with permissions
* [Sub-Users] Fix Creation
* [Cron] Add basics
* Add basic auth and messages
* Add basic buttons
* WIP on issue/353
* WIP on issue/353
* Add Database page
* Update Database Page
* Start of Backup Page
* Composer Update
* Changes
* Send input
* Remove this includes
* Better offline handling
* Consolidate top nav config
* Update Backups Page
* Update Backups
* Change name
* Add Assign All, Layout Fixes.
* conflict
* update schedule pages
* fix phpstan
* update pint.json
* add cron presets to schedule
* fix tests
* fix task creation
* schedules: disable task creation if limit is reached & disable backup action if backup limit is 0
* update activity pages
* update resources
* Update Edit User
TODO: actually save permissions when they're changed.
TODO: Figure out why Control does not update it's state... but the rest do...
* .... Sure it works.
TODO: Update permissions when you save editing a sub user.
* user: update canAccessPanel & canAccessTenant
* add helper to convert bytes into readable format
* very basic file explorer
* files: fix some stuff & remove dummy data
* files: better error handling
* files: basic file editor
* files: add some actions
* File manager updates
* files: fix paths
* Revery Composer Upgrade, Fixes SQLite
* fix: Pint (#517)
feat: MenuItems to and from admin
* Update File Editing
Updated File Editing to its own page,
Added Permission checks for file manager.
Co-authored-by: Boy132 <Boy132@users.noreply.github.com>
* add enum for editor langs
* files: add upload & pull actions
* fix build
* files: handle images
* Update to Filament v3.2.98
* files: add remaining actions
* use `authorize` instead of `hidden`
* fix canAccessTenant
* update date columns
* files: testing & fixes
* Fix File Names
Co-authored-by: lancepioch <git@lance.sh>
* Combine Pull/Upload
* Fix BulkDelete
* Uncontained tabs
* Hide Lang Selection, Move Actions
* Update Monaco, more custom
* Add livewire config
livewire limits uploads to 12MB... who knows why...
Fixed uploading a single files failing
* files: fix record url
* basic setup for settings & startup page
* make abstract class for simple app pages
* Basic Startup Page
* Update nav sort
* small cleanup
* startup: fix shouldHideComponent & getSelectOptionsFromRules
* startup: fix non editable fields & set default value
* startup: add todo for save button
* Save Variables after update & off click
Variables update when the user clicks off the input.
* Notifications are cool
* Add rule validation
* Sort variables by sortid
* pint
* Settings Page + Startup Changes
* settings: cleanup
* refactor: use server model for ServerFormPage (formerly known as SimplePage)
* Use Repeater for variables
* Add Network, Remove breadcrumbs
* Add paginated to file explorer
* Fix updating variables
* Add link to go to new client area
* fix after merge
* Add graphs to console page
Graphs still need to get the data from the web socket.
* fix pint & phpstan
* fix authorizeAccess for EditFiles and Startup page
* Fix rules on startup page
* Update console size
* Fix node name
* add "global search" to files list
requires https://github.com/pelican-dev/wings/pull/44
* remove debug dummy data
* update view action on ListServers
* enable SPA mode for app panel
* remove colors from app panel
they are defined globally in AppServiceProvider
* update global search ui a bit
(to be replaced with a custom page that is similar to the list files table)
* add own page for global search
untested - and route needs cleanup (if possible)
* fix File getRows
* remove "path" from SearchFiles (for now)
* fix caching for searched files
* add title and breadcrumbs to global search page
* make cpu & memory charts on console page working
* fix phpstan
* add missing import
* cleanup console views & widgets
* add overview stats to console
* don't be so lazy, console!
* make history working
* decode data to get array
* add missing On
* fix json_decode
* change polling to 1 sec
* hide "0" cpu/ memory
* add data to network chart
* Remove data labels
* fix data on network chart
* fix data on network chart (2nd try)
* WIP Network Stats
* Remove test
* Change MaxWidth
* run pint
* fix phpstan
* Fix storeStats cast
* make $data a string
this time for real
* update visible check for "admin" menu item
* remove account widget
* rebrand "Dashboard" to "Server List"
WIP - doesn't look good but is somewhat working
* fix canAccessPanel
* separate server list into own panel
* change path to avoid conflicts with old client area (and remove sidebar width)
* display correct icon and color on server list entries
* show total memory if server is offline
* replace custom server list page with ListRecords page
* fix tests
* fix namespace
* remove "open" button and make whole column clickable
* Update EditProfile
* run pint
* fix access to server list
* add new login page to panels
* fix next_run_at for new schedules
* use new DateTimeColumn
* add own column for file bytes
* return to server list when clicking title
* fix console loading
* handle server with "conflict state"
* add banner if server is in "conflict state"
* fix phpstan
* update docker image select
* fix permission checks on Settings & Startup pages
* fix query for activity log page
* fix activity log not being logged
* adjust ListActivities
* fix phpstan
* fix pint
* fix profile menu item link on server panel
* add ip tooltip to activity logs (and role permission)
* change backup icon
* update navigation sort
* general code cleanup
* more cleanup
* Disable Restart/Stop if server is offline
* Change rename notification
* Remove negation on abort_unless
* Add notification on save
* Single disabled closure & comment unused import
* Add required to Server Name & Nullable to description
* mutateFormDataBeforeSave doesn't work since we use forceFill
* Fix web socket connection not existing.
* Fix some subuser permissions
* add permission checks to resources
* do not allow self-deletion
* Update editing file permissions
* Fix of the previous fix
* add service for subuser updating
* Only allow save if they have file_update
* Remove unused import
* Update backup delete button
* Add Delete, remove bulks
* Update Database page
* Use Allocation Permissions
* add canAccess check to startup
* Add Permission checks to Settings page
* add service for subuser deletion
* Remove Kill permission
* Updates
* fix move files
* add redirects
* fix phpstan
* activity: remove properties from tans for now
* If alias, use that, else ip
---------
Co-authored-by: notCharles <charles@pelican.dev>
Co-authored-by: Boy132 <mail@boy132.de>
Co-authored-by: Senna <62171904+Poseidon281@users.noreply.github.com>
Co-authored-by: Boy132 <Boy132@users.noreply.github.com>
Co-authored-by: RMartinOscar <40749467+RMartinOscar@users.noreply.github.com>
2024-11-30 22:13:45 -05:00
|
|
|
'seeIps',
|
|
|
|
|
],
|
2025-10-08 06:18:20 -04:00
|
|
|
'panelLog' => [
|
|
|
|
|
'view',
|
|
|
|
|
],
|
2025-12-20 00:32:13 +01:00
|
|
|
'plugin' => [
|
|
|
|
|
'viewList',
|
|
|
|
|
'create',
|
|
|
|
|
'update',
|
|
|
|
|
'delete',
|
|
|
|
|
],
|
2024-09-21 12:27:41 +02:00
|
|
|
];
|
|
|
|
|
|
2025-11-27 22:51:57 +00:00
|
|
|
public const MODEL_ICONS = [
|
2026-01-27 11:36:07 +01:00
|
|
|
'health' => TablerIcon::Heart,
|
|
|
|
|
'activityLog' => TablerIcon::Stack,
|
|
|
|
|
'panelLog' => TablerIcon::FileInfo,
|
2025-11-27 22:51:57 +00:00
|
|
|
];
|
|
|
|
|
|
2025-04-04 09:30:45 +02:00
|
|
|
/** @var array<string, array<string>> */
|
|
|
|
|
protected static array $customPermissions = [];
|
|
|
|
|
|
|
|
|
|
/** @param array<string, array<string>> $customPermissions */
|
|
|
|
|
public static function registerCustomPermissions(array $customPermissions): void
|
|
|
|
|
{
|
|
|
|
|
static::$customPermissions = [
|
|
|
|
|
...static::$customPermissions,
|
|
|
|
|
...$customPermissions,
|
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static function registerCustomDefaultPermissions(string $model): void
|
|
|
|
|
{
|
|
|
|
|
$permissions = [];
|
|
|
|
|
|
|
|
|
|
foreach (RolePermissionPrefixes::cases() as $prefix) {
|
|
|
|
|
$permissions[] = $prefix->value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static::registerCustomPermissions([
|
|
|
|
|
$model => $permissions,
|
|
|
|
|
]);
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-27 11:36:07 +01:00
|
|
|
/** @var array<string, string|BackedEnum> */
|
2025-11-27 22:51:57 +00:00
|
|
|
protected static array $customModelIcons = [];
|
|
|
|
|
|
2026-01-27 11:36:07 +01:00
|
|
|
public static function registerCustomModelIcon(string $model, string|BackedEnum $icon): void
|
2025-11-27 22:51:57 +00:00
|
|
|
{
|
|
|
|
|
static::$customModelIcons[$model] = $icon;
|
|
|
|
|
}
|
|
|
|
|
|
2025-04-04 09:30:45 +02:00
|
|
|
/** @return array<string, array<string>> */
|
|
|
|
|
public static function getPermissionList(): array
|
|
|
|
|
{
|
|
|
|
|
$allPermissions = [];
|
|
|
|
|
|
|
|
|
|
// Standard permissions for our default model
|
|
|
|
|
foreach (RolePermissionModels::cases() as $model) {
|
|
|
|
|
$allPermissions[$model->value] ??= [];
|
|
|
|
|
|
|
|
|
|
foreach (RolePermissionPrefixes::cases() as $prefix) {
|
|
|
|
|
array_push($allPermissions[$model->value], $prefix->value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (array_key_exists($model->value, Role::MODEL_SPECIFIC_PERMISSIONS)) {
|
|
|
|
|
foreach (static::MODEL_SPECIFIC_PERMISSIONS[$model->value] as $permission) {
|
|
|
|
|
array_push($allPermissions[$model->value], $permission);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Special permissions for our default models
|
|
|
|
|
foreach (static::SPECIAL_PERMISSIONS as $model => $prefixes) {
|
|
|
|
|
$allPermissions[$model] ??= [];
|
|
|
|
|
|
|
|
|
|
foreach ($prefixes as $prefix) {
|
|
|
|
|
array_push($allPermissions[$model], $prefix);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Custom third party permissions
|
|
|
|
|
foreach (static::$customPermissions as $model => $prefixes) {
|
|
|
|
|
$allPermissions[$model] ??= [];
|
|
|
|
|
|
|
|
|
|
foreach ($prefixes as $prefix) {
|
|
|
|
|
array_push($allPermissions[$model], $prefix);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
foreach ($allPermissions as $model => $permissions) {
|
|
|
|
|
$allPermissions[$model] = array_unique($permissions);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $allPermissions;
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-27 11:36:07 +01:00
|
|
|
public static function getModelIcon(string $model): null|string|BackedEnum
|
2025-11-27 22:51:57 +00:00
|
|
|
{
|
|
|
|
|
$customModels = array_merge(static::MODEL_ICONS, static::$customModelIcons);
|
|
|
|
|
|
|
|
|
|
if (array_key_exists($model, $customModels)) {
|
|
|
|
|
return $customModels[$model];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$model = ucwords($model);
|
|
|
|
|
|
|
|
|
|
if (class_exists($class = '\\App\\Filament\\Admin\\Resources\\' . $model . 's\\' . $model . 'Resource')) {
|
|
|
|
|
return $class::getNavigationIcon();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (class_exists($class = '\\App\\Filament\\Admin\\Pages\\' . $model)) {
|
|
|
|
|
return $class::getNavigationIcon();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (class_exists($class = '\\App\\Filament\\Server\\Resources\\' . $model . 's\\' . $model . 'Resource')) {
|
|
|
|
|
return $class::getNavigationIcon();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-21 12:27:41 +02:00
|
|
|
public function isRootAdmin(): bool
|
|
|
|
|
{
|
|
|
|
|
return $this->name === self::ROOT_ADMIN;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static function getRootAdmin(): self
|
|
|
|
|
{
|
|
|
|
|
/** @var self $role */
|
2024-10-29 18:29:25 +01:00
|
|
|
$role = self::findOrCreate(self::ROOT_ADMIN, self::DEFAULT_GUARD_NAME);
|
2024-09-21 12:27:41 +02:00
|
|
|
|
|
|
|
|
return $role;
|
|
|
|
|
}
|
2025-05-05 12:58:55 +02:00
|
|
|
|
|
|
|
|
public function nodes(): BelongsToMany
|
|
|
|
|
{
|
|
|
|
|
return $this->belongsToMany(Node::class, NodeRole::class);
|
|
|
|
|
}
|
2024-09-21 12:27:41 +02:00
|
|
|
}
|