* @license http://www.gnu.org/licenses/gpl-2.0.txt GPL, version 2 * @author Icinga Development Team * */ // {{{ICINGA_LICENSE_HEADER}}} namespace Icinga\Application; use DateTimeZone; use Exception; use Zend_Config; use Icinga\Application\Modules\Manager as ModuleManager; use Icinga\Application\Config; use Icinga\Data\ResourceFactory; use Icinga\Exception\ConfigurationError; use Icinga\Exception\NotReadableError; use Icinga\Logger\Logger; use Icinga\Util\DateTimeFactory; use Icinga\Util\Translator; /** * This class bootstraps a thin Icinga application layer * * Usage example for CLI: * * use Icinga\Application\Cli; * Cli::start(); * * * Usage example for Icinga Web application: * * use Icinga\Application\Web; * Web::start()->dispatch(); * * * Usage example for Icinga-Web 1.x compatibility mode: * * use Icinga\Application\LegacyWeb; * LegacyWeb::start()->setIcingaWebBasedir(ICINGAWEB_BASEDIR)->dispatch(); * */ abstract class ApplicationBootstrap { /** * Icinga auto loader * * @var Loader */ private $loader; /** * Library directory * * @var string */ private $libDir; /** * Config object * * @var Zend_Config */ protected $config; /** * Configuration directory * * @var string */ private $configDir; /** * Application directory * * @var string */ private $appDir; /** * Module manager * * @var ModuleManager */ private $moduleManager; /** * Flag indicates we're on cli environment * * @var bool */ protected $isCli = false; /** * Flag indicates we're on web environment * * @var bool */ protected $isWeb = false; /** * Constructor */ protected function __construct($configDir = null) { $this->libDir = realpath(__DIR__ . '/../..'); if (!defined('ICINGA_LIBDIR')) { define('ICINGA_LIBDIR', $this->libDir); } // TODO: Make appdir configurable for packagers $this->appDir = realpath($this->libDir. '/../application'); if (!defined('ICINGA_APPDIR')) { define('ICINGA_APPDIR', $this->appDir); } if ($configDir === null) { if (array_key_exists('ICINGAWEB_CONFIGDIR', $_SERVER)) { $configDir = $_SERVER['ICINGAWEB_CONFIGDIR']; } else { $configDir = '/etc/icingaweb'; } } $this->configDir = realpath($configDir); $this->setupAutoloader(); $this->setupZendAutoloader(); Benchmark::measure('Bootstrap, autoloader registered'); Icinga::setApp($this); require_once dirname(__FILE__) . '/functions.php'; } /** * Bootstrap interface method for concrete bootstrap objects * * @return mixed */ abstract protected function bootstrap(); /** * Getter for module manager * * @return ModuleManager */ public function getModuleManager() { return $this->moduleManager; } /** * Getter for class loader * * @return Loader */ public function getLoader() { return $this->loader; } /** * Getter for configuration object * * @return Config */ public function getConfig() { return $this->config; } /** * Flag indicates we're on cli environment * * @return bool */ public function isCli() { return $this->isCli; } /** * Flag indicates we're on web environment * * @return bool */ public function isWeb() { return $this->isWeb; } /** * Getter for application dir * * Optional append sub directory * * @param string $subdir optional subdir * * @return string */ public function getApplicationDir($subdir = null) { return $this->getDirWithSubDir($this->appDir, $subdir); } /** * Getter for config dir * * @param string $subdir * * @return string */ public function getConfigDir($subdir = null) { return $this->getDirWithSubDir($this->configDir, $subdir); } /** * Get the path to the bootstrapping directory. * * This is usually /public for Web and EmbeddedWeb * * @return string */ public function getBootstrapDirecory() { return dirname($_SERVER['SCRIPT_FILENAME']); } /** * Helper to glue directories together * * @param string $dir * @param string $subdir * * @return string */ private function getDirWithSubDir($dir, $subdir = null) { if ($subdir !== null) { $dir .= '/' . ltrim($subdir, '/'); } return $dir; } /** * Starting concrete bootstrap classes * * @param string $configDir * * @return ApplicationBootstrap */ public static function start($configDir = null) { $application = new static($configDir); $application->bootstrap(); return $application; } /** * Setup Icinga auto loader * * @return self */ public function setupAutoloader() { require $this->libDir . '/Icinga/Application/Loader.php'; $this->loader = new Loader(); $this->loader->registerNamespace('Icinga', $this->libDir. '/Icinga'); $this->loader->registerNamespace('Icinga\\Form', $this->appDir. '/forms'); $this->loader->register(); return $this; } /** * Register the Zend Autoloader * * @return self */ protected function setupZendAutoloader() { require_once 'Zend/Loader/Autoloader.php'; \Zend_Loader_Autoloader::getInstance(); // Unfortunately this is needed to get the Zend Plugin loader working: set_include_path( implode( PATH_SEPARATOR, array($this->libDir, get_include_path()) ) ); return $this; } /** * Setup module manager * * @return self */ protected function setupModuleManager() { $this->moduleManager = new ModuleManager( $this, $this->configDir . '/enabledModules', explode( ':', $this->config->global !== null ? $this->config->global->get('modulePath', ICINGA_APPDIR . '/../modules') : ICINGA_APPDIR . '/../modules' ) ); return $this; } /** * Load all enabled modules * * @return self */ protected function loadEnabledModules() { try { $this->moduleManager->loadEnabledModules(); } catch (NotReadableError $e) { Logger::error(new Exception('Cannot load enabled modules. An exception was thrown:', 0, $e)); } return $this; } /** * Setup default logging * * @return self */ protected function setupLogging() { Logger::create( new Zend_Config( array( 'enable' => true, 'level' => Logger::$ERROR, 'type' => 'syslog', 'facility' => 'LOG_USER', 'application' => 'Icinga Web' ) ) ); return $this; } /** * Load Configuration * * @return self */ protected function loadConfig() { Config::$configDir = $this->configDir; try { $this->config = Config::app(); } catch (NotReadableError $e) { Logger::error(new Exception('Cannot load application configuration. An exception was thrown:', 0, $e)); $this->config = new Zend_Config(array()); } return $this; } /** * Error handling configuration * * @return self */ protected function setupErrorHandling() { error_reporting(E_ALL | E_NOTICE); ini_set('display_startup_errors', 1); ini_set('display_errors', 1); return $this; } /** * Set up logger * * @return self */ protected function setupLogger() { if ($this->config->logging !== null) { try { Logger::create($this->config->logging); } catch (ConfigurationError $e) { Logger::error($e); } } return $this; } /** * Set up the resource factory * * @return self */ protected function setupResourceFactory() { try { $config = Config::app('resources'); ResourceFactory::setConfig($config); } catch (NotReadableError $e) { Logger::error( new Exception('Cannot load resource configuration. An exception was thrown:', 0, $e) ); } return $this; } /** * Setup default timezone * * @return self * @throws ConfigurationError if the timezone in config.ini isn't valid */ protected function setupTimezone() { $timeZoneString = $this->config->global !== null ? $this->config->global->get('timezone', 'UTC') : 'UTC'; try { $tz = new DateTimeZone($timeZoneString); } catch (Exception $e) { throw new ConfigurationError(t('Invalid timezone') . ' "' . $timeZoneString . '"'); } date_default_timezone_set($timeZoneString); DateTimeFactory::setConfig(array('timezone' => $tz)); return $this; } /** * Setup internationalization using gettext * * Uses the language defined in the global config or the default one * * @return self */ protected function setupInternationalization() { try { Translator::setupLocale( $this->config->global !== null ? $this->config->global->get('language', Translator::DEFAULT_LOCALE) : Translator::DEFAULT_LOCALE ); } catch (Exception $error) { Logger::error($error); } $localeDir = $this->getApplicationDir('locale'); if (file_exists($localeDir) && is_dir($localeDir)) { Translator::registerDomain('icinga', $localeDir); } return $this; } }