mirror of
				https://github.com/Icinga/icingaweb2.git
				synced 2025-10-30 19:04:10 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			219 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			219 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /* Icinga Web 2 | (c) 2013-2015 Icinga Development Team | GPLv2+ */
 | |
| 
 | |
| namespace Icinga\Web;
 | |
| 
 | |
| use Exception;
 | |
| use Icinga\Application\Logger;
 | |
| 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();
 | |
| 
 | |
|     /**
 | |
|      * Namespace prefix
 | |
|      *
 | |
|      * @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';
 | |
| 
 | |
|     /**
 | |
|      * Reset object state
 | |
|      */
 | |
|     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
 | |
|      *
 | |
|      * @return  mixed
 | |
|      */
 | |
|     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) {
 | |
|             Logger::debug(
 | |
|                 'Hook "%s" (%s) (%s) failed, will be unloaded: %s',
 | |
|                 $name,
 | |
|                 $key,
 | |
|                 $class,
 | |
|                 $e->getMessage()
 | |
|             );
 | |
|             // TODO: Persist unloading for "some time" or "current session"
 | |
|             unset(self::$hooks[$name][$key]);
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
|         self::assertValidHook($instance, $name);
 | |
|         self::$instances[$name][$key] = $instance;
 | |
|         return $instance;
 | |
|     }
 | |
| 
 | |
|     protected static function splitHookName($name)
 | |
|     {
 | |
|         $sep = '\\';
 | |
|         if (false === $module = strpos($name, $sep)) {
 | |
|             return array(null, $name);
 | |
|         }
 | |
|         return array(
 | |
|             substr($name, 0, $module),
 | |
|             substr($name, $module + 1)
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Test for a valid class name
 | |
|      *
 | |
|      * @param   mixed   $instance
 | |
|      * @param   string  $name
 | |
|      *
 | |
|      * @throws  ProgrammingError
 | |
|      */
 | |
|     private static function assertValidHook($instance, $name)
 | |
|     {
 | |
|         $suffix = self::$classSuffix; // 'Hook'
 | |
|         $base = self::$BASE_NS;       // 'Icinga\\Web\\Hook\\'
 | |
| 
 | |
|         list($module, $name) = self::splitHookName($name);
 | |
| 
 | |
|         if ($module === null) {
 | |
|             $base_class = $base . ucfirst($name) . 'Hook';
 | |
| 
 | |
|             // I'm unsure whether this makes sense. Unused and Wrong.
 | |
|             if (strpos($base_class, $suffix) === false) {
 | |
|                 $base_class .= $suffix;
 | |
|             }
 | |
|         } else {
 | |
|             $base_class = 'Icinga\\Module\\'
 | |
|                         . ucfirst($module)
 | |
|                         . '\\Web\\Hook\\'
 | |
|                         . ucfirst($name)
 | |
|                         . $suffix;
 | |
|         }
 | |
| 
 | |
|         if (!$instance instanceof $base_class) {
 | |
|             throw new ProgrammingError(
 | |
|                 '%s is not an instance of %s',
 | |
|                 get_class($instance),
 | |
|                 $base_class
 | |
|             );
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return all instances of a specific name
 | |
|      *
 | |
|      * @param   string  $name   One of the predefined hook names
 | |
|      *
 | |
|      * @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];
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the first hook
 | |
|      *
 | |
|      * @param   string  $name   One of the predefined hook names
 | |
|      *
 | |
|      * @return  null|mixed
 | |
|      */
 | |
|     public static function first($name)
 | |
|     {
 | |
|         if (self::has($name)) {
 | |
|             return self::createInstance($name, key(self::$hooks[$name]));
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * 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 register($name, $key, $class)
 | |
|     {
 | |
|         if (!isset(self::$hooks[$name])) {
 | |
|             self::$hooks[$name] = array();
 | |
|         }
 | |
| 
 | |
|         self::$hooks[$name][$key] = $class;
 | |
|     }
 | |
| }
 |