Dispatcher.class.php 10 KB

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