CheckActionRouteBehavior.class.php 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  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. defined('THINK_PATH') or exit();
  12. /**
  13. * 系统行为扩展:操作路由检测
  14. * @category Think
  15. * @package Think
  16. * @subpackage Behavior
  17. * @author liu21st <liu21st@gmail.com>
  18. */
  19. class CheckActionRouteBehavior extends Behavior {
  20. // 行为扩展的执行入口必须是run
  21. public function run(&$config){
  22. // 优先检测是否存在PATH_INFO
  23. $regx = trim($_SERVER['PATH_INFO'],'/');
  24. if(empty($regx)) return ;
  25. // 路由定义文件优先于config中的配置定义
  26. // 路由处理
  27. $routes = $config['routes'];
  28. if(!empty($routes)) {
  29. $depr = C('URL_PATHINFO_DEPR');
  30. // 分隔符替换 确保路由定义使用统一的分隔符
  31. $regx = str_replace($depr,'/',$regx);
  32. $regx = substr_replace($regx,'',0,strlen(__URL__));
  33. foreach ($routes as $rule=>$route){
  34. if(0===strpos($rule,'/') && preg_match($rule,$regx,$matches)) { // 正则路由
  35. return C('ACTION_NAME',$this->parseRegex($matches,$route,$regx));
  36. }else{ // 规则路由
  37. $len1 = substr_count($regx,'/');
  38. $len2 = substr_count($rule,'/');
  39. if($len1>=$len2) {
  40. if('$' == substr($rule,-1,1)) {// 完整匹配
  41. if($len1 != $len2) {
  42. continue;
  43. }else{
  44. $rule = substr($rule,0,-1);
  45. }
  46. }
  47. $match = $this->checkUrlMatch($regx,$rule);
  48. if($match) return C('ACTION_NAME',$this->parseRule($rule,$route,$regx));
  49. }
  50. }
  51. }
  52. }
  53. }
  54. // 检测URL和规则路由是否匹配
  55. private function checkUrlMatch($regx,$rule) {
  56. $m1 = explode('/',$regx);
  57. $m2 = explode('/',$rule);
  58. $match = true; // 是否匹配
  59. foreach ($m2 as $key=>$val){
  60. if(':' == substr($val,0,1)) {// 动态变量
  61. if(strpos($val,'\\')) {
  62. $type = substr($val,-1);
  63. if('d'==$type && !is_numeric($m1[$key])) {
  64. $match = false;
  65. break;
  66. }
  67. }elseif(strpos($val,'^')){
  68. $array = explode('|',substr(strstr($val,'^'),1));
  69. if(in_array($m1[$key],$array)) {
  70. $match = false;
  71. break;
  72. }
  73. }
  74. }elseif(0 !== strcasecmp($val,$m1[$key])){
  75. $match = false;
  76. break;
  77. }
  78. }
  79. return $match;
  80. }
  81. // 解析规范的路由地址
  82. // 地址格式 操作?参数1=值1&参数2=值2...
  83. private function parseUrl($url) {
  84. $var = array();
  85. if(false !== strpos($url,'?')) { // 操作?参数1=值1&参数2=值2...
  86. $info = parse_url($url);
  87. $path = $info['path'];
  88. parse_str($info['query'],$var);
  89. }else{ // 操作
  90. $path = $url;
  91. }
  92. $var[C('VAR_ACTION')] = $path;
  93. return $var;
  94. }
  95. // 解析规则路由
  96. // '路由规则'=>'操作?额外参数1=值1&额外参数2=值2...'
  97. // '路由规则'=>array('操作','额外参数1=值1&额外参数2=值2...')
  98. // '路由规则'=>'外部地址'
  99. // '路由规则'=>array('外部地址','重定向代码')
  100. // 路由规则中 :开头 表示动态变量
  101. // 外部地址中可以用动态变量 采用 :1 :2 的方式
  102. // 'news/:month/:day/:id'=>array('News/read?cate=1','status=1'),
  103. // 'new/:id'=>array('/new.php?id=:1',301), 重定向
  104. private function parseRule($rule,$route,$regx) {
  105. // 获取路由地址规则
  106. $url = is_array($route)?$route[0]:$route;
  107. // 获取URL地址中的参数
  108. $paths = explode('/',$regx);
  109. // 解析路由规则
  110. $matches = array();
  111. $rule = explode('/',$rule);
  112. foreach ($rule as $item){
  113. if(0===strpos($item,':')) { // 动态变量获取
  114. if($pos = strpos($item,'^') ) {
  115. $var = substr($item,1,$pos-1);
  116. }elseif(strpos($item,'\\')){
  117. $var = substr($item,1,-2);
  118. }else{
  119. $var = substr($item,1);
  120. }
  121. $matches[$var] = array_shift($paths);
  122. }else{ // 过滤URL中的静态变量
  123. array_shift($paths);
  124. }
  125. }
  126. if(0=== strpos($url,'/') || 0===strpos($url,'http')) { // 路由重定向跳转
  127. if(strpos($url,':')) { // 传递动态参数
  128. $values = array_values($matches);
  129. $url = preg_replace('/:(\d+)/e','$values[\\1-1]',$url);
  130. }
  131. header("Location: $url", true,(is_array($route) && isset($route[1]))?$route[1]:301);
  132. exit;
  133. }else{
  134. // 解析路由地址
  135. $var = $this->parseUrl($url);
  136. // 解析路由地址里面的动态参数
  137. $values = array_values($matches);
  138. foreach ($var as $key=>$val){
  139. if(0===strpos($val,':')) {
  140. $var[$key] = $values[substr($val,1)-1];
  141. }
  142. }
  143. $var = array_merge($matches,$var);
  144. // 解析剩余的URL参数
  145. if($paths) {
  146. preg_replace('@(\w+)\/([^\/]+)@e', '$var[strtolower(\'\\1\')]=strip_tags(\'\\2\');', implode('/',$paths));
  147. }
  148. // 解析路由自动传人参数
  149. if(is_array($route) && isset($route[1])) {
  150. parse_str($route[1],$params);
  151. $var = array_merge($var,$params);
  152. }
  153. $action = $var[C('VAR_ACTION')];
  154. unset($var[C('VAR_ACTION')]);
  155. $_GET = array_merge($var,$_GET);
  156. return $action;
  157. }
  158. }
  159. // 解析正则路由
  160. // '路由正则'=>'[分组/模块/操作]?参数1=值1&参数2=值2...'
  161. // '路由正则'=>array('[分组/模块/操作]?参数1=值1&参数2=值2...','额外参数1=值1&额外参数2=值2...')
  162. // '路由正则'=>'外部地址'
  163. // '路由正则'=>array('外部地址','重定向代码')
  164. // 参数值和外部地址中可以用动态变量 采用 :1 :2 的方式
  165. // '/new\/(\d+)\/(\d+)/'=>array('News/read?id=:1&page=:2&cate=1','status=1'),
  166. // '/new\/(\d+)/'=>array('/new.php?id=:1&page=:2&status=1','301'), 重定向
  167. private function parseRegex($matches,$route,$regx) {
  168. // 获取路由地址规则
  169. $url = is_array($route)?$route[0]:$route;
  170. $url = preg_replace('/:(\d+)/e','$matches[\\1]',$url);
  171. if(0=== strpos($url,'/') || 0===strpos($url,'http')) { // 路由重定向跳转
  172. header("Location: $url", true,(is_array($route) && isset($route[1]))?$route[1]:301);
  173. exit;
  174. }else{
  175. // 解析路由地址
  176. $var = $this->parseUrl($url);
  177. // 解析剩余的URL参数
  178. $regx = substr_replace($regx,'',0,strlen($matches[0]));
  179. if($regx) {
  180. preg_replace('@(\w+)\/([^,\/]+)@e', '$var[strtolower(\'\\1\')]=strip_tags(\'\\2\');', $regx);
  181. }
  182. // 解析路由自动传人参数
  183. if(is_array($route) && isset($route[1])) {
  184. parse_str($route[1],$params);
  185. $var = array_merge($var,$params);
  186. }
  187. $action = $var[C('VAR_ACTION')];
  188. unset($var[C('VAR_ACTION')]);
  189. $_GET = array_merge($var,$_GET);
  190. }
  191. return $action;
  192. }
  193. }