-
-
Save Maxlab/bfe026c48840a7c963bbe3debc8defcd to your computer and use it in GitHub Desktop.
See comments
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<?php | |
namespace App\Console\Commands\Foundation; | |
use App\Console\Commands\Command; | |
use App\User; | |
use Illuminate\Contracts\Auth\Access\Gate; | |
class AbilitiesListCommand extends Command | |
{ | |
/** | |
* The name and signature of the console command. | |
* | |
* @var string | |
*/ | |
protected $signature = 'ability:list | |
{user? : The ID of the user for ability checking} | |
{--M|model=* : The `ModelClass:id` for checking policy methods}'; | |
/** | |
* The console command description. | |
* | |
* @var string | |
*/ | |
protected $description = 'List all registered abilities'; | |
/** | |
* @var \Illuminate\Contracts\Auth\Access\Gate | |
*/ | |
protected $gate; | |
/** | |
* @var | |
*/ | |
protected $user; | |
/** | |
* List of model for checking policy methods. | |
* | |
* @var array | |
*/ | |
protected $models = []; | |
/** | |
* {@inheritdoc} | |
*/ | |
public function handle(Gate $gate) | |
{ | |
$this->initGate($gate); | |
$this->initModels(); | |
$this->output->title('Abilities list'); | |
collect() | |
->merge($this->extractBaseAbilities()) | |
->merge($this->extractPoliciesAbilities()) | |
->each([$this, 'outputAbility']); | |
$this->output->newLine(); | |
} | |
/** | |
* @param object $ability | |
*/ | |
public function outputAbility($ability) | |
{ | |
if (null === $this->user) { | |
$this->line("* {$ability->ability}"); | |
} elseif (null === $ability->resolved) { | |
$this->line("[?] {$ability->ability}"); | |
} elseif (true === $ability->resolved) { | |
$this->info("[+] {$ability->ability}"); | |
} else { | |
$this->warn("[-] {$ability->ability}"); | |
} | |
} | |
/** | |
* @param Gate $gate | |
*/ | |
protected function initGate(Gate $gate) | |
{ | |
$this->gate = $gate; | |
if (! $this->hasArgument('user')) { | |
return; | |
} | |
$this->user = User::find($this->argument('user')); | |
$this->gate = $this->gate->forUser($this->user); | |
} | |
/** | |
* | |
*/ | |
protected function initModels() | |
{ | |
$models = (array) $this->option('model'); | |
foreach ($models as $model) { | |
list($class, $id) = explode(':', $model, 2); | |
$this->models[$class] = compact('class', 'id'); | |
} | |
} | |
/** | |
* @return \Illuminate\Support\Collection | |
*/ | |
protected function extractBaseAbilities() | |
{ | |
$reflected_gate = new \ReflectionObject($this->gate); | |
$abilities = $reflected_gate->getProperty('abilities'); | |
$abilities->setAccessible(true); | |
return collect($abilities->getValue($this->gate)) | |
->keys() | |
->transform(function ($ability) { | |
return $this->makeResolvedBaseAbility($ability); | |
}); | |
} | |
/** | |
* @return \Illuminate\Support\Collection | |
*/ | |
protected function extractPoliciesAbilities() | |
{ | |
$reflected_gate = new \ReflectionObject($this->gate); | |
$abilities = collect(); | |
$policies = $reflected_gate->getProperty('policies'); | |
$policies->setAccessible(true); | |
$policies = $policies->getValue($this->gate); | |
foreach ($policies as $class => $policy) { | |
$abilities = $abilities->merge($this->extractPolicyAbilities($class, $policy)); | |
} | |
return $abilities; | |
} | |
/** | |
* @param string $modelClass | |
* @param string $policyClass | |
* @return \Illuminate\Support\Collection | |
*/ | |
protected function extractPolicyAbilities($modelClass, $policyClass) | |
{ | |
$abilities = collect(); | |
$reflected_policy = new \ReflectionClass($policyClass); | |
$methods = $reflected_policy->getMethods(\ReflectionMethod::IS_PUBLIC); | |
foreach ($methods as $method) { | |
$ability = str_replace_first('_', '-', snake_case($method->getName(), '_')); | |
$abilities->push($this->makeResolvedPolicyAbility($modelClass, $policyClass, $ability)); | |
} | |
return $abilities; | |
} | |
/** | |
* @param string $ability | |
* @return object | |
*/ | |
protected function makeResolvedBaseAbility($ability) | |
{ | |
$resolved = null; | |
if (null !== $this->user) { | |
$resolved = $this->resolveBaseAbility($ability); | |
} | |
return (object) compact('ability', 'resolved'); | |
} | |
/** | |
* @param string $modelClass | |
* @param string $policyClass | |
* @param string $ability | |
* @return object | |
*/ | |
protected function makeResolvedPolicyAbility($modelClass, $policyClass, $ability) | |
{ | |
$resolved = null; | |
if (null !== $this->user) { | |
$resolved = $this->resolvePolicyAbility($modelClass, $policyClass, $ability); | |
} | |
$ability = strtolower(class_basename($modelClass)) . ':' . $ability; | |
return (object) compact('ability', 'resolved'); | |
} | |
/** | |
* @param string $ability | |
* @return bool|null | |
*/ | |
protected function resolveBaseAbility($ability) | |
{ | |
try { | |
return $this->gate->check($ability); | |
} catch (\Throwable $e) { | |
return null; | |
} | |
} | |
/** | |
* @param string $modelClass | |
* @param string $policyClass | |
* @param string $ability | |
* @return bool|null | |
*/ | |
protected function resolvePolicyAbility($modelClass, $policyClass, $ability) | |
{ | |
$base_class = class_basename($modelClass); | |
$formatted_class = strtolower($base_class); | |
try { | |
return $this->gate->check($ability, $modelClass); | |
} catch (\Throwable $e) { | |
$model = $this->getModel($modelClass, [$base_class, $formatted_class]); | |
if (null === $model) { | |
return null; | |
} | |
try { | |
return $this->gate->check($ability, $model); | |
} catch (\Throwable $e) { | |
return null; | |
} | |
} | |
return null; | |
} | |
/** | |
* @param $class | |
* @param $aliases | |
* @return null|\Illuminate\Database\Eloquent\Model | |
*/ | |
protected function getModel($class, $aliases) | |
{ | |
$classes = array_merge([$class], $aliases); | |
foreach ($classes as $class_alias) { | |
if (isset($this->models[$class_alias])) { | |
$instance = $this->models[$class_alias]['instance'] ?? $class::find($this->models[$class_alias]['id']); | |
return $this->models[$class_alias]['instance'] = $instance; | |
} | |
} | |
return null; | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<?php | |
namespace App\Console\Commands\Foundation; | |
use App\Console\Commands\Command; | |
use Cron\CronExpression; | |
use Illuminate\Console\Scheduling\Schedule; | |
use Illuminate\Support\Collection; | |
class ScheduleListCommand extends Command | |
{ | |
/** | |
* The name and signature of the console command. | |
* | |
* @var string | |
*/ | |
protected $signature = 'schedule:list {--s|time-sort : Sort list by datetime of nex call}'; | |
/** | |
* The console command description. | |
* | |
* @var string | |
*/ | |
protected $description = 'List all schedule commands'; | |
/** | |
* The schedule instance. | |
* | |
* @var \Illuminate\Console\Scheduling\Schedule | |
*/ | |
protected $schedule; | |
/** | |
* Create a new command instance. | |
* | |
* @param Schedule $schedule | |
*/ | |
public function __construct(Schedule $schedule) | |
{ | |
$this->schedule = $schedule; | |
parent::__construct(); | |
} | |
/** | |
* Execute the console command. | |
* | |
* @return mixed | |
*/ | |
public function handle() | |
{ | |
$events = $this->schedule->events(); | |
$table = new Collection(); | |
foreach ($events as $event) { | |
$table[] = [ | |
'command' => $this->prepareCommandToDisplay($event->getSummaryForDisplay()), | |
'expression' => $event->getExpression(), | |
'next' => $this->getCommandNextRunTime($event->getExpression()) | |
]; | |
} | |
if ($this->option('time-sort')) { | |
$table = $table->sortBy('next'); | |
} | |
$table->transform(function ($command) { | |
$command['next'] = $command['next']->format('H:i:s Y-m-d'); | |
return $command; | |
}); | |
$this->table(['command', 'expression', 'next time'], $table->toArray()); | |
} | |
/** | |
* Possible change command text for simple view. | |
* | |
* @param $eventCommand | |
* @return mixed | |
*/ | |
protected function prepareCommandToDisplay($eventCommand) | |
{ | |
return $eventCommand; | |
} | |
/** | |
* Get next time command run datetime. | |
* | |
* @param $eventCronExpression | |
* @return \DateTime | |
*/ | |
protected function getCommandNextRunTime($eventCronExpression) | |
{ | |
return CronExpression::factory($eventCronExpression)->getNextRunDate(); | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment