Viewing file: PermissionRegistrar.php (11.71 KB) -rw-r--r-- Select action/file-type: (+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
<?php
namespace Spatie\Permission;
use Illuminate\Cache\CacheManager; use Illuminate\Contracts\Auth\Access\Authorizable; use Illuminate\Contracts\Auth\Access\Gate; use Illuminate\Contracts\Cache\Repository; use Illuminate\Contracts\Cache\Store; use Illuminate\Database\Eloquent\Collection; use Illuminate\Database\Eloquent\Model; use Spatie\Permission\Contracts\Permission; use Spatie\Permission\Contracts\Role;
class PermissionRegistrar { protected Repository $cache;
protected CacheManager $cacheManager;
protected string $permissionClass;
protected string $roleClass;
/** @var Collection|array|null */ protected $permissions;
public string $pivotRole;
public string $pivotPermission;
/** @var \DateInterval|int */ public $cacheExpirationTime;
public bool $teams;
public string $teamsKey;
protected string|int|null $teamId = null;
public string $cacheKey;
private array $cachedRoles = [];
private array $alias = [];
private array $except = [];
private array $wildcardPermissionsIndex = [];
/** * PermissionRegistrar constructor. */ public function __construct(CacheManager $cacheManager) { $this->permissionClass = config('permission.models.permission'); $this->roleClass = config('permission.models.role');
$this->cacheManager = $cacheManager; $this->initializeCache(); }
public function initializeCache(): void { $this->cacheExpirationTime = config('permission.cache.expiration_time') ?: \DateInterval::createFromDateString('24 hours');
$this->teams = config('permission.teams', false); $this->teamsKey = config('permission.column_names.team_foreign_key', 'team_id');
$this->cacheKey = config('permission.cache.key');
$this->pivotRole = config('permission.column_names.role_pivot_key') ?: 'role_id'; $this->pivotPermission = config('permission.column_names.permission_pivot_key') ?: 'permission_id';
$this->cache = $this->getCacheStoreFromConfig(); }
protected function getCacheStoreFromConfig(): Repository { // the 'default' fallback here is from the permission.php config file, // where 'default' means to use config(cache.default) $cacheDriver = config('permission.cache.store', 'default');
// when 'default' is specified, no action is required since we already have the default instance if ($cacheDriver === 'default') { return $this->cacheManager->store(); }
// if an undefined cache store is specified, fallback to 'array' which is Laravel's closest equiv to 'none' if (! \array_key_exists($cacheDriver, config('cache.stores'))) { $cacheDriver = 'array'; }
return $this->cacheManager->store($cacheDriver); }
/** * Set the team id for teams/groups support, this id is used when querying permissions/roles * * @param int|string|\Illuminate\Database\Eloquent\Model|null $id */ public function setPermissionsTeamId($id): void { if ($id instanceof \Illuminate\Database\Eloquent\Model) { $id = $id->getKey(); } $this->teamId = $id; }
/** * @return int|string|null */ public function getPermissionsTeamId() { return $this->teamId; }
/** * Register the permission check method on the gate. * We resolve the Gate fresh here, for benefit of long-running instances. */ public function registerPermissions(Gate $gate): bool { $gate->before(function (Authorizable $user, string $ability, array &$args = []) { if (is_string($args[0] ?? null) && ! class_exists($args[0])) { $guard = array_shift($args); } if (method_exists($user, 'checkPermissionTo')) { return $user->checkPermissionTo($ability, $guard ?? null) ?: null; } });
return true; }
/** * Flush the cache. */ public function forgetCachedPermissions() { $this->permissions = null; $this->forgetWildcardPermissionIndex();
return $this->cache->forget($this->cacheKey); }
public function forgetWildcardPermissionIndex(?Model $record = null): void { if ($record) { unset($this->wildcardPermissionsIndex[get_class($record)][$record->getKey()]);
return; }
$this->wildcardPermissionsIndex = []; }
public function getWildcardPermissionIndex(Model $record): array { if (isset($this->wildcardPermissionsIndex[get_class($record)][$record->getKey()])) { return $this->wildcardPermissionsIndex[get_class($record)][$record->getKey()]; }
return $this->wildcardPermissionsIndex[get_class($record)][$record->getKey()] = app($record->getWildcardClass(), ['record' => $record])->getIndex(); }
/** * Clear already-loaded permissions collection. * This is only intended to be called by the PermissionServiceProvider on boot, * so that long-running instances like Octane or Swoole don't keep old data in memory. */ public function clearPermissionsCollection(): void { $this->permissions = null; $this->wildcardPermissionsIndex = []; }
/** * @deprecated * * @alias of clearPermissionsCollection() */ public function clearClassPermissions() { $this->clearPermissionsCollection(); }
/** * Load permissions from cache * And turns permissions array into a \Illuminate\Database\Eloquent\Collection */ private function loadPermissions(): void { if ($this->permissions) { return; }
$this->permissions = $this->cache->remember( $this->cacheKey, $this->cacheExpirationTime, fn () => $this->getSerializedPermissionsForCache() );
$this->alias = $this->permissions['alias'];
$this->hydrateRolesCache();
$this->permissions = $this->getHydratedPermissionCollection();
$this->cachedRoles = $this->alias = $this->except = []; }
/** * Get the permissions based on the passed params. */ public function getPermissions(array $params = [], bool $onlyOne = false): Collection { $this->loadPermissions();
$method = $onlyOne ? 'first' : 'filter';
$permissions = $this->permissions->$method(static function ($permission) use ($params) { foreach ($params as $attr => $value) { if ($permission->getAttribute($attr) != $value) { return false; } }
return true; });
if ($onlyOne) { $permissions = new Collection($permissions ? [$permissions] : []); }
return $permissions; }
public function getPermissionClass(): string { return $this->permissionClass; }
public function setPermissionClass($permissionClass) { $this->permissionClass = $permissionClass; config()->set('permission.models.permission', $permissionClass); app()->bind(Permission::class, $permissionClass);
return $this; }
public function getRoleClass(): string { return $this->roleClass; }
public function setRoleClass($roleClass) { $this->roleClass = $roleClass; config()->set('permission.models.role', $roleClass); app()->bind(Role::class, $roleClass);
return $this; }
public function getCacheRepository(): Repository { return $this->cache; }
public function getCacheStore(): Store { return $this->cache->getStore(); }
protected function getPermissionsWithRoles(): Collection { return $this->permissionClass::select()->with('roles')->get(); }
/** * Changes array keys with alias */ private function aliasedArray($model): array { return collect(is_array($model) ? $model : $model->getAttributes())->except($this->except) ->keyBy(fn ($value, $key) => $this->alias[$key] ?? $key) ->all(); }
/** * Array for cache alias */ private function aliasModelFields($newKeys = []): void { $i = 0; $alphas = ! count($this->alias) ? range('a', 'h') : range('j', 'p');
foreach (array_keys($newKeys->getAttributes()) as $value) { if (! isset($this->alias[$value])) { $this->alias[$value] = $alphas[$i++] ?? $value; } }
$this->alias = array_diff_key($this->alias, array_flip($this->except)); }
/* * Make the cache smaller using an array with only required fields */ private function getSerializedPermissionsForCache(): array { $this->except = config('permission.cache.column_names_except', ['created_at', 'updated_at', 'deleted_at']);
$permissions = $this->getPermissionsWithRoles() ->map(function ($permission) { if (! $this->alias) { $this->aliasModelFields($permission); }
return $this->aliasedArray($permission) + $this->getSerializedRoleRelation($permission); })->all(); $roles = array_values($this->cachedRoles); $this->cachedRoles = [];
return ['alias' => array_flip($this->alias)] + compact('permissions', 'roles'); }
private function getSerializedRoleRelation($permission): array { if (! $permission->roles->count()) { return []; }
if (! isset($this->alias['roles'])) { $this->alias['roles'] = 'r'; $this->aliasModelFields($permission->roles[0]); }
return [ 'r' => $permission->roles->map(function ($role) { if (! isset($this->cachedRoles[$role->getKey()])) { $this->cachedRoles[$role->getKey()] = $this->aliasedArray($role); }
return $role->getKey(); })->all(), ]; }
private function getHydratedPermissionCollection(): Collection { $permissionInstance = (new ($this->getPermissionClass())())->newInstance([], true);
return Collection::make(array_map( fn ($item) => (clone $permissionInstance) ->setRawAttributes($this->aliasedArray(array_diff_key($item, ['r' => 0])), true) ->setRelation('roles', $this->getHydratedRoleCollection($item['r'] ?? [])), $this->permissions['permissions'] )); }
private function getHydratedRoleCollection(array $roles): Collection { return Collection::make(array_values( array_intersect_key($this->cachedRoles, array_flip($roles)) )); }
private function hydrateRolesCache(): void { $roleInstance = (new ($this->getRoleClass())())->newInstance([], true);
array_map(function ($item) use ($roleInstance) { $role = (clone $roleInstance) ->setRawAttributes($this->aliasedArray($item), true); $this->cachedRoles[$role->getKey()] = $role; }, $this->permissions['roles']);
$this->permissions['roles'] = []; }
public static function isUid($value): bool { if (! is_string($value) || empty(trim($value))) { return false; }
// check if is UUID/GUID $uid = preg_match('/^[\da-f]{8}-[\da-f]{4}-[\da-f]{4}-[\da-f]{4}-[\da-f]{12}$/iD', $value) > 0; if ($uid) { return true; }
// check if is ULID $ulid = strlen($value) == 26 && strspn($value, '0123456789ABCDEFGHJKMNPQRSTVWXYZabcdefghjkmnpqrstvwxyz') == 26 && $value[0] <= '7'; if ($ulid) { return true; }
return false; } }
|