2 // vim: foldmethod=marker
6 * @author ICHII Takashi <ichii386@schweetheart.jp>
7 * @author Kazuhiro Hosoi <hosoi@gree.co.jp>
8 * @license http://www.opensource.org/licenses/bsd-license.php The BSD License
17 * @author ICHII Takashi <ichii386@schweetheart.jp>
18 * @author Kazuhiro Hosoi <hosoi@gree.co.jp>
28 /** @var object Ethna_Controller コントローラオブジェクト */
31 /** @var object Ethna_Controller コントローラオブジェクト($controllerの省略形) */
34 /** @var object Ethna_Logger ログオブジェクト */
37 /** @var array プラグインのオブジェクト(インスタンス)を保存する配列 */
38 var $obj_registry = array();
40 /** @var array プラグインのクラス名、ソースファイル名を保存する配列 */
41 var $src_registry = array();
43 /** @var array 検索対象ディレクトリを,プラグインの優先順に保存する配列 */
44 var $_dirlist = array();
50 * Ethna_Pluginのコンストラクタ
53 * @param object Ethna_Controller $controller コントローラオブジェクト
55 function Ethna_Plugin(&$controller)
57 $this->controller =& $controller;
58 $this->ctl =& $this->controller;
62 $this->_loadPluginDirList();
68 * LogWriterはpluginなので、pluginインスタンス作成時点では
72 * @param object Ethna_Logger $logger ログオブジェクト
74 function setLogger(&$logger)
76 if ($this->logger === null && is_object($logger)) {
77 $this->logger =& $logger;
82 // {{{ プラグイン呼び出しインタフェース
87 * @param string $type プラグインの種類
88 * @param string $name プラグインの名前
89 * @return object プラグインのインスタンス
91 function &getPlugin($type, $name)
93 return $this->_getPlugin($type, $name);
97 * ある種類 ($type) のプラグイン ($name) の全リストを取得
100 * @param string $type プラグインの種類
101 * @return array プラグインオブジェクトの配列
103 function getPluginList($type)
105 $plugin_list = array();
107 $this->searchAllPluginSrc($type);
108 if (isset($this->src_registry[$type]) == false) {
111 foreach ($this->src_registry[$type] as $name => $value) {
112 if (is_null($value)) {
115 $plugin_list[$name] =& $this->getPlugin($type, $name);
121 // {{{ obj_registry のアクセサ
123 * プラグインのインスタンスをレジストリから取得
126 * @param string $type プラグインの種類
127 * @param string $name プラグインの名前
128 * @return object プラグインのインスタンス
130 function &_getPlugin($type, $name)
132 if (isset($this->obj_registry[$type]) == false) {
133 $this->obj_registry[$type] = array();
135 // プラグインの親クラスを(存在すれば)読み込み
136 list($class, $file) = $this->getPluginNaming($type, null);
137 $dir = $this->_searchPluginSrcDir($type, null);
138 if (!Ethna::isError($dir)) {
139 $this->_includePluginSrc($class, $dir, $file, true);
143 // key がないときはプラグインをロードする
144 if (array_key_exists($name, $this->obj_registry[$type]) == false) {
145 $this->_loadPlugin($type, $name);
148 // null のときはロードに失敗している
149 if (is_null($this->obj_registry[$type][$name])) {
150 return Ethna::raiseWarning('plugin [type=%s, name=%s] is not found',
151 E_PLUGIN_NOTFOUND, $type, $name);
155 return $this->obj_registry[$type][$name];
159 * プラグインをincludeしてnewし,レジストリに登録
162 * @param string $type プラグインの種類
163 * @param string $name プラグインの名前
165 function _loadPlugin($type, $name)
168 $plugin_src_registry = $this->_getPluginSrc($type, $name);
169 if (is_null($plugin_src_registry)) {
170 $this->obj_registry[$type][$name] = null;
173 list($plugin_class, $plugin_dir, $plugin_file) = $plugin_src_registry;
176 $r =& $this->_includePluginSrc($plugin_class, $plugin_dir, $plugin_file);
177 if (Ethna::isError($r)) {
178 $this->obj_registry[$type][$name] = null;
183 $instance =& new $plugin_class($this->controller, $type, $name);
184 if (is_object($instance) == false
185 || strcasecmp(get_class($instance), $plugin_class) != 0) {
186 $this->logger->log(LOG_WARNING, 'plugin [%s::%s] instantiation failed', $type, $name);
187 $this->obj_registry[$type][$name] = null;
190 $this->obj_registry[$type][$name] =& $instance;
194 * プラグインのインスタンスをレジストリから消す
197 * @param string $type プラグインの種類
198 * @param string $name プラグインの名前
200 function _unloadPlugin($type, $name)
202 unset($this->obj_registry[$type][$name]);
207 * プラグインのインスタンスをレジストリから消す
210 * @param string $type プラグインの種類
211 * @param string $name プラグインの名前
213 function _loadPluginDirList()
215 $this->_dirlist[] = $this->controller->getDirectory('plugin');
218 $include_path_list = explode(PATH_SEPARATOR, get_include_path());
220 // Communiy based libraries
221 $extlib_dir = implode(DIRECTORY_SEPARATOR, array('Ethna', 'extlib', 'Plugin'));
223 $class_dir = implode(DIRECTORY_SEPARATOR, array('Ethna', 'class', 'Plugin'));
224 foreach ($include_path_list as $include_path) {
225 if (is_dir($include_path . DIRECTORY_SEPARATOR . $extlib_dir)) {
226 $this->_dirlist[] = $include_path . DIRECTORY_SEPARATOR . $extlib_dir;
228 if (is_dir($include_path . DIRECTORY_SEPARATOR . $class_dir)) {
229 $this->_dirlist[] = $include_path . DIRECTORY_SEPARATOR . $class_dir;
234 // {{{ src_registry のアクセサ
236 * プラグインのソースファイル名とクラス名をレジストリから取得
239 * @param string $type プラグインの種類
240 * @param string $name プラグインの名前
241 * @return array ソースファイル名とクラス名からなる配列
243 function _getPluginSrc($type, $name)
245 if (isset($this->src_registry[$type]) == false) {
246 $this->src_registry[$type] = array();
249 // key がないときはプラグインの検索をする
250 if (array_key_exists($name, $this->src_registry[$type]) == false) {
251 $this->_searchPluginSrc($type, $name);
255 return $this->src_registry[$type][$name];
261 * プラグインのクラス名、ディレクトリ、ファイル名を決定
264 * @param string $type プラグインの種類
265 * @param string $name プラグインの名前 (nullのときは親クラス)
266 * @param string $appid アプリケーションID (廃止予定)
267 * @return array プラグインのクラス名、ファイル名の配列
269 function getPluginNaming($type, $name = null, $appid = 'Ethna')
271 $ext = $this->ctl->getExt('php');
273 $plugin_class_name = array(
279 if ($name !== null) {
280 $plugin_class_name[] = $name;
286 $class = implode('_', $plugin_class_name);
287 $file = "{$name}.{$ext}";
289 return array($class, $file);
293 * プラグインのソースを include する
296 * @param string $class クラス名
297 * @param string $dir ディレクトリ名
298 * @param string $file ファイル名
299 * @param bool $parent 親クラスかどうかのフラグ
300 * @return true|Ethna_Error
302 function &_includePluginSrc($class, $dir, $file, $parent = false)
305 if (class_exists($class)) {
309 $file = $dir . '/' . $file;
310 if (file_exists_ex($file) === false) {
311 if ($parent === false) {
312 return Ethna::raiseWarning('plugin file is not found: [%s]',
313 E_PLUGIN_NOTFOUND, $file);
321 if (class_exists($class) === false) {
322 if ($parent === false) {
323 return Ethna::raiseWarning('plugin class [%s] is not defined',
324 E_PLUGIN_NOTFOUND, $class);
330 if ($parent === false) {
331 $this->logger->log(LOG_DEBUG, 'plugin class [%s] is defined', $class);
337 * プラグインのソースディレクトリを決定する
339 * @param string $type プラグインの種類
340 * @param string $name プラグインの名前 (nullのときは親クラス)
341 * @retur string directory
343 function _searchPluginSrcDir($type, $name)
345 list(, $file) = $this->getPluginNaming($type, $name);
348 if ($name !== null) {
349 $dir_prefix = DIRECTORY_SEPARATOR . $type;
353 foreach ($this->_dirlist as $dir) {
356 if (file_exists($dir . DIRECTORY_SEPARATOR . $file)) {
361 return Ethna::raiseWarning('plugin file is not found in search directories: [%s]',
362 E_PLUGIN_NOTFOUND, $file);
366 * アプリケーション, extlib, Ethna の順でプラグインのソースを検索する
369 * @param string $type プラグインの種類
370 * @param string $name プラグインの名前
371 * @return array class, dir, file
373 function _searchPluginSrc($type, $name)
375 list($class, $file) = $this->getPluginNaming($type, $name);
376 if (class_exists($class)) {
377 // すでにクラスが存在する場合は特別にスキップ
378 if (isset($this->src_registry[$type]) == false) {
379 $this->src_registry[$type] = array();
383 $dir = $this->_searchPluginSrcDir($type, $name);
385 if (Ethna::isError($dir)) {
386 $this->src_registry[$type][$name] = null;
390 if (file_exists("{$dir}/{$file}")) {
391 $this->logger->log(LOG_DEBUG, 'plugin file is found in search: [%s/%s]',
393 if (isset($this->src_registry[$type]) == false) {
394 $this->src_registry[$type] = array();
396 $this->src_registry[$type][$name] = array($class, $dir, $file);
400 // 見つからなかった場合 (nullで記憶しておく)
401 $this->logger->log(LOG_WARNING, 'plugin file for [type=%s, name=%s] is not found in search', $type, $name);
402 $this->src_registry[$type][$name] = null;
406 * プラグインの種類 ($type) をすべて検索する
411 function searchAllPluginType()
413 $type_list = array();
414 foreach($this->_dirlist as $dir) {
415 $type_dir= glob($dir . DIRECTORY_SEPARATOR . "*", GLOB_ONLYDIR);
419 foreach ($type_dir as $dir) {
420 if ($type_dir{0} != '.') {
421 $type_list[basename($dir)] = 0;
425 return array_keys($type_list);
429 * 指定された $type のプラグイン (のソース) をすべて検索する
432 * @param string $type プラグインの種類
434 function searchAllPluginSrc($type)
436 // 後で見付かったもので上書きするので $this->appid_list の逆順とする
437 $name_list = array();
438 $ext = $this->ctl->getExt('php');
440 foreach($this->_dirlist as $dir) {
441 $files = glob($dir . DIRECTORY_SEPARATOR . $type . DIRECTORY_SEPARATOR . "/*." . $ext);
443 $this->logger->log(LOG_DEBUG, 'cannot open plugin directory: [%s/%s]', $dir, $type);
446 $this->logger->log(LOG_DEBUG, 'plugin directory opened: [%s]', $dir);
447 foreach ($files as $plugin_file) {
448 $plugin_name = substr(basename($plugin_file), 0, - strlen($ext) - 1);
449 $name_list[$plugin_name] = 0;
453 foreach (array_keys($name_list) as $name) {
455 $this->_searchPluginSrc($type, $name);
460 // {{{ static な include メソッド
462 * Ethna 本体付属のプラグインのソースを include する
465 * @param string $type プラグインの種類
466 * @param string $name プラグインの名前
469 function includeEthnaPlugin($type, $name)
471 Ethna_Plugin::includePlugin($type, $name, 'Ethna');
475 * プラグインのソースを include する
478 * @param string $type プラグインの種類
479 * @param string $name プラグインの名前
480 * @param string $appid アプリケーションID
483 function includePlugin($type, $name, $appid = null)
485 $ctl =& Ethna_Controller::getInstance();
486 $plugin =& $ctl->getPlugin();
488 if ($appid === null) {
489 $appid = $ctl->getAppId();
492 list($class, $file) = $plugin->getPluginNaming($type, $name);
493 $dir = $this->_searchPluginSrcDir($type, $name);
494 $plugin->_includePluginSrc($class, $dir, $file);