Dispatcher.class.php 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410
  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. /**
  19. * URL映射到控制器
  20. * @access public
  21. * @return void
  22. */
  23. public static function dispatch()
  24. {
  25. $varPath = C('VAR_PATHINFO');
  26. $urlCase = C('URL_CASE_INSENSITIVE');
  27. if (isset($_GET[$varPath])) { // 判断URL里面是否有兼容模式参数
  28. $_SERVER['PATH_INFO'] = $_GET[$varPath];
  29. unset($_GET[$varPath]);
  30. } elseif (IS_CLI) { // CLI模式下 index.php module/controller/action/params/...
  31. $_SERVER['PATH_INFO'] = isset($_SERVER['argv'][1]) ? $_SERVER['argv'][1] : '';
  32. }
  33. // 开启子域名部署
  34. if (C('APP_SUB_DOMAIN_DEPLOY')) {
  35. $rules = C('APP_SUB_DOMAIN_RULES');
  36. if (isset($rules[$_SERVER['HTTP_HOST']])) { // 完整域名或者IP配置
  37. define('APP_DOMAIN', $_SERVER['HTTP_HOST']); // 当前完整域名
  38. $rule = $rules[APP_DOMAIN];
  39. } else {
  40. if (strpos(C('APP_DOMAIN_SUFFIX'), '.')) { // com.cn net.cn
  41. $domain = array_slice(explode('.', $_SERVER['HTTP_HOST']), 0, -3);
  42. } else {
  43. $domain = array_slice(explode('.', $_SERVER['HTTP_HOST']), 0, -2);
  44. }
  45. if (!empty($domain)) {
  46. $subDomain = implode('.', $domain);
  47. define('SUB_DOMAIN', $subDomain); // 当前完整子域名
  48. $domain2 = array_pop($domain); // 二级域名
  49. if ($domain) {
  50. // 存在三级域名
  51. $domain3 = array_pop($domain);
  52. }
  53. if (isset($rules[$subDomain])) {
  54. // 子域名
  55. $rule = $rules[$subDomain];
  56. } elseif (isset($rules['*.' . $domain2]) && !empty($domain3)) {
  57. // 泛三级域名
  58. $rule = $rules['*.' . $domain2];
  59. $panDomain = $domain3;
  60. } elseif (isset($rules['*']) && !empty($domain2) && 'www' != $domain2) {
  61. // 泛二级域名
  62. $rule = $rules['*'];
  63. $panDomain = $domain2;
  64. }
  65. }
  66. }
  67. if (!empty($rule)) {
  68. // 子域名部署规则 '子域名'=>array('模块名[/控制器名]','var1=a&var2=b');
  69. if (is_array($rule)) {
  70. list($rule, $vars) = $rule;
  71. }
  72. $array = explode('/', $rule);
  73. // 模块绑定
  74. define('BIND_MODULE', array_shift($array));
  75. // 控制器绑定
  76. if (!empty($array)) {
  77. $controller = array_shift($array);
  78. if ($controller) {
  79. define('BIND_CONTROLLER', $controller);
  80. }
  81. }
  82. if (isset($vars)) {
  83. // 传入参数
  84. parse_str($vars, $parms);
  85. if (isset($panDomain)) {
  86. $pos = array_search('*', $parms);
  87. if (false !== $pos) {
  88. // 泛域名作为参数
  89. $parms[$pos] = $panDomain;
  90. }
  91. }
  92. $_GET = array_merge($_GET, $parms);
  93. }
  94. }
  95. }
  96. // 分析PATHINFO信息
  97. if (!isset($_SERVER['PATH_INFO'])) {
  98. $types = explode(',', C('URL_PATHINFO_FETCH'));
  99. foreach ($types as $type) {
  100. if (0 === strpos($type, ':')) {
  101. // 支持函数判断
  102. $_SERVER['PATH_INFO'] = call_user_func(substr($type, 1));
  103. break;
  104. } elseif (!empty($_SERVER[$type])) {
  105. $_SERVER['PATH_INFO'] = (0 === strpos($_SERVER[$type], $_SERVER['SCRIPT_NAME'])) ?
  106. substr($_SERVER[$type], strlen($_SERVER['SCRIPT_NAME'])) : $_SERVER[$type];
  107. break;
  108. }
  109. }
  110. }
  111. $depr = C('URL_PATHINFO_DEPR');
  112. define('MODULE_PATHINFO_DEPR', $depr);
  113. if (empty($_SERVER['PATH_INFO'])) {
  114. $_SERVER['PATH_INFO'] = '';
  115. define('__INFO__', '');
  116. define('__EXT__', '');
  117. $paths = array();
  118. } else {
  119. // URL后缀
  120. define('__EXT__', strtolower(pathinfo($_SERVER['PATH_INFO'], PATHINFO_EXTENSION)));
  121. // 检查禁止访问的URL后缀
  122. if ($denySuffix = C('URL_DENY_SUFFIX')) {
  123. if (in_array(__EXT__, explode('|', strtolower(str_replace('.', '', $denySuffix))))) {
  124. send_http_status(404);
  125. exit;
  126. }
  127. }
  128. define('__INFO__', trim($_SERVER['PATH_INFO'], '/'));
  129. // 去除URL后缀
  130. $_SERVER['PATH_INFO'] = preg_replace('/\.' . __EXT__ . '$/i', '', __INFO__);
  131. $paths = explode($depr, trim($_SERVER['PATH_INFO'], $depr));
  132. }
  133. // URL常量
  134. define('__SELF__', strip_tags($_SERVER[C('URL_REQUEST_URI')]));
  135. // 获取模块名称
  136. define('MODULE_NAME', self::getModule($paths));
  137. // 检测模块是否存在
  138. if (MODULE_NAME && is_dir(APP_PATH . MODULE_NAME)) {
  139. // 定义当前模块路径
  140. define('MODULE_PATH', APP_PATH . MODULE_NAME . '/');
  141. // 定义当前模块的模版缓存路径
  142. C('CACHE_PATH', CACHE_PATH . MODULE_NAME . '/');
  143. // 定义当前模块的日志目录
  144. C('LOG_PATH', realpath(LOG_PATH) . '/' . MODULE_NAME . '/');
  145. // 模块配置文件开始载入检测位
  146. Hook::listen('module_check');
  147. // 加载模块配置文件
  148. if (is_file(MODULE_PATH . 'Conf/config' . CONF_EXT)) {
  149. C(load_config(MODULE_PATH . 'Conf/config' . CONF_EXT));
  150. }
  151. // 加载应用模式对应的配置文件
  152. if ('common' != APP_MODE && is_file(MODULE_PATH . 'Conf/config_' . APP_MODE . CONF_EXT)) {
  153. C(load_config(MODULE_PATH . 'Conf/config_' . APP_MODE . CONF_EXT));
  154. }
  155. // 当前应用状态对应的配置文件
  156. if (APP_STATUS && is_file(MODULE_PATH . 'Conf/' . APP_STATUS . CONF_EXT)) {
  157. C(load_config(MODULE_PATH . 'Conf/' . APP_STATUS . CONF_EXT));
  158. }
  159. // 加载模块别名定义
  160. if (is_file(MODULE_PATH . 'Conf/alias.php')) {
  161. Think::addMap(include MODULE_PATH . 'Conf/alias.php');
  162. }
  163. // 加载模块tags文件定义
  164. if (is_file(MODULE_PATH . 'Conf/tags.php')) {
  165. Hook::import(include MODULE_PATH . 'Conf/tags.php');
  166. }
  167. // 加载模块函数文件
  168. if (is_file(MODULE_PATH . 'Common/function.php')) {
  169. include MODULE_PATH . 'Common/function.php';
  170. }
  171. // 加载模块的扩展配置文件
  172. load_ext_file(MODULE_PATH);
  173. // 模块配置文件加载完成检测位
  174. Hook::listen('module_config');
  175. } else {
  176. E(L('_MODULE_NOT_EXIST_') . ':' . MODULE_NAME);
  177. }
  178. if (!defined('__APP__')) {
  179. $urlMode = C('URL_MODEL');
  180. if (URL_COMPAT == $urlMode) {
  181. // 兼容模式判断
  182. define('PHP_FILE', _PHP_FILE_ . '?' . $varPath . '=');
  183. } elseif (URL_REWRITE == $urlMode) {
  184. $url = dirname(_PHP_FILE_);
  185. if ('/' == $url || '\\' == $url) {
  186. $url = '';
  187. }
  188. define('PHP_FILE', $url);
  189. } else {
  190. define('PHP_FILE', _PHP_FILE_);
  191. }
  192. // 当前应用地址
  193. define('__APP__', strip_tags(PHP_FILE));
  194. }
  195. // 模块URL地址
  196. $moduleName = defined('MODULE_ALIAS') ? MODULE_ALIAS : MODULE_NAME;
  197. define('__MODULE__', (defined('BIND_MODULE') || !C('MULTI_MODULE')) ? __APP__ : __APP__ . '/' . ($urlCase ? strtolower($moduleName) : $moduleName));
  198. // 获取控制器和操作名
  199. define('CONTROLLER_NAME', self::getController($paths, $urlCase));
  200. define('ACTION_NAME', self::getAction($paths, $urlCase));
  201. if ($paths) {
  202. // 解析剩余的URL参数
  203. $var = array();
  204. if (C('URL_PARAMS_BIND') && 1 == C('URL_PARAMS_BIND_TYPE')) {
  205. // URL参数按顺序绑定变量
  206. $var = $paths;
  207. } else {
  208. preg_replace_callback('/(\w+)\/([^\/]+)/', function ($match) use (&$var) {
  209. $var[$match[1]] = strip_tags($match[2]);
  210. }, implode('/', $paths));
  211. }
  212. $_GET = array_merge($var, $_GET);
  213. }
  214. // 获取控制器的命名空间(路径)
  215. define('CONTROLLER_PATH', self::getSpace($urlCase));
  216. // 当前控制器的UR地址
  217. $controllerName = defined('CONTROLLER_ALIAS') ? CONTROLLER_ALIAS : CONTROLLER_NAME;
  218. define('__CONTROLLER__', __MODULE__ . $depr . (defined('BIND_CONTROLLER') ? '' : ($urlCase ? parse_name($controllerName) : $controllerName)));
  219. // 当前操作的URL地址
  220. define('__ACTION__', __CONTROLLER__ . $depr . (defined('ACTION_ALIAS') ? ACTION_ALIAS : ACTION_NAME));
  221. //保证$_REQUEST正常取值
  222. $_REQUEST = array_merge($_POST, $_GET);
  223. }
  224. /**
  225. * 获得控制器的命名空间路径 便于插件机制访问
  226. * @param boolean $urlCase 是否转换成小写
  227. * @return string
  228. */
  229. private static function getSpace($urlCase)
  230. {
  231. $var = C('VAR_ADDON');
  232. $space = !empty($_GET[$var]) ? strip_tags($_GET[$var]) : '';
  233. unset($_GET[$var]);
  234. return $space;
  235. }
  236. /**
  237. * 获得实际的控制器名称
  238. * @param array $paths path_info数组
  239. * @param boolean $urlCase 是否转换成小写
  240. * @return string
  241. */
  242. private static function getController(&$paths, $urlCase)
  243. {
  244. if (defined('BIND_CONTROLLER')) {
  245. return BIND_CONTROLLER;
  246. } else {
  247. if ($paths && C('URL_ROUTER_ON') && Route::check($paths)) {
  248. $depr = C('URL_PATHINFO_DEPR');
  249. $paths = explode($depr, trim($_SERVER['PATH_INFO'], $depr));
  250. }
  251. if ($paths) {
  252. // PATH_INFO检测标签位
  253. Hook::listen('path_info');
  254. if (C('CONTROLLER_LEVEL') > 1) { // 控制器层次
  255. $controller = implode('/', array_slice($paths, 0, C('CONTROLLER_LEVEL')));
  256. $paths = array_slice($paths, C('CONTROLLER_LEVEL'));
  257. } else {
  258. $controller = array_shift($paths);
  259. }
  260. } else {
  261. $var = C('VAR_CONTROLLER');
  262. if (!empty($_GET[$var])) {
  263. $controller = $_GET[$var];
  264. unset($_GET[$var]);
  265. } else {
  266. $controller = C('DEFAULT_CONTROLLER');
  267. }
  268. }
  269. }
  270. if ($maps = C('URL_CONTROLLER_MAP')) {
  271. if (isset($maps[strtolower($controller)])) {
  272. // 记录当前别名
  273. define('CONTROLLER_ALIAS', strtolower($controller));
  274. // 获取实际的控制器名
  275. return ucfirst($maps[CONTROLLER_ALIAS]);
  276. } elseif (array_search(strtolower($controller), $maps)) {
  277. // 禁止访问原始控制器
  278. return '';
  279. }
  280. }
  281. if ($urlCase) {
  282. // URL地址不区分大小写
  283. // 智能识别方式 user_type 识别到 UserTypeController 控制器
  284. $controller = parse_name($controller, 1);
  285. }
  286. return strip_tags(ucfirst($controller));
  287. }
  288. /**
  289. * 获得实际的操作名称
  290. * @param array $paths path_info数组
  291. * @param boolean $urlCase 是否转换成小写
  292. * @return string
  293. */
  294. private static function getAction(&$paths, $urlCase)
  295. {
  296. if (defined('BIND_ACTION')) {
  297. return BIND_ACTION;
  298. } else {
  299. if ($paths) {
  300. $action = array_shift($paths);
  301. } else {
  302. $var = C('VAR_ACTION');
  303. if (!empty($_GET[$var])) {
  304. $action = $_GET[$var];
  305. unset($_GET[$var]);
  306. } elseif (!empty($_POST[$var])) {
  307. $action = $_POST[$var];
  308. unset($_POST[$var]);
  309. } else {
  310. $action = C('DEFAULT_ACTION');
  311. }
  312. }
  313. }
  314. if ($maps = C('URL_ACTION_MAP')) {
  315. if (isset($maps[strtolower(CONTROLLER_NAME)])) {
  316. $maps = $maps[strtolower(CONTROLLER_NAME)];
  317. if (isset($maps[strtolower($action)])) {
  318. // 记录当前别名
  319. define('ACTION_ALIAS', strtolower($action));
  320. // 获取实际的操作名
  321. if (is_array($maps[ACTION_ALIAS])) {
  322. parse_str($maps[ACTION_ALIAS][1], $vars);
  323. $_GET = array_merge($_GET, $vars);
  324. return $maps[ACTION_ALIAS][0];
  325. } else {
  326. return $maps[ACTION_ALIAS];
  327. }
  328. } elseif (array_search(strtolower($action), $maps)) {
  329. // 禁止访问原始操作
  330. return '';
  331. }
  332. }
  333. }
  334. return strip_tags($urlCase ? strtolower($action) : $action);
  335. }
  336. /**
  337. * 获得实际的模块名称
  338. * @param array $paths path_info数组
  339. * @return string
  340. */
  341. private static function getModule(&$paths)
  342. {
  343. if (defined('BIND_MODULE')) {
  344. return BIND_MODULE;
  345. } else {
  346. // 检查路由
  347. if ($paths && C('URL_ROUTER_ON') && Route::check($paths)) {
  348. $paths = explode(MODULE_PATHINFO_DEPR, trim($_SERVER['PATH_INFO'], MODULE_PATHINFO_DEPR));
  349. }
  350. if ($paths && C('MULTI_MODULE')) { // 获取模块名
  351. $allowList = C('MODULE_ALLOW_LIST'); // 允许的模块列表
  352. if (empty($allowList) || (is_array($allowList) && in_array_case($paths[0], $allowList))) {
  353. $module = array_shift($paths);
  354. $_SERVER['PATH_INFO'] = implode(MODULE_PATHINFO_DEPR, $paths);
  355. }
  356. } else {
  357. $var = C('VAR_MODULE');
  358. if (!empty($_GET[$var])) {
  359. $module = $_GET[$var];
  360. unset($_GET[$var]);
  361. }
  362. }
  363. if (empty($module)) {
  364. $module = C('DEFAULT_MODULE');
  365. }
  366. }
  367. if ($maps = C('URL_MODULE_MAP')) {
  368. if (isset($maps[strtolower($module)])) {
  369. // 记录当前别名
  370. define('MODULE_ALIAS', strtolower($module));
  371. // 获取实际的模块名
  372. return ucfirst($maps[MODULE_ALIAS]);
  373. } elseif (array_search(strtolower($module), $maps) || in_array_case($module, C('MODULE_DENY_LIST'))) {
  374. // 禁止访问原始模块
  375. return '';
  376. }
  377. }
  378. return strip_tags(ucfirst($module));
  379. }
  380. }