app = $app; $this->modulePaths = $availableDirs; $this->enableDir = $enabledDir; } /** * Query interface for the module manager * * @return SimpleQuery */ public function select() { $source = new ArrayDatasource($this->getModuleInfo()); return $source->select(); } /** * Check for enabled modules * * Update the internal $enabledDirs property with the enabled modules. * * @throws ConfigurationError If module dir does not exist, is not a directory or not readable */ private function detectEnabledModules() { if (! file_exists($parent = dirname($this->enableDir))) { return; } if (! is_readable($parent)) { throw new NotReadableError( 'Cannot read enabled modules. Module directory\'s parent directory "%s" is not readable', $parent ); } if (! file_exists($this->enableDir)) { return; } if (! is_dir($this->enableDir)) { throw new NotReadableError( 'Cannot read enabled modules. Module directory "%s" is not a directory', $this->enableDir ); } if (! is_readable($this->enableDir)) { throw new NotReadableError( 'Cannot read enabled modules. Module directory "%s" is not readable', $this->enableDir ); } if (($dh = opendir($this->enableDir)) !== false) { $this->enabledDirs = array(); while (($file = readdir($dh)) !== false) { if ($file[0] === '.' || $file === 'README') { continue; } $link = $this->enableDir . DIRECTORY_SEPARATOR . $file; if (! is_link($link)) { Logger::warning( 'Found invalid module in enabledModule directory "%s": "%s" is not a symlink', $this->enableDir, $link ); continue; } $dir = realpath($link); if (! file_exists($dir) || !is_dir($dir)) { Logger::warning( 'Found invalid module in enabledModule directory "%s": "%s" points to non existing path "%s"', $this->enableDir, $link, $dir ); continue; } $this->enabledDirs[$file] = $dir; ksort($this->enabledDirs); } closedir($dh); } } /** * Try to set all enabled modules in loaded sate * * @return $this * @see Manager::loadModule() */ public function loadEnabledModules() { foreach ($this->listEnabledModules() as $name) { $this->loadModule($name); } return $this; } /** * Try to load the module and register it in the application * * @param string $name The name of the module to load * @param mixed $basedir Optional module base directory * * @return $this */ public function loadModule($name, $basedir = null) { if ($this->hasLoaded($name)) { return $this; } $module = null; if ($basedir === null) { $module = new Module($this->app, $name, $this->getModuleDir($name)); } else { $module = new Module($this->app, $name, $basedir); } $module->register(); $this->loadedModules[$name] = $module; return $this; } /** * Set the given module to the enabled state * * @param string $name The module to enable * * @return $this * @throws ConfigurationError When trying to enable a module that is not installed * @throws SystemPermissionException When insufficient permissions for the application exist */ public function enableModule($name) { if (! $this->hasInstalled($name)) { throw new ConfigurationError( 'Cannot enable module "%s". Module is not installed.', $name ); } clearstatcache(true); $target = $this->installedBaseDirs[$name]; $link = $this->enableDir . DIRECTORY_SEPARATOR . $name; if (! is_dir($this->enableDir) && !@mkdir($this->enableDir, 02770, true)) { $error = error_get_last(); throw new SystemPermissionException( 'Failed to create enabledModules directory "%s" (%s)', $this->enableDir, $error['message'] ); } elseif (! is_writable($this->enableDir)) { throw new SystemPermissionException( 'Cannot enable module "%s". Check the permissions for the enabledModules directory: %s', $name, $this->enableDir ); } if (file_exists($link) && is_link($link)) { return $this; } if (! @symlink($target, $link)) { $error = error_get_last(); if (strstr($error["message"], "File exists") === false) { throw new SystemPermissionException( 'Cannot enable module "%s" at %s due to file system errors. ' . 'Please check path and mounting points because this is not a permission error. ' . 'Primary error was: %s', $name, $this->enableDir, $error['message'] ); } } $this->enabledDirs[$name] = $link; $this->loadModule($name); return $this; } /** * Disable the given module and remove its enabled state * * @param string $name The name of the module to disable * * @return $this * * @throws ConfigurationError When the module is not installed or it's not a symlink * @throws SystemPermissionException When insufficient permissions for the application exist */ public function disableModule($name) { if (! $this->hasEnabled($name)) { throw new ConfigurationError( 'Cannot disable module "%s". Module is not installed.', $name ); } if (! is_writable($this->enableDir)) { throw new SystemPermissionException( 'Cannot disable module "%s". Check the permissions for the enabledModules directory: %s', $name, $this->enableDir ); } $link = $this->enableDir . DIRECTORY_SEPARATOR . $name; if (! file_exists($link)) { throw new ConfigurationError( 'Cannot disable module "%s". Module is not installed.', $name ); } if (! is_link($link)) { throw new ConfigurationError( 'Cannot disable module %s at %s. ' . 'It looks like you have installed this module manually and moved it to your module folder. ' . 'In order to dynamically enable and disable modules, you have to create a symlink to ' . 'the enabledModules folder.', $name, $this->enableDir ); } if (file_exists($link) && is_link($link)) { if (! @unlink($link)) { $error = error_get_last(); throw new SystemPermissionException( 'Cannot enable module "%s" at %s due to file system errors. ' . 'Please check path and mounting points because this is not a permission error. ' . 'Primary error was: %s', $name, $this->enableDir, $error['message'] ); } } unset($this->enabledDirs[$name]); return $this; } /** * Return the directory of the given module as a string, optionally with a given sub directoy * * @param string $name The module name to return the module directory of * @param string $subdir The sub directory to append to the path * * @return string * * @throws ProgrammingError When the module is not installed or existing */ public function getModuleDir($name, $subdir = '') { if ($this->hasLoaded($name)) { return $this->getModule($name)->getBaseDir() . $subdir; } if ($this->hasEnabled($name)) { return $this->enabledDirs[$name]. $subdir; } if ($this->hasInstalled($name)) { return $this->installedBaseDirs[$name] . $subdir; } throw new ProgrammingError( 'Trying to access uninstalled module dir: %s', $name ); } /** * Return true when the module with the given name is installed, otherwise false * * @param string $name The module to check for being installed * * @return bool */ public function hasInstalled($name) { if (!count($this->installedBaseDirs)) { $this->detectInstalledModules(); } return array_key_exists($name, $this->installedBaseDirs); } /** * Return true when the given module is in enabled state, otherwise false * * @param string $name The module to check for being enabled * * @return bool */ public function hasEnabled($name) { return array_key_exists($name, $this->enabledDirs); } /** * Return true when the module is in loaded state, otherwise false * * @param string $name The module to check for being loaded * * @return bool */ public function hasLoaded($name) { return array_key_exists($name, $this->loadedModules); } /** * Get the currently loaded modules * * @return Module[] */ public function getLoadedModules() { return $this->loadedModules; } /** * Get a module * * @param string $name Name of the module * @param bool $assertLoaded Whether or not to throw an exception if the module hasn't been loaded * * @return Module * @throws ProgrammingError If the module hasn't been loaded */ public function getModule($name, $assertLoaded = true) { if ($this->hasLoaded($name)) { return $this->loadedModules[$name]; } elseif (! (bool) $assertLoaded) { return new Module($this->app, $name, $this->getModuleDir($name)); } throw new ProgrammingError( 'Can\'t access module %s because it hasn\'t been loaded', $name ); } /** * Return an array containing information objects for each available module * * Each entry has the following fields * * name, name of the module as a string * * path, path where the module is located as a string * * enabled, whether the module is enabled or not as a boolean * * loaded, whether the module is loaded or not as a boolean * * @return array */ public function getModuleInfo() { $info = array(); $enabled = $this->listEnabledModules(); foreach ($enabled as $name) { $info[$name] = (object) array( 'name' => $name, 'path' => $this->enabledDirs[$name], 'enabled' => true, 'loaded' => $this->hasLoaded($name) ); } $installed = $this->listInstalledModules(); foreach ($installed as $name) { $info[$name] = (object) array( 'name' => $name, 'path' => $this->installedBaseDirs[$name], 'enabled' => $this->hasEnabled($name), 'loaded' => $this->hasLoaded($name) ); } return $info; } /** * Return an array containing all enabled module names as strings * * @return array */ public function listEnabledModules() { if (count($this->enabledDirs) === 0) { $this->detectEnabledModules(); } return array_keys($this->enabledDirs); } /** * Return an array containing all loaded module names as strings * * @return array */ public function listLoadedModules() { return array_keys($this->loadedModules); } /** * Return an array of module names from installed modules * * Calls detectInstalledModules() if no module discovery has been performed yet * * @return array * * @see detectInstalledModules() */ public function listInstalledModules() { if (!count($this->installedBaseDirs)) { $this->detectInstalledModules(); } if (count($this->installedBaseDirs)) { return array_keys($this->installedBaseDirs); } return array(); } /** * Detect installed modules from every path provided in modulePaths * * @param array $availableDirs Installed modules location * * @return $this */ public function detectInstalledModules(array $availableDirs = null) { $modulePaths = $availableDirs !== null ? $availableDirs : $this->modulePaths; foreach ($modulePaths as $basedir) { $canonical = realpath($basedir); if ($canonical === false) { Logger::warning('Module path "%s" does not exist', $basedir); continue; } if (!is_dir($canonical)) { Logger::error('Module path "%s" is not a directory', $canonical); continue; } if (!is_readable($canonical)) { Logger::error('Module path "%s" is not readable', $canonical); continue; } if (($dh = opendir($canonical)) !== false) { while (($file = readdir($dh)) !== false) { if ($file[0] === '.') { continue; } if (is_dir($canonical . '/' . $file)) { if (! array_key_exists($file, $this->installedBaseDirs)) { $this->installedBaseDirs[$file] = $canonical . '/' . $file; } else { Logger::debug( 'Module "%s" already exists in installation path "%s" and is ignored.', $canonical . '/' . $file, $this->installedBaseDirs[$file] ); } } } closedir($dh); } } ksort($this->installedBaseDirs); return $this; } /** * Get the directories where to look for installed modules * * @return array */ public function getModuleDirs() { return $this->modulePaths; } }