source: branches/camp/camp-2_13-plugin/data/class/pages/admin/ownersstore/LC_Page_Admin_OwnersStore.php @ 22668

Revision 22668, 40.7 KB checked in by adachi, 11 years ago (diff)

#2181 トランザクションの範囲を変更

  • Property svn:eol-style set to LF
  • Property svn:keywords set to Id
  • Property svn:mime-type set to text/x-httpd-php; charset=UTF-8
Line 
1<?php
2/*
3 * This file is part of EC-CUBE
4 *
5 * Copyright(c) 2000-2013 LOCKON CO.,LTD. All Rights Reserved.
6 *
7 * http://www.lockon.co.jp/
8 *
9 * This program is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU General Public License
11 * as published by the Free Software Foundation; either version 2
12 * of the License, or (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
22 */
23
24// {{{ requires
25require_once CLASS_EX_REALDIR . 'page_extends/admin/LC_Page_Admin_Ex.php';
26
27/**
28 * オーナーズストア:プラグイン管理 のページクラス.
29 *
30 * @package Page
31 * @author LOCKON CO.,LTD.
32 * @version $Id$
33 */
34class LC_Page_Admin_OwnersStore extends LC_Page_Admin_Ex
35{
36
37    // }}}
38    // {{{ functions
39
40    /**
41     * Page を初期化する.
42     *
43     * @return void
44     */
45    function init()
46    {
47        parent::init();
48        $this->tpl_mainpage = 'ownersstore/plugin.tpl';
49        $this->tpl_subno    = 'index';
50        $this->tpl_mainno   = 'ownersstore';
51        $this->tpl_maintitle = 'オーナーズストア';
52        $this->tpl_subtitle = 'プラグイン管理';
53    }
54
55    /**
56     * Page のプロセス.
57     *
58     * @return void
59     */
60    function process()
61    {
62        $this->action();
63        $this->sendResponse();
64    }
65
66    /**
67     * Page のアクション.
68     *
69     * @return void
70     */
71    function action()
72    {
73        // パラメーター管理クラス
74        $objFormParam = new SC_FormParam_Ex();
75        $mode = $this->getMode();
76        // パラメーター情報の初期化
77        $this->initParam($objFormParam, $mode);
78        $objFormParam->setParam($_POST);
79
80        switch ($mode) {
81            // インストール
82            case 'install':
83                $file_key = 'plugin_file';
84                $this->arrErr = $this->checkUploadFile($file_key);
85                if ($this->isError($this->arrErr) === false) {
86                    $archive_file_name = $_FILES[$file_key]['name'];
87                    // インストール処理.
88                    $this->arrErr = $this->installPlugin($archive_file_name, 'plugin_file');
89                    if ($this->isError($this->arrErr) === false) {
90                        // コンパイルファイルのクリア処理
91                        SC_Utils_Ex::clearCompliedTemplate();
92                        $this->tpl_onload = "alert('プラグインをインストールしました。');";
93                    }
94                }
95                break;
96            // 削除
97            case 'uninstall':
98                // エラーチェック
99                $this->arrErr = $objFormParam->checkError();
100                if ($this->isError($this->arrErr) === false) {
101                    $plugin_id = $objFormParam->getValue('plugin_id');
102                    $plugin = SC_Plugin_Util_Ex::getPluginByPluginId($plugin_id);
103                    $this->arrErr = $this->uninstallPlugin($plugin);
104                    if ($this->isError($this->arrErr) === false) {
105                        // TODO 全プラグインのインスタンスを保持したまま後続処理が実行されるので、全てのインスタンスを解放する。
106                        unset($GLOBALS['_SC_Helper_Plugin_instance']);
107                        // コンパイルファイルのクリア処理
108                        SC_Utils_Ex::clearCompliedTemplate();
109                        $this->tpl_onload = "alert('" . $plugin['plugin_name'] ."を削除しました。');";
110                    }
111                }
112                break;
113            // 有効化
114            case 'enable':
115                // エラーチェック
116                $this->arrErr = $objFormParam->checkError();
117                if ($this->isError($this->arrErr) === false) {
118                    $plugin_id = $objFormParam->getValue('plugin_id');
119                    // プラグイン取得.
120                    $plugin = SC_Plugin_Util_Ex::getPluginByPluginId($plugin_id);
121                    $this->arrErr = $this->enablePlugin($plugin);
122                    if ($this->isError($this->arrErr) === false) {
123                        // TODO 全プラグインのインスタンスを保持したまま後続処理が実行されるので、全てのインスタンスを解放する。
124                        unset($GLOBALS['_SC_Helper_Plugin_instance']);
125                        // コンパイルファイルのクリア処理
126                        SC_Utils_Ex::clearCompliedTemplate();
127                        $this->tpl_onload = "alert('" . $plugin['plugin_name'] . "を有効にしました。');";
128                    }
129                }
130                break;
131            // 無効化
132            case 'disable':
133                // エラーチェック
134                $this->arrErr = $objFormParam->checkError();
135                if ($this->isError($this->arrErr) === false) {
136                    $plugin_id = $objFormParam->getValue('plugin_id');
137                    // プラグイン取得.
138                    $plugin = SC_Plugin_Util_Ex::getPluginByPluginId($plugin_id);
139                    $this->arrErr = $this->disablePlugin($plugin);
140                    if ($this->isError($this->arrErr) === false) {
141                        // TODO 全プラグインのインスタンスを保持したまま後続処理が実行されるので、全てのインスタンスを解放する。
142                        unset($GLOBALS['_SC_Helper_Plugin_instance']);
143                        // コンパイルファイルのクリア処理
144                        SC_Utils_Ex::clearCompliedTemplate();
145                        $this->tpl_onload = "alert('" . $plugin['plugin_name'] . "を無効にしました。');";
146                    }
147                }
148                break;
149            // アップデート.
150            case 'update':
151                // エラーチェック
152                $this->arrErr = $objFormParam->checkError();
153                if ($this->isError($this->arrErr) === false) {
154                    $plugin_id = $objFormParam->getValue('plugin_id');
155                    $plugin = SC_Plugin_Util_Ex::getPluginByPluginId($plugin_id);
156                    $target_plugin_code = $plugin['plugin_code']; // アップデート対象のプラグインコード
157                    $this->arrErr = $this->checkUploadFile($target_plugin_code);
158
159                    if ($this->isError($this->arrErr) === false) {
160                        $update_plugin_file = $_FILES[$target_plugin_code];
161                        $update_plugin_file_name = $update_plugin_file['name']; // アップデートファイルのファイル名.
162                        // インストール処理.
163                        $target_plugin = SC_Plugin_Util_Ex::getPluginByPluginCode($target_plugin_code);
164                        $this->arrErr = $this->updatePlugin($target_plugin, $update_plugin_file_name, $target_plugin_code);
165                        if ($this->isError($this->arrErr) === false) {
166                            // コンパイルファイルのクリア処理
167                            SC_Utils_Ex::clearCompliedTemplate();
168                            $this->tpl_onload = "alert('プラグインをアップデートしました。');";
169                        }
170                    }
171                }
172                break;
173            // 優先度.
174            case 'priority':
175                // エラーチェック
176                $arrErr = $objFormParam->checkError();
177                $plugin_id = $objFormParam->getValue('plugin_id');
178                if ($this->isError($arrErr) === false) {
179                    // 優先度の更新
180                    $priority = $objFormParam->getValue('priority');
181                    $this->updatePriority($plugin_id, $priority);
182                    // コンパイルファイルのクリア処理
183                    SC_Utils_Ex::clearCompliedTemplate();
184                } else {
185                    // エラーメッセージを詰め直す.
186                    $this->arrErr['priority'][$plugin_id] = $arrErr['priority'];
187                }
188
189                break;
190            default:
191                break;
192        }
193        // DBからプラグイン情報を取得
194        $plugins = SC_Plugin_Util_Ex::getAllPlugin();
195
196        foreach ($plugins as $key => $plugin) {
197            // ロゴファイルへのパスを生成(ロゴが無い場合はNO_IMAGEを表示)
198            if (file_exists(PLUGIN_HTML_REALDIR . $plugins[$key]['plugin_code'] . '/logo.png') === true){
199                $plugins[$key]['logo'] = ROOT_URLPATH . 'plugin/' . $plugins[$key]['plugin_code'] . '/logo.png';
200            } else {
201                $plugins[$key]['logo'] = IMAGE_SAVE_URLPATH . 'noimage_plugin_list.png';
202            }
203
204            // 設定ファイルがあるかを判定.
205            $plugins[$key]['config_flg'] = $this->isContainsFile(PLUGIN_UPLOAD_REALDIR . $plugin['plugin_code'], 'config.php');
206            if ($plugins[$key]['enable'] === PLUGIN_ENABLE_TRUE) {
207                // 競合するプラグインがあるかを判定.
208                //$plugins[$key]['conflict_message']= $this->checkConflictPlugin($plugin['plugin_id']);
209                $plugins[$key]['conflict_message'] = SC_Plugin_Util_Ex::checkConflictPlugin($plugin['plugin_id']);
210            }
211        }
212        $this->plugins = $plugins;
213    }
214
215    /**
216     * デストラクタ.
217     *
218     * @return void
219     */
220    function destroy()
221    {
222        parent::destroy();
223    }
224
225    /**
226     * パラメーター初期化.
227     *
228     * @param SC_FormParam_Ex $objFormParam
229     * @param string $mode モード
230     * @return void
231     */
232    function initParam(&$objFormParam, $mode)
233    {
234        $objFormParam->addParam('mode', 'mode', INT_LEN, '', array('ALPHA_CHECK', 'MAX_LENGTH_CHECK'));
235        $objFormParam->addParam('plugin_id', 'plugin_id', INT_LEN, '', array('NUM_CHECK', 'MAX_LENGTH_CHECK'));
236        if ($mode === 'priority') {
237            $objFormParam->addParam('優先度', 'priority', INT_LEN, '', array('EXIST_CHECK', 'NUM_CHECK', 'MAX_LENGTH_CHECK'));
238        }
239    }
240
241    /**
242     * ファイルパラメーター初期化.
243     *
244     * @param SC_UploadFile_Ex $objUpFile SC_UploadFileのインスタンス.
245     * @param string $key 登録するキー.
246     * @return void
247     */
248    function initUploadFile(&$objUpFile, $key)
249    {
250        $objUpFile->addFile('プラグインファイル', $key, explode(',', PLUGIN_EXTENSION), FILE_SIZE, true, 0, 0, false);
251    }
252
253    /**
254     * ファイルが指定されている事をチェックします.
255     *
256     * @param string $file ファイル
257     * @param string $file_key ファイルキー
258     * @return array エラー情報を格納した連想配列.
259     */
260    function checkUploadFile($file_key)
261    {
262        $objErr = new SC_CheckError_Ex();
263        // 拡張子チェック
264        $objErr->doFunc(array('プラグインファイル', $file_key, explode(',', PLUGIN_EXTENSION)), array('FILE_EXT_CHECK'));
265        // ファイルサイズチェック
266        $objErr->doFunc(array('プラグインファイル', $file_key, FILE_SIZE), array('FILE_SIZE_CHECK'));
267        // ファイル名チェック
268        $objErr->doFunc(array('プラグインファイル', $file_key), array('FILE_NAME_CHECK'));
269
270        return $objErr->arrErr;
271    }
272
273    /**
274     * 既にインストールされているプラグインかを判定します.
275     *
276     * @param string $plugin_code プラグインコード
277     * @return boolean インストール済の場合true インストールされていない場合false
278     */
279    function isInstalledPlugin($plugin_code)
280    {
281        $plugin = SC_Plugin_Util_Ex::getPluginByPluginCode($plugin_code);
282        if (!empty($plugin)) {
283            return true;
284        }
285        return false;
286    }
287
288    /**
289     * ファイル名からプラグインコードを取得する.
290     *
291     * ファイル名を「.」で配列に分解.
292     * 配列内から拡張子として格納される可能性のある「tar」「gz」を除外すし、再度結合する.
293     *
294     * @param string $file_name ファイル名
295     * @return string $plugin_code プラグインコード.
296     */
297    function getPluginCode($file_name)
298    {
299        // 分解
300        $array_ext = explode('.', $file_name);
301        $array_file_name = array_diff($array_ext, array('tar','gz'));
302        // 結合
303        $plugin_code = implode('.', $array_file_name);
304        return $plugin_code;
305    }
306
307    /**
308     * プラグイン保存ディレクトリのパスを取得する.
309     *
310     * @param string $plugin_code プラグインコード
311     * @return string $plugin_dir_path プラグイン保存ディレクトリのパス.
312     */
313    function getPluginDir($plugin_code)
314    {
315        $plugin_dir_path = PLUGIN_UPLOAD_REALDIR . $plugin_code . '/';
316        return $plugin_dir_path;
317    }
318
319    /**
320     * プラグインHTMLディレクトリのパスを取得する.
321     *
322     * @param string $plugin_code プラグインコード
323     * @return string $plugin_dir_path プラグイン保存ディレクトリのパス.
324     */
325    function getHtmlPluginDir($plugin_code)
326    {
327        $plugin_html_dir_path = PLUGIN_HTML_REALDIR . $plugin_code . '/';
328        return $plugin_html_dir_path;
329    }
330
331    /**
332     * プラグインファイルのパスを取得する.
333     *
334     * @param string $plugin_code プラグインコード
335     * @param string $plugin_class プラグインクラス名
336     * @return string $plugin_file_path クラスファイルのパス.
337     */
338    function getPluginFilePath($plugin_code , $plugin_class)
339    {
340        $plugin_file_path = $this->getPluginDir($plugin_code) . $plugin_class . '.php';
341        return $plugin_file_path;
342    }
343
344    /**
345     * プラグインをインストールします.
346     *
347     * @param string $archive_file_name アーカイブファイル名.
348     * @param string $key キー.
349     * @return array エラー情報を格納した連想配列.
350     */
351    function installPlugin($archive_file_name, $key)
352    {
353        $objQuery =& SC_Query_Ex::getSingletonInstance();
354        $objQuery->begin();
355       
356        // 一時展開ディレクトリにファイルがある場合は事前に削除.
357        $arrFileHash = SC_Helper_FileManager_Ex::sfGetFileList(DOWNLOADS_TEMP_PLUGIN_INSTALL_DIR);
358        if (count($arrFileHash) > 0) {
359            SC_Helper_FileManager_Ex::deleteFile(DOWNLOADS_TEMP_PLUGIN_INSTALL_DIR, false);
360        }
361
362        //シンタックスエラーがあるtar.gzをアップ後、削除するとたまにディレクトリが消えるので追加
363        $this->makeDir(PLUGIN_UPLOAD_REALDIR);
364
365        $arrErr = array();
366        // 必須拡張モジュールのチェック
367        $arrErr = SC_Plugin_Util_Ex::checkExtension($key);
368        if ($this->isError($arrErr) === true) {
369            return $arrErr;
370        }
371        // ファイルをチェックし一時展開用ディレクトリに展開します.
372        $arrErr = $this->unpackPluginFile($archive_file_name, DOWNLOADS_TEMP_PLUGIN_INSTALL_DIR, $key);
373        if ($this->isError($arrErr) === true) {
374            return $arrErr;
375        }
376        // plugin_infoを読み込み.
377        $arrErr = $this->requirePluginFile(DOWNLOADS_TEMP_PLUGIN_INSTALL_DIR . 'plugin_info.php', $key);
378        if ($this->isError($arrErr) === true) {
379            $this->rollBack(DOWNLOADS_TEMP_PLUGIN_INSTALL_DIR);
380            return $arrErr;
381        }
382
383        // リフレクションオブジェクトを生成.
384        $objReflection = new ReflectionClass('plugin_info');
385        $arrPluginInfo = $this->getPluginInfo($objReflection);
386        // プラグインクラスに必須となるパラメータが正常に定義されているかチェックします.
387        $arrErr = $this->checkPluginConstants($objReflection, DOWNLOADS_TEMP_PLUGIN_INSTALL_DIR);
388        if ($this->isError($arrErr) === true) {
389            $this->rollBack(DOWNLOADS_TEMP_PLUGIN_INSTALL_DIR);
390            return $arrErr;
391        }
392
393        // 既にインストールされていないかを判定.
394        if ($this->isInstalledPlugin($arrPluginInfo['PLUGIN_CODE']) === true) {
395            $this->rollBack(DOWNLOADS_TEMP_PLUGIN_INSTALL_DIR);
396            $arrErr['plugin_file'] = '※ ' . $arrPluginInfo['PLUGIN_NAME'] . 'は既にインストールされています。<br/>';
397            return $arrErr;
398        }
399
400        // プラグイン情報をDB登録
401        if ($this->registerData($arrPluginInfo) === false) {
402            $this->rollBack(DOWNLOADS_TEMP_PLUGIN_INSTALL_DIR);
403            $arrErr['plugin_file'] = '※ DB登録に失敗しました。<br/>';
404            return $arrErr;
405        }
406
407        // プラグイン保存ディレクトリを作成し、一時展開用ディレクトリから移動します.
408        $plugin_dir_path = $this->getPluginDir($arrPluginInfo['PLUGIN_CODE']);
409        $this->makeDir($plugin_dir_path);
410        SC_Utils_Ex::copyDirectory(DOWNLOADS_TEMP_PLUGIN_INSTALL_DIR, $plugin_dir_path);
411
412        // プラグイン情報を取得
413        $plugin = SC_Plugin_Util_Ex::getPluginByPluginCode($arrPluginInfo['PLUGIN_CODE']);
414
415        // クラスファイルを読み込み.
416        $plugin_class_file_path = $this->getPluginFilePath($plugin['plugin_code'], $plugin['class_name']);
417        $arrErr = $this->requirePluginFile($plugin_class_file_path, $key);
418        if ($this->isError($arrErr) === true) {
419            $this->rollBack(DOWNLOADS_TEMP_PLUGIN_INSTALL_DIR, $plugin['plugin_id']);
420            return $arrErr;
421        }
422        // プラグインhtmlディレクトリ作成
423        $plugin_html_dir_path = $this->getHtmlPluginDir($plugin['plugin_code']);
424        $this->makeDir($plugin_html_dir_path);
425
426        $arrErr = $this->execPlugin($plugin, $plugin['class_name'], 'install');
427        if ($this->isError($arrErr) === true) {
428            $this->rollBack(DOWNLOADS_TEMP_PLUGIN_INSTALL_DIR, $plugin['plugin_id'], $plugin_html_dir_path);
429            return $arrErr;
430        }
431
432        $objQuery->commit();
433       
434        // 不要なファイルの削除
435        SC_Helper_FileManager_Ex::deleteFile(DOWNLOADS_TEMP_PLUGIN_INSTALL_DIR, false);
436        return $arrErr;
437    }
438
439    /**
440     * ロールバック処理
441     * インストール失敗時などに不要な一時ファイルを削除します.
442     *
443     * @param string $temp_dir インストール・アップデート時の一時展開用ディレクトリのパス.
444     * @param string $plugin_id プラグインID.
445     * @param string $plugin_html_dir_path プラグイン毎に生成されるhtmlディレクトリのパス.
446     */
447    function rollBack($temp_dir, $plugin_id = '', $plugin_html_dir_path ='')
448    {
449        // 一時ディレクトリを削除.
450        SC_Helper_FileManager_Ex::deleteFile($temp_dir, false);
451        // DBからプラグイン情報を削除
452        if (empty($plugin_id) === false) {
453            SC_Plugin_Util_Ex::deletePluginByPluginId($plugin_id);
454        }
455        // htmlディレクトリを削除
456        if (empty($plugin_html_dir_path) === false) {
457            SC_Helper_FileManager_Ex::deleteFile($plugin_html_dir_path, true);
458        }
459    }
460
461    /**
462     * プラグイン情報を取得します.
463     *
464     * @param ReflectionClass $objReflection
465     * @return array プラグイン情報の配列
466     */
467    function getPluginInfo(ReflectionClass $objReflection)
468    {
469        $arrStaticProps = $objReflection->getStaticProperties();
470        $arrConstants   = $objReflection->getConstants();
471
472        $arrPluginInfoKey = array(
473            'PLUGIN_CODE',
474            'PLUGIN_NAME',
475            'CLASS_NAME',
476            'PLUGIN_VERSION',
477            'COMPLIANT_VERSION',
478            'AUTHOR',
479            'DESCRIPTION',
480            'PLUGIN_SITE_URL',
481            'AUTHOR_SITE_URL',
482            'HOOK_POINTS',
483        );
484        $arrPluginInfo = array();
485        foreach ($arrPluginInfoKey as $key) {
486            // クラス変数での定義を優先
487            if (isset($arrStaticProps[$key])) {
488                $arrPluginInfo[$key] = $arrStaticProps[$key];
489            // クラス変数定義がなければ, クラス定数での定義を読み込み.
490            } elseif ($arrConstants[$key]) {
491                $arrPluginInfo[$key] = $arrConstants[$key];
492            } else {
493                $arrPluginInfo[$key] = null;
494            }
495        }
496        return $arrPluginInfo;
497    }
498
499    /**
500     * プラグインクラス内の定数をチェックします.
501     *
502     * @param ReflectionClass $objReflection リフレクションオブジェクト
503     * @param string $dir_path チェックするプラグインディレクトリ
504     * @return array エラー情報を格納した連想配列.
505     */
506    function checkPluginConstants(ReflectionClass $objReflection, $dir_path)
507    {
508        $arrErr = array();
509        // プラグイン情報を取得
510        $arrPluginInfo = $this->getPluginInfo($objReflection);
511
512        if (!isset($arrPluginInfo['PLUGIN_CODE'])) {
513            $arrErr['plugin_file'] = '※ PLUGIN_CODEが定義されていません。<br/>';
514            return $arrErr;
515        }
516        if (!isset($arrPluginInfo['PLUGIN_NAME'])) {
517            $arrErr['plugin_file'] = '※ PLUGIN_NAMEが定義されていません。<br/>';
518            return $arrErr;
519        }
520        if (!isset($arrPluginInfo['CLASS_NAME'])) {
521            $arrErr['plugin_file'] = '※ CLASS_NAMEが定義されていません。<br/>';
522            return $arrErr;
523        }
524        $plugin_class_file_path = $dir_path . $arrPluginInfo['CLASS_NAME'] . '.php';
525        if (file_exists($plugin_class_file_path) === false) {
526            $arrErr['plugin_file'] = '※ CLASS_NAMEが正しく定義されていません。<br/>';
527            return $arrErr;
528        }
529        if (!isset($arrPluginInfo['PLUGIN_VERSION'])) {
530            $arrErr['plugin_file'] = '※ PLUGIN_VERSIONが定義されていません。<br/>';
531            return $arrErr;
532        }
533        if (!isset($arrPluginInfo['COMPLIANT_VERSION'])) {
534            $arrErr['plugin_file'] = '※ COMPLIANT_VERSIONが定義されていません。<br/>';
535            return $arrErr;
536        }
537        if (!isset($arrPluginInfo['AUTHOR'])) {
538            $arrErr['plugin_file'] = '※ AUTHORが定義されていません。<br/>';
539            return $arrErr;
540        }
541        if (!isset($arrPluginInfo['DESCRIPTION'])) {
542            $arrErr['plugin_file'] = '※ DESCRIPTIONが定義されていません。<br/>';
543            return $arrErr;
544        }
545        $objErr = new SC_CheckError_Ex($arrPluginInfo);
546        $objErr->doFunc(array('PLUGIN_CODE', 'PLUGIN_CODE', STEXT_LEN), array('MAX_LENGTH_CHECK','GRAPH_CHECK'));
547        $objErr->doFunc(array('PLUGIN_NAME', 'PLUGIN_NAME', STEXT_LEN), array('MAX_LENGTH_CHECK'));
548        $objErr->doFunc(array('CLASS_NAME', 'CLASS_NAME', STEXT_LEN), array('MAX_LENGTH_CHECK','GRAPH_CHECK'));
549        $objErr->doFunc(array('PLUGIN_VERSION', 'PLUGIN_VERSION', STEXT_LEN), array('MAX_LENGTH_CHECK'));
550        $objErr->doFunc(array('COMPLIANT_VERSION', 'COMPLIANT_VERSION', STEXT_LEN), array('MAX_LENGTH_CHECK'));
551        $objErr->doFunc(array('AUTHOR', 'AUTHOR', STEXT_LEN), array('MAX_LENGTH_CHECK'));
552        $objErr->doFunc(array('DESCRIPTION', 'DESCRIPTION', MTEXT_LEN), array('MAX_LENGTH_CHECK'));
553        if (isset($arrPluginInfo['PLUGIN_SITE_URL'])) {
554            $objErr->doFunc(array('PLUGIN_SITE_URL', 'PLUGIN_SITE_URL', URL_LEN), array('MAX_LENGTH_CHECK','GRAPH_CHECK'));
555        }
556        if (isset($arrPluginInfo['AUTHOR_SITE_URL'])) {
557            $objErr->doFunc(array('AUTHOR_SITE_URL', 'AUTHOR_SITE_URL', URL_LEN), array('MAX_LENGTH_CHECK','GRAPH_CHECK'));
558        }
559        // エラー内容を出力用の配列にセットします.
560        if ($this->isError($objErr->arrErr)) {
561            $arrErr['plugin_file'] = '';
562            foreach ($objErr->arrErr as $error) {
563                    $arrErr['plugin_file'] .= $error;
564            }
565        }
566        return $arrErr;
567    }
568
569    /**
570     * プラグインをアップデートします.
571     *
572     * @param array $target_plugin アップデートするプラグイン情報の配列.
573     * @param string $upload_file_name アップロードファイル名.
574     * @return array エラー情報を格納した連想配列.
575     */
576    function updatePlugin($target_plugin, $upload_file_name)
577    {
578        // アップデート前に不要なファイルを消しておきます.
579        SC_Helper_FileManager_Ex::deleteFile(DOWNLOADS_TEMP_PLUGIN_UPDATE_DIR, false);
580
581        $arrErr = array();
582
583        // ファイルをチェックし展開します.
584        $arrErr = $this->unpackPluginFile($upload_file_name, DOWNLOADS_TEMP_PLUGIN_UPDATE_DIR, $target_plugin['plugin_code']);
585        if ($this->isError($arrErr) === true) {
586            return $arrErr;
587        }
588        // plugin_infoを読み込み.
589        $arrErr = $this->requirePluginFile(DOWNLOADS_TEMP_PLUGIN_UPDATE_DIR . 'plugin_info.php', $target_plugin['plugin_code']);
590        if ($this->isError($arrErr) === true) {
591            $this->rollBack(DOWNLOADS_TEMP_PLUGIN_INSTALL_DIR);
592            return $arrErr;
593        }
594        // リフレクションオブジェクトを生成.
595        $objReflection = new ReflectionClass('plugin_info');
596        $arrPluginInfo = $this->getPluginInfo($objReflection);
597        if ($arrPluginInfo['PLUGIN_CODE'] != $target_plugin['plugin_code']) {
598            $arrErr[$target_plugin['plugin_code']] = '※ プラグインコードが一致しません。<br/>';
599            return $arrErr;
600        }
601
602        // plugin_update.phpを読み込み.
603        $arrErr = $this->requirePluginFile(DOWNLOADS_TEMP_PLUGIN_UPDATE_DIR . 'plugin_update.php', $target_plugin['plugin_code']);
604        if ($this->isError($arrErr) === true) {
605            $this->rollBack(DOWNLOADS_TEMP_PLUGIN_UPDATE_DIR);
606            return $arrErr;
607        }
608        // プラグインクラスファイルのUPDATE処理を実行.
609        $arrErr = $this->execPlugin($target_plugin, 'plugin_update', 'update');
610
611        // 保存ディレクトリの削除.
612        SC_Helper_FileManager_Ex::deleteFile(DOWNLOADS_TEMP_PLUGIN_UPDATE_DIR, false);
613
614        return $arrErr;
615    }
616
617    /**
618     * ファイルをアップロードし、解凍先のディレクトリに解凍します.
619     *
620     * @param string $unpack_file_name 解凍ファイル名
621     * @param string $unpack_dir_path 解凍先ディレクトリパス
622     * @param string $file_key ファイルキー
623     * @return array エラー情報を格納した連想配列.
624     */
625    function unpackPluginFile($unpack_file_name, $unpack_dir_path, $file_key)
626    {
627        $arrErr = array();
628        // 解凍ディレクトリディレクトリを作成し、一時ディレクトリからファイルを移動
629        $objUpFile = new SC_UploadFile_Ex(PLUGIN_TEMP_REALDIR, $unpack_dir_path);
630        $this->initUploadFile($objUpFile, $file_key);
631        $arrErr = $objUpFile->makeTempFile($file_key, false);
632        if ($this->isError($arrErr) === true) {
633            return $arrErr;
634        }
635
636        // 正常にアップロードされているかをチェック.
637        $arrErr = $objUpFile->checkExists($file_key);
638        if ($this->isError($arrErr) === true) {
639            return $arrErr;
640        }
641        $objUpFile->moveTempFile();
642        // 解凍
643        $unpack_file_path = $unpack_dir_path . $unpack_file_name;
644        if (!$this->unpackPluginArchive($unpack_file_path)) {
645            $arrErr['plugin_file'] = '※ 解凍に失敗しました。<br/>';
646            return $arrErr;
647        }
648        return $arrErr;
649    }
650
651    /**
652     * プラグインをアンインストールします.
653     *
654     * @param array $plugin プラグイン情報を確認した連想配列.
655     * @return array エラー情報を格納した連想配列.
656     */
657    function uninstallPlugin($plugin)
658    {
659        $arrErr = array();
660        // プラグインファイルを読み込みます.
661        $plugin_class_file_path = $this->getPluginFilePath($plugin['plugin_code'], $plugin['class_name']);
662        $arrErr = $this->requirePluginFile($plugin_class_file_path, 'plugin_error');
663        if ($this->isError($arrErr) === true) {
664            return $arrErr;
665        }
666
667        // プラグインが有効な場合に無効化処理を実行
668        if ($plugin['enable'] == PLUGIN_ENABLE_TRUE){
669            // 無効化処理を実行します.
670            $arrErr = $this->execPlugin($plugin, $plugin['class_name'], 'disable');
671            if ($this->isError($arrErr) === true) {
672                return $arrErr;
673            }
674            // プラグインを無効にします.
675            $this->updatePluginEnable($plugin['plugin_id'], PLUGIN_ENABLE_FALSE);
676        }
677
678        // アンインストール処理を実行します.
679        $arrErr = $this->execPlugin($plugin, $plugin['class_name'], 'uninstall');
680        // プラグインの削除処理.
681        $arrErr = $this->deletePlugin($plugin['plugin_id'], $plugin['plugin_code']);
682
683        return $arrErr;
684    }
685
686    /**
687     * プラグインを有効にします.
688     *
689     * @param array $plugin プラグイン情報を確認した連想配列.
690     * @return array $arrErr エラー情報を格納した連想配列.
691     */
692    function enablePlugin($plugin)
693    {
694        $arrErr = array();
695        // クラスファイルを読み込み.
696        $plugin_class_file_path = $this->getPluginFilePath($plugin['plugin_code'], $plugin['class_name']);
697        $arrErr = $this->requirePluginFile($plugin_class_file_path, 'plugin_error');
698        if ($this->isError($arrErr) === true) {
699            return $arrErr;
700        }
701        // 有効化処理を実行します.
702        $arrErr = $this->execPlugin($plugin, $plugin['class_name'], 'enable');
703        if ($this->isError($arrErr) === true) {
704            return $arrErr;
705        }
706        // プラグインを有効にします.
707        $this->updatePluginEnable($plugin['plugin_id'], PLUGIN_ENABLE_TRUE);
708
709        return $arrErr;
710    }
711
712    /**
713     * プラグインを無効にします.
714     *
715     * @param array $plugin プラグイン情報を確認した連想配列.
716     * @return array $arrErr エラー情報を格納した連想配列.
717     */
718    function disablePlugin($plugin)
719    {
720        $arrErr = array();
721        // クラスファイルを読み込み.
722        $plugin_class_file_path =$this->getPluginFilePath($plugin['plugin_code'], $plugin['class_name']);
723        $arrErr = $this->requirePluginFile($plugin_class_file_path, 'plugin_error');
724        if ($this->isError($arrErr) === true) {
725            return $arrErr;
726        }
727
728        // 無効化処理を実行します.
729        $arrErr = $this->execPlugin($plugin, $plugin['class_name'], 'disable');
730        if ($this->isError($arrErr) === true) {
731            return $arrErr;
732        }
733        // プラグインを無効にします.
734        $this->updatePluginEnable($plugin['plugin_id'], PLUGIN_ENABLE_FALSE);
735
736        return $arrErr;
737    }
738
739    /**
740     * 優先度を更新します.
741     *
742     * @param int $plugin_id プラグインID
743     * @param int $priority 優先度
744     * @return integer 更新件数
745     */
746    function updatePriority($plugin_id, $priority)
747    {
748        $objQuery =& SC_Query_Ex::getSingletonInstance();
749        // UPDATEする値を作成する。
750        $sqlval['priority'] = $priority;
751        $sqlval['update_date'] = 'CURRENT_TIMESTAMP';
752        $where = 'plugin_id = ?';
753        // UPDATEの実行
754        $ret = $objQuery->update('dtb_plugin', $sqlval, $where, array($plugin_id));
755        return $ret;
756    }
757
758    /**
759     * プラグイン情報をDB登録.
760     *
761     * @param array $arrPluginInfo プラグイン情報を格納した連想配列.
762     * @return array エラー情報を格納した連想配列.
763     */
764    function registerData($arrPluginInfo)
765    {
766
767        // プラグイン情報をDB登録.
768        $objQuery =& SC_Query_Ex::getSingletonInstance();
769        $arr_sqlval_plugin = array();
770        $plugin_id = $objQuery->nextVal('dtb_plugin_plugin_id');
771        $arr_sqlval_plugin['plugin_id'] = $plugin_id;
772        $arr_sqlval_plugin['plugin_name'] = $arrPluginInfo['PLUGIN_NAME'];
773        $arr_sqlval_plugin['plugin_code'] = $arrPluginInfo['PLUGIN_CODE'];
774        $arr_sqlval_plugin['class_name'] = $arrPluginInfo['CLASS_NAME'];
775        $arr_sqlval_plugin['author'] = $arrPluginInfo['AUTHOR'];
776        // AUTHOR_SITE_URLが定義されているか判定.
777        $author_site_url = $arrPluginInfo['AUTHOR_SITE_URL'];
778        if ($author_site_url !== null) {
779            $arr_sqlval_plugin['author_site_url'] = $arrPluginInfo['AUTHOR_SITE_URL'];
780        }
781        // PLUGIN_SITE_URLが定義されているか判定.
782        $plugin_site_url = $arrPluginInfo['PLUGIN_SITE_URL'];
783        if ($plugin_site_url !== null) {
784            $arr_sqlval_plugin['plugin_site_url'] = $plugin_site_url;
785        }
786        $arr_sqlval_plugin['plugin_version'] = $arrPluginInfo['PLUGIN_VERSION'];
787        $arr_sqlval_plugin['compliant_version'] = $arrPluginInfo['COMPLIANT_VERSION'];
788        $arr_sqlval_plugin['plugin_description'] = $arrPluginInfo['DESCRIPTION'];
789        $arr_sqlval_plugin['priority'] = 0;
790        $arr_sqlval_plugin['enable'] = PLUGIN_ENABLE_FALSE;
791        $arr_sqlval_plugin['update_date'] = 'CURRENT_TIMESTAMP';
792        $objQuery->insert('dtb_plugin', $arr_sqlval_plugin);
793
794        // フックポイントをDB登録.
795        $hook_point = $arrPluginInfo['HOOK_POINTS'];
796        if ($hook_point !== null) {
797            /**
798             * FIXME コードが重複しているため、要修正
799             */
800            // フックポイントが配列で定義されている場合
801            if (is_array($hook_point)) {
802                foreach ($hook_point as $h) {
803                    $arr_sqlval_plugin_hookpoint = array();
804                    $id = $objQuery->nextVal('dtb_plugin_hookpoint_plugin_hookpoint_id');
805                    $arr_sqlval_plugin_hookpoint['plugin_hookpoint_id'] = $id;
806                    $arr_sqlval_plugin_hookpoint['plugin_id'] = $plugin_id;
807                    $arr_sqlval_plugin_hookpoint['hook_point'] = $h[0];
808                    $arr_sqlval_plugin_hookpoint['callback'] = $h[1];
809                    $arr_sqlval_plugin_hookpoint['update_date'] = 'CURRENT_TIMESTAMP';
810                    $objQuery->insert('dtb_plugin_hookpoint', $arr_sqlval_plugin_hookpoint);
811                }
812            // 文字列定義の場合
813            } else {
814                $array_hook_point = explode(',', $hook_point);
815                foreach ($array_hook_point as $h) {
816                    $arr_sqlval_plugin_hookpoint = array();
817                    $id = $objQuery->nextVal('dtb_plugin_hookpoint_plugin_hookpoint_id');
818                    $arr_sqlval_plugin_hookpoint['plugin_hookpoint_id'] = $id;
819                    $arr_sqlval_plugin_hookpoint['plugin_id'] = $plugin_id;
820                    $arr_sqlval_plugin_hookpoint['hook_point'] = $h;
821                    $arr_sqlval_plugin_hookpoint['update_date'] = 'CURRENT_TIMESTAMP';
822                    $objQuery->insert('dtb_plugin_hookpoint', $arr_sqlval_plugin_hookpoint);
823                }
824            }
825        }
826    }
827
828    /**
829     * ファイルを読み込む.
830     *
831     * @param string $file_path クラスのpath
832     * @param string $key エラー情報のキー.
833     * @return array $arrErr エラー情報を格納した連想配列.
834     */
835    function requirePluginFile($file_path, $key)
836    {
837        $arrErr = array();
838        if (file_exists($file_path)) {
839            require_once $file_path;
840        } else {
841            $arrErr[$key] = '※ ' . $file_path .'の読み込みに失敗しました。<br/>';
842        }
843        return $arrErr;
844    }
845
846    /**
847     * インスタンスを生成し、指定のメソッドを実行する.
848     *
849     * @param object $obj インスタンス
850     * @param string $class_name クラス名
851     * @param string $exec_func 実行するメソッド名.
852     * @return array $arrErr エラー情報を格納した連想配列.
853     *
854     */
855    function execPlugin($obj, $class_name, $exec_func)
856    {
857        $objPluginInstaller = new SC_Plugin_Installer($obj);
858
859        $arrErr = array();
860        if (method_exists($class_name, $exec_func) === true) {
861            $ret = call_user_func_array(
862                    array($class_name, $exec_func),
863                    array($obj, $objPluginInstaller));
864            if (!(is_null($ret) || $ret === true)) {
865                $arrErr[$obj['plugin_code']] = $ret;
866            }
867        } else {
868            $arrErr['plugin_error'] = '※ ' . $class_name . '.php に' . $exec_func . 'が見つかりません。<br/>';
869        }
870
871        $objPluginInstaller->execInstall();
872        return $arrErr;
873    }
874
875    /**
876     * プラグインアーカイブを解凍する.
877     *
878     * @param string $path アーカイブパス
879     * @return boolean Archive_Tar::extractModify()のエラー
880     */
881    function unpackPluginArchive($path)
882    {
883        // 圧縮フラグTRUEはgzip解凍をおこなう
884        $tar = new Archive_Tar($path, true);
885
886        $dir = dirname($path);
887        $file_name = basename($path);
888
889        // 拡張子を切り取る
890        $unpacking_name = preg_replace("/(\.tar|\.tar\.gz)$/", '', $file_name);
891
892        // 指定されたフォルダ内に解凍する
893        $result = $tar->extractModify($dir. '/', $unpacking_name);
894        GC_Utils_Ex::gfPrintLog('解凍:' . $dir.'/'.$file_name.'->'.$dir.'/'.$unpacking_name);
895        // 解凍元のファイルを削除する.
896        unlink($path);
897
898        return $result;
899    }
900
901    /**
902     * plugin_idをキーにdtb_pluginのstatusを更新します.
903     *
904     * @param int $plugin_id プラグインID
905     * @param int $enable_flg 有効フラグ
906     * @return integer 更新件数
907     */
908    function updatePluginEnable($plugin_id, $enable_flg)
909    {
910        $objQuery =& SC_Query_Ex::getSingletonInstance();
911        // UPDATEする値を作成する。
912        $sqlval['enable'] = $enable_flg;
913        $sqlval['update_date'] = 'CURRENT_TIMESTAMP';
914        $where = 'plugin_id = ?';
915        // UPDATEの実行
916        $ret = $objQuery->update('dtb_plugin', $sqlval, $where, array($plugin_id));
917        return $ret;
918    }
919
920    /**
921     * plugin_idをキーにdtb_plugin, dtb_plugin_hookpointから物理削除します.
922     *
923     * @param int $plugin_id プラグインID.
924     * @param string $plugin_code プラグインコード.
925     * @return array $arrErr エラー情報を格納した連想配列.
926     */
927    function deletePlugin($plugin_id, $plugin_code)
928    {
929        $arrErr = array();
930        $objQuery =& SC_Query_Ex::getSingletonInstance();
931        $objQuery->begin();
932
933        SC_Plugin_Util_Ex::deletePluginByPluginId($plugin_id);
934
935        if (SC_Helper_FileManager_Ex::deleteFile($this->getPluginDir($plugin_code)) === false) {
936            // TODO エラー処理
937        }
938
939        if (SC_Helper_FileManager_Ex::deleteFile($this->getHtmlPluginDir($plugin_code)) === false) {
940            // TODO エラー処理
941        }
942
943        $objQuery->commit();
944
945        return $arrErr;
946    }
947
948    /**
949     * ファイルがあるかを判定します.
950     *
951     * @param string $plugin_dir 対象ディレクトリ.
952     * @param string $file_name ファイル名.
953     * @return boolean
954     */
955    function isContainsFile($plugin_dir, $file_name)
956    {
957        if (file_exists($plugin_dir) && is_dir($plugin_dir)) {
958            if ($handle = opendir($plugin_dir)) {
959                while (($item = readdir($handle)) !== false) {
960                    if ($item === $file_name) return true;
961                }
962            }
963            closedir($handle);
964        }
965        return false;
966    }
967
968    /**
969     * アーカイブ内に指定のファイルが存在するかを判定します.
970     *
971     * @param Archive_Tar $tar_obj
972     * @param string $file_path 判定するファイルパス
973     * @return boolean
974     */
975    function checkContainsFile($tar_obj, $file_path)
976    {
977        // ファイル一覧を取得
978        $arrayFile = $tar_obj->listContent();
979        foreach ($arrayFile as  $value) {
980            if ($value['filename'] === $file_path) return true;
981        }
982        return false;
983    }
984
985    /**
986     * ディレクトリを作成します.
987     *
988     * @param string $dir_path 作成するディレクトリのパス
989     * @return void
990     */
991    function makeDir($dir_path)
992    {
993        // ディレクトリ作成
994        if (!file_exists($dir_path)) {
995            mkdir($dir_path);
996        }
997    }
998
999    /**
1000     * エラー情報が格納されているか判定します.
1001     *
1002     * @param array $arrErr エラー情報を格納した連想配列.
1003     * @return boolean.
1004     */
1005    function isError($error)
1006    {
1007        if (is_array($error) && count($error) > 0) {
1008            return true;
1009        }
1010        return false;
1011    }
1012
1013}
Note: See TracBrowser for help on using the repository browser.