PageRenderTime 46ms CodeModel.GetById 16ms RepoModel.GetById 0ms app.codeStats 0ms

/xianpipa/ThinkPHP/Mode/Lite/Dispatcher.class.php

https://gitlab.com/fangfangchen/xianpipa
PHP | 264 lines | 218 code | 13 blank | 33 comment | 41 complexity | 030815068a512ef67a8a965a6873aab8 MD5 | raw file
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | ThinkPHP [ WE CAN DO IT JUST THINK IT ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2006-2014 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. namespace Think;
  12. /**
  13. * ThinkPHP内置的Dispatcher类
  14. * 完成URL解析、路由和调度
  15. */
  16. class Dispatcher {
  17. /**
  18. * URL映射到控制器
  19. * @access public
  20. * @return void
  21. */
  22. static public function dispatch() {
  23. $varPath = C('VAR_PATHINFO');
  24. $varModule = C('VAR_MODULE');
  25. $varController = C('VAR_CONTROLLER');
  26. $varAction = C('VAR_ACTION');
  27. $urlCase = C('URL_CASE_INSENSITIVE');
  28. if(isset($_GET[$varPath])) { // 判断URL里面是否有兼容模式参数
  29. $_SERVER['PATH_INFO'] = $_GET[$varPath];
  30. unset($_GET[$varPath]);
  31. }elseif(IS_CLI){ // CLI模式下 index.php module/controller/action/params/...
  32. $_SERVER['PATH_INFO'] = isset($_SERVER['argv'][1]) ? $_SERVER['argv'][1] : '';
  33. }
  34. // 开启子域名部署
  35. if(C('APP_SUB_DOMAIN_DEPLOY')) {
  36. $rules = C('APP_SUB_DOMAIN_RULES');
  37. if(isset($rules[$_SERVER['HTTP_HOST']])) { // 完整域名或者IP配置
  38. define('APP_DOMAIN',$_SERVER['HTTP_HOST']); // 当前完整域名
  39. $rule = $rules[APP_DOMAIN];
  40. }else{
  41. if(strpos(C('APP_DOMAIN_SUFFIX'),'.')){ // com.cn net.cn
  42. $domain = array_slice(explode('.', $_SERVER['HTTP_HOST']), 0, -3);
  43. }else{
  44. $domain = array_slice(explode('.', $_SERVER['HTTP_HOST']), 0, -2);
  45. }
  46. if(!empty($domain)) {
  47. $subDomain = implode('.', $domain);
  48. define('SUB_DOMAIN',$subDomain); // 当前完整子域名
  49. $domain2 = array_pop($domain); // 二级域名
  50. if($domain) { // 存在三级域名
  51. $domain3 = array_pop($domain);
  52. }
  53. if(isset($rules[$subDomain])) { // 子域名
  54. $rule = $rules[$subDomain];
  55. }elseif(isset($rules['*.' . $domain2]) && !empty($domain3)){ // 泛三级域名
  56. $rule = $rules['*.' . $domain2];
  57. $panDomain = $domain3;
  58. }elseif(isset($rules['*']) && !empty($domain2) && 'www' != $domain2 ){ // 泛二级域名
  59. $rule = $rules['*'];
  60. $panDomain = $domain2;
  61. }
  62. }
  63. }
  64. if(!empty($rule)) {
  65. // 子域名部署规则 '子域名'=>array('模块名','var1=a&var2=b');
  66. if(is_array($rule)){
  67. list($rule,$vars) = $rule;
  68. }
  69. $array = explode('/',$rule);
  70. // 模块绑定
  71. define('BIND_MODULE',array_shift($array));
  72. if(isset($vars)) { // 传入参数
  73. parse_str($vars,$parms);
  74. if(isset($panDomain)){
  75. $pos = array_search('*', $parms);
  76. if(false !== $pos) {
  77. // 泛域名作为参数
  78. $parms[$pos] = $panDomain;
  79. }
  80. }
  81. $_GET = array_merge($_GET,$parms);
  82. }
  83. }
  84. }
  85. // 分析PATHINFO信息
  86. if(!isset($_SERVER['PATH_INFO'])) {
  87. $types = explode(',',C('URL_PATHINFO_FETCH'));
  88. foreach ($types as $type){
  89. if(0===strpos($type,':')) {// 支持函数判断
  90. $_SERVER['PATH_INFO'] = call_user_func(substr($type,1));
  91. break;
  92. }elseif(!empty($_SERVER[$type])) {
  93. $_SERVER['PATH_INFO'] = (0 === strpos($_SERVER[$type],$_SERVER['SCRIPT_NAME']))?
  94. substr($_SERVER[$type], strlen($_SERVER['SCRIPT_NAME'])) : $_SERVER[$type];
  95. break;
  96. }
  97. }
  98. }
  99. $depr = C('URL_PATHINFO_DEPR');
  100. define('MODULE_PATHINFO_DEPR', $depr);
  101. if(empty($_SERVER['PATH_INFO'])) {
  102. $_SERVER['PATH_INFO'] = '';
  103. define('__INFO__','');
  104. define('__EXT__','');
  105. }else{
  106. define('__INFO__',trim($_SERVER['PATH_INFO'],'/'));
  107. // URL后缀
  108. define('__EXT__', strtolower(pathinfo($_SERVER['PATH_INFO'],PATHINFO_EXTENSION)));
  109. $_SERVER['PATH_INFO'] = __INFO__;
  110. if(!defined('BIND_MODULE') && (!C('URL_ROUTER_ON') || !Route::check())){
  111. if (__INFO__ ){ // 获取模块名
  112. $paths = explode($depr,__INFO__,2);
  113. $allowList = C('MODULE_ALLOW_LIST'); // 允许的模块列表
  114. $module = preg_replace('/\.' . __EXT__ . '$/i', '',$paths[0]);
  115. if( empty($allowList) || (is_array($allowList) && in_array_case($module, $allowList))){
  116. $_GET[$varModule] = $module;
  117. $_SERVER['PATH_INFO'] = isset($paths[1])?$paths[1]:'';
  118. }
  119. }
  120. }
  121. }
  122. // URL常量
  123. define('__SELF__',strip_tags($_SERVER[C('URL_REQUEST_URI')]));
  124. // 获取模块名称
  125. define('MODULE_NAME', defined('BIND_MODULE')? BIND_MODULE : self::getModule($varModule));
  126. // 检测模块是否存在
  127. if( MODULE_NAME && (defined('BIND_MODULE') || !in_array_case(MODULE_NAME,C('MODULE_DENY_LIST')) ) && is_dir(APP_PATH.MODULE_NAME)){
  128. // 定义当前模块路径
  129. define('MODULE_PATH', APP_PATH.MODULE_NAME.'/');
  130. // 定义当前模块的模版缓存路径
  131. C('CACHE_PATH',CACHE_PATH.MODULE_NAME.'/');
  132. // 定义当前模块的日志目录
  133. C('LOG_PATH', realpath(LOG_PATH).'/'.MODULE_NAME.'/');
  134. // 加载模块配置文件
  135. if(is_file(MODULE_PATH.'Conf/config'.CONF_EXT))
  136. C(load_config(MODULE_PATH.'Conf/config'.CONF_EXT));
  137. // 加载模块别名定义
  138. if(is_file(MODULE_PATH.'Conf/alias.php'))
  139. Think::addMap(include MODULE_PATH.'Conf/alias.php');
  140. // 加载模块函数文件
  141. if(is_file(MODULE_PATH.'Common/function.php'))
  142. include MODULE_PATH.'Common/function.php';
  143. }else{
  144. E(L('_MODULE_NOT_EXIST_').':'.MODULE_NAME);
  145. }
  146. if(!defined('__APP__')){
  147. $urlMode = C('URL_MODEL');
  148. if($urlMode == URL_COMPAT ){// 兼容模式判断
  149. define('PHP_FILE',_PHP_FILE_.'?'.$varPath.'=');
  150. }elseif($urlMode == URL_REWRITE ) {
  151. $url = dirname(_PHP_FILE_);
  152. if($url == '/' || $url == '\\')
  153. $url = '';
  154. define('PHP_FILE',$url);
  155. }else {
  156. define('PHP_FILE',_PHP_FILE_);
  157. }
  158. // 当前应用地址
  159. define('__APP__',strip_tags(PHP_FILE));
  160. }
  161. // 模块URL地址
  162. $moduleName = defined('MODULE_ALIAS')? MODULE_ALIAS : MODULE_NAME;
  163. define('__MODULE__',defined('BIND_MODULE') ? __APP__ : __APP__.'/'.($urlCase ? strtolower($moduleName) : $moduleName));
  164. if('' != $_SERVER['PATH_INFO'] && (!C('URL_ROUTER_ON') || !Route::check()) ){ // 检测路由规则 如果没有则按默认规则调度URL
  165. // 检查禁止访问的URL后缀
  166. if(C('URL_DENY_SUFFIX') && preg_match('/\.('.trim(C('URL_DENY_SUFFIX'),'.').')$/i', $_SERVER['PATH_INFO'])){
  167. send_http_status(404);
  168. exit;
  169. }
  170. // 去除URL后缀
  171. $_SERVER['PATH_INFO'] = preg_replace(C('URL_HTML_SUFFIX')? '/\.('.trim(C('URL_HTML_SUFFIX'),'.').')$/i' : '/\.'.__EXT__.'$/i', '', $_SERVER['PATH_INFO']);
  172. $depr = C('URL_PATHINFO_DEPR');
  173. $paths = explode($depr,trim($_SERVER['PATH_INFO'],$depr));
  174. $_GET[$varController] = array_shift($paths);
  175. // 获取操作
  176. $_GET[$varAction] = array_shift($paths);
  177. // 解析剩余的URL参数
  178. $var = array();
  179. if(C('URL_PARAMS_BIND') && 1 == C('URL_PARAMS_BIND_TYPE')){
  180. // URL参数按顺序绑定变量
  181. $var = $paths;
  182. }else{
  183. preg_replace_callback('/(\w+)\/([^\/]+)/', function($match) use(&$var){$var[$match[1]]=strip_tags($match[2]);}, implode('/',$paths));
  184. }
  185. $_GET = array_merge($var,$_GET);
  186. }
  187. // 获取控制器和操作名
  188. define('CONTROLLER_NAME', self::getController($varController,$urlCase));
  189. define('ACTION_NAME', self::getAction($varAction,$urlCase));
  190. // 当前控制器的UR地址
  191. define('__CONTROLLER__',__MODULE__.$depr.( $urlCase ? parse_name(CONTROLLER_NAME) : CONTROLLER_NAME ) );
  192. // 当前操作的URL地址
  193. define('__ACTION__',__CONTROLLER__.$depr.ACTION_NAME);
  194. //保证$_REQUEST正常取值
  195. $_REQUEST = array_merge($_POST,$_GET);
  196. }
  197. /**
  198. * 获得实际的控制器名称
  199. */
  200. static private function getController($var,$urlCase) {
  201. $controller = (!empty($_GET[$var])? $_GET[$var]:C('DEFAULT_CONTROLLER'));
  202. unset($_GET[$var]);
  203. if($urlCase) {
  204. // URL地址不区分大小写
  205. // 智能识别方式 user_type 识别到 UserTypeController 控制器
  206. $controller = parse_name($controller,1);
  207. }
  208. return strip_tags(ucfirst($controller));
  209. }
  210. /**
  211. * 获得实际的操作名称
  212. */
  213. static private function getAction($var,$urlCase) {
  214. $action = !empty($_POST[$var]) ?
  215. $_POST[$var] :
  216. (!empty($_GET[$var])?$_GET[$var]:C('DEFAULT_ACTION'));
  217. unset($_POST[$var],$_GET[$var]);
  218. return strip_tags( $urlCase? strtolower($action) : $action );
  219. }
  220. /**
  221. * 获得实际的模块名称
  222. */
  223. static private function getModule($var) {
  224. $module = (!empty($_GET[$var])?$_GET[$var]:C('DEFAULT_MODULE'));
  225. unset($_GET[$var]);
  226. if($maps = C('URL_MODULE_MAP')) {
  227. if(isset($maps[strtolower($module)])) {
  228. // 记录当前别名
  229. define('MODULE_ALIAS',strtolower($module));
  230. // 获取实际的模块名
  231. return ucfirst($maps[MODULE_ALIAS]);
  232. }elseif(array_search(strtolower($module),$maps)){
  233. // 禁止访问原始模块
  234. return '';
  235. }
  236. }
  237. return strip_tags(ucfirst($module));
  238. }
  239. }