123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339 |
- <?php
- // +----------------------------------------------------------------------
- // | ThinkPHP [ WE CAN DO IT JUST THINK IT ]
- // +----------------------------------------------------------------------
- // | Copyright (c) 2006-2014 http://thinkphp.cn All rights reserved.
- // +----------------------------------------------------------------------
- // | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
- // +----------------------------------------------------------------------
- // | Author: liu21st <liu21st@gmail.com>
- // +----------------------------------------------------------------------
- namespace Think;
- /**
- * ThinkPHP内置的Dispatcher类
- * 完成URL解析、路由和调度
- */
- class Dispatcher {
- /**
- * URL映射到控制器
- * @access public
- * @return void
- */
- static public function dispatch() {
- $varPath = C('VAR_PATHINFO');
- $varAddon = C('VAR_ADDON');
- $varModule = C('VAR_MODULE');
- $varController = C('VAR_CONTROLLER');
- $varAction = C('VAR_ACTION');
- $urlCase = C('URL_CASE_INSENSITIVE');
- if(isset($_GET[$varPath])) { // 判断URL里面是否有兼容模式参数
- $_SERVER['PATH_INFO'] = $_GET[$varPath];
- unset($_GET[$varPath]);
- }elseif(IS_CLI){ // CLI模式下 index.php module/controller/action/params/...
- $_SERVER['PATH_INFO'] = isset($_SERVER['argv'][1]) ? $_SERVER['argv'][1] : '';
- }
- // 开启子域名部署
- if(C('APP_SUB_DOMAIN_DEPLOY')) {
- $rules = C('APP_SUB_DOMAIN_RULES');
- if(isset($rules[$_SERVER['HTTP_HOST']])) { // 完整域名或者IP配置
- define('APP_DOMAIN',$_SERVER['HTTP_HOST']); // 当前完整域名
- $rule = $rules[APP_DOMAIN];
- }else{
- if(strpos(C('APP_DOMAIN_SUFFIX'),'.')){ // com.cn net.cn
- $domain = array_slice(explode('.', $_SERVER['HTTP_HOST']), 0, -3);
- }else{
- $domain = array_slice(explode('.', $_SERVER['HTTP_HOST']), 0, -2);
- }
- if(!empty($domain)) {
- $subDomain = implode('.', $domain);
- define('SUB_DOMAIN',$subDomain); // 当前完整子域名
- $domain2 = array_pop($domain); // 二级域名
- if($domain) { // 存在三级域名
- $domain3 = array_pop($domain);
- }
- if(isset($rules[$subDomain])) { // 子域名
- $rule = $rules[$subDomain];
- }elseif(isset($rules['*.' . $domain2]) && !empty($domain3)){ // 泛三级域名
- $rule = $rules['*.' . $domain2];
- $panDomain = $domain3;
- }elseif(isset($rules['*']) && !empty($domain2) && 'www' != $domain2 ){ // 泛二级域名
- $rule = $rules['*'];
- $panDomain = $domain2;
- }
- }
- }
- if(!empty($rule)) {
- // 子域名部署规则 '子域名'=>array('模块名[/控制器名]','var1=a&var2=b');
- if(is_array($rule)){
- list($rule,$vars) = $rule;
- }
- $array = explode('/',$rule);
- // 模块绑定
- define('BIND_MODULE',array_shift($array));
- // 控制器绑定
- if(!empty($array)) {
- $controller = array_shift($array);
- if($controller){
- define('BIND_CONTROLLER',$controller);
- }
- }
- if(isset($vars)) { // 传入参数
- parse_str($vars,$parms);
- if(isset($panDomain)){
- $pos = array_search('*', $parms);
- if(false !== $pos) {
- // 泛域名作为参数
- $parms[$pos] = $panDomain;
- }
- }
- $_GET = array_merge($_GET,$parms);
- }
- }
- }
- // 分析PATHINFO信息
- if(!isset($_SERVER['PATH_INFO'])) {
- $types = explode(',',C('URL_PATHINFO_FETCH'));
- foreach ($types as $type){
- if(0===strpos($type,':')) {// 支持函数判断
- $_SERVER['PATH_INFO'] = call_user_func(substr($type,1));
- break;
- }elseif(!empty($_SERVER[$type])) {
- $_SERVER['PATH_INFO'] = (0 === strpos($_SERVER[$type],$_SERVER['SCRIPT_NAME']))?
- substr($_SERVER[$type], strlen($_SERVER['SCRIPT_NAME'])) : $_SERVER[$type];
- break;
- }
- }
- }
- $depr = C('URL_PATHINFO_DEPR');
- define('MODULE_PATHINFO_DEPR', $depr);
- if(empty($_SERVER['PATH_INFO'])) {
- $_SERVER['PATH_INFO'] = '';
- define('__INFO__','');
- define('__EXT__','');
- }else{
- define('__INFO__',trim($_SERVER['PATH_INFO'],'/'));
- // URL后缀
- define('__EXT__', strtolower(pathinfo($_SERVER['PATH_INFO'],PATHINFO_EXTENSION)));
- $_SERVER['PATH_INFO'] = __INFO__;
- if(!defined('BIND_MODULE') && (!C('URL_ROUTER_ON') || !Route::check())){
- if (__INFO__ && C('MULTI_MODULE')){ // 获取模块名
- $paths = explode($depr,__INFO__,2);
- $allowList = C('MODULE_ALLOW_LIST'); // 允许的模块列表
- $module = preg_replace('/\.' . __EXT__ . '$/i', '',$paths[0]);
- if( empty($allowList) || (is_array($allowList) && in_array_case($module, $allowList))){
- $_GET[$varModule] = $module;
- $_SERVER['PATH_INFO'] = isset($paths[1])?$paths[1]:'';
- }
- }
- }
- }
- // URL常量
- define('__SELF__',strip_tags($_SERVER[C('URL_REQUEST_URI')]));
- // 获取模块名称
- define('MODULE_NAME', defined('BIND_MODULE')? BIND_MODULE : self::getModule($varModule));
-
- // 检测模块是否存在
- if( MODULE_NAME && (defined('BIND_MODULE') || !in_array_case(MODULE_NAME,C('MODULE_DENY_LIST')) ) && is_dir(APP_PATH.MODULE_NAME)){
- // 定义当前模块路径
- define('MODULE_PATH', APP_PATH.MODULE_NAME.'/');
- // 定义当前模块的模版缓存路径
- C('CACHE_PATH',CACHE_PATH.MODULE_NAME.'/');
- // 定义当前模块的日志目录
- C('LOG_PATH', realpath(LOG_PATH).'/'.MODULE_NAME.'/');
- // 模块检测
- Hook::listen('module_check');
- // 加载模块配置文件
- if(is_file(MODULE_PATH.'Conf/config'.CONF_EXT))
- C(load_config(MODULE_PATH.'Conf/config'.CONF_EXT));
- // 加载应用模式对应的配置文件
- if('common' != APP_MODE && is_file(MODULE_PATH.'Conf/config_'.APP_MODE.CONF_EXT))
- C(load_config(MODULE_PATH.'Conf/config_'.APP_MODE.CONF_EXT));
- // 当前应用状态对应的配置文件
- if(APP_STATUS && is_file(MODULE_PATH.'Conf/'.APP_STATUS.CONF_EXT))
- C(load_config(MODULE_PATH.'Conf/'.APP_STATUS.CONF_EXT));
- // 加载模块别名定义
- if(is_file(MODULE_PATH.'Conf/alias.php'))
- Think::addMap(include MODULE_PATH.'Conf/alias.php');
- // 加载模块tags文件定义
- if(is_file(MODULE_PATH.'Conf/tags.php'))
- Hook::import(include MODULE_PATH.'Conf/tags.php');
- // 加载模块函数文件
- if(is_file(MODULE_PATH.'Common/function.php'))
- include MODULE_PATH.'Common/function.php';
-
- $urlCase = C('URL_CASE_INSENSITIVE');
- // 加载模块的扩展配置文件
- load_ext_file(MODULE_PATH);
- }else{
- E(L('_MODULE_NOT_EXIST_').':'.MODULE_NAME);
- }
- if(!defined('__APP__')){
- $urlMode = C('URL_MODEL');
- if($urlMode == URL_COMPAT ){// 兼容模式判断
- define('PHP_FILE',_PHP_FILE_.'?'.$varPath.'=');
- }elseif($urlMode == URL_REWRITE ) {
- $url = dirname(_PHP_FILE_);
- if($url == '/' || $url == '\\')
- $url = '';
- define('PHP_FILE',$url);
- }else {
- define('PHP_FILE',_PHP_FILE_);
- }
- // 当前应用地址
- define('__APP__',strip_tags(PHP_FILE));
- }
- // 模块URL地址
- $moduleName = defined('MODULE_ALIAS')? MODULE_ALIAS : MODULE_NAME;
- define('__MODULE__',(defined('BIND_MODULE') || !C('MULTI_MODULE'))? __APP__ : __APP__.'/'.($urlCase ? strtolower($moduleName) : $moduleName));
- if('' != $_SERVER['PATH_INFO'] && (!C('URL_ROUTER_ON') || !Route::check()) ){ // 检测路由规则 如果没有则按默认规则调度URL
- Hook::listen('path_info');
- // 检查禁止访问的URL后缀
- if(C('URL_DENY_SUFFIX') && preg_match('/\.('.trim(C('URL_DENY_SUFFIX'),'.').')$/i', $_SERVER['PATH_INFO'])){
- send_http_status(404);
- exit;
- }
-
- // 去除URL后缀
- $_SERVER['PATH_INFO'] = preg_replace(C('URL_HTML_SUFFIX')? '/\.('.trim(C('URL_HTML_SUFFIX'),'.').')$/i' : '/\.'.__EXT__.'$/i', '', $_SERVER['PATH_INFO']);
- $depr = C('URL_PATHINFO_DEPR');
- $paths = explode($depr,trim($_SERVER['PATH_INFO'],$depr));
- if(!defined('BIND_CONTROLLER')) {// 获取控制器
- if(C('CONTROLLER_LEVEL')>1){// 控制器层次
- $_GET[$varController] = implode('/',array_slice($paths,0,C('CONTROLLER_LEVEL')));
- $paths = array_slice($paths, C('CONTROLLER_LEVEL'));
- }else{
- $_GET[$varController] = array_shift($paths);
- }
- }
- // 获取操作
- if(!defined('BIND_ACTION')){
- $_GET[$varAction] = array_shift($paths);
- }
- // 解析剩余的URL参数
- $var = array();
- if(C('URL_PARAMS_BIND') && 1 == C('URL_PARAMS_BIND_TYPE')){
- // URL参数按顺序绑定变量
- $var = $paths;
- }else{
- preg_replace_callback('/(\w+)\/([^\/]+)/', function($match) use(&$var){$var[$match[1]]=strip_tags($match[2]);}, implode('/',$paths));
- }
- $_GET = array_merge($var,$_GET);
- }
- // 获取控制器的命名空间(路径)
- define('CONTROLLER_PATH', self::getSpace($varAddon,$urlCase));
- // 获取控制器和操作名
- define('CONTROLLER_NAME', defined('BIND_CONTROLLER')? BIND_CONTROLLER : self::getController($varController,$urlCase));
- define('ACTION_NAME', defined('BIND_ACTION')? BIND_ACTION : self::getAction($varAction,$urlCase));
- // 当前控制器的UR地址
- $controllerName = defined('CONTROLLER_ALIAS')? CONTROLLER_ALIAS : CONTROLLER_NAME;
- define('__CONTROLLER__',__MODULE__.$depr.(defined('BIND_CONTROLLER')? '': ( $urlCase ? parse_name($controllerName) : $controllerName )) );
- // 当前操作的URL地址
- define('__ACTION__',__CONTROLLER__.$depr.(defined('ACTION_ALIAS')?ACTION_ALIAS:ACTION_NAME));
- //保证$_REQUEST正常取值
- $_REQUEST = array_merge($_POST,$_GET,$_COOKIE); // -- 加了$_COOKIE. 保证哦..
- }
- /**
- * 获得控制器的命名空间路径 便于插件机制访问
- */
- static private function getSpace($var,$urlCase) {
- $space = !empty($_GET[$var])?strip_tags($_GET[$var]):'';
- unset($_GET[$var]);
- return $space;
- }
- /**
- * 获得实际的控制器名称
- */
- static private function getController($var,$urlCase) {
- $controller = (!empty($_GET[$var])? $_GET[$var]:C('DEFAULT_CONTROLLER'));
- unset($_GET[$var]);
- if($maps = C('URL_CONTROLLER_MAP')) {
- if(isset($maps[strtolower($controller)])) {
- // 记录当前别名
- define('CONTROLLER_ALIAS',strtolower($controller));
- // 获取实际的控制器名
- return ucfirst($maps[CONTROLLER_ALIAS]);
- }elseif(array_search(strtolower($controller),$maps)){
- // 禁止访问原始控制器
- return '';
- }
- }
- if($urlCase) {
- // URL地址不区分大小写
- // 智能识别方式 user_type 识别到 UserTypeController 控制器
- $controller = parse_name($controller,1);
- }
- return strip_tags(ucfirst($controller));
- }
- /**
- * 获得实际的操作名称
- */
- static private function getAction($var,$urlCase) {
- $action = !empty($_POST[$var]) ?
- $_POST[$var] :
- (!empty($_GET[$var])?$_GET[$var]:C('DEFAULT_ACTION'));
- unset($_POST[$var],$_GET[$var]);
- if($maps = C('URL_ACTION_MAP')) {
- if(isset($maps[strtolower(CONTROLLER_NAME)])) {
- $maps = $maps[strtolower(CONTROLLER_NAME)];
- if(isset($maps[strtolower($action)])) {
- // 记录当前别名
- define('ACTION_ALIAS',strtolower($action));
- // 获取实际的操作名
- if(is_array($maps[ACTION_ALIAS])){
- parse_str($maps[ACTION_ALIAS][1],$vars);
- $_GET = array_merge($_GET,$vars);
- return $maps[ACTION_ALIAS][0];
- }else{
- return $maps[ACTION_ALIAS];
- }
-
- }elseif(array_search(strtolower($action),$maps)){
- // 禁止访问原始操作
- return '';
- }
- }
- }
- return strip_tags( $urlCase? strtolower($action) : $action );
- }
- /**
- * 获得实际的模块名称
- */
- static private function getModule($var) {
- $module = (!empty($_GET[$var])?$_GET[$var]:C('DEFAULT_MODULE'));
- unset($_GET[$var]);
- if($maps = C('URL_MODULE_MAP')) {
- if(isset($maps[strtolower($module)])) {
- // 记录当前别名
- define('MODULE_ALIAS',strtolower($module));
- // 获取实际的模块名
- return ucfirst($maps[MODULE_ALIAS]);
- }elseif(array_search(strtolower($module),$maps)){
- // 禁止访问原始模块
- return '';
- }
- }
- return strip_tags(ucfirst($module));
- }
- }
|