PageRenderTime 38ms CodeModel.GetById 13ms RepoModel.GetById 0ms app.codeStats 0ms

/Lib/Core/Dispatcher.class.php

https://gitlab.com/colin.luo/shbs
PHP | 201 lines | 175 code | 2 blank | 24 comment | 8 complexity | 7fd24df0a2289cd01fcb232ebc13d7c8 MD5 | raw file
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | ThinkPHP [ WE CAN DO IT JUST THINK IT ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2006-2012 http://thinkphp.cn All rights reserved.
  6. // +----------------------------------------------------------------------
  7. // | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
  8. // +----------------------------------------------------------------------
  9. // | Author: liu21st <liu21st@gmail.com>
  10. // +----------------------------------------------------------------------
  11. /**
  12. * ThinkPHP内置的Dispatcher类
  13. * 完成URL解析、路由和调度
  14. * @category Think
  15. * @package Think
  16. * @subpackage Core
  17. * @author liu21st <liu21st@gmail.com>
  18. */
  19. class Dispatcher {
  20. /**
  21. * URL映射到控制器
  22. * @access public
  23. * @return void
  24. */
  25. static public function dispatch() {
  26. $urlMode = C('URL_MODEL');
  27. if(!empty($_GET[C('VAR_PATHINFO')])) { // 判断URL里面是否有兼容模式参数
  28. $_SERVER['PATH_INFO'] = $_GET[C('VAR_PATHINFO')];
  29. unset($_GET[C('VAR_PATHINFO')]);
  30. }
  31. if($urlMode == URL_COMPAT ){
  32. // 兼容模式判断
  33. define('PHP_FILE',_PHP_FILE_.'?'.C('VAR_PATHINFO').'=');
  34. }elseif($urlMode == URL_REWRITE ) {
  35. //当前项目地址
  36. $url = dirname(_PHP_FILE_);
  37. if($url == '/' || $url == '\\')
  38. $url = '';
  39. define('PHP_FILE',$url);
  40. }else {
  41. //当前项目地址
  42. define('PHP_FILE',_PHP_FILE_);
  43. }
  44. // 开启子域名部署
  45. if(C('APP_SUB_DOMAIN_DEPLOY')) {
  46. $rules = C('APP_SUB_DOMAIN_RULES');
  47. $subDomain = strtolower(substr($_SERVER['HTTP_HOST'],0,strpos($_SERVER['HTTP_HOST'],'.')));
  48. define('SUB_DOMAIN',$subDomain); // 二级域名定义
  49. if($subDomain && isset($rules[$subDomain])) {
  50. $rule = $rules[$subDomain];
  51. }elseif(isset($rules['*'])){ // 泛域名支持
  52. if('www' != $subDomain && !in_array($subDomain,C('APP_SUB_DOMAIN_DENY'))) {
  53. $rule = $rules['*'];
  54. }
  55. }
  56. if(!empty($rule)) {
  57. // 子域名部署规则 '子域名'=>array('分组名/[模块名]','var1=a&var2=b');
  58. $array = explode('/',$rule[0]);
  59. $module = array_pop($array);
  60. if(!empty($module)) {
  61. $_GET[C('VAR_MODULE')] = $module;
  62. $domainModule = true;
  63. }
  64. if(!empty($array)) {
  65. $_GET[C('VAR_GROUP')] = array_pop($array);
  66. $domainGroup = true;
  67. }
  68. if(isset($rule[1])) { // 传入参数
  69. parse_str($rule[1],$parms);
  70. $_GET = array_merge($_GET,$parms);
  71. }
  72. }
  73. }
  74. // 分析PATHINFO信息
  75. if(empty($_SERVER['PATH_INFO'])) {
  76. $types = explode(',',C('URL_PATHINFO_FETCH'));
  77. foreach ($types as $type){
  78. if(0===strpos($type,':')) {// 支持函数判断
  79. $_SERVER['PATH_INFO'] = call_user_func(substr($type,1));
  80. break;
  81. }elseif(!empty($_SERVER[$type])) {
  82. $_SERVER['PATH_INFO'] = (0 === strpos($_SERVER[$type],$_SERVER['SCRIPT_NAME']))?
  83. substr($_SERVER[$type], strlen($_SERVER['SCRIPT_NAME'])) : $_SERVER[$type];
  84. break;
  85. }
  86. }
  87. }
  88. $depr = C('URL_PATHINFO_DEPR');
  89. if(!empty($_SERVER['PATH_INFO'])) {
  90. tag('path_info');
  91. $part = pathinfo($_SERVER['PATH_INFO']);
  92. define('__EXT__', isset($part['extension'])?strtolower($part['extension']):'');
  93. if(C('URL_HTML_SUFFIX')) {
  94. $_SERVER['PATH_INFO'] = preg_replace('/\.('.trim(C('URL_HTML_SUFFIX'),'.').')$/i', '', $_SERVER['PATH_INFO']);
  95. }elseif(__EXT__) {
  96. $_SERVER['PATH_INFO'] = preg_replace('/.'.__EXT__.'$/i','',$_SERVER['PATH_INFO']);
  97. }
  98. if(!self::routerCheck()){ // 检测路由规则 如果没有则按默认规则调度URL
  99. $paths = explode($depr,trim($_SERVER['PATH_INFO'],'/'));
  100. if(C('VAR_URL_PARAMS')) {
  101. // 直接通过$_GET['_URL_'][1] $_GET['_URL_'][2] 获取URL参数 方便不用路由时参数获取
  102. $_GET[C('VAR_URL_PARAMS')] = $paths;
  103. }
  104. $var = array();
  105. if (C('APP_GROUP_LIST') && !isset($_GET[C('VAR_GROUP')])){
  106. $var[C('VAR_GROUP')] = in_array(strtolower($paths[0]),explode(',',strtolower(C('APP_GROUP_LIST'))))? array_shift($paths) : '';
  107. if(C('APP_GROUP_DENY') && in_array(strtolower($var[C('VAR_GROUP')]),explode(',',strtolower(C('APP_GROUP_DENY'))))) {
  108. // 禁止直接访问分组
  109. exit;
  110. }
  111. }
  112. if(!isset($_GET[C('VAR_MODULE')])) {// 还没有定义模块名称
  113. $var[C('VAR_MODULE')] = array_shift($paths);
  114. }
  115. $var[C('VAR_ACTION')] = array_shift($paths);
  116. // 解析剩余的URL参数
  117. preg_replace('@(\w+)\/([^\/]+)@e', '$var[\'\\1\']=strip_tags(\'\\2\');', implode('/',$paths));
  118. $_GET = array_merge($var,$_GET);
  119. }
  120. define('__INFO__',$_SERVER['PATH_INFO']);
  121. }
  122. // 获取分组 模块和操作名称
  123. if (C('APP_GROUP_LIST')) {
  124. define('GROUP_NAME', self::getGroup(C('VAR_GROUP')));
  125. }
  126. define('MODULE_NAME',self::getModule(C('VAR_MODULE')));
  127. define('ACTION_NAME',self::getAction(C('VAR_ACTION')));
  128. // URL常量
  129. define('__SELF__',strip_tags($_SERVER['REQUEST_URI']));
  130. // 当前项目地址
  131. define('__APP__',strip_tags(PHP_FILE));
  132. // 当前模块和分组地址
  133. if(defined('GROUP_NAME')) {
  134. define('__GROUP__',(!empty($domainGroup) || strtolower(GROUP_NAME) == strtolower(C('DEFAULT_GROUP')) )?__APP__ : __APP__.'/'.GROUP_NAME);
  135. define('__URL__',!empty($domainModule)?__GROUP__.$depr : __GROUP__.$depr.MODULE_NAME);
  136. }else{
  137. define('__URL__',!empty($domainModule)?__APP__.'/' : __APP__.'/'.MODULE_NAME);
  138. }
  139. // 当前操作地址
  140. define('__ACTION__',__URL__.$depr.ACTION_NAME);
  141. //保证$_REQUEST正常取值
  142. $_REQUEST = array_merge($_POST,$_GET);
  143. }
  144. /**
  145. * 路由检测
  146. * @access public
  147. * @return void
  148. */
  149. static public function routerCheck() {
  150. $return = false;
  151. // 路由检测标签
  152. tag('route_check',$return);
  153. return $return;
  154. }
  155. /**
  156. * 获得实际的模块名称
  157. * @access private
  158. * @return string
  159. */
  160. static private function getModule($var) {
  161. $module = (!empty($_GET[$var])? $_GET[$var]:C('DEFAULT_MODULE'));
  162. unset($_GET[$var]);
  163. if(C('URL_CASE_INSENSITIVE')) {
  164. // URL地址不区分大小写
  165. // 智能识别方式 index.php/user_type/index/ 识别到 UserTypeAction 模块
  166. $module = ucfirst(parse_name($module,1));
  167. }
  168. return strip_tags($module);
  169. }
  170. /**
  171. * 获得实际的操作名称
  172. * @access private
  173. * @return string
  174. */
  175. static private function getAction($var) {
  176. $action = !empty($_POST[$var]) ?
  177. $_POST[$var] :
  178. (!empty($_GET[$var])?$_GET[$var]:C('DEFAULT_ACTION'));
  179. unset($_POST[$var],$_GET[$var]);
  180. return strip_tags(C('URL_CASE_INSENSITIVE')?strtolower($action):$action);
  181. }
  182. /**
  183. * 获得实际的分组名称
  184. * @access private
  185. * @return string
  186. */
  187. static private function getGroup($var) {
  188. $group = (!empty($_GET[$var])?$_GET[$var]:C('DEFAULT_GROUP'));
  189. unset($_GET[$var]);
  190. return strip_tags(C('URL_CASE_INSENSITIVE') ?ucfirst(strtolower($group)):$group);
  191. }
  192. }