Dispatcher.class.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  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(isset($_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. if(isset($rules[$_SERVER['HTTP_HOST']])) { // 完整域名或者IP配置
  48. $rule = $rules[$_SERVER['HTTP_HOST']];
  49. }else{
  50. $subDomain = strtolower(substr($_SERVER['HTTP_HOST'],0,strpos($_SERVER['HTTP_HOST'],'.')));
  51. define('SUB_DOMAIN',$subDomain); // 二级域名定义
  52. if($subDomain && isset($rules[$subDomain])) {
  53. $rule = $rules[$subDomain];
  54. }elseif(isset($rules['*'])){ // 泛域名支持
  55. if('www' != $subDomain && !in_array($subDomain,C('APP_SUB_DOMAIN_DENY'))) {
  56. $rule = $rules['*'];
  57. }
  58. }
  59. }
  60. if(!empty($rule)) {
  61. // 子域名部署规则 '子域名'=>array('分组名/[模块名]','var1=a&var2=b');
  62. $array = explode('/',$rule[0]);
  63. $module = array_pop($array);
  64. if(!empty($module)) {
  65. $_GET[C('VAR_MODULE')] = $module;
  66. $domainModule = true;
  67. }
  68. if(!empty($array)) {
  69. $_GET[C('VAR_GROUP')] = array_pop($array);
  70. $domainGroup = true;
  71. }
  72. if(isset($rule[1])) { // 传入参数
  73. parse_str($rule[1],$parms);
  74. $_GET = array_merge($_GET,$parms);
  75. }
  76. }
  77. }
  78. // 分析PATHINFO信息
  79. if(!isset($_SERVER['PATH_INFO'])) {
  80. $types = explode(',',C('URL_PATHINFO_FETCH'));
  81. foreach ($types as $type){
  82. if(0===strpos($type,':')) {// 支持函数判断
  83. $_SERVER['PATH_INFO'] = call_user_func(substr($type,1));
  84. break;
  85. }elseif(!empty($_SERVER[$type])) {
  86. $_SERVER['PATH_INFO'] = (0 === strpos($_SERVER[$type],$_SERVER['SCRIPT_NAME']))?
  87. substr($_SERVER[$type], strlen($_SERVER['SCRIPT_NAME'])) : $_SERVER[$type];
  88. break;
  89. }
  90. }
  91. }
  92. $depr = C('URL_PATHINFO_DEPR');
  93. if(!empty($_SERVER['PATH_INFO'])) {
  94. tag('path_info');
  95. $part = pathinfo($_SERVER['PATH_INFO']);
  96. define('__EXT__', isset($part['extension'])?strtolower($part['extension']):'');
  97. if(__EXT__){
  98. if(C('URL_DENY_SUFFIX') && preg_match('/\.('.trim(C('URL_DENY_SUFFIX'),'.').')$/i', $_SERVER['PATH_INFO'])){
  99. send_http_status(404);
  100. exit;
  101. }
  102. if(C('URL_HTML_SUFFIX')) {
  103. $_SERVER['PATH_INFO'] = preg_replace('/\.('.trim(C('URL_HTML_SUFFIX'),'.').')$/i', '', $_SERVER['PATH_INFO']);
  104. }else{
  105. $_SERVER['PATH_INFO'] = preg_replace('/.'.__EXT__.'$/i','',$_SERVER['PATH_INFO']);
  106. }
  107. }
  108. if(!self::routerCheck()){ // 检测路由规则 如果没有则按默认规则调度URL
  109. $paths = explode($depr,trim($_SERVER['PATH_INFO'],'/'));
  110. if(C('VAR_URL_PARAMS')) {
  111. // 直接通过$_GET['_URL_'][1] $_GET['_URL_'][2] 获取URL参数 方便不用路由时参数获取
  112. $_GET[C('VAR_URL_PARAMS')] = $paths;
  113. }
  114. $var = array();
  115. if (C('APP_GROUP_LIST') && !isset($_GET[C('VAR_GROUP')])){
  116. $var[C('VAR_GROUP')] = in_array(strtolower($paths[0]),explode(',',strtolower(C('APP_GROUP_LIST'))))? array_shift($paths) : '';
  117. if(C('APP_GROUP_DENY') && in_array(strtolower($var[C('VAR_GROUP')]),explode(',',strtolower(C('APP_GROUP_DENY'))))) {
  118. // 禁止直接访问分组
  119. exit;
  120. }
  121. }
  122. if(!isset($_GET[C('VAR_MODULE')])) {// 还没有定义模块名称
  123. $var[C('VAR_MODULE')] = array_shift($paths);
  124. }
  125. $var[C('VAR_ACTION')] = array_shift($paths);
  126. // 解析剩余的URL参数
  127. preg_replace('@(\w+)\/([^\/]+)@e', '$var[\'\\1\']=strip_tags(\'\\2\');', implode('/',$paths));
  128. $_GET = array_merge($var,$_GET);
  129. }
  130. define('__INFO__',$_SERVER['PATH_INFO']);
  131. }else{
  132. define('__INFO__','');
  133. }
  134. // URL常量
  135. define('__SELF__',strip_tags($_SERVER['REQUEST_URI']));
  136. // 当前项目地址
  137. define('__APP__',strip_tags(PHP_FILE));
  138. // 获取分组 模块和操作名称
  139. if (C('APP_GROUP_LIST')) {
  140. define('GROUP_NAME', self::getGroup(C('VAR_GROUP')));
  141. // 分组URL地址
  142. define('__GROUP__',(!empty($domainGroup) || strtolower(GROUP_NAME) == strtolower(C('DEFAULT_GROUP')) )?__APP__ : __APP__.'/'.(C('URL_CASE_INSENSITIVE') ? strtolower(GROUP_NAME) : GROUP_NAME));
  143. }
  144. // 定义项目基础加载路径
  145. define('BASE_LIB_PATH', (defined('GROUP_NAME') && C('APP_GROUP_MODE')==1) ? APP_PATH.C('APP_GROUP_PATH').'/'.GROUP_NAME.'/' : LIB_PATH);
  146. if(defined('GROUP_NAME')) {
  147. C('CACHE_PATH',CACHE_PATH.GROUP_NAME.'/');
  148. if(1 == C('APP_GROUP_MODE')){ // 独立分组模式
  149. $config_path = BASE_LIB_PATH.'Conf/';
  150. $common_path = BASE_LIB_PATH.'Common/';
  151. }else{ // 普通分组模式
  152. $config_path = CONF_PATH.GROUP_NAME.'/';
  153. $common_path = COMMON_PATH.GROUP_NAME.'/';
  154. }
  155. // 加载分组配置文件
  156. if(is_file($config_path.'config.php'))
  157. C(include $config_path.'config.php');
  158. // 加载分组别名定义
  159. if(is_file($config_path.'alias.php'))
  160. alias_import(include $config_path.'alias.php');
  161. // 加载分组tags文件定义
  162. if(is_file($config_path.'tags.php'))
  163. C('tags', include $config_path.'tags.php');
  164. // 加载分组函数文件
  165. if(is_file($common_path.'function.php'))
  166. include $common_path.'function.php';
  167. }else{
  168. C('CACHE_PATH',CACHE_PATH);
  169. }
  170. define('MODULE_NAME',self::getModule(C('VAR_MODULE')));
  171. define('ACTION_NAME',self::getAction(C('VAR_ACTION')));
  172. // 当前模块和分组地址
  173. $moduleName = defined('MODULE_ALIAS')?MODULE_ALIAS:MODULE_NAME;
  174. if(defined('GROUP_NAME')) {
  175. define('__URL__',!empty($domainModule)?__GROUP__.$depr : __GROUP__.$depr.( C('URL_CASE_INSENSITIVE') ? strtolower($moduleName) : $moduleName ) );
  176. }else{
  177. define('__URL__',!empty($domainModule)?__APP__.'/' : __APP__.'/'.( C('URL_CASE_INSENSITIVE') ? strtolower($moduleName) : $moduleName) );
  178. }
  179. // 当前操作地址
  180. define('__ACTION__',__URL__.$depr.(defined('ACTION_ALIAS')?ACTION_ALIAS:ACTION_NAME));
  181. //保证$_REQUEST正常取值
  182. $_REQUEST = array_merge($_POST,$_GET);
  183. }
  184. /**
  185. * 路由检测
  186. * @access public
  187. * @return void
  188. */
  189. static public function routerCheck() {
  190. $return = false;
  191. // 路由检测标签
  192. tag('route_check',$return);
  193. return $return;
  194. }
  195. /**
  196. * 获得实际的模块名称
  197. * @access private
  198. * @return string
  199. */
  200. static private function getModule($var) {
  201. $module = (!empty($_GET[$var])? $_GET[$var]:C('DEFAULT_MODULE'));
  202. unset($_GET[$var]);
  203. if($maps = C('URL_MODULE_MAP')) {
  204. if(isset($maps[strtolower($module)])) {
  205. // 记录当前别名
  206. define('MODULE_ALIAS',strtolower($module));
  207. // 获取实际的模块名
  208. return $maps[MODULE_ALIAS];
  209. }elseif(array_search(strtolower($module),$maps)){
  210. // 禁止访问原始模块
  211. return '';
  212. }
  213. }
  214. if(C('URL_CASE_INSENSITIVE')) {
  215. // URL地址不区分大小写
  216. // 智能识别方式 index.php/user_type/index/ 识别到 UserTypeAction 模块
  217. $module = ucfirst(parse_name($module,1));
  218. }
  219. return strip_tags($module);
  220. }
  221. /**
  222. * 获得实际的操作名称
  223. * @access private
  224. * @return string
  225. */
  226. static private function getAction($var) {
  227. $action = !empty($_POST[$var]) ?
  228. $_POST[$var] :
  229. (!empty($_GET[$var])?$_GET[$var]:C('DEFAULT_ACTION'));
  230. unset($_POST[$var],$_GET[$var]);
  231. if($maps = C('URL_ACTION_MAP')) {
  232. if(isset($maps[strtolower(MODULE_NAME)])) {
  233. $maps = $maps[strtolower(MODULE_NAME)];
  234. if(isset($maps[strtolower($action)])) {
  235. // 记录当前别名
  236. define('ACTION_ALIAS',strtolower($action));
  237. // 获取实际的操作名
  238. return $maps[ACTION_ALIAS];
  239. }elseif(array_search(strtolower($action),$maps)){
  240. // 禁止访问原始操作
  241. return '';
  242. }
  243. }
  244. }
  245. return strip_tags(C('URL_CASE_INSENSITIVE')?strtolower($action):$action);
  246. }
  247. /**
  248. * 获得实际的分组名称
  249. * @access private
  250. * @return string
  251. */
  252. static private function getGroup($var) {
  253. $group = (!empty($_GET[$var])?$_GET[$var]:C('DEFAULT_GROUP'));
  254. unset($_GET[$var]);
  255. return strip_tags(C('URL_CASE_INSENSITIVE') ?ucfirst(strtolower($group)):$group);
  256. }
  257. }