2021-10-20 13:16:43 +02:00
|
|
|
<?php
|
|
|
|
|
2021-11-03 16:38:58 +01:00
|
|
|
/* Icinga Web 2 | (c) 2021 Icinga GmbH | GPLv2+ */
|
|
|
|
|
2021-10-20 13:16:43 +02:00
|
|
|
namespace Icinga\Module\Migrate\Clicommands;
|
|
|
|
|
|
|
|
use Icinga\Application\Config;
|
|
|
|
use Icinga\Application\Icinga;
|
|
|
|
use Icinga\Application\Logger;
|
|
|
|
use Icinga\Cli\Command;
|
|
|
|
use Icinga\Data\ConfigObject;
|
|
|
|
use Icinga\Exception\NotReadableError;
|
|
|
|
use Icinga\Exception\NotWritableError;
|
|
|
|
use Icinga\Module\Icingadb\Compat\UrlMigrator;
|
|
|
|
use Icinga\Util\DirectoryIterator;
|
|
|
|
use Icinga\Web\Request;
|
|
|
|
use ipl\Web\Filter\QueryString;
|
|
|
|
use ipl\Web\Url;
|
|
|
|
|
|
|
|
class NavigationCommand extends Command
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Migrate local user monitoring navigation items to the Icinga DB Web actions
|
|
|
|
*
|
|
|
|
* USAGE
|
|
|
|
*
|
|
|
|
* icingacli migrate navigation [options]
|
|
|
|
*
|
|
|
|
* OPTIONS:
|
|
|
|
*
|
|
|
|
* --user=<username> Migrate monitoring navigation items only for
|
|
|
|
* the given user. (Default *)
|
|
|
|
*
|
|
|
|
* --delete Remove the legacy files after successfully
|
|
|
|
* migrated the navigation items.
|
|
|
|
*/
|
|
|
|
public function indexAction()
|
|
|
|
{
|
|
|
|
$moduleManager = Icinga::app()->getModuleManager();
|
|
|
|
if (! $moduleManager->hasEnabled('icingadb')) {
|
|
|
|
Logger::error('Icinga DB module is not enabled. Please verify that the module is installed and enabled.');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$preferencesPath = Config::resolvePath('preferences');
|
|
|
|
$sharedNavigation = Config::resolvePath('navigation');
|
|
|
|
if (! file_exists($preferencesPath) && ! file_exists($sharedNavigation)) {
|
|
|
|
Logger::info('There are no local user navigation items to migrate');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$rc = 0;
|
2021-12-07 17:42:21 +01:00
|
|
|
$user = $this->params->get('user');
|
2021-10-20 13:16:43 +02:00
|
|
|
$directories = new DirectoryIterator($preferencesPath);
|
|
|
|
|
|
|
|
foreach ($directories as $directory) {
|
2021-12-07 17:42:21 +01:00
|
|
|
$username = $user;
|
2021-10-20 13:16:43 +02:00
|
|
|
if ($username !== null && $directories->key() !== $username) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($username === null) {
|
|
|
|
$username = $directories->key();
|
|
|
|
}
|
|
|
|
|
|
|
|
$hostActions = $this->readFromIni($directory . '/host-actions.ini', $rc);
|
|
|
|
$serviceActions = $this->readFromIni($directory . '/service-actions.ini', $rc);
|
|
|
|
|
|
|
|
Logger::info('Migrating monitoring navigation items for user "%s" to the Icinga DB Web actions', $username);
|
|
|
|
|
|
|
|
if (! $hostActions->isEmpty()) {
|
|
|
|
$this->migrateNavigationItems($hostActions, $directory . '/icingadb-host-actions.ini', $rc);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! $serviceActions->isEmpty()) {
|
|
|
|
$this->migrateNavigationItems($serviceActions, $directory . '/icingadb-service-actions.ini', $rc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start migrating shared navigation items
|
|
|
|
$hostActions = $this->readFromIni($sharedNavigation . '/host-actions.ini', $rc);
|
|
|
|
$serviceActions = $this->readFromIni($sharedNavigation . '/service-actions.ini', $rc);
|
|
|
|
|
|
|
|
Logger::info('Migrating shared monitoring navigation items to the Icinga DB Web actions');
|
|
|
|
|
|
|
|
if (! $hostActions->isEmpty()) {
|
|
|
|
$this->migrateNavigationItems($hostActions, $sharedNavigation . '/icingadb-host-actions.ini', $rc);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! $serviceActions->isEmpty()) {
|
|
|
|
$this->migrateNavigationItems($serviceActions, $sharedNavigation . '/icingadb-service-actions.ini', $rc);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($rc > 0) {
|
|
|
|
Logger::error('Failed to migrate some monitoring navigation items');
|
|
|
|
exit($rc);
|
|
|
|
}
|
|
|
|
|
|
|
|
Logger::info('Successfully migrated all local user monitoring navigation items');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Migrate the given config to the given new config path
|
|
|
|
*
|
|
|
|
* @param Config $config
|
|
|
|
* @param string $path
|
|
|
|
* @param int $rc
|
|
|
|
*/
|
|
|
|
private function migrateNavigationItems($config, $path, &$rc)
|
|
|
|
{
|
|
|
|
$deleteLegacyFiles = $this->params->get('delete');
|
|
|
|
$newConfig = $this->readFromIni($path, $rc);
|
|
|
|
$counter = 1;
|
|
|
|
|
|
|
|
/** @var ConfigObject $configObject */
|
|
|
|
foreach ($config->getConfigObject() as $configObject) {
|
|
|
|
// Change the config type from "host-action" to icingadb's new action
|
|
|
|
if (strpos($path, 'icingadb-host-actions') !== false) {
|
|
|
|
$configObject->type = 'icingadb-host-action';
|
|
|
|
} else {
|
|
|
|
$configObject->type = 'icingadb-service-action';
|
|
|
|
}
|
|
|
|
|
|
|
|
$urlString = $configObject->get('url');
|
|
|
|
if ($urlString !== null) {
|
|
|
|
$url = Url::fromPath($urlString, [], new Request());
|
|
|
|
|
|
|
|
try {
|
|
|
|
$urlString = UrlMigrator::transformUrl($url)->getAbsoluteUrl();
|
|
|
|
$configObject->url = rawurldecode($urlString);
|
|
|
|
} catch (\InvalidArgumentException $err) {
|
|
|
|
// Do nothing
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$legacyFilter = $configObject->get('filter');
|
|
|
|
if ($legacyFilter !== null) {
|
|
|
|
$filter = QueryString::parse($legacyFilter);
|
|
|
|
$filter = UrlMigrator::transformFilter($filter);
|
|
|
|
if ($filter !== false) {
|
|
|
|
$configObject->filter = rawurldecode(QueryString::render($filter));
|
|
|
|
} else {
|
|
|
|
unset($configObject->filter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$section = $config->key();
|
|
|
|
while ($newConfig->hasSection($section)) {
|
|
|
|
$section = $config->key() . $counter++;
|
|
|
|
}
|
|
|
|
|
|
|
|
$newConfig->setSection($section, $configObject);
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
if (! $newConfig->isEmpty()) {
|
|
|
|
$newConfig->saveIni();
|
|
|
|
|
|
|
|
// Remove the legacy file only if explicitly requested
|
|
|
|
if ($deleteLegacyFiles) {
|
|
|
|
unlink($config->getConfigFile());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (NotWritableError $error) {
|
|
|
|
Logger::error('%s: %s', $error->getMessage(), $error->getPrevious()->getMessage());
|
|
|
|
$rc = 256;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the navigation items config from the given ini path
|
|
|
|
*
|
|
|
|
* @param string $path Absolute path of the ini file
|
|
|
|
* @param int $rc The return code used to exit the action
|
|
|
|
*
|
|
|
|
* @return Config
|
|
|
|
*/
|
|
|
|
private function readFromIni($path, &$rc)
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
$config = Config::fromIni($path);
|
|
|
|
} catch (NotReadableError $error) {
|
|
|
|
if ($error->getPrevious() !== null) {
|
|
|
|
Logger::error('%s: %s', $error->getMessage(), $error->getPrevious()->getMessage());
|
|
|
|
} else {
|
|
|
|
Logger::error($error->getMessage());
|
|
|
|
}
|
|
|
|
|
|
|
|
$config = new Config();
|
|
|
|
$rc = 128;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $config;
|
|
|
|
}
|
|
|
|
}
|