在MVC框架中,一般都提供对自定义URL的支持。也就是将所有的请求都重定向到index.php,然后PHP程序通过URL配置规则来确定执行什么样的逻辑。
随着产品业务逻辑越来越复杂,页面模块化程度越来越高,一个URL要执行多个逻辑功能。通常的情况就是在对应的Action里分别调用不同模块的接口,然后输出到模版。但这种方式对代码的维护和管理非常不方便,不能很快速的定位到一个URL执行了哪些逻辑。
ActionChain就是用来解决这个问题的,它可以在一个URL里指定按顺序执行多个Action,每个Action可以是相互独立的。这样可以极大的方便了代码的阅读性和维护性。
THinkPHP是一个国人开发的已经有多年历史的框架,已经有越来越多的中小型产品使用这个框架开发。并且我也用这个框架开发了多个应用。
这个框架提供了大量的非常快捷的访问接口,这也是我为什么喜欢这个框架的原因。并且相对于Zend, Yii, CI等框架,要简洁的多。
http://thinkphp.cn/ 可以通过这个URL了解更多的信息
ThinkPHP虽然理论上是支持ActionChain模式的,但是设计上没有考虑安全问题,导致无法使用。如:
http://serverName/appName/User/action1:action2:action3/
那么会依次执行 UserAction 癿 action1 、action2 和 action3 方法,并且当前操作名称是最后一个操作。
由于ActionChain是支持在URL上指定的,导致访问者可以很轻松的绕过某些Action,这样就有严重的安全问题,比如:
action1是检测用户是否已经登录的功能,action2是拉取某个特定模块信息的功能,action3是...
如果用户将URL里的action1去掉,直接导致是否已经登录的检测去掉了。
既然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 = ACTION_CHAIN;
$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
Copyright© 2013-2020
All Rights Reserved 京ICP备2023019179号-8