Dispatcher.class.php 10 KB

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