Url.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | ThinkPHP [ WE CAN DO IT JUST THINK ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2006~2018 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. class Url
  13. {
  14. /**
  15. * ROOT地址
  16. * @var string
  17. */
  18. protected $root;
  19. /**
  20. * 绑定检查
  21. * @var bool
  22. */
  23. protected $bindCheck;
  24. /**
  25. * 应用对象
  26. * @var App
  27. */
  28. protected $app;
  29. public function __construct(App $app)
  30. {
  31. $this->app = $app;
  32. if (is_file($app->getRuntimePath() . 'route.php')) {
  33. // 读取路由映射文件
  34. $app['route']->setName(include $app->getRuntimePath() . 'route.php');
  35. }
  36. }
  37. /**
  38. * URL生成 支持路由反射
  39. * @access public
  40. * @param string $url 路由地址
  41. * @param string|array $vars 参数(支持数组和字符串)a=val&b=val2... ['a'=>'val1', 'b'=>'val2']
  42. * @param string|bool $suffix 伪静态后缀,默认为true表示获取配置值
  43. * @param boolean|string $domain 是否显示域名 或者直接传入域名
  44. * @return string
  45. */
  46. public function build($url = '', $vars = '', $suffix = true, $domain = false)
  47. {
  48. // 解析URL
  49. if (0 === strpos($url, '[') && $pos = strpos($url, ']')) {
  50. // [name] 表示使用路由命名标识生成URL
  51. $name = substr($url, 1, $pos - 1);
  52. $url = 'name' . substr($url, $pos + 1);
  53. }
  54. if (false === strpos($url, '://') && 0 !== strpos($url, '/')) {
  55. $info = parse_url($url);
  56. $url = !empty($info['path']) ? $info['path'] : '';
  57. if (isset($info['fragment'])) {
  58. // 解析锚点
  59. $anchor = $info['fragment'];
  60. if (false !== strpos($anchor, '?')) {
  61. // 解析参数
  62. list($anchor, $info['query']) = explode('?', $anchor, 2);
  63. }
  64. if (false !== strpos($anchor, '@')) {
  65. // 解析域名
  66. list($anchor, $domain) = explode('@', $anchor, 2);
  67. }
  68. } elseif (strpos($url, '@') && false === strpos($url, '\\')) {
  69. // 解析域名
  70. list($url, $domain) = explode('@', $url, 2);
  71. }
  72. }
  73. // 解析参数
  74. if (is_string($vars)) {
  75. // aaa=1&bbb=2 转换成数组
  76. parse_str($vars, $vars);
  77. }
  78. if ($url) {
  79. $rule = $this->app['route']->getName(isset($name) ? $name : $url . (isset($info['query']) ? '?' . $info['query'] : ''));
  80. if (is_null($rule) && isset($info['query'])) {
  81. $rule = $this->app['route']->getName($url);
  82. // 解析地址里面参数 合并到vars
  83. parse_str($info['query'], $params);
  84. $vars = array_merge($params, $vars);
  85. unset($info['query']);
  86. }
  87. }
  88. if (!empty($rule) && $match = $this->getRuleUrl($rule, $vars)) {
  89. // 匹配路由命名标识
  90. $url = $match[0];
  91. if (!empty($match[1])) {
  92. $host = $this->app['config']->get('app_host') ?: $this->app['request']->host();
  93. if ($domain || $match[1] != $host) {
  94. $domain = $match[1];
  95. }
  96. }
  97. if (!is_null($match[2])) {
  98. $suffix = $match[2];
  99. }
  100. } elseif (!empty($rule) && isset($name)) {
  101. throw new \InvalidArgumentException('route name not exists:' . $name);
  102. } else {
  103. // 检查别名路由
  104. $alias = $this->app['route']->getAlias();
  105. $matchAlias = false;
  106. if ($alias) {
  107. // 别名路由解析
  108. foreach ($alias as $key => $val) {
  109. if (is_array($val)) {
  110. $val = $val[0];
  111. }
  112. if (0 === strpos($url, $val)) {
  113. $url = $key . substr($url, strlen($val));
  114. $matchAlias = true;
  115. break;
  116. }
  117. }
  118. }
  119. if (!$matchAlias) {
  120. // 路由标识不存在 直接解析
  121. $url = $this->parseUrl($url);
  122. }
  123. if (isset($info['query'])) {
  124. // 解析地址里面参数 合并到vars
  125. parse_str($info['query'], $params);
  126. $vars = array_merge($params, $vars);
  127. }
  128. }
  129. // 检测URL绑定
  130. if (!$this->bindCheck) {
  131. $bind = $this->app['route']->getBind();
  132. if ($bind && 0 === strpos($url, $bind)) {
  133. $url = substr($url, strlen($bind) + 1);
  134. }
  135. }
  136. // 还原URL分隔符
  137. $depr = $this->app['config']->get('pathinfo_depr');
  138. $url = str_replace('/', $depr, $url);
  139. // URL后缀
  140. if ('/' == substr($url, -1) || '' == $url) {
  141. $suffix = '';
  142. } else {
  143. $suffix = $this->parseSuffix($suffix);
  144. }
  145. // 锚点
  146. $anchor = !empty($anchor) ? '#' . $anchor : '';
  147. // 参数组装
  148. if (!empty($vars)) {
  149. // 添加参数
  150. if ($this->app['config']->get('url_common_param')) {
  151. $vars = http_build_query($vars);
  152. $url .= $suffix . '?' . $vars . $anchor;
  153. } else {
  154. $paramType = $this->app['config']->get('url_param_type');
  155. foreach ($vars as $var => $val) {
  156. if ('' !== trim($val)) {
  157. if ($paramType) {
  158. $url .= $depr . urlencode($val);
  159. } else {
  160. $url .= $depr . $var . $depr . urlencode($val);
  161. }
  162. }
  163. }
  164. $url .= $suffix . $anchor;
  165. }
  166. } else {
  167. $url .= $suffix . $anchor;
  168. }
  169. // 检测域名
  170. $domain = $this->parseDomain($url, $domain);
  171. // URL组装
  172. $url = $domain . rtrim($this->root ?: $this->app['request']->root(), '/') . '/' . ltrim($url, '/');
  173. $this->bindCheck = false;
  174. return $url;
  175. }
  176. // 直接解析URL地址
  177. protected function parseUrl($url)
  178. {
  179. $request = $this->app['request'];
  180. if (0 === strpos($url, '/')) {
  181. // 直接作为路由地址解析
  182. $url = substr($url, 1);
  183. } elseif (false !== strpos($url, '\\')) {
  184. // 解析到类
  185. $url = ltrim(str_replace('\\', '/', $url), '/');
  186. } elseif (0 === strpos($url, '@')) {
  187. // 解析到控制器
  188. $url = substr($url, 1);
  189. } else {
  190. // 解析到 模块/控制器/操作
  191. $module = $request->module();
  192. $module = $module ? $module . '/' : '';
  193. $controller = $request->controller();
  194. if ('' == $url) {
  195. $action = $request->action();
  196. } else {
  197. $path = explode('/', $url);
  198. $action = array_pop($path);
  199. $controller = empty($path) ? $controller : array_pop($path);
  200. $module = empty($path) ? $module : array_pop($path) . '/';
  201. }
  202. if ($this->app['config']->get('url_convert')) {
  203. $action = strtolower($action);
  204. $controller = Loader::parseName($controller);
  205. }
  206. $url = $module . $controller . '/' . $action;
  207. }
  208. return $url;
  209. }
  210. // 检测域名
  211. protected function parseDomain(&$url, $domain)
  212. {
  213. if (!$domain) {
  214. return '';
  215. }
  216. if (true === $domain) {
  217. // 自动判断域名
  218. $domain = $this->app['config']->get('app_host') ?: $this->app['request']->host();
  219. $rootDomain = $this->app['config']->get('url_domain_root');
  220. $domains = $this->app['route']->getDomains();
  221. if ($domains) {
  222. $route_domain = array_keys($domains);
  223. foreach ($route_domain as $domain_prefix) {
  224. if (0 === strpos($domain_prefix, '*.') && strpos($domain, ltrim($domain_prefix, '*.')) !== false) {
  225. foreach ($domains as $key => $rule) {
  226. $rule = is_array($rule) ? $rule[0] : $rule;
  227. if (is_string($rule) && false === strpos($key, '*') && 0 === strpos($url, $rule)) {
  228. $url = ltrim($url, $rule);
  229. $domain = $key;
  230. // 生成对应子域名
  231. if (!empty($rootDomain)) {
  232. $domain .= $rootDomain;
  233. }
  234. break;
  235. } elseif (false !== strpos($key, '*')) {
  236. if (!empty($rootDomain)) {
  237. $domain .= $rootDomain;
  238. }
  239. break;
  240. }
  241. }
  242. }
  243. }
  244. }
  245. }
  246. if (false !== strpos($domain, '://')) {
  247. $scheme = '';
  248. } else {
  249. $scheme = $this->app['request']->isSsl() || $this->app['config']->get('is_https') ? 'https://' : 'http://';
  250. }
  251. return $scheme . $domain;
  252. }
  253. // 解析URL后缀
  254. protected function parseSuffix($suffix)
  255. {
  256. if ($suffix) {
  257. $suffix = true === $suffix ? $this->app['config']->get('url_html_suffix') : $suffix;
  258. if ($pos = strpos($suffix, '|')) {
  259. $suffix = substr($suffix, 0, $pos);
  260. }
  261. }
  262. return (empty($suffix) || 0 === strpos($suffix, '.')) ? $suffix : '.' . $suffix;
  263. }
  264. // 匹配路由地址
  265. public function getRuleUrl($rule, &$vars = [])
  266. {
  267. foreach ($rule as $item) {
  268. list($url, $pattern, $domain, $suffix) = $item;
  269. if (empty($pattern)) {
  270. return [rtrim($url, '$'), $domain, $suffix];
  271. }
  272. $type = $this->app['config']->get('url_common_param');
  273. foreach ($pattern as $key => $val) {
  274. if (isset($vars[$key])) {
  275. $url = str_replace(['[:' . $key . ']', '[:' . $key . '$]', '<' . $key . '?>$', '<' . $key . '?>', ':' . $key . '$', ':' . $key . '', '<' . $key . '>$', '<' . $key . '>'], $type ? $vars[$key] : urlencode($vars[$key]), $url);
  276. unset($vars[$key]);
  277. $result = [$url, $domain, $suffix];
  278. } elseif (2 == $val) {
  279. $url = str_replace(['/[:' . $key . ']', '/[:' . $key . '$]', '[:' . $key . ']', '[:' . $key . '$]', '<' . $key . '?>$', '<' . $key . '?>'], '', $url);
  280. $result = [$url, $domain, $suffix];
  281. } else {
  282. break;
  283. }
  284. }
  285. if (isset($result)) {
  286. return $result;
  287. }
  288. }
  289. return false;
  290. }
  291. // 指定当前生成URL地址的root
  292. public function root($root)
  293. {
  294. $this->root = $root;
  295. $this->app['request']->root($root);
  296. }
  297. }