| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261 | <?php/** * Hoa * * * @license * * New BSD License * * Copyright © 2007-2017, Hoa community. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: *     * Redistributions of source code must retain the above copyright *       notice, this list of conditions and the following disclaimer. *     * Redistributions in binary form must reproduce the above copyright *       notice, this list of conditions and the following disclaimer in the *       documentation and/or other materials provided with the distribution. *     * Neither the name of the Hoa nor the names of its contributors may be *       used to endorse or promote products derived from this software without *       specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS AND CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */namespace Hoa\Consistency;/** * Class Hoa\Consistency\Autoloader. * * This class is a PSR-4 compliant autoloader. * * @copyright  Copyright © 2007-2017 Hoa community * @license    New BSD License */class Autoloader{    /**     * Namespace prefixes to base directories.     *     * @var array     */    protected $_namespacePrefixesToBaseDirectories = [];    /**     * Add a base directory for a namespace prefix.     *     * @param   string  $prefix           Namespace prefix.     * @param   string  $baseDirectory    Base directory for this prefix.     * @param   bool    $prepend          Whether the prefix is prepended or     *                                    appended to the prefix' stack.     * @return  void     */    public function addNamespace($prefix, $baseDirectory, $prepend = false)    {        $prefix        = trim($prefix, '\\') . '\\';        $baseDirectory = rtrim($baseDirectory, DIRECTORY_SEPARATOR) . DIRECTORY_SEPARATOR;        if (false === isset($this->_namespacePrefixesToBaseDirectories[$prefix])) {            $this->_namespacePrefixesToBaseDirectories[$prefix] = [];        }        if (true === $prepend) {            array_unshift(                $this->_namespacePrefixesToBaseDirectories[$prefix],                $baseDirectory            );        } else {            array_push(                $this->_namespacePrefixesToBaseDirectories[$prefix],                $baseDirectory            );        }        return;    }    /**     * Try to load the entity file for a given entity name.     *     * @param   string  $entity    Entity name to load.     * @return  bool     */    public function load($entity)    {        $entityPrefix     = $entity;        $hasBaseDirectory = false;        while (false !== $pos = strrpos($entityPrefix, '\\')) {            $currentEntityPrefix = substr($entity, 0, $pos + 1);            $entityPrefix        = rtrim($currentEntityPrefix, '\\');            $entitySuffix        = substr($entity, $pos + 1);            $entitySuffixAsPath  = str_replace('\\', '/', $entitySuffix);            if (false === $this->hasBaseDirectory($currentEntityPrefix)) {                continue;            }            $hasBaseDirectory = true;            foreach ($this->getBaseDirectories($currentEntityPrefix) as $baseDirectory) {                $file = $baseDirectory . $entitySuffixAsPath . '.php';                if (false !== $this->requireFile($file)) {                    return $file;                }            }        }        if (true    === $hasBaseDirectory &&            $entity === Consistency::getEntityShortestName($entity) &&            false   !== $pos = strrpos($entity, '\\')) {            return $this->runAutoloaderStack(                $entity . '\\' . substr($entity, $pos + 1)            );        }        return null;    }    /**     * Require a file if exists.     *     * @param   string  $filename    File name.     * @return  bool     */    public function requireFile($filename)    {        if (false === file_exists($filename)) {            return false;        }        require $filename;        return true;    }    /**     * Check whether at least one base directory exists for a namespace prefix.     *     * @param   string  $namespacePrefix    Namespace prefix.     * @return  bool     */    public function hasBaseDirectory($namespacePrefix)    {        return isset($this->_namespacePrefixesToBaseDirectories[$namespacePrefix]);    }    /**     * Get declared base directories for a namespace prefix.     *     * @param   string  $namespacePrefix    Namespace prefix.     * @return  array     */    public function getBaseDirectories($namespacePrefix)    {        if (false === $this->hasBaseDirectory($namespacePrefix)) {            return [];        }        return $this->_namespacePrefixesToBaseDirectories[$namespacePrefix];    }    /**     * Get loaded classes.     *     * @return  array     */    public static function getLoadedClasses()    {        return get_declared_classes();    }    /**     * Run the entire autoloader stack with a specific entity.     *     * @param   string  $entity    Entity name to load.     * @return  void     */    public function runAutoloaderStack($entity)    {        return spl_autoload_call($entity);    }    /**     * Register the autoloader.     *     * @param   bool  $prepend    Prepend this autoloader to the stack or not.     * @return  bool     */    public function register($prepend = false)    {        return spl_autoload_register([$this, 'load'], true, $prepend);    }    /**     * Unregister the autoloader.     *     * @return  bool     */    public function unregister()    {        return spl_autoload_unregister([$this, 'load']);    }    /**     * Get all registered autoloaders (not only from this library).     *     * @return  array     */    public function getRegisteredAutoloaders()    {        return spl_autoload_functions();    }    /**     * Dynamic new, a simple factory.     * It loads and constructs a class, with provided arguments.     *     * @param   bool     $classname    Classname.     * @param   array    $arguments    Arguments for the constructor.     * @return  object     */    public static function dnew($classname, array $arguments = [])    {        $classname = ltrim($classname, '\\');        if (false === Consistency::entityExists($classname, false)) {            spl_autoload_call($classname);        }        $class = new \ReflectionClass($classname);        if (empty($arguments) || false === $class->hasMethod('__construct')) {            return $class->newInstance();        }        return $class->newInstanceArgs($arguments);    }}/** * Autoloader. */$autoloader = new Autoloader();$autoloader->addNamespace('Hoa', dirname(__DIR__));$autoloader->register();
 |