08月10, 2011

如何让ThinkPHP支持ActionChain

什么是ActionChain

在MVC框架中,一般都提供对自定义URL的支持。也就是将所有的请求都重定向到index.php,然后PHP程序通过URL配置规则来确定执行什么样的逻辑。

随着产品业务逻辑越来越复杂,页面模块化程度越来越高,一个URL要执行多个逻辑功能。通常的情况就是在对应的Action里分别调用不同模块的接口,然后输出到模版。但这种方式对代码的维护和管理非常不方便,不能很快速的定位到一个URL执行了哪些逻辑。

ActionChain就是用来解决这个问题的,它可以在一个URL里指定按顺序执行多个Action,每个Action可以是相互独立的。这样可以极大的方便了代码的阅读性和维护性。

什么是ThinkPHP框架

THinkPHP是一个国人开发的已经有多年历史的框架,已经有越来越多的中小型产品使用这个框架开发。并且我也用这个框架开发了多个应用。

这个框架提供了大量的非常快捷的访问接口,这也是我为什么喜欢这个框架的原因。并且相对于Zend, Yii, CI等框架,要简洁的多。

http://thinkphp.cn/ 可以通过这个URL了解更多的信息

ThinkPHP对ActionChain的支持

ThinkPHP虽然理论上是支持ActionChain模式的,但是设计上没有考虑安全问题,导致无法使用。如:

http://serverName/appName/User/action1:action2:action3/

那么会依次执行 UserAction 癿 action1 、action2 和 action3 方法,并且当前操作名称是最后一个操作。

由于ActionChain是支持在URL上指定的,导致访问者可以很轻松的绕过某些Action,这样就有严重的安全问题,比如:

action1是检测用户是否已经登录的功能,action2是拉取某个特定模块信息的功能,action3是...

如果用户将URL里的action1去掉,直接导致是否已经登录的检测去掉了。

如何让ThinkPHP完美的支持ActionChain

既然ThinkPHP原生提供的ActionChain有安全问题,那如果想使用ActionChain功能的化必须对ThinkPHP进行改造。

由于ThinkPHP支持自定义路由功能,所以改造非常简单,下面讲讲具体的改造过程:

支持的ActionChain在路由的配置里是如何表现的?

array(  //各个页面速度对比
    '/^page\/compare/',
    'index/initVar,analytic/pageCompare',
    '',
    'tpl=analytic_page_compare'
)

不同的action可以用逗号隔开,并且可以指定不同的Controller,还可以指定不同的Group。(但不推荐不同的Group之间的调用)

1、修改ThinkPHP/Lib/Think/Util/Dispatcher.class.php,将parseUrl方法修改如下:

static private function parseUrl($route) {
        if (strpos($route, ',') !== false){
            return array(C('VAR_ACTION_CHAIN') => $route);
        }
        $array   =  explode('/',$route);
        $var  =  array();
        $var[C('VAR_ACTION')] = array_pop($array);
        $var[C('VAR_MODULE')] = array_pop($array);
        if(!empty($array)) $var[C('VAR_GROUP')]  = array_pop($array);
        return $var;
    }

2、在Dispatcher.class.php文件添加如下的方法:

 static private function getActionChain($var){
        $actionChain   = !empty($_GET[$var]) ? $_GET[$var] : '';
        unset($_GET[$var]);
        return strtolower($actionChain);
    }
3、修改Dispatcher.class.php

在 define('ACTION_NAME',self::getAction(C('VAR_ACTION')));后面加上

define('ACTION_CHAIN', self::getActionChain(C('VAR_ACTION_CHAIN')));
4、修改ThinkPHP/Lib/Think/Core/App.class.php,修改run方法

static public function run() {
        App::init();
        // 记录应用初始化时间
        if(C('SHOW_RUN_TIME')) G('initTime');

    $view = Think::instance('View');
    if ($_GET['tpl']){
        $view->assign('tpl', $_GET['tpl']);
    }
    if (ACTION_CHAIN){
        $module = App::actionChainExec();
    }else{
        $module = App::exec();
    }
    $tpl = $view->get('tpl');
    if ($module && $tpl){
        $module->display($tpl);
    }
    // 保存日志记录
    if(C('LOG_RECORD')) Log::save();
    return true;
}</pre>

5、在App.class.php文件添加actionChainExec方法

static function actionChainExec(){
    $actionChain = ACTIONCHAIN;
    $actionList = explode(',', $actionChain);
    foreach ($actionList as $item){
        $items = explode('/', $item);
        if (count($items) < 2) continue;
        if (count($items) == 2){
            array_unshift($items, strtolower(GROUP_NAME));
        }
        $module = A(ucfirst($items[0]).C('APP_GROUP_DEPR'). ucfirst($items[1]));
        if ($module){
            $result = $module->$items2;
            if ($result === false){
                return $module;
            }
        }else{
            throw_exception(L('_MODULE_NOT_EXIST').$items[1]);
        }
    }
    return $module;
}
这样差不多就改完了,当然如果你按照这种方式修改比较麻烦,且容易出问题,这里提供一份下载的版本。

下载支持ActionChain的ThinkPHP

本文链接:http://welefen.com/post/how-to-make-thinkphp-support-actionchain.html

-- EOF --

Comments

评论加载中...

注:如果长时间无法加载,请针对 disq.us | disquscdn.com | disqus.com 启用代理。