icingaweb2/library/Icinga/Web/Hook.php

239 lines
5.7 KiB
PHP
Raw Normal View History

<?php
2013-08-20 19:33:15 +02:00
// {{{ICINGA_LICENSE_HEADER}}}
// {{{ICINGA_LICENSE_HEADER}}}
namespace Icinga\Web;
use Exception;
use Icinga\Logger\Logger;
2014-02-18 18:48:13 +01:00
use Icinga\Exception\ProgrammingError;
/**
* Icinga Web Hook registry
*
* Modules making use of predefined hooks have to use this registry
*
* Usage:
* <code>
* Hook::register('grapher', 'My\\Grapher\\Class');
* </code>
*/
class Hook
{
/**
* Our hook name registry
*
* @var array
*/
protected static $hooks = array();
/**
* Hooks that have already been instantiated
*
* @var array
*/
protected static $instances = array();
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\\';
/**
* Append this string to base class
*
* All base classes renamed to *Hook
*
* @var string
*/
public static $classSuffix = '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\\';
}
/**
* Whether someone registered itself for the given hook name
*
* @param string $name One of the predefined hook names
*
* @return bool
*/
public static function has($name)
{
return array_key_exists($name, self::$hooks);
}
/**
* Create or return an instance of a given hook
*
* TODO: Should return some kind of a hook interface
*
* @param string $name One of the predefined hook names
* @param string $key The identifier of a specific subtype
2013-08-20 19:33:15 +02:00
*
* @return mixed
2013-06-28 11:16:35 +02:00
*/
public static function createInstance($name, $key)
{
if (!self::has($name, $key)) {
return null;
}
2013-06-28 11:16:35 +02:00
if (isset(self::$instances[$name][$key])) {
return self::$instances[$name][$key];
}
2013-06-28 11:16:35 +02:00
$class = self::$hooks[$name][$key];
try {
$instance = new $class();
} catch (Exception $e) {
Logger::debug(
2013-06-28 11:16:35 +02:00
'Hook "%s" (%s) (%s) failed, will be unloaded: %s',
$name,
$key,
$class,
$e->getMessage()
);
// TODO: Persist unloading for "some time" or "current session"
2013-06-28 11:16:35 +02:00
unset(self::$hooks[$name][$key]);
return null;
}
2013-06-28 11:16:35 +02:00
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 mixed $instance
* @param string $name
*
* @throws ProgrammingError
2013-06-28 11:16:35 +02:00
*/
private static function assertValidHook($instance, $name)
2013-06-28 11:16:35 +02:00
{
$base_class = self::$BASE_NS . ucfirst($name) . 'Hook';
if (strpos($base_class, self::$classSuffix) === false) {
$base_class .= self::$classSuffix;
}
2013-06-28 11:16:35 +02:00
if (!$instance instanceof $base_class) {
throw new ProgrammingError(
'%s is not an instance of %s',
get_class($instance),
$base_class
2013-06-28 11:16:35 +02:00
);
}
}
/**
* Return all instances of a specific name
2013-08-20 19:33:15 +02:00
*
* @param string $name One of the predefined hook names
2013-08-20 19:33:15 +02:00
*
* @return array
2013-06-28 11:16:35 +02:00
*/
public static function all($name)
{
if (!self::has($name)) {
return array();
}
2013-06-28 11:16:35 +02:00
foreach (self::$hooks[$name] as $key => $hook) {
if (self::createInstance($name, $key) === null) {
return array();
}
}
2013-06-28 11:16:35 +02:00
return self::$instances[$name];
}
/**
2013-08-20 19:33:15 +02:00
* Get the first hook
*
* @param string $name One of the predefined hook names
2013-08-20 19:33:15 +02:00
*
* @return null|mixed
2013-06-28 11:16:35 +02:00
*/
public static function first($name)
{
if (self::has($name)) {
return self::createInstance($name, key(self::$hooks[$name]));
}
2013-06-28 11:16:35 +02:00
}
/**
* Register your hook
*
* Alias for Hook::registerClass()
*
* @see Hook::registerClass()
*/
2013-06-28 11:16:35 +02:00
public static function register($name, $key, $class)
{
self::registerClass($name, $key, $class);
}
/**
* Register a class
*
* @param string $name One of the predefined hook names
* @param string $key The identifier of a specific subtype
* @param string $class Your class name, must inherit one of the
* classes in the Icinga/Web/Hook folder
*/
public static function registerClass($name, $key, $class)
{
if (!class_exists($class)) {
throw new ProgrammingError(
'"%s" is not an existing class',
$class
);
}
if (!isset(self::$hooks[$name])) {
self::$hooks[$name] = array();
}
self::$hooks[$name][$key] = $class;
}
/**
* Register an object
2013-08-20 19:33:15 +02:00
*
* @param string $name One of the predefined hook names
* @param string $key The identifier of a specific subtype
* @param object $object The instantiated hook to register
2013-08-20 19:33:15 +02:00
*
* @throws ProgrammingError
*/
public static function registerObject($name, $key, $object)
{
if (!is_object($object)) {
throw new ProgrammingError(
'"%s" is not an instantiated class',
$object
);
}
if (!isset(self::$instances[$name])) {
self::$instances[$name] = array();
}
self::$instances[$name][$key] = $object;
self::registerClass($name, $key, get_class($object));
}
}