1
0
Fork 0
mirror of https://github.com/Oreolek/ifhub.club.git synced 2024-07-03 06:55:03 +03:00
ifhub.club/engine/classes/Engine.class.php

641 lines
19 KiB
PHP
Raw Normal View History

<?php
2008-09-21 09:36:57 +03:00
/*-------------------------------------------------------
*
* LiveStreet Engine Social Networking
* Copyright © 2008 Mzhelskiy Maxim
*
*--------------------------------------------------------
*
* Official site: www.livestreet.ru
* Contact e-mail: rus.engine@gmail.com
*
* GNU General Public License, version 2:
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
*
---------------------------------------------------------
*/
2009-07-26 16:43:16 +03:00
set_include_path(get_include_path().PATH_SEPARATOR.dirname(__FILE__));
require_once(Config::Get('path.root.engine').'/lib/internal/ProfilerSimple/Profiler.class.php');
2009-07-26 16:43:16 +03:00
require_once("Object.class.php");
require_once("Plugin.class.php");
2009-07-26 16:43:16 +03:00
require_once("Block.class.php");
require_once("Hook.class.php");
require_once("Module.class.php");
require_once("Router.class.php");
require_once("Entity.class.php");
require_once("Mapper.class.php");
2008-09-21 09:36:57 +03:00
/**
* Основной класс движка, который позволяет напрямую обращаться к любому модулю
*
*/
class Engine extends Object {
static protected $oInstance=null;
protected $aModules=array();
protected $aPlugins=array();
2008-09-21 09:36:57 +03:00
protected $aConfigModule;
public $iTimeLoadModule=0;
/**
* Массив содержит меппер кастомизации сущностей
*
* @var arrat
*/
static protected $aEntityCustoms=array();
2008-09-21 09:36:57 +03:00
/**
* При создании объекта делаем инициализацию
*
*/
protected function __construct() {
2009-07-26 16:43:16 +03:00
if (get_magic_quotes_gpc()) {
func_stripslashes($_REQUEST);
2010-02-13 15:30:16 +02:00
func_stripslashes($_GET);
func_stripslashes($_POST);
func_stripslashes($_COOKIE);
2009-07-26 16:43:16 +03:00
}
2008-09-21 09:36:57 +03:00
}
/**
* Ограничиваем объект только одним экземпляром
*
* @return Engine
*/
static public function getInstance() {
if (isset(self::$oInstance) and (self::$oInstance instanceof self)) {
return self::$oInstance;
} else {
self::$oInstance= new self();
return self::$oInstance;
}
}
/**
* Инициализация
*
*/
2009-07-26 16:43:16 +03:00
public function Init() {
/**
* Загружаем плагины
*/
$this->LoadPlugins();
/**
* Инициализируем хуки
*/
$this->InitHooks();
/**
* Загружаем модули автозагрузки
*/
2009-07-26 16:43:16 +03:00
$this->LoadModules();
/**
* Инициализируем загруженные модули
*/
2009-07-26 16:43:16 +03:00
$this->InitModules();
/**
* Инициализируем загруженные плагины
*/
$this->InitPlugins();
/**
* Запускаем хуки для события завершения инициализации Engine
*/
$this->Hook_Run('engine_init_complete');
2009-07-26 16:43:16 +03:00
}
/**
* Завершение работы модуля
*
*/
public function Shutdown() {
$this->ShutdownModules();
2008-09-21 09:36:57 +03:00
}
/**
* Производит инициализацию всех модулей
*
*/
2009-07-26 16:43:16 +03:00
protected function InitModules() {
2008-09-21 09:36:57 +03:00
foreach ($this->aModules as $oModule) {
2010-02-04 21:44:19 +02:00
if(!$oModule->isInit()) {
/**
* Замеряем время инициализации модуля
*/
$oProfiler=ProfilerSimple::getInstance();
$iTimeId=$oProfiler->Start('InitModule',get_class($oModule));
2010-02-04 21:26:38 +02:00
$oModule->Init();
$oModule->SetInit();
2010-02-04 21:44:19 +02:00
$oProfiler->Stop($iTimeId);
2010-02-04 21:26:38 +02:00
}
2008-09-21 09:36:57 +03:00
}
}
/**
* Завершаем работу всех модулей
*
*/
2009-07-26 16:43:16 +03:00
protected function ShutdownModules() {
foreach ($this->aModules as $sKey => $oModule) {
/**
* Замеряем время shutdown`a модуля
*/
$oProfiler=ProfilerSimple::getInstance();
$iTimeId=$oProfiler->Start('ShutdownModule',get_class($oModule));
2008-09-21 09:36:57 +03:00
$oModule->Shutdown();
$oProfiler->Stop($iTimeId);
2008-09-21 09:36:57 +03:00
}
}
/**
* Выполняет загрузку модуля по его названию
*
* @param string $sModuleName
* @param bool $bInit - инициализировать модуль или нет
* @return Module
2008-09-21 09:36:57 +03:00
*/
public function LoadModule($sModuleName,$bInit=false) {
$tm1=microtime(true);
if(!preg_match('/^Plugin([\w]+)_([\w]+)$/i',$sModuleName,$aMatches)){
if ($this->isFileExists(Config::Get('path.root.engine')."/modules/".strtolower($sModuleName)."/".$sModuleName.".class.php")) {
require_once(Config::Get('path.root.engine')."/modules/".strtolower($sModuleName)."/".$sModuleName.".class.php");
} elseif ($this->isFileExists(Config::Get('path.root.server')."/classes/modules/".strtolower($sModuleName)."/".$sModuleName.".class.php")) {
require_once(Config::Get('path.root.server')."/classes/modules/".strtolower($sModuleName)."/".$sModuleName.".class.php");
} else {
throw new Exception("Can not find module class - ".$sModuleName);
}
/**
* Проверяем наличие кастомного класса. Также можно переопределить системный модуль
*/
$sPrefixCustom='';
if ($this->isFileExists(Config::Get('path.root.server')."/classes/modules/".strtolower($sModuleName)."/".$sModuleName.".class.custom.php")) {
require_once(Config::Get('path.root.server')."/classes/modules/".strtolower($sModuleName)."/".$sModuleName.".class.custom.php");
$sPrefixCustom='_custom';
}
/**
* Определяем имя класса
*/
$sModuleNameClass='Module'.$sModuleName.$sPrefixCustom;
2008-09-21 09:36:57 +03:00
} else {
/**
* Это модуль плагина
*/
$sModuleFile = Config::Get('path.root.server').'/plugins/'.strtolower($aMatches[1]).'/classes/modules/'.strtolower($aMatches[2]).'/'.$aMatches[2].'.class.php';
if($this->isFileExists($sModuleFile)) {
require_once($sModuleFile);
} else {
throw new Exception("Can not find module class - ".$sModuleName);
}
/**
* Определяем имя класса
*/
$sModuleNameClass='Plugin'.$aMatches[1].'_Module'.$aMatches[2];
}
/**
* Создаем объект модуля
*/
$oModule=new $sModuleNameClass($this);
2010-02-13 15:38:19 +02:00
if ($bInit or $sModuleName=='Cache') {
2008-09-21 09:36:57 +03:00
$oModule->Init();
$oModule->SetInit();
2008-09-21 09:36:57 +03:00
}
$this->aModules[$sModuleName]=$oModule;
$tm2=microtime(true);
$this->iTimeLoadModule+=$tm2-$tm1;
dump("load $sModuleName - \t\t".($tm2-$tm1)."");
return $oModule;
}
/**
* Загружает все используемые модули и передает им в конструктор ядро
*
*/
protected function LoadModules() {
$this->LoadConfig();
2008-09-21 09:36:57 +03:00
foreach ($this->aConfigModule['autoLoad'] as $sModuleName) {
2010-02-13 15:09:48 +02:00
if (!isset($this->aModules[$sModuleName])) {
$this->LoadModule($sModuleName);
}
2008-09-21 09:36:57 +03:00
}
}
/**
* Выполняет загрузку конфигов
*
*/
protected function LoadConfig() {
$this->aConfigModule = Config::Get('module');
}
2009-05-09 20:04:04 +03:00
/**
* Регистрирует хуки из /classes/hooks/
*
*/
2009-07-26 16:43:16 +03:00
protected function InitHooks() {
$sDirHooks=Config::Get('path.root.server').'/classes/hooks/';
$aFiles=glob($sDirHooks.'Hook*.class.php');
if($aFiles and count($aFiles)) {
foreach ($aFiles as $sFile) {
if (preg_match("/Hook([\w]+)\.class\.php$/i",basename($sFile),$aMatch)) {
require_once($sFile);
$sClassName='Hook'.$aMatch[1];
$oHook=new $sClassName;
$oHook->RegisterHook();
}
}
}
/**
* Подгружаем хуки активных плагинов
*/
$this->InitPluginHooks();
}
/**
* Инициализация хуков активированных плагинов
*
*/
protected function InitPluginHooks() {
if($aPluginList = @file(Config::Get('path.root.server').'/plugins/plugins.dat')) {
$aPluginList=array_map('trim',$aPluginList);
$aFiles=array();
$sDirHooks=Config::Get('path.root.server').'/plugins/';
foreach ($aPluginList as $sPluginName) {
$aFiles=glob($sDirHooks.$sPluginName.'/classes/hooks/Hook*.class.php');
if($aFiles and count($aFiles)) {
foreach ($aFiles as $sFile) {
if (preg_match("/Hook([\w]+)\.class\.php$/i",basename($sFile),$aMatch)) {
require_once($sFile);
2010-03-06 21:34:00 +02:00
$sPluginName = ucfirst($sPluginName);
$sClassName="Plugin{$sPluginName}_Hook{$aMatch[1]}";
$oHook=new $sClassName;
$oHook->RegisterHook();
}
}
2009-05-09 20:04:04 +03:00
}
}
}
}
/**
* Загрузка плагинов и делегирование
*
*/
protected function LoadPlugins() {
if($aPluginList = @file(Config::Get('path.root.server').'/plugins/plugins.dat')) {
$aPluginList=array_map('trim',$aPluginList);
foreach ($aPluginList as $sPluginName) {
$sDirPlugins=Config::Get('path.root.server').'/plugins/';
2010-02-13 15:09:48 +02:00
$sPluginNameClass='Plugin'.ucfirst($sPluginName);
$sFile="{$sDirPlugins}{$sPluginName}/{$sPluginNameClass}.class.php";
if(is_file($sFile)) {
2010-02-13 15:09:48 +02:00
require_once($sFile);
$sClassName="{$sPluginNameClass}";
$oPlugin=new $sClassName;
$oPlugin->Delegate();
$this->aPlugins[$sPluginName]=$oPlugin;
}
}
}
}
/**
* Инициализация активированных плагинов
*
*/
protected function InitPlugins() {
foreach ($this->aPlugins as $oPlugin) {
$oPlugin->Init();
}
}
/**
* Возвращает список активных плагинов
*
* @return unknown
*/
public function GetPlugins() {
return $this->aPlugins;
}
/**
* Проверяет файл на существование, если используется кеширование memcache то кеширует результат работы
*
* @param string $sFile
* @return mixed
*/
2009-07-26 16:43:16 +03:00
public function isFileExists($sFile,$iTime=3600) { return file_exists($sFile);
if (strpos($sFile,'/Cache.class.')!==false) {
return file_exists($sFile);
}
if (SYS_CACHE_USE and SYS_CACHE_TYPE==SYS_CACHE_TYPE_MEMORY) {
if (false === ($data = $this->Cache_Get("file_exists_{$sFile}"))) {
$data=file_exists($sFile);
$this->Cache_Set((int)$data, "file_exists_{$sFile}", array(), $iTime);
}
return $data;
} else {
return file_exists($sFile);
}
}
2008-09-21 09:36:57 +03:00
/**
* Вызывает метод нужного модуля
*
* @param string $sName
* @param array $aArgs
* @return unknown
*/
public function _CallModule($sName,$aArgs) {
2008-09-21 09:36:57 +03:00
$sArgs='';
$aStrArgs=array();
foreach ($aArgs as $sKey => $arg) {
$aStrArgs[]='$aArgs['.$sKey.']';
}
$sArgs=join(',',$aStrArgs);
list($oModule,$sModuleName,$sMethod)=$this->GetModule($sName);
2008-09-21 09:36:57 +03:00
if (!method_exists($oModule,$sMethod)) {
throw new Exception("The module has no required method: ".$sModuleName.'->'.$sMethod.'()');
}
2009-07-26 16:43:16 +03:00
$oProfiler=ProfilerSimple::getInstance();
$iTimeId=$oProfiler->Start('callModule',$sModuleName.'->'.$sMethod.'()');
$sModuleName=strtolower($sModuleName);
$aResultHook=array();
if (!in_array($sModuleName,array('plugin','hook'))) {
2010-03-28 00:53:39 +02:00
$aResultHook=$this->_CallModule('Hook_Run',array('module_'.$sModuleName.'_'.strtolower($sMethod).'_before',&$aArgs));
}
if (array_key_exists('delegate_result',$aResultHook)) {
$result=$aResultHook['delegate_result'];
} else {
2010-03-30 23:28:29 +03:00
$aArgsRef=array();
foreach ($aArgs as $key=>$v) {
$aArgsRef[]=&$aArgs[$key];
}
$result=call_user_func_array(array($oModule,$sMethod),$aArgsRef);
}
if (!in_array($sModuleName,array('plugin','hook'))) {
$this->Hook_Run('module_'.$sModuleName.'_'.strtolower($sMethod).'_after',array('result'=>$result));
}
2009-07-26 16:43:16 +03:00
$oProfiler->Stop($iTimeId);
return $result;
2008-09-21 09:36:57 +03:00
}
/**
* Возвращает объект модуля, имя модуля и имя вызванного метода
*
* @param string $sName
* @return array
*/
public function GetModule($sName) {
$aName=explode("_",$sName);
if(count($aName)==2) {
$sModuleName=$aName[0];
$sMethod=$aName[1];
} else {
$sModuleName=$aName[0].'_'.$aName[1];
$sMethod=$aName[2];
}
/**
* Подхватыем делегат модуля (в случае наличия такового)
*/
if(!in_array($sModuleName,array('Plugin','Hook'))) $sModuleName=$this->Plugin_GetDelegate('module',$sModuleName);
if (isset($this->aModules[$sModuleName])) {
$oModule=$this->aModules[$sModuleName];
} else {
$oModule=$this->LoadModule($sModuleName,true);
}
return array($oModule,$sModuleName,$sMethod);
}
2009-07-26 16:43:16 +03:00
public function getStats() {
return array('sql'=>$this->Database_GetStats(),'cache'=>$this->Cache_GetStats(),'engine'=>array('time_load_module'=>round($this->iTimeLoadModule,3)));
}
2008-09-21 09:36:57 +03:00
/**
* Ставим хук на вызов неизвестного метода и считаем что хотели вызвать метод какого либо модуля
*
* @param string $sName
* @param array $aArgs
* @return unknown
*/
public function __call($sName,$aArgs) {
return $this->_CallModule($sName,$aArgs);
}
/**
* Блокируем копирование/клонирование объекта роутинга
*
*/
protected function __clone() {
}
/**
* Получает объект маппера
*
* @param string $sClassName
* @param string $sName
* @return mixed
*/
public static function GetMapper($sClassName,$sName=null,$oConnect=null) {
if (preg_match("/^(?:Plugin\w+_)?Module(\w+)$/i",$sClassName,$aMatch)) {
if (!$sName) {
$sName=$aMatch[1];
}
$sClass=$sClassName.'_Mapper'.$sName;
if (!$oConnect) {
$oConnect=Engine::getInstance()->Database_GetConnect();
}
return new $sClass($oConnect);
}
return null;
}
/**
* Создает объект сущности, контролируя варианты кастомизации
*
* @param string $sName
* @param mixed $aParams
* @return mixed
*/
public static function GetEntity($sName,$aParams=array()) {
/**
* Сущности, имеющие такое же название как модуль,
* можно вызывать сокращенно. Например, вместо User_User -> User
*/
switch (substr_count($sName,'_')) {
case 0:
$sEntity = $sModule = $sName;
break;
case 1:
list($sModule,$sEntity) = explode('_',$sName,2);
/**
* Обслуживание короткой записи сущностей плагинов
* PluginTest_Test -> PluginTest_ModuleTest_EntityTest
*/
if(substr($sModule,0,6)=='Plugin' and strlen($sModule)>6) {
$sPlugin = substr($sModule,6);
$sModule = $sEntity;
}
break;
case 2:
/**
* Entity плагина
*/
if(substr($sName,0,6)=='Plugin') {
list($sPlugin,$sModule,$sEntity)=explode('_',$sName);
$sPlugin = substr($sPlugin,6);
} else {
throw new Exception("Unknown entity '{$sName}' given.");
}
break;
default:
throw new Exception("Unknown entity '{$sName}' given.");
}
/**
* Проверяем наличие сущности в меппере кастомизации
*/
if(array_key_exists($sName,self::$aEntityCustoms)) {
$sEntity = (self::$aEntityCustoms[$sName]=='custom')
? $sEntity.'_custom'
: $sEntity;
} else {
$sFileDefaultClass=isset($sPlugin)
? Config::get('path.root.server').'/plugins/'.strtolower($sPlugin).'/classes/modules/'.strtolower($sModule).'/entity/'.$sEntity.'.entity.class.php'
: Config::get('path.root.server').'/classes/modules/'.strtolower($sModule).'/entity/'.$sEntity.'.entity.class.php';
$sFileCustomClass=isset($sPlugin)
? Config::get('path.root.server').'/plugins/'.strtolower($sPlugin).'/classes/modules/'.strtolower($sModule).'/entity/'.$sEntity.'.entity.class.custom.php'
: Config::get('path.root.server').'/classes/modules/'.strtolower($sModule).'/entity/'.$sEntity.'.entity.class.custom.php';
/**
* Пытаемся найти кастомизированную сущность
*/
if(file_exists($sFileCustomClass)) {
$sFileClass=$sFileCustomClass;
$sEntity.='_custom';
self::$aEntityCustoms[$sName]='custom';
} elseif(file_exists($sFileDefaultClass)) {
$sFileClass=$sFileDefaultClass;
self::$aEntityCustoms[$sName]='default';
} else {
throw new Exception('Entity class not found');
return null;
}
/**
* Подгружаем нужный файл
*/
require_once($sFileClass);
}
$sClass=isset($sPlugin)
? 'Plugin'.$sPlugin.'_Module'.$sModule.'_Entity'.$sEntity
: 'Module'.$sModule.'_Entity'.$sEntity;
/**
* Определяем наличие делегата сущности
* Делегирование указывается только в полной форме!
*/
$sClass=self::getInstance()->Plugin_GetDelegate('entity',$sClass);
$oEntity=new $sClass($aParams);
return $oEntity;
}
2008-09-21 09:36:57 +03:00
}
/**
* Автозагрузка кслассов
*
* @param unknown_type $sClassName
*/
function __autoload($sClassName) {
/**
* Если класс подходит под шаблон класса сущности то загружаем его
2008-09-21 09:36:57 +03:00
*/
if (preg_match("/^Module(\w+)\_Entity(\w+)$/i",$sClassName,$aMatch)) {
2008-09-21 09:36:57 +03:00
$tm1=microtime(true);
$sFileClass= (substr($aMatch[2],-7)=='_custom')
? Config::get('path.root.server').'/classes/modules/'.strtolower($aMatch[1]).'/entity/'.substr($aMatch[2],0,strlen($aMatch[2])-7).'.entity.class.custom.php'
: Config::get('path.root.server').'/classes/modules/'.strtolower($aMatch[1]).'/entity/'.$aMatch[2].'.entity.class.php';
2008-09-21 09:36:57 +03:00
if (file_exists($sFileClass)) {
require_once($sFileClass);
$tm2=microtime(true);
dump($sClassName." - \t\t".($tm2-$tm1));
}
}
/**
* Если класс подходит под шаблон класса сущности плагина
*/
if (preg_match("/^Plugin(\w+)\_Module(\w+)\_Entity(\w+)$/i",$sClassName,$aMatch)) {
$tm1=microtime(true);
$sFileClass= Config::get('path.root.server').'/plugins/'.strtolower($aMatch[1]).'/classes/modules/'.strtolower($aMatch[2]).'/entity/'.$aMatch[3].'.entity.class.php';
if (file_exists($sFileClass)) {
require_once($sFileClass);
$tm2=microtime(true);
dump($sClassName." - \t\t".($tm2-$tm1));
}
}
/**
* Если класс подходит под шаблон модуля, то загружаем его
*/
if(preg_match("/^Module(\w+)$/i",$sClassName,$aMatch)) {
$sName = ucfirst(strtolower($aMatch[1]));
$sFileClass= (substr($sName,-7)=='_custom')
2009-11-25 16:10:54 +02:00
? Config::get('path.root.server').'/classes/modules/'.strtolower($sName).'/'.substr($sName,0,strlen($sName)-7).'.class.custom.php'
: Config::get('path.root.server').'/classes/modules/'.strtolower($sName).'/'.$sName.'.class.php';
if (file_exists($sFileClass)) {
require_once($sFileClass);
} else {
$sFileClass = str_replace('/classes/modules/','/engine/modules/',$sFileClass);
if(file_exists($sFileClass)) require_once($sFileClass);
}
}
/**
* Если класс подходит под шаблон класса маппера, то загружаем его
*/
if (preg_match("/^Module(\w+)\_Mapper(\w+)$/i",$sClassName,$aMatch)) {
$sFileClass=Config::get('path.root.server').'/classes/modules/'.strtolower($aMatch[1]).'/mapper/'.$aMatch[2].'.mapper.class.php';
if (file_exists($sFileClass)) {
require_once($sFileClass);
}
}
/**
* Если класс подходит под шаблон класса маппера плагина, то загружаем его
*/
if (preg_match("/^Plugin(\w+)\_Module(\w+)\_Mapper(\w+)$/i",$sClassName,$aMatch)) {
$sFileClass=Config::get('path.root.server').'/plugins/'.strtolower($aMatch[1]).'/classes/modules/'.strtolower($aMatch[2]).'/mapper/'.$aMatch[2].'.mapper.class.php';
if (file_exists($sFileClass)) {
require_once($sFileClass);
}
}
2008-09-21 09:36:57 +03:00
}
?>