mirror of
https://github.com/fusionpbx/fusionpbx.git
synced 2025-12-30 00:53:50 +00:00
* remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove class_exists wrapper for class definitions * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove include statement of class file * remove closing tag * remove invalid method params * remove closing tag * remove closing tag * Update auto_loader to load each class file in the project Update the auto_loader class to use an include statement on each file in the project to load the class within the file. This will allow mismatched names within the file to be loaded and mapped according to the declaration instead of the filename. The class is then checked against the parsed classes from the PHP engine so that namespaces are available and mapped to the file they were declared in. An update was also made to the search algorithm used to find a file that was not already loaded by collapsing the array to have only valid matches to increase performance on a cache miss. Logging within the auto_loader has been moved to a function. Multiple files were modified to allow the include statement. When the class has the `if(class_exists())` statement, the auto_loader is called to check for the class. This caused an infinite loop scenario so all wrappers have been removed. The auto_loader will now break the loop by directly modifying the internal classes array instead of trying to restart with the 'reload_classes' method. - APCu is used to cache classes so any loading of the classes is done only once. To clear the APCu cache, restart php-fpm or call the auto_loader::clear_cache() function. - Cache file is used when APCu is not available. To clear the cache remove it from the tmp folder or call the auto_loader::clear_cache() function. - All classes must no longer have a class_exists wrapper to benefit from the performance boost. - Classes should not be directly included when the auto_loader is used. * remove include statement of class file * Update destinations.php
324 lines
9.1 KiB
PHP
324 lines
9.1 KiB
PHP
<?php
|
|
|
|
/*
|
|
FusionPBX
|
|
Version: MPL 1.1
|
|
|
|
The contents of this file are subject to the Mozilla Public License Version
|
|
1.1 (the "License"); you may not use this file except in compliance with
|
|
the License. You may obtain a copy of the License at
|
|
http://www.mozilla.org/MPL/
|
|
|
|
Software distributed under the License is distributed on an "AS IS" basis,
|
|
WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
for the specific language governing rights and limitations under the
|
|
License.
|
|
|
|
The Original Code is FusionPBX
|
|
|
|
The Initial Developer of the Original Code is
|
|
Mark J Crane <markjcrane@fusionpbx.com>
|
|
Portions created by the Initial Developer are Copyright (C) 2008-2024
|
|
the Initial Developer. All Rights Reserved.
|
|
|
|
Contributor(s):
|
|
Mark J Crane <markjcrane@fusionpbx.com>
|
|
*/
|
|
|
|
/**
|
|
* Auto Loader class
|
|
* Searches for project files when a class is required. Debugging mode can be set using:
|
|
* - export DEBUG=1
|
|
* OR
|
|
* - debug=true is appended to the url
|
|
*/
|
|
class auto_loader {
|
|
|
|
const FILE = 'autoloader_cache.php';
|
|
const CACHE_KEY = 'autoloader_classes';
|
|
|
|
private $classes;
|
|
|
|
/**
|
|
* Tracks the APCu extension for caching to RAM drive across requests
|
|
* @var bool
|
|
*/
|
|
private $apcu_enabled;
|
|
|
|
/**
|
|
* Cache path and file name
|
|
* @var string
|
|
*/
|
|
private static $cache_file = null;
|
|
|
|
public function __construct($project_path = '') {
|
|
|
|
//set if we can use RAM cache
|
|
$this->apcu_enabled = function_exists('apcu_enabled') && apcu_enabled();
|
|
|
|
//set cache location
|
|
self::$cache_file = sys_get_temp_dir() . DIRECTORY_SEPARATOR . self::FILE;
|
|
|
|
//classes must be loaded before this object is registered
|
|
if (!$this->load_cache()) {
|
|
//cache miss so load them
|
|
$this->reload_classes($project_path);
|
|
//update the cache after loading classes array
|
|
$this->update_cache();
|
|
}
|
|
//register this object to load any unknown classes
|
|
spl_autoload_register(array($this, 'loader'));
|
|
}
|
|
|
|
public function update_cache(string $file = ''): bool {
|
|
//guard against writing an empty file
|
|
if (empty($this->classes)) {
|
|
return false;
|
|
}
|
|
|
|
//update RAM cache when available
|
|
if ($this->apcu_enabled) {
|
|
$success = apcu_store(self::CACHE_KEY, $this->classes);
|
|
//do not save to drive when we are using apcu
|
|
if ($success) return true;
|
|
}
|
|
|
|
//ensure we have somewhere to put the file
|
|
if (empty($file)) {
|
|
$file = self::$cache_file;
|
|
}
|
|
|
|
//export the classes array using PHP engine
|
|
$data = var_export($this->classes, true);
|
|
|
|
//put the array in a form that it can be loaded directly to an array
|
|
$result = file_put_contents($file, "<?php\n return " . $data . ";\n");
|
|
if ($result !== false) {
|
|
return true;
|
|
}
|
|
|
|
//file failed to save - send error to syslog when debugging
|
|
$error_array = error_get_last();
|
|
self::log(LOG_WARNING, $error_array['message'] ?? '');
|
|
|
|
return false;
|
|
}
|
|
|
|
public function load_cache(string $file = ''): bool {
|
|
$this->classes = [];
|
|
|
|
//use apcu when available
|
|
if ($this->apcu_enabled && apcu_exists(self::CACHE_KEY)) {
|
|
$this->classes = apcu_fetch(self::CACHE_KEY, $success);
|
|
if ($success) return true;
|
|
}
|
|
|
|
//use a standard file
|
|
if (empty($file)) {
|
|
$file = self::$cache_file;
|
|
}
|
|
|
|
//use PHP engine to parse it
|
|
if (file_exists($file)) {
|
|
$this->classes = include $file;
|
|
}
|
|
|
|
//catch edge case of first time using apcu cache
|
|
if ($this->apcu_enabled) {
|
|
apcu_store(self::CACHE_KEY, $this->classes);
|
|
}
|
|
|
|
//return true when we have classes and false if the array is still empty
|
|
return !empty($this->classes);
|
|
}
|
|
|
|
public function reload_classes($project_path = '') {
|
|
//set project path using magic dir constant
|
|
if (empty($project_path)) {
|
|
$project_path = dirname(__DIR__, 2);
|
|
}
|
|
|
|
//build the array of all locations for classes in specific order
|
|
$search_path = [
|
|
$project_path . '/resources/interfaces/*.php',
|
|
$project_path . '/resources/traits/*.php',
|
|
$project_path . '/resources/classes/*.php',
|
|
$project_path . '/*/*/resources/interfaces/*.php',
|
|
$project_path . '/*/*/resources/traits/*.php',
|
|
$project_path . '/*/*/resources/classes/*.php',
|
|
$project_path . '/core/authentication/resources/classes/plugins/*.php',
|
|
];
|
|
|
|
//get all php files for each path
|
|
$files = [];
|
|
foreach ($search_path as $path) {
|
|
$files = array_merge($files, glob($path));
|
|
}
|
|
|
|
//reset the current array
|
|
$this->classes = [];
|
|
|
|
//store PHP language declared classes, interfaces, and traits
|
|
$curr_classes = get_declared_classes();
|
|
$curr_interfaces = get_declared_interfaces();
|
|
$curr_traits = get_declared_traits();
|
|
|
|
//store the class name (key) and the path (value)
|
|
foreach ($files as $file) {
|
|
|
|
//include the new class
|
|
try {
|
|
include_once $file;
|
|
} catch (Exception $e) {
|
|
//report the error
|
|
self::log(LOG_ERR, "Exception while trying to include file '$file': " . $e->getMessage());
|
|
continue;
|
|
}
|
|
|
|
//get the new classes
|
|
$new_classes = get_declared_classes();
|
|
$new_interfaces = get_declared_interfaces();
|
|
$new_traits = get_declared_traits();
|
|
|
|
//check for a new class
|
|
$classes = array_diff($new_classes, $curr_classes);
|
|
if (!empty($classes)) {
|
|
foreach ($classes as $class) {
|
|
$this->classes[$class] = $file;
|
|
}
|
|
//overwrite previous array with new values
|
|
$curr_classes = $new_classes;
|
|
}
|
|
|
|
//check for a new interface
|
|
$interfaces = array_diff($new_interfaces, $curr_interfaces);
|
|
if (!empty($interfaces)) {
|
|
foreach ($interfaces as $interface) {
|
|
$this->classes[$interface] = $file;
|
|
}
|
|
//overwrite previous array with new values
|
|
$curr_interfaces = $new_interfaces;
|
|
}
|
|
|
|
//check for a new trait
|
|
$traits = array_diff($new_traits, $curr_traits);
|
|
if (!empty($traits)) {
|
|
foreach ($traits as $trait) {
|
|
$this->classes[$trait] = $file;
|
|
}
|
|
//overwrite previous array with new values
|
|
$curr_traits = $new_traits;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* The loader is set to private because only the PHP engine should be calling this method
|
|
* @param string $class_name The class name that needs to be loaded
|
|
* @return bool True if the class is loaded or false when the class is not found
|
|
* @access private
|
|
*/
|
|
private function loader($class_name): bool {
|
|
|
|
//sanitize the class name
|
|
$class_name = preg_replace('[^a-zA-Z0-9_]', '', $class_name);
|
|
|
|
//find the path using the class_name as the key in the classes array
|
|
if (isset($this->classes[$class_name])) {
|
|
//include the class or interface
|
|
include_once $this->classes[$class_name];
|
|
|
|
//return boolean
|
|
return true;
|
|
}
|
|
|
|
//Smarty has it's own autoloader so reject the request
|
|
if ($class_name === 'Smarty_Autoloader') {
|
|
return false;
|
|
}
|
|
|
|
//cache miss
|
|
self::log(LOG_WARNING, "class '$class_name' not found in cache");
|
|
|
|
//set project path using magic dir constant
|
|
$project_path = dirname(__DIR__, 2);
|
|
|
|
//build the search path array
|
|
$search_path[] = glob($project_path . "/resources/interfaces/" . $class_name . ".php");
|
|
$search_path[] = glob($project_path . "/resources/traits/" . $class_name . ".php");
|
|
$search_path[] = glob($project_path . "/resources/classes/" . $class_name . ".php");
|
|
$search_path[] = glob($project_path . "/*/*/resources/interfaces/" . $class_name . ".php");
|
|
$search_path[] = glob($project_path . "/*/*/resources/traits/" . $class_name . ".php");
|
|
$search_path[] = glob($project_path . "/*/*/resources/classes/" . $class_name . ".php");
|
|
|
|
//collapse all entries to only the matched entry
|
|
$matches = array_filter($search_path);
|
|
if (!empty($matches)) {
|
|
$path = array_pop($matches)[0];
|
|
|
|
//include the class, interface, or trait
|
|
include_once $path;
|
|
|
|
//inject the class in to the array
|
|
$this->classes[$class_name] = $path;
|
|
|
|
//update the cache with new classes
|
|
$this->update_cache();
|
|
|
|
//return boolean
|
|
return true;
|
|
}
|
|
|
|
//send to syslog when debugging
|
|
self::log(LOG_ERR, "class '$class_name' not found name");
|
|
|
|
//return boolean
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Returns a list of classes loaded by the auto_loader. If no classes have been loaded an empty array is returned.
|
|
* @return array List of classes loaded by the auto_loader or empty array
|
|
*/
|
|
public function get_class_list(): array {
|
|
if (!empty($this->classes)) {
|
|
return $this->classes;
|
|
}
|
|
return [];
|
|
}
|
|
|
|
public static function clear_cache(string $file = '') {
|
|
|
|
//check for apcu cache
|
|
if (function_exists('apcu_enabled') && apcu_enabled()) {
|
|
apcu_delete(self::CACHE_KEY);
|
|
}
|
|
|
|
//set default file
|
|
if (empty(self::$cache_file)) {
|
|
self::$cache_file = sys_get_temp_dir() . DIRECTORY_SEPARATOR . self::FILE;
|
|
}
|
|
|
|
//set file to clear
|
|
if (empty($file)) {
|
|
$file = self::$cache_file;
|
|
}
|
|
|
|
//remove the file when it exists
|
|
if (file_exists($file)) {
|
|
@unlink($file);
|
|
$error_array = error_get_last();
|
|
//send to syslog when debugging with either environment variable or debug in the url
|
|
self::log(LOG_WARNING, $error_array['message'] ?? '');
|
|
}
|
|
}
|
|
|
|
private static function log(int $level, string $message): void {
|
|
if (filter_var($_REQUEST['debug'] ?? false, FILTER_VALIDATE_BOOL) || filter_var(getenv('DEBUG') ?? false, FILTER_VALIDATE_BOOL)) {
|
|
openlog("PHP", LOG_PID | LOG_PERROR, LOG_LOCAL0);
|
|
syslog($level, "[auto_loader] " . $message);
|
|
closelog();
|
|
}
|
|
}
|
|
}
|