2013-10-22 14:26:45 +02:00
|
|
|
<?php
|
2016-02-08 15:41:00 +01:00
|
|
|
/* Icinga Web 2 | (c) 2013 Icinga Development Team | GPLv2+ */
|
2013-10-22 14:26:45 +02:00
|
|
|
|
|
|
|
namespace Icinga\Cli;
|
|
|
|
|
|
|
|
use Icinga\Application\ApplicationBootstrap as App;
|
2015-07-24 16:19:20 +02:00
|
|
|
use Icinga\Exception\IcingaException;
|
2016-02-25 17:31:09 +01:00
|
|
|
use Icinga\Exception\NotReadableError;
|
2013-10-22 14:26:45 +02:00
|
|
|
use Icinga\Exception\ProgrammingError;
|
|
|
|
use Icinga\Cli\Params;
|
|
|
|
use Icinga\Cli\Screen;
|
2014-01-22 13:08:20 +01:00
|
|
|
use Icinga\Cli\Command;
|
2013-10-22 14:26:45 +02:00
|
|
|
use Icinga\Cli\Documentation;
|
|
|
|
use Exception;
|
|
|
|
|
|
|
|
/**
|
2014-03-06 13:08:11 +01:00
|
|
|
*
|
2013-10-22 14:26:45 +02:00
|
|
|
*/
|
|
|
|
class Loader
|
|
|
|
{
|
|
|
|
protected $app;
|
|
|
|
|
|
|
|
protected $docs;
|
|
|
|
|
|
|
|
protected $commands;
|
|
|
|
|
|
|
|
protected $modules;
|
|
|
|
|
|
|
|
protected $moduleCommands = array();
|
|
|
|
|
|
|
|
protected $coreAppDir;
|
|
|
|
|
|
|
|
protected $screen;
|
|
|
|
|
|
|
|
protected $moduleName;
|
|
|
|
|
|
|
|
protected $commandName;
|
|
|
|
|
|
|
|
protected $actionName; // Should this better be moved to the Command?
|
|
|
|
|
|
|
|
/**
|
|
|
|
* [$command] = $class;
|
|
|
|
*/
|
|
|
|
protected $commandClassMap = array();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* [$command] = $file;
|
|
|
|
*/
|
|
|
|
protected $commandFileMap = array();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* [$module][$command] = $class;
|
|
|
|
*/
|
|
|
|
protected $moduleClassMap = array();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* [$module][$command] = $file;
|
|
|
|
*/
|
|
|
|
protected $moduleFileMap = array();
|
|
|
|
|
|
|
|
protected $commandInstances = array();
|
|
|
|
|
|
|
|
protected $moduleInstances = array();
|
|
|
|
|
|
|
|
protected $lastSuggestions = array();
|
|
|
|
|
|
|
|
public function __construct(App $app)
|
|
|
|
{
|
|
|
|
$this->app = $app;
|
2014-11-14 16:01:40 +01:00
|
|
|
$this->coreAppDir = $app->getApplicationDir('clicommands');
|
2013-10-22 14:26:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Screen shortcut
|
|
|
|
*
|
|
|
|
* @return Screen
|
|
|
|
*/
|
|
|
|
protected function screen()
|
|
|
|
{
|
|
|
|
if ($this->screen === null) {
|
2020-11-30 15:31:22 +01:00
|
|
|
$this->screen = Screen::instance(STDERR);
|
2013-10-22 14:26:45 +02:00
|
|
|
}
|
2020-11-30 15:31:22 +01:00
|
|
|
|
2013-10-22 14:26:45 +02:00
|
|
|
return $this->screen;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Documentation shortcut
|
|
|
|
*
|
|
|
|
* @return Documentation
|
|
|
|
*/
|
|
|
|
protected function docs()
|
|
|
|
{
|
|
|
|
if ($this->docs === null) {
|
|
|
|
$this->docs = new Documentation($this->app);
|
|
|
|
}
|
|
|
|
return $this->docs;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Show given message and exit
|
|
|
|
*
|
|
|
|
* @param string $msg message to show
|
|
|
|
*/
|
|
|
|
public function fail($msg)
|
|
|
|
{
|
2020-11-24 16:47:59 +01:00
|
|
|
fprintf(STDERR, "%s: %s\n", $this->screen()->colorize('ERROR', 'red'), $msg);
|
2013-10-22 14:26:45 +02:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2013-10-22 15:47:31 +02:00
|
|
|
public function getModuleName()
|
|
|
|
{
|
|
|
|
return $this->moduleName;
|
|
|
|
}
|
|
|
|
|
2014-06-05 01:04:11 +02:00
|
|
|
public function setModuleName($name)
|
|
|
|
{
|
|
|
|
$this->moduleName = $name;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2013-10-22 15:47:31 +02:00
|
|
|
public function getCommandName()
|
|
|
|
{
|
|
|
|
return $this->commandName;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getActionName()
|
|
|
|
{
|
|
|
|
return $this->actionName;
|
|
|
|
}
|
|
|
|
|
2013-10-22 14:26:45 +02:00
|
|
|
public function getCommandInstance($command)
|
|
|
|
{
|
|
|
|
if (! array_key_exists($command, $this->commandInstances)) {
|
|
|
|
$this->assertCommandExists($command);
|
|
|
|
require_once $this->commandFileMap[$command];
|
|
|
|
$className = $this->commandClassMap[$command];
|
|
|
|
$this->commandInstances[$command] = new $className(
|
|
|
|
$this->app,
|
|
|
|
null,
|
|
|
|
$command,
|
|
|
|
null,
|
|
|
|
false
|
2014-02-21 14:07:32 +01:00
|
|
|
);
|
2013-10-22 14:26:45 +02:00
|
|
|
}
|
|
|
|
return $this->commandInstances[$command];
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getModuleCommandInstance($module, $command)
|
|
|
|
{
|
|
|
|
if (! array_key_exists($command, $this->moduleInstances[$module])) {
|
|
|
|
$this->assertModuleCommandExists($module, $command);
|
|
|
|
require_once $this->moduleFileMap[$module][$command];
|
|
|
|
$className = $this->moduleClassMap[$module][$command];
|
|
|
|
$this->moduleInstances[$module][$command] = new $className(
|
|
|
|
$this->app,
|
|
|
|
$module,
|
|
|
|
$command,
|
|
|
|
null,
|
|
|
|
false
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return $this->moduleInstances[$module][$command];
|
|
|
|
}
|
|
|
|
|
2014-01-22 12:45:42 +01:00
|
|
|
public function getLastSuggestions()
|
|
|
|
{
|
|
|
|
return $this->lastSuggestions;
|
|
|
|
}
|
|
|
|
|
2013-10-22 14:26:45 +02:00
|
|
|
public function showLastSuggestions()
|
|
|
|
{
|
|
|
|
if (! empty($this->lastSuggestions)) {
|
|
|
|
foreach ($this->lastSuggestions as & $s) {
|
|
|
|
$s = $this->screen()->colorize($s, 'lightblue');
|
|
|
|
}
|
2020-11-24 16:47:59 +01:00
|
|
|
fprintf(
|
|
|
|
STDERR,
|
2013-10-22 14:26:45 +02:00
|
|
|
"Did you mean %s?\n",
|
|
|
|
implode(" or ", $this->lastSuggestions)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function parseParams(Params $params = null)
|
|
|
|
{
|
|
|
|
if ($params === null) {
|
|
|
|
$params = $this->app->getParams();
|
|
|
|
}
|
2014-06-05 01:04:11 +02:00
|
|
|
|
|
|
|
if ($this->moduleName === null) {
|
2014-06-05 01:37:36 +02:00
|
|
|
$first = $params->shift();
|
|
|
|
if (! $first) {
|
|
|
|
return;
|
|
|
|
}
|
2014-06-05 01:04:11 +02:00
|
|
|
$found = $this->resolveName($first);
|
|
|
|
} else {
|
|
|
|
$found = $this->moduleName;
|
|
|
|
}
|
2013-10-22 14:26:45 +02:00
|
|
|
if (! $found) {
|
|
|
|
$msg = "There is no such module or command: '$first'";
|
2020-11-24 16:47:59 +01:00
|
|
|
fprintf(STDERR, "%s: %s\n", $this->screen()->colorize('ERROR', 'red'), $msg);
|
2013-10-22 14:26:45 +02:00
|
|
|
$this->showLastSuggestions();
|
2020-11-24 16:47:59 +01:00
|
|
|
fwrite(STDERR, "\n");
|
2013-10-22 14:26:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$obj = null;
|
|
|
|
if ($this->hasCommand($found)) {
|
|
|
|
$this->commandName = $found;
|
|
|
|
$obj = $this->getCommandInstance($this->commandName);
|
|
|
|
} elseif ($this->hasModule($found)) {
|
|
|
|
$this->moduleName = $found;
|
|
|
|
$command = $this->resolveModuleCommandName($found, $params->shift());
|
|
|
|
if ($command) {
|
|
|
|
$this->commandName = $command;
|
|
|
|
$obj = $this->getModuleCommandInstance(
|
|
|
|
$this->moduleName,
|
|
|
|
$this->commandName
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ($obj !== null) {
|
|
|
|
$action = $this->resolveObjectActionName(
|
|
|
|
$obj,
|
|
|
|
$params->getStandalone()
|
|
|
|
);
|
|
|
|
if ($obj->hasActionName($action)) {
|
|
|
|
$this->actionName = $action;
|
|
|
|
$params->shift();
|
|
|
|
} elseif ($obj->hasDefaultActionName()) {
|
|
|
|
$this->actionName = $obj->getDefaultActionName();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function handleParams(Params $params = null)
|
|
|
|
{
|
|
|
|
$this->parseParams($params);
|
|
|
|
$this->dispatch();
|
|
|
|
}
|
|
|
|
|
2014-02-14 14:19:56 +01:00
|
|
|
public function dispatch(Params $overrideParams = null)
|
2013-10-22 14:26:45 +02:00
|
|
|
{
|
|
|
|
if ($this->commandName === null) {
|
2020-11-24 16:47:59 +01:00
|
|
|
fwrite(STDERR, $this->docs()->usage($this->moduleName));
|
2013-10-22 14:26:45 +02:00
|
|
|
return false;
|
|
|
|
} elseif ($this->actionName === null) {
|
2020-11-24 16:47:59 +01:00
|
|
|
fwrite(STDERR, $this->docs()->usage($this->moduleName, $this->commandName));
|
2013-10-22 14:26:45 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
if ($this->moduleName) {
|
|
|
|
$this->app->getModuleManager()->loadModule($this->moduleName);
|
|
|
|
$obj = $this->getModuleCommandInstance(
|
|
|
|
$this->moduleName,
|
|
|
|
$this->commandName
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$obj = $this->getCommandInstance($this->commandName);
|
|
|
|
}
|
2014-02-14 14:19:56 +01:00
|
|
|
if ($overrideParams !== null) {
|
|
|
|
$obj->setParams($overrideParams);
|
|
|
|
}
|
2013-10-22 14:26:45 +02:00
|
|
|
$obj->init();
|
|
|
|
return $obj->{$this->actionName . 'Action'}();
|
|
|
|
} catch (Exception $e) {
|
2014-01-22 13:08:20 +01:00
|
|
|
if ($obj && $obj instanceof Command && $obj->showTrace()) {
|
2020-11-24 16:47:59 +01:00
|
|
|
fwrite(STDERR, $this->formatTrace($e->getTrace()));
|
2014-01-22 13:08:20 +01:00
|
|
|
}
|
2015-07-24 15:58:32 +02:00
|
|
|
|
2015-07-24 16:19:20 +02:00
|
|
|
$this->fail(IcingaException::describe($e));
|
2013-10-22 14:26:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function searchMatch($needle, $haystack)
|
|
|
|
{
|
2022-01-19 12:08:28 +01:00
|
|
|
if ($needle === null) {
|
|
|
|
$needle = '';
|
|
|
|
}
|
|
|
|
|
2015-08-21 13:34:03 +02:00
|
|
|
$this->lastSuggestions = preg_grep(sprintf('/^%s.*$/', preg_quote($needle, '/')), $haystack);
|
|
|
|
$match = array_search($needle, $haystack, true);
|
|
|
|
if (false !== $match) {
|
|
|
|
return $haystack[$match];
|
2013-10-22 14:26:45 +02:00
|
|
|
}
|
2016-02-10 13:28:34 +01:00
|
|
|
if (count($this->lastSuggestions) === 1) {
|
|
|
|
$lastSuggestions = array_values($this->lastSuggestions);
|
|
|
|
return $lastSuggestions[0];
|
|
|
|
}
|
2013-10-22 14:26:45 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function resolveName($name)
|
|
|
|
{
|
|
|
|
return $this->searchMatch(
|
|
|
|
$name,
|
|
|
|
array_merge($this->listCommands(), $this->listModules())
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function resolveCommandName($name)
|
|
|
|
{
|
|
|
|
return $this->searchMatch($name, $this->listCommands());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function resolveModuleName($name)
|
|
|
|
{
|
|
|
|
return $this->searchMatch($name, $this->listModules());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function resolveModuleCommandName($module, $name)
|
|
|
|
{
|
|
|
|
return $this->searchMatch($name, $this->listModuleCommands($module));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function resolveObjectActionName($obj, $name)
|
|
|
|
{
|
|
|
|
return $this->searchMatch($name, $obj->listActions());
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function assertModuleExists($module)
|
|
|
|
{
|
|
|
|
if (! $this->hasModule($module)) {
|
|
|
|
throw new ProgrammingError(
|
2014-08-26 11:15:19 +02:00
|
|
|
'There is no such module: %s',
|
|
|
|
$module
|
2013-10-22 14:26:45 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function assertCommandExists($command)
|
|
|
|
{
|
|
|
|
if (! $this->hasCommand($command)) {
|
|
|
|
throw new ProgrammingError(
|
2014-08-26 11:15:19 +02:00
|
|
|
'There is no such command: %s',
|
|
|
|
$command
|
2013-10-22 14:26:45 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function assertModuleCommandExists($module, $command)
|
|
|
|
{
|
|
|
|
$this->assertModuleExists($module);
|
|
|
|
if (! $this->hasModuleCommand($module, $command)) {
|
|
|
|
throw new ProgrammingError(
|
2014-08-26 11:15:19 +02:00
|
|
|
'The module \'%s\' has no such command: %s',
|
|
|
|
$module,
|
|
|
|
$command
|
2013-10-22 14:26:45 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-22 13:08:20 +01:00
|
|
|
protected function formatTrace($trace)
|
|
|
|
{
|
|
|
|
$output = array();
|
2014-02-21 14:07:32 +01:00
|
|
|
foreach ($trace as $i => $step) {
|
2014-01-22 13:08:20 +01:00
|
|
|
$object = '';
|
|
|
|
if (isset($step['object']) && is_object($step['object'])) {
|
|
|
|
$object = sprintf('[%s]', get_class($step['object'])) . $step['type'];
|
|
|
|
} elseif (! empty($step['object'])) {
|
|
|
|
$object = (string) $step['object'] . $step['type'];
|
|
|
|
}
|
2022-01-19 10:56:19 +01:00
|
|
|
if (isset($step['args']) && is_array($step['args'])) {
|
2014-02-21 14:07:32 +01:00
|
|
|
foreach ($step['args'] as & $arg) {
|
|
|
|
if (is_object($arg)) {
|
2014-01-22 13:08:20 +01:00
|
|
|
$arg = sprintf('[%s]', get_class($arg));
|
|
|
|
}
|
|
|
|
if (is_string($arg)) {
|
|
|
|
$arg = preg_replace('~\n~', '\n', $arg);
|
2014-02-21 14:07:32 +01:00
|
|
|
if (strlen($arg) > 50) {
|
|
|
|
$arg = substr($arg, 0, 47) . '...';
|
|
|
|
}
|
2014-01-22 13:08:20 +01:00
|
|
|
$arg = "'" . $arg . "'";
|
|
|
|
}
|
2014-02-21 14:07:32 +01:00
|
|
|
if ($arg === null) {
|
|
|
|
$arg = 'NULL';
|
|
|
|
}
|
|
|
|
if (is_bool($arg)) {
|
|
|
|
$arg = $arg ? 'TRUE' : 'FALSE';
|
|
|
|
}
|
2014-01-22 13:08:20 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$step['args'] = array();
|
|
|
|
}
|
|
|
|
$args = $step['args'];
|
|
|
|
foreach ($args as & $v) {
|
|
|
|
if (is_array($v)) {
|
|
|
|
$v = var_export($v, 1);
|
|
|
|
} else {
|
|
|
|
$v = (string) $v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$output[$i] = sprintf(
|
|
|
|
'#%d %s:%d %s%s(%s)',
|
|
|
|
$i,
|
|
|
|
isset($step['file']) ? preg_replace(
|
|
|
|
'~.+/library/~',
|
|
|
|
'library/',
|
|
|
|
$step['file']
|
|
|
|
) : '[unknown file]',
|
|
|
|
isset($step['line']) ? $step['line'] : '0',
|
|
|
|
$object,
|
|
|
|
$step['function'],
|
|
|
|
implode(', ', $args)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return implode(PHP_EOL, $output) . PHP_EOL;
|
|
|
|
}
|
|
|
|
|
2013-10-22 14:26:45 +02:00
|
|
|
public function hasCommand($name)
|
|
|
|
{
|
|
|
|
return in_array($name, $this->listCommands());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function hasModule($name)
|
|
|
|
{
|
|
|
|
return in_array($name, $this->listModules());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function hasModuleCommand($module, $name)
|
|
|
|
{
|
|
|
|
return in_array($name, $this->listModuleCommands($module));
|
|
|
|
}
|
|
|
|
|
|
|
|
public function listModules()
|
|
|
|
{
|
|
|
|
if ($this->modules === null) {
|
|
|
|
$this->modules = array();
|
2016-02-25 17:31:09 +01:00
|
|
|
try {
|
|
|
|
$this->modules = array_unique(array_merge(
|
|
|
|
$this->app->getModuleManager()->listEnabledModules(),
|
|
|
|
$this->app->getModuleManager()->listLoadedModules()
|
|
|
|
));
|
|
|
|
} catch (NotReadableError $e) {
|
|
|
|
$this->fail($e->getMessage());
|
|
|
|
}
|
2013-10-22 14:26:45 +02:00
|
|
|
}
|
|
|
|
return $this->modules;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function retrieveCommandsFromDir($dirname)
|
|
|
|
{
|
|
|
|
$commands = array();
|
|
|
|
if (! @file_exists($dirname) || ! is_readable($dirname)) {
|
|
|
|
return $commands;
|
|
|
|
}
|
|
|
|
|
|
|
|
$base = opendir($dirname);
|
|
|
|
if ($base === false) {
|
|
|
|
return $commands;
|
|
|
|
}
|
|
|
|
while (false !== ($dir = readdir($base))) {
|
|
|
|
if ($dir[0] === '.') {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (preg_match('~^([A-Za-z0-9]+)Command\.php$~', $dir, $m)) {
|
|
|
|
$cmd = strtolower($m[1]);
|
|
|
|
$commands[] = $cmd;
|
|
|
|
}
|
2019-02-20 05:48:04 +01:00
|
|
|
}
|
|
|
|
closedir($base);
|
|
|
|
sort($commands);
|
2013-10-22 14:26:45 +02:00
|
|
|
return $commands;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function listCommands()
|
|
|
|
{
|
|
|
|
if ($this->commands === null) {
|
|
|
|
$this->commands = array();
|
|
|
|
$ns = 'Icinga\\Clicommands\\';
|
|
|
|
$this->commands = $this->retrieveCommandsFromDir($this->coreAppDir);
|
|
|
|
foreach ($this->commands as $cmd) {
|
|
|
|
$this->commandClassMap[$cmd] = $ns . ucfirst($cmd) . 'Command';
|
|
|
|
$this->commandFileMap[$cmd] = $this->coreAppDir . '/' . ucfirst($cmd) . 'Command.php';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $this->commands;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function listModuleCommands($module)
|
|
|
|
{
|
|
|
|
if (! array_key_exists($module, $this->moduleCommands)) {
|
|
|
|
$ns = 'Icinga\\Module\\' . ucfirst($module) . '\\Clicommands\\';
|
|
|
|
$this->assertModuleExists($module);
|
|
|
|
$manager = $this->app->getModuleManager();
|
2014-01-22 12:36:11 +01:00
|
|
|
$manager->loadModule($module);
|
2013-10-22 14:26:45 +02:00
|
|
|
$dir = $manager->getModuleDir($module) . '/application/clicommands';
|
|
|
|
$this->moduleCommands[$module] = $this->retrieveCommandsFromDir($dir);
|
|
|
|
$this->moduleInstances[$module] = array();
|
|
|
|
foreach ($this->moduleCommands[$module] as $cmd) {
|
|
|
|
$this->moduleClassMap[$module][$cmd] = $ns . ucfirst($cmd) . 'Command';
|
|
|
|
$this->moduleFileMap[$module][$cmd] = $dir . '/' . ucfirst($cmd) . 'Command.php';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $this->moduleCommands[$module];
|
|
|
|
}
|
|
|
|
}
|