Files
espocrm/application/Espo/Core/Utils/Autoload/NamespaceLoader.php
Yuri Kuznetsov b67c2dc625 refactoring
2021-05-06 15:09:21 +03:00

237 lines
6.6 KiB
PHP

<?php
/************************************************************************
* This file is part of EspoCRM.
*
* EspoCRM - Open Source CRM application.
* Copyright (C) 2014-2021 Yurii Kuznietsov, Taras Machyshyn, Oleksii Avramenko
* Website: https://www.espocrm.com
*
* EspoCRM is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* EspoCRM is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
*
* The interactive user interfaces in modified source and object code versions
* of this program must display Appropriate Legal Notices, as required under
* Section 5 of the GNU General Public License version 3.
*
* In accordance with Section 7(b) of the GNU General Public License version 3,
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
************************************************************************/
namespace Espo\Core\Utils\Autoload;
use Espo\Core\{
Utils\Util,
Utils\Config,
Utils\DataCache,
Utils\Log,
};
use Composer\Autoload\ClassLoader;
use Exception;
class NamespaceLoader
{
private $classLoader;
private $namespaces;
private $autoloadFilePath = 'vendor/autoload.php';
private $namespacesPaths = [
'psr-4' => 'vendor/composer/autoload_psr4.php',
'psr-0' => 'vendor/composer/autoload_namespaces.php',
'classmap' => 'vendor/composer/autoload_classmap.php',
];
private $methodNameMap = [
'psr-4' => 'addPsr4',
'psr-0' => 'add',
'classmap' => 'addClassMap',
];
private $vendorNamespaces;
private $cacheKey = 'autoloadVendorNamespaces';
private $config;
private $dataCache;
private $log;
public function __construct(Config $config, DataCache $dataCache, Log $log)
{
$this->config = $config;
$this->dataCache = $dataCache;
$this->log = $log;
$this->classLoader = new ClassLoader();
}
public function register(array $data)
{
$this->addListToClassLoader($this->classLoader, $data);
$this->classLoader->register(true);
}
protected function loadNamespaces($basePath = '')
{
$namespaces = [];
foreach ($this->namespacesPaths as $type => $path) {
$mapFile = Util::concatPath($basePath, $path);
if (!file_exists($mapFile)) {
continue;
}
$map = require($mapFile);
if (!empty($map) && is_array($map)) {
$namespaces[$type] = $map;
}
}
return $namespaces;
}
protected function getNamespaces()
{
if (!$this->namespaces) {
$this->namespaces = $this->loadNamespaces();
}
return $this->namespaces;
}
protected function getNamespaceList($type, $default = [])
{
$namespaces = $this->getNamespaces();
if (isset($namespaces[$type])) {
return array_keys($namespaces[$type]);
}
return $default;
}
protected function addNamespace($type, $name, $path)
{
if (!$this->namespaces) {
$this->getNamespaces();
}
$this->namespaces[$type][$name] = (array) $path;
}
protected function hasNamespace($type, $name)
{
if (in_array($name, $this->getNamespaceList($type))) {
return true;
}
if (!preg_match('/\\\$/', $name)) {
$name = $name . '\\';
if (in_array($name, $this->getNamespaceList($type))) {
return true;
}
}
return false;
}
protected function addListToClassLoader(ClassLoader $classLoader, array $list, bool $skipVendorNamespaces = false)
{
foreach ($this->methodNameMap as $type => $methodName) {
if (!isset($list[$type])) {
continue;
}
if (!method_exists($classLoader, $methodName)) {
$this->log->warning('Autoload: ClassLoader method ['. $methodName .'] is not found.');
continue;
}
foreach ($list[$type] as $prefix => $path) {
if (!$skipVendorNamespaces) {
$vendorNamespaces = $this->getVendorNamespaces($path);
if (!empty($vendorNamespaces)) {
$this->addListToClassLoader($classLoader, $vendorNamespaces, true);
}
}
if (!$this->hasNamespace($type, $prefix)) {
try {
$classLoader->$methodName($prefix, $path);
$this->addNamespace($type, $prefix, $path);
}
catch (Exception $e) {
$this->log->warning('Autoload: error adding the namespace ['. $prefix .']');
}
}
}
}
}
protected function getVendorNamespaces($path)
{
$useCache = $this->config->get('useCache');
if (!isset($this->vendorNamespaces)) {
$this->vendorNamespaces = [];
if ($useCache && $this->dataCache->has($this->cacheKey)) {
$this->vendorNamespaces = $this->dataCache->get($this->cacheKey);
if (!is_array($this->vendorNamespaces)) {
$this->vendorNamespaces = [];
}
}
}
if (!array_key_exists($path, $this->vendorNamespaces)) {
$vendorPath = $this->findVendorPath($path);
if ($vendorPath) {
$this->vendorNamespaces[$path] = $this->loadNamespaces($vendorPath);
if ($useCache) {
$this->dataCache->store($this->cacheKey, $this->vendorNamespaces);
}
}
}
if (isset($this->vendorNamespaces[$path])) {
return $this->vendorNamespaces[$path];
}
}
protected function findVendorPath($path)
{
$vendor = Util::concatPath($path, $this->autoloadFilePath);
if (file_exists($vendor)) {
return $path;
}
$parentDir = dirname($path);
if (!empty($parentDir) && $parentDir !== '.') {
return $this->findVendorPath($parentDir);
}
}
}