mirror of
				https://github.com/Icinga/icingaweb2.git
				synced 2025-10-25 01:14:26 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			338 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			338 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /* Icinga Web 2 | (c) 2013 Icinga Development Team | GPLv2+ */
 | |
| 
 | |
| namespace Icinga\Application;
 | |
| 
 | |
| use Zend_Loader_Autoloader;
 | |
| 
 | |
| /**
 | |
|  * PSR-4 class loader
 | |
|  */
 | |
| class ClassLoader
 | |
| {
 | |
|     /**
 | |
|      * Namespace separator
 | |
|      */
 | |
|     const NAMESPACE_SEPARATOR = '\\';
 | |
| 
 | |
|     /**
 | |
|      * Icinga Web 2 module namespace prefix
 | |
|      */
 | |
|     const MODULE_PREFIX = 'Icinga\\Module\\';
 | |
| 
 | |
|     /**
 | |
|      * Icinga Web 2 module namespace prefix length
 | |
|      *
 | |
|      * Helps to make substr/strpos operations even faster
 | |
|      */
 | |
|     const MODULE_PREFIX_LENGTH = 14;
 | |
| 
 | |
|     /**
 | |
|      * A hardcoded class/subdir map for application ns prefixes
 | |
|      *
 | |
|      * When a module registers with an application directory, those
 | |
|      * namespace prefixes (after the module prefix) will be looked up
 | |
|      * in the corresponding application subdirectories
 | |
|      *
 | |
|      * @var array
 | |
|      */
 | |
|     protected $applicationPrefixes = array(
 | |
|         'Clicommands' => 'clicommands',
 | |
|         'Controllers' => 'controllers',
 | |
|         'Forms'       => 'forms'
 | |
|     );
 | |
| 
 | |
|     /**
 | |
|      * Whether we already instantiated the ZF autoloader
 | |
|      *
 | |
|      * @var boolean
 | |
|      */
 | |
|     protected $gotZend = false;
 | |
| 
 | |
|     /**
 | |
|      * Namespaces
 | |
|      *
 | |
|      * @var array
 | |
|      */
 | |
|     private $namespaces = array();
 | |
| 
 | |
|     /**
 | |
|      * Application directories
 | |
|      *
 | |
|      * @var array
 | |
|      */
 | |
|     private $applicationDirectories = array();
 | |
| 
 | |
|     /**
 | |
|      * Register a base directory for a namespace prefix
 | |
|      *
 | |
|      * Application directory is optional and provides additional lookup
 | |
|      * logic for hardcoded namespaces like "Forms"
 | |
|      *
 | |
|      * @param   string  $namespace
 | |
|      * @param   string  $directory
 | |
|      * @param   string  $appDirectory
 | |
|      *
 | |
|      * @return  $this
 | |
|      */
 | |
|     public function registerNamespace($namespace, $directory, $appDirectory = null)
 | |
|     {
 | |
|         $this->namespaces[$namespace] = $directory;
 | |
| 
 | |
|         if ($appDirectory !== null) {
 | |
|             $this->applicationDirectories[$namespace] = $appDirectory;
 | |
|         }
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Test whether a namespace exists
 | |
|      *
 | |
|      * @param   string $namespace
 | |
|      *
 | |
|      * @return  bool
 | |
|      */
 | |
|     public function hasNamespace($namespace)
 | |
|     {
 | |
|         return array_key_exists($namespace, $this->namespaces);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the source file of the given class or interface
 | |
|      *
 | |
|      * @param   string      $class Name of the class or interface
 | |
|      *
 | |
|      * @return  string|null
 | |
|      */
 | |
|     public function getSourceFile($class)
 | |
|     {
 | |
|         if ($file = $this->getModuleSourceFile($class)) {
 | |
|             return $file;
 | |
|         }
 | |
| 
 | |
|         foreach ($this->namespaces as $namespace => $dir) {
 | |
|             if ($class === strstr($class, $namespace)) {
 | |
|                 return $this->buildClassFilename($class, $namespace);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return null;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the source file of the given module class or interface
 | |
|      *
 | |
|      * @param   string      $class Module class or interface name
 | |
|      *
 | |
|      * @return  string|null
 | |
|      */
 | |
|     protected function getModuleSourceFile($class)
 | |
|     {
 | |
|         if (! $this->classBelongsToModule($class)) {
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
|         $modules = Icinga::app()->getModuleManager();
 | |
|         $namespace = $this->extractModuleNamespace($class);
 | |
| 
 | |
|         if ($this->hasNamespace($namespace)) {
 | |
| 
 | |
|             return $this->buildClassFilename($class, $namespace);
 | |
| 
 | |
|         } elseif (! $modules->loadedAllEnabledModules()) {
 | |
| 
 | |
|             $moduleName = $this->extractModuleName($class);
 | |
| 
 | |
|             if ($modules->hasEnabled($moduleName)) {
 | |
|                 $modules->loadModule($moduleName);
 | |
| 
 | |
|                 return $this->buildClassFilename($class, $namespace);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return null;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Extract the Icinga module namespace from a given namespaced class name
 | |
|      *
 | |
|      * Does no validation, prefix must have been checked before
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     protected function extractModuleNamespace($class)
 | |
|     {
 | |
|         return substr(
 | |
|             $class,
 | |
|             0,
 | |
|             strpos($class, self::NAMESPACE_SEPARATOR, self::MODULE_PREFIX_LENGTH + 1)
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Extract the Icinga module name from a given namespaced class name
 | |
|      *
 | |
|      * Does no validation, prefix must have been checked before
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     protected function extractModuleName($class)
 | |
|     {
 | |
|         return lcfirst(
 | |
|             substr(
 | |
|                 $class,
 | |
|                 self::MODULE_PREFIX_LENGTH,
 | |
|                 strpos(
 | |
|                     $class,
 | |
|                     self::NAMESPACE_SEPARATOR,
 | |
|                     self::MODULE_PREFIX_LENGTH + 1
 | |
|                 ) - self::MODULE_PREFIX_LENGTH
 | |
|             )
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Whether the given class name belongs to a module namespace
 | |
|      *
 | |
|      * @return boolean
 | |
|      */
 | |
|     protected function classBelongsToModule($class)
 | |
|     {
 | |
|         return substr($class, 0, self::MODULE_PREFIX_LENGTH) === self::MODULE_PREFIX;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Prepare a filename string for the given class
 | |
|      *
 | |
|      * Expects the given namespace to be registered with a path name
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     protected function buildClassFilename($class, $namespace)
 | |
|     {
 | |
|         $relNs = substr($class, strlen($namespace) + 1);
 | |
| 
 | |
|         if ($this->namespaceHasApplictionDirectory($namespace)) {
 | |
|             $prefixSeparator = strpos($relNs, self::NAMESPACE_SEPARATOR);
 | |
|             $prefix = substr($relNs, 0, $prefixSeparator);
 | |
| 
 | |
|             if ($this->isApplicationPrefix($prefix)) {
 | |
|                 return $this->applicationDirectories[$namespace]
 | |
|                     . DIRECTORY_SEPARATOR
 | |
|                     . $this->applicationPrefixes[$prefix]
 | |
|                     . $this->classToRelativePhpFilename(substr($relNs, $prefixSeparator));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $this->namespaces[$namespace] . DIRECTORY_SEPARATOR . $this->classToRelativePhpFilename($relNs);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return the relative file name for the given (namespaces) class
 | |
|      *
 | |
|      * @param  string $class
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     protected function classToRelativePhpFilename($class)
 | |
|     {
 | |
|         return str_replace(
 | |
|             self::NAMESPACE_SEPARATOR,
 | |
|             DIRECTORY_SEPARATOR,
 | |
|             $class
 | |
|         ) . '.php';
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Whether given prefix (Forms, Controllers...) makes part of "application"
 | |
|      *
 | |
|      * @param  string $prefix
 | |
|      *
 | |
|      * @return boolean
 | |
|      */
 | |
|     protected function isApplicationPrefix($prefix)
 | |
|     {
 | |
|         return array_key_exists($prefix, $this->applicationPrefixes);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Whether the given namespace registered an application directory
 | |
|      *
 | |
|      * @return boolean
 | |
|      */
 | |
|     protected function namespaceHasApplictionDirectory($namespace)
 | |
|     {
 | |
|         return array_key_exists($namespace, $this->applicationDirectories);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Require ZF autoloader
 | |
|      *
 | |
|      * @return Zend_Loader_Autoloader
 | |
|      */
 | |
|     protected function requireZendAutoloader()
 | |
|     {
 | |
|         require_once 'Zend/Loader/Autoloader.php';
 | |
|         $this->gotZend = true;
 | |
|         return Zend_Loader_Autoloader::getInstance();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Load the given class or interface
 | |
|      *
 | |
|      * @param   string  $class  Name of the class or interface
 | |
|      *
 | |
|      * @return  bool            Whether the class or interface has been loaded
 | |
|      */
 | |
|     public function loadClass($class)
 | |
|     {
 | |
|         // We are aware of the Zend_ prefix and lazyload it's autoloader.
 | |
|         // Return as fast as possible if we already did so.
 | |
|         if (substr($class, 0, 5) === 'Zend_') {
 | |
|             if (! $this->gotZend) {
 | |
|                 $zendLoader = $this->requireZendAutoloader();
 | |
|                 if (version_compare(PHP_VERSION, '7.0.0') >= 0) {
 | |
|                     // PHP7 seems to remember the autoload function stack before auto-loading. Thus
 | |
|                     // autoload functions registered during autoload never get called
 | |
|                     return $zendLoader::autoload($class);
 | |
|                 }
 | |
|             }
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         if ($file = $this->getSourceFile($class)) {
 | |
|             if (file_exists($file)) {
 | |
|                 require $file;
 | |
|                 return true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Register {@link loadClass()} as an autoloader
 | |
|      */
 | |
|     public function register()
 | |
|     {
 | |
|         spl_autoload_register(array($this, 'loadClass'));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Unregister {@link loadClass()} as an autoloader
 | |
|      */
 | |
|     public function unregister()
 | |
|     {
 | |
|         spl_autoload_unregister(array($this, 'loadClass'));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Unregister this as an autoloader
 | |
|      */
 | |
|     public function __destruct()
 | |
|     {
 | |
|         $this->unregister();
 | |
|     }
 | |
| }
 |