123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298 |
- <?php
- // +----------------------------------------------------------------------
- // | ThinkPHP [ WE CAN DO IT JUST THINK ]
- // +----------------------------------------------------------------------
- // | Copyright (c) 2006~2018 http://thinkphp.cn All rights reserved.
- // +----------------------------------------------------------------------
- // | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
- // +----------------------------------------------------------------------
- // | Author: liu21st <liu21st@gmail.com>
- // +----------------------------------------------------------------------
- namespace think\route;
- use think\Route;
- class RuleItem extends Rule
- {
- /**
- * 路由规则
- * @var string
- */
- protected $name;
- /**
- * 路由地址
- * @var string|\Closure
- */
- protected $route;
- /**
- * 请求类型
- * @var string
- */
- protected $method;
- /**
- * 架构函数
- * @access public
- * @param Route $router 路由实例
- * @param RuleGroup $group 路由所属分组对象
- * @param string|array $name 路由规则
- * @param string $method 请求类型
- * @param string|\Closure $route 路由地址
- * @param array $option 路由参数
- * @param array $pattern 变量规则
- */
- public function __construct(Route $router, RuleGroup $group, $name, $route, $method = '*', $option = [], $pattern = [])
- {
- $this->router = $router;
- $this->parent = $group;
- $this->route = $route;
- $this->method = $method;
- $this->option = $option;
- $this->pattern = $pattern;
- $this->setRule($name);
- }
- /**
- * 路由规则预处理
- * @access public
- * @param string $rule 路由规则
- * @return void
- */
- public function setRule($rule)
- {
- if ('$' == substr($rule, -1, 1)) {
- // 是否完整匹配
- $rule = substr($rule, 0, -1);
- $this->option['complete_match'] = true;
- }
- $this->name($rule);
- }
- /**
- * 获取当前路由地址
- * @access public
- * @return mixed
- */
- public function getRoute()
- {
- return $this->route;
- }
- /**
- * 设置为自动路由
- * @access public
- * @return $this
- */
- public function isAuto()
- {
- $this->parent->setAutoRule($this);
- return $this;
- }
- /**
- * 设置为MISS路由
- * @access public
- * @return $this
- */
- public function isMiss()
- {
- $this->parent->setMissRule($this);
- return $this;
- }
- /**
- * 检测路由
- * @access public
- * @param Request $request 请求对象
- * @param string $url 访问地址
- * @param string $depr 路径分隔符
- * @param bool $completeMatch 路由是否完全匹配
- * @return Dispatch
- */
- public function check($request, $url, $depr = '/', $completeMatch = false)
- {
- if ($this->parent && $prefix = $this->parent->getFullName()) {
- $this->name = $prefix . ($this->name ? '/' . ltrim($this->name, '/') : '');
- }
- if ($dispatch = $this->checkCrossDomain($request)) {
- // 允许跨域
- return $dispatch;
- }
- // 检查参数有效性
- if (!$this->checkOption($this->option, $request)) {
- return false;
- }
- // 合并分组参数
- $this->mergeGroupOptions();
- $option = $this->option;
- if (!empty($option['append'])) {
- $request->route($option['append']);
- }
- // 是否区分 / 地址访问
- if (!empty($option['remove_slash']) && '/' != $this->name) {
- $this->name = rtrim($this->name, '/');
- $url = rtrim($url, '|');
- }
- // 检查前置行为
- if (isset($option['before']) && false === $this->checkBefore($option['before'])) {
- return false;
- }
- if (isset($option['ext'])) {
- // 路由ext参数 优先于系统配置的URL伪静态后缀参数
- $url = preg_replace('/\.(' . $request->ext() . ')$/i', '', $url);
- }
- return $this->checkRule($request, $url, $depr, $completeMatch, $option);
- }
- /**
- * 检测路由规则
- * @access private
- * @param Request $request 请求对象
- * @param string $url URL地址
- * @param string $depr URL分隔符(全局)
- * @param bool $completeMatch 路由是否完全匹配
- * @param array $option 路由参数
- * @return array|false
- */
- private function checkRule($request, $url, $depr, $completeMatch = false, $option = [])
- {
- // 检查完整规则定义
- if (isset($this->pattern['__url__']) && !preg_match(0 === strpos($this->pattern['__url__'], '/') ? $this->pattern['__url__'] : '/^' . $this->pattern['__url__'] . '/', str_replace('|', $depr, $url))) {
- return false;
- }
- // 检查路由的参数分隔符
- if (isset($option['param_depr'])) {
- $url = str_replace(['|', $option['param_depr']], [$depr, '|'], $url);
- }
- $len1 = substr_count($url, '|');
- $len2 = substr_count($this->name, '/');
- // 多余参数是否合并
- $merge = !empty($option['merge_extra_vars']) ? true : false;
- if ($merge && $len1 > $len2) {
- $url = str_replace('|', $depr, $url);
- $url = implode('|', explode($depr, $url, $len2 + 1));
- }
- if (isset($option['complete_match'])) {
- $completeMatch = $option['complete_match'];
- }
- if ($len1 >= $len2 || strpos($this->name, '[')) {
- // 完整匹配
- if ($completeMatch && (!$merge && $len1 != $len2 && (false === strpos($this->name, '[') || $len1 > $len2 || $len1 < $len2 - substr_count($this->name, '[')))) {
- return false;
- }
- $pattern = array_merge($this->parent->getPattern(), $this->pattern);
- if (false !== $match = $this->match($url, $pattern)) {
- // 匹配到路由规则
- return $this->parseRule($request, $this->name, $this->route, $url, $option, $match);
- }
- }
- return false;
- }
- /**
- * 检测URL和规则路由是否匹配
- * @access private
- * @param string $url URL地址
- * @param array $pattern 变量规则
- * @return array|false
- */
- private function match($url, $pattern)
- {
- $m2 = explode('/', $this->name);
- $m1 = explode('|', $url);
- $var = [];
- foreach ($m2 as $key => $val) {
- // val中定义了多个变量 <id><name>
- if (false !== strpos($val, '<') && preg_match_all('/<(\w+(\??))>/', $val, $matches)) {
- $value = [];
- $replace = [];
- foreach ($matches[1] as $name) {
- if (strpos($name, '?')) {
- $name = substr($name, 0, -1);
- $replace[] = '(' . (isset($pattern[$name]) ? $pattern[$name] : '\w+') . ')?';
- } else {
- $replace[] = '(' . (isset($pattern[$name]) ? $pattern[$name] : '\w+') . ')';
- }
- $value[] = $name;
- }
- $val = str_replace($matches[0], $replace, $val);
- if (preg_match('/^' . $val . '$/', isset($m1[$key]) ? $m1[$key] : '', $match)) {
- array_shift($match);
- foreach ($value as $k => $name) {
- if (isset($match[$k])) {
- $var[$name] = $match[$k];
- }
- }
- continue;
- } else {
- return false;
- }
- }
- if (0 === strpos($val, '[:')) {
- // 可选参数
- $val = substr($val, 1, -1);
- $optional = true;
- } else {
- $optional = false;
- }
- if (0 === strpos($val, ':')) {
- // URL变量
- $name = substr($val, 1);
- if (!$optional && !isset($m1[$key])) {
- return false;
- }
- if (isset($m1[$key]) && isset($pattern[$name])) {
- // 检查变量规则
- if ($pattern[$name] instanceof \Closure) {
- $result = call_user_func_array($pattern[$name], [$m1[$key]]);
- if (false === $result) {
- return false;
- }
- } elseif (!preg_match(0 === strpos($pattern[$name], '/') ? $pattern[$name] : '/^' . $pattern[$name] . '$/', $m1[$key])) {
- return false;
- }
- }
- $var[$name] = isset($m1[$key]) ? $m1[$key] : '';
- } elseif (!isset($m1[$key]) || 0 !== strcasecmp($val, $m1[$key])) {
- return false;
- }
- }
- // 成功匹配后返回URL中的动态变量数组
- return $var;
- }
- }
|