2013-06-03 17:02:08 +02:00
|
|
|
<?php
|
2013-08-20 19:33:15 +02:00
|
|
|
// {{{ICINGA_LICENSE_HEADER}}}
|
2013-06-27 13:33:46 +02:00
|
|
|
/**
|
2013-10-23 15:10:33 +02:00
|
|
|
* This file is part of Icinga Web 2.
|
2013-08-20 19:33:15 +02:00
|
|
|
*
|
2013-10-23 15:10:33 +02:00
|
|
|
* Icinga Web 2 - Head for multiple monitoring backends.
|
2013-08-20 19:33:15 +02:00
|
|
|
* Copyright (C) 2013 Icinga Development Team
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
2013-10-23 15:10:33 +02:00
|
|
|
* @copyright 2013 Icinga Development Team <info@icinga.org>
|
|
|
|
* @license http://www.gnu.org/licenses/gpl-2.0.txt GPL, version 2
|
|
|
|
* @author Icinga Development Team <info@icinga.org>
|
|
|
|
*
|
2013-06-27 13:33:46 +02:00
|
|
|
*/
|
2013-08-20 19:33:15 +02:00
|
|
|
// {{{ICINGA_LICENSE_HEADER}}}
|
|
|
|
|
2013-06-03 17:02:08 +02:00
|
|
|
namespace Icinga\Web;
|
2013-06-07 13:29:11 +02:00
|
|
|
|
2014-02-18 18:48:13 +01:00
|
|
|
use Icinga\Application\Logger as Log;
|
|
|
|
use Icinga\Exception\ProgrammingError;
|
|
|
|
use stdClass;
|
2013-06-03 17:02:08 +02:00
|
|
|
|
2013-06-27 13:42:48 +02:00
|
|
|
/**
|
2013-06-27 13:33:46 +02:00
|
|
|
* Icinga Web Hook registry
|
2013-06-27 13:42:48 +02:00
|
|
|
*
|
2013-06-27 13:33:46 +02:00
|
|
|
* Modules making use of predefined hooks have to use this registry
|
2013-06-27 13:42:48 +02:00
|
|
|
*
|
2013-06-27 13:33:46 +02:00
|
|
|
* Usage:
|
|
|
|
* <code>
|
|
|
|
* Hook::register('grapher', 'My\\Grapher\\Class');
|
|
|
|
* </code>
|
2013-06-27 13:42:48 +02:00
|
|
|
*/
|
2013-06-03 17:02:08 +02:00
|
|
|
class Hook
|
|
|
|
{
|
2013-06-07 13:29:11 +02:00
|
|
|
/**
|
2013-06-27 13:33:46 +02:00
|
|
|
* Our hook name registry
|
|
|
|
*
|
2013-06-07 13:29:11 +02:00
|
|
|
* @var array
|
|
|
|
*/
|
2013-06-03 17:02:08 +02:00
|
|
|
protected static $hooks = array();
|
2013-06-07 13:29:11 +02:00
|
|
|
|
|
|
|
/**
|
2013-06-27 13:33:46 +02:00
|
|
|
* Hooks that have already been instantiated
|
|
|
|
*
|
2013-06-07 13:29:11 +02:00
|
|
|
* @var array
|
|
|
|
*/
|
2013-06-03 17:02:08 +02:00
|
|
|
protected static $instances = array();
|
2013-06-07 13:29:11 +02:00
|
|
|
|
2013-06-28 11:16:35 +02:00
|
|
|
/**
|
2013-08-20 19:33:15 +02:00
|
|
|
* Namespace prefix
|
|
|
|
*
|
2013-06-28 11:16:35 +02:00
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
public static $BASE_NS = 'Icinga\\Web\\Hook\\';
|
|
|
|
|
2013-08-20 19:33:15 +02:00
|
|
|
/**
|
|
|
|
* Reset object state
|
|
|
|
*/
|
2013-06-28 11:16:35 +02:00
|
|
|
public static function clean()
|
|
|
|
{
|
|
|
|
self::$hooks = array();
|
|
|
|
self::$instances = array();
|
|
|
|
self::$BASE_NS = 'Icinga\\Web\\Hook\\';
|
|
|
|
}
|
|
|
|
|
2013-06-07 13:29:11 +02:00
|
|
|
/**
|
2013-06-27 13:33:46 +02:00
|
|
|
* Whether someone registered itself for the given hook name
|
|
|
|
*
|
2013-08-20 19:33:15 +02:00
|
|
|
* @param string $name One of the predefined hook names
|
2013-06-07 13:29:11 +02:00
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
2013-06-27 13:33:46 +02:00
|
|
|
public static function has($name)
|
2013-06-03 17:02:08 +02:00
|
|
|
{
|
2013-06-27 13:33:46 +02:00
|
|
|
return array_key_exists($name, self::$hooks);
|
2013-06-03 17:02:08 +02:00
|
|
|
}
|
|
|
|
|
2013-06-07 13:29:11 +02:00
|
|
|
/**
|
2013-06-27 13:33:46 +02:00
|
|
|
* Get the first registered instance for the given hook name
|
|
|
|
*
|
|
|
|
* TODO: Multiple instances are not handled yet
|
|
|
|
* TODO: Should return some kind of a hook interface
|
|
|
|
*
|
|
|
|
* @param string $name One of the predefined hook names
|
|
|
|
*
|
|
|
|
* @return mixed
|
2013-08-20 19:33:15 +02:00
|
|
|
* @throws ProgrammingError
|
2013-06-07 13:29:11 +02:00
|
|
|
*/
|
2013-06-27 13:33:46 +02:00
|
|
|
public static function get($name)
|
2013-06-03 17:02:08 +02:00
|
|
|
{
|
2013-06-27 13:33:46 +02:00
|
|
|
if (! self::has($name)) {
|
2013-06-03 17:02:08 +02:00
|
|
|
return null;
|
|
|
|
}
|
2013-06-27 13:33:46 +02:00
|
|
|
if (! array_key_exists($name, self::$instances)) {
|
|
|
|
$class = self::$hooks[$name];
|
|
|
|
try {
|
|
|
|
$obj = new $class();
|
|
|
|
} catch (\Exception $e) {
|
|
|
|
// TODO: Persist unloading for "some time" or "current session"
|
|
|
|
Log::debug(
|
|
|
|
'Hook "%s" (%s) failed, will be unloaded: %s',
|
|
|
|
$name,
|
|
|
|
$class,
|
|
|
|
$e->getMessage()
|
|
|
|
);
|
|
|
|
unset(self::$hooks[$name]);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
$base_class = 'Icinga\\Web\\Hook\\' . ucfirst($name);
|
|
|
|
if (! $obj instanceof $base_class) {
|
|
|
|
throw new ProgrammingError(
|
|
|
|
sprintf(
|
|
|
|
'%s is not an instance of %s',
|
|
|
|
get_class($obj),
|
|
|
|
$base_class
|
|
|
|
)
|
|
|
|
);
|
2013-06-07 13:29:11 +02:00
|
|
|
}
|
2013-06-27 13:33:46 +02:00
|
|
|
self::$instances[$name] = $obj;
|
|
|
|
|
2013-06-03 17:02:08 +02:00
|
|
|
}
|
|
|
|
return self::$instances[$name];
|
|
|
|
}
|
|
|
|
|
2013-06-28 11:16:35 +02:00
|
|
|
/**
|
|
|
|
* Create or return an instance of the hook
|
2013-08-20 19:33:15 +02:00
|
|
|
*
|
2013-06-28 11:16:35 +02:00
|
|
|
* @param string $name
|
|
|
|
* @param string $key
|
2013-08-20 19:33:15 +02:00
|
|
|
* @return stdClass
|
|
|
|
*
|
2013-06-28 11:16:35 +02:00
|
|
|
*/
|
|
|
|
public static function createInstance($name, $key)
|
|
|
|
{
|
|
|
|
if (!self::has($name, $key)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
if (isset(self::$instances[$name][$key])) {
|
|
|
|
return self::$instances[$name][$key];
|
|
|
|
}
|
|
|
|
$class = self::$hooks[$name][$key];
|
|
|
|
try {
|
|
|
|
$instance = new $class();
|
|
|
|
} catch (\Exception $e) {
|
|
|
|
Log::debug(
|
|
|
|
'Hook "%s" (%s) (%s) failed, will be unloaded: %s',
|
|
|
|
$name,
|
|
|
|
$key,
|
|
|
|
$class,
|
|
|
|
$e->getMessage()
|
|
|
|
);
|
|
|
|
unset(self::$hooks[$name][$key]);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
self::assertValidHook($instance, $name);
|
|
|
|
self::$instances[$name][$key] = $instance;
|
|
|
|
return $instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test for a valid class name
|
2013-08-20 19:33:15 +02:00
|
|
|
*
|
|
|
|
* @param stdClass $instance
|
2013-06-28 11:16:35 +02:00
|
|
|
* @param string $name
|
2013-08-20 19:33:15 +02:00
|
|
|
* @throws ProgrammingError
|
2013-06-28 11:16:35 +02:00
|
|
|
*/
|
|
|
|
private static function assertValidHook(&$instance, $name)
|
|
|
|
{
|
|
|
|
$base_class = self::$BASE_NS . ucfirst($name);
|
|
|
|
if (!$instance instanceof $base_class) {
|
|
|
|
throw new ProgrammingError(
|
|
|
|
sprintf(
|
|
|
|
'%s is not an instance of %s',
|
|
|
|
get_class($instance),
|
|
|
|
$base_class
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return all instances of a specific name
|
2013-08-20 19:33:15 +02:00
|
|
|
*
|
2013-06-28 11:16:35 +02:00
|
|
|
* @param string $name
|
2013-08-20 19:33:15 +02:00
|
|
|
*
|
2013-06-28 11:16:35 +02:00
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public static function all($name)
|
|
|
|
{
|
|
|
|
if (!self::has($name)) {
|
|
|
|
return array();
|
|
|
|
}
|
|
|
|
foreach (self::$hooks[$name] as $key => $hook) {
|
|
|
|
if (self::createInstance($name, $key) === null) {
|
|
|
|
return array();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return self::$instances[$name];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2013-08-20 19:33:15 +02:00
|
|
|
* Get the first hook
|
|
|
|
*
|
2013-06-28 11:16:35 +02:00
|
|
|
* @param string $name
|
2013-08-20 19:33:15 +02:00
|
|
|
*
|
|
|
|
* @return stdClass
|
2013-06-28 11:16:35 +02:00
|
|
|
*/
|
|
|
|
public static function first($name)
|
|
|
|
{
|
|
|
|
return self::createInstance($name, key(self::$hooks[$name]));
|
|
|
|
}
|
|
|
|
|
2013-06-07 13:29:11 +02:00
|
|
|
/**
|
2013-06-27 13:33:46 +02:00
|
|
|
* Register your hook
|
|
|
|
*
|
|
|
|
* @param string $name One of the predefined hook names
|
2013-08-20 19:33:15 +02:00
|
|
|
* @param string $key
|
2013-06-27 13:33:46 +02:00
|
|
|
* @param string $class Your class name, must inherit one of the classes
|
|
|
|
* in the Icinga/Web/Hook folder
|
|
|
|
*
|
2013-06-07 13:29:11 +02:00
|
|
|
*/
|
2013-06-28 11:16:35 +02:00
|
|
|
public static function register($name, $key, $class)
|
2013-06-03 17:02:08 +02:00
|
|
|
{
|
2013-06-27 13:42:48 +02:00
|
|
|
self::registerClass($name, $key, $class);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Register a class
|
|
|
|
*
|
|
|
|
* @param string $name
|
|
|
|
* @param string $key
|
|
|
|
* @param string $class
|
|
|
|
*/
|
|
|
|
public static function registerClass($name, $key, $class)
|
|
|
|
{
|
|
|
|
if (!isset(self::$hooks[$name])) {
|
|
|
|
self::$hooks[$name] = array();
|
|
|
|
}
|
|
|
|
|
2013-06-03 17:02:08 +02:00
|
|
|
self::$hooks[$name][$key] = $class;
|
|
|
|
}
|
2013-06-27 13:42:48 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Register an object
|
2013-08-20 19:33:15 +02:00
|
|
|
*
|
2013-06-27 13:42:48 +02:00
|
|
|
* @param string $name
|
|
|
|
* @param string $key
|
|
|
|
* @param object $object
|
2013-08-20 19:33:15 +02:00
|
|
|
*
|
|
|
|
* @throws ProgrammingError
|
2013-06-27 13:42:48 +02:00
|
|
|
*/
|
|
|
|
public static function registerObject($name, $key, $object)
|
|
|
|
{
|
|
|
|
if (!is_object($object)) {
|
|
|
|
throw new ProgrammingError('object is not an instantiated class');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isset(self::$instances[$name])) {
|
|
|
|
self::$instances[$name] = array();
|
|
|
|
}
|
|
|
|
|
|
|
|
self::$instances[$name][$key] =& $object;
|
|
|
|
self::registerClass($name, $key, get_class($object));
|
|
|
|
}
|
2013-06-03 17:02:08 +02:00
|
|
|
}
|