作者 sgj
1 个管道 的构建 通过 耗费 5 秒

Merge remote-tracking branch 'origin/master'

1 <?php return array ( 1 <?php return array (
  2 + '子分类/:id' =>
  3 + array (
  4 + 0 => 'portal/Article/index?cid=21',
  5 + 1 =>
  6 + array (
  7 + ),
  8 + 2 =>
  9 + array (
  10 + 'id' => '\d+',
  11 + 'cid' => '\d+',
  12 + ),
  13 + ),
  14 + '三级分类2/:id' =>
  15 + array (
  16 + 0 => 'portal/Article/index?cid=22',
  17 + 1 =>
  18 + array (
  19 + ),
  20 + 2 =>
  21 + array (
  22 + 'id' => '\d+',
  23 + 'cid' => '\d+',
  24 + ),
  25 + ),
  26 + '子分嗯呢/:id' =>
  27 + array (
  28 + 0 => 'portal/Article/index?cid=23',
  29 + 1 =>
  30 + array (
  31 + ),
  32 + 2 =>
  33 + array (
  34 + 'id' => '\d+',
  35 + 'cid' => '\d+',
  36 + ),
  37 + ),
  38 + '子分类' =>
  39 + array (
  40 + 0 => 'portal/List/index?id=21',
  41 + 1 =>
  42 + array (
  43 + ),
  44 + 2 =>
  45 + array (
  46 + 'id' => '\d+',
  47 + ),
  48 + ),
  49 + '三级分类2' =>
  50 + array (
  51 + 0 => 'portal/List/index?id=22',
  52 + 1 =>
  53 + array (
  54 + ),
  55 + 2 =>
  56 + array (
  57 + 'id' => '\d+',
  58 + ),
  59 + ),
  60 + '子分嗯呢' =>
  61 + array (
  62 + 0 => 'portal/List/index?id=23',
  63 + 1 =>
  64 + array (
  65 + ),
  66 + 2 =>
  67 + array (
  68 + 'id' => '\d+',
  69 + ),
  70 + ),
2 ); 71 );
1 -<?php  
2 -// +----------------------------------------------------------------------  
3 -// | ThinkPHP [ WE CAN DO IT JUST THINK ]  
4 -// +----------------------------------------------------------------------  
5 -// | Copyright (c) 2006~2017 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 -namespace think;  
13 -  
14 -use think\exception\ClassNotFoundException;  
15 -use think\exception\HttpException;  
16 -use think\exception\HttpResponseException;  
17 -use think\exception\RouteNotFoundException;  
18 -  
19 -/**  
20 - * App 应用管理  
21 - * @author liu21st <liu21st@gmail.com>  
22 - */  
23 -class App  
24 -{  
25 - /**  
26 - * @var bool 是否初始化过  
27 - */  
28 - protected static $init = false;  
29 -  
30 - /**  
31 - * @var string 当前模块路径  
32 - */  
33 - public static $modulePath;  
34 -  
35 - /**  
36 - * @var bool 应用调试模式  
37 - */  
38 - public static $debug = true;  
39 -  
40 - /**  
41 - * @var string 应用类库命名空间  
42 - */  
43 - public static $namespace = 'app';  
44 -  
45 - /**  
46 - * @var bool 应用类库后缀  
47 - */  
48 - public static $suffix = false;  
49 -  
50 - /**  
51 - * @var bool 应用路由检测  
52 - */  
53 - protected static $routeCheck;  
54 -  
55 - /**  
56 - * @var bool 严格路由检测  
57 - */  
58 - protected static $routeMust;  
59 -  
60 - protected static $dispatch;  
61 - protected static $file = [];  
62 -  
63 - /**  
64 - * 执行应用程序  
65 - * @access public  
66 - * @param Request $request Request对象  
67 - * @return Response  
68 - * @throws Exception  
69 - */  
70 - public static function run(Request $request = null)  
71 - {  
72 - is_null($request) && $request = Request::instance();  
73 -  
74 - try {  
75 - $config = self::initCommon();  
76 - if (defined('BIND_MODULE')) {  
77 - // 模块/控制器绑定  
78 - BIND_MODULE && Route::bind(BIND_MODULE);  
79 - } elseif ($config['auto_bind_module']) {  
80 - // 入口自动绑定  
81 - $name = pathinfo($request->baseFile(), PATHINFO_FILENAME);  
82 - if ($name && 'index' != $name && is_dir(APP_PATH . $name)) {  
83 - Route::bind($name);  
84 - }  
85 - }  
86 -  
87 - $request->filter($config['default_filter']);  
88 -  
89 - // 默认语言  
90 - Lang::range($config['default_lang']);  
91 - if ($config['lang_switch_on']) {  
92 - // 开启多语言机制 检测当前语言  
93 - Lang::detect();  
94 - }  
95 - $request->langset(Lang::range());  
96 -  
97 - // 加载系统语言包  
98 - Lang::load([  
99 - THINK_PATH . 'lang' . DS . $request->langset() . EXT,  
100 - APP_PATH . 'lang' . DS . $request->langset() . EXT,  
101 - ]);  
102 -  
103 - // 获取应用调度信息  
104 - $dispatch = self::$dispatch;  
105 - if (empty($dispatch)) {  
106 - // 进行URL路由检测  
107 - $dispatch = self::routeCheck($request, $config);  
108 - }  
109 - // 记录当前调度信息  
110 - $request->dispatch($dispatch);  
111 -  
112 - // 记录路由和请求信息  
113 - if (self::$debug) {  
114 - Log::record('[ ROUTE ] ' . var_export($dispatch, true), 'info');  
115 - Log::record('[ HEADER ] ' . var_export($request->header(), true), 'info');  
116 - Log::record('[ PARAM ] ' . var_export($request->param(), true), 'info');  
117 - }  
118 -  
119 - // 监听app_begin  
120 - Hook::listen('app_begin', $dispatch);  
121 - // 请求缓存检查  
122 - $request->cache($config['request_cache'], $config['request_cache_expire'], $config['request_cache_except']);  
123 -  
124 - $data = self::exec($dispatch, $config);  
125 - } catch (HttpResponseException $exception) {  
126 - $data = $exception->getResponse();  
127 - }  
128 -  
129 - // 清空类的实例化  
130 - Loader::clearInstance();  
131 -  
132 - // 输出数据到客户端  
133 - if ($data instanceof Response) {  
134 - $response = $data;  
135 - } elseif (!is_null($data)) {  
136 - // 默认自动识别响应输出类型  
137 - $isAjax = $request->isAjax();  
138 - $type = $isAjax ? Config::get('default_ajax_return') : Config::get('default_return_type');  
139 - $response = Response::create($data, $type);  
140 - } else {  
141 - $response = Response::create();  
142 - }  
143 -  
144 - // 监听app_end  
145 - Hook::listen('app_end', $response);  
146 -  
147 - return $response;  
148 - }  
149 -  
150 - /**  
151 - * 设置当前请求的调度信息  
152 - * @access public  
153 - * @param array|string $dispatch 调度信息  
154 - * @param string $type 调度类型  
155 - * @return void  
156 - */  
157 - public static function dispatch($dispatch, $type = 'module')  
158 - {  
159 - self::$dispatch = ['type' => $type, $type => $dispatch];  
160 - }  
161 -  
162 - /**  
163 - * 执行函数或者闭包方法 支持参数调用  
164 - * @access public  
165 - * @param string|array|\Closure $function 函数或者闭包  
166 - * @param array $vars 变量  
167 - * @return mixed  
168 - */  
169 - public static function invokeFunction($function, $vars = [])  
170 - {  
171 - $reflect = new \ReflectionFunction($function);  
172 - $args = self::bindParams($reflect, $vars);  
173 - // 记录执行信息  
174 - self::$debug && Log::record('[ RUN ] ' . $reflect->__toString(), 'info');  
175 - return $reflect->invokeArgs($args);  
176 - }  
177 -  
178 - /**  
179 - * 调用反射执行类的方法 支持参数绑定  
180 - * @access public  
181 - * @param string|array $method 方法  
182 - * @param array $vars 变量  
183 - * @return mixed  
184 - */  
185 - public static function invokeMethod($method, $vars = [])  
186 - {  
187 - if (is_array($method)) {  
188 - $class = is_object($method[0]) ? $method[0] : self::invokeClass($method[0]);  
189 - $reflect = new \ReflectionMethod($class, $method[1]);  
190 - } else {  
191 - // 静态方法  
192 - $reflect = new \ReflectionMethod($method);  
193 - }  
194 - $args = self::bindParams($reflect, $vars);  
195 -  
196 - self::$debug && Log::record('[ RUN ] ' . $reflect->class . '->' . $reflect->name . '[ ' . $reflect->getFileName() . ' ]', 'info');  
197 - return $reflect->invokeArgs(isset($class) ? $class : null, $args);  
198 - }  
199 -  
200 - /**  
201 - * 调用反射执行类的实例化 支持依赖注入  
202 - * @access public  
203 - * @param string $class 类名  
204 - * @param array $vars 变量  
205 - * @return mixed  
206 - */  
207 - public static function invokeClass($class, $vars = [])  
208 - {  
209 - $reflect = new \ReflectionClass($class);  
210 - $constructor = $reflect->getConstructor();  
211 - if ($constructor) {  
212 - $args = self::bindParams($constructor, $vars);  
213 - } else {  
214 - $args = [];  
215 - }  
216 - return $reflect->newInstanceArgs($args);  
217 - }  
218 -  
219 - /**  
220 - * 绑定参数  
221 - * @access private  
222 - * @param \ReflectionMethod|\ReflectionFunction $reflect 反射类  
223 - * @param array $vars 变量  
224 - * @return array  
225 - */  
226 - private static function bindParams($reflect, $vars = [])  
227 - {  
228 - if (empty($vars)) {  
229 - // 自动获取请求变量  
230 - if (Config::get('url_param_type')) {  
231 - $vars = Request::instance()->route();  
232 - } else {  
233 - $vars = Request::instance()->param();  
234 - }  
235 - }  
236 - $args = [];  
237 - if ($reflect->getNumberOfParameters() > 0) {  
238 - // 判断数组类型 数字数组时按顺序绑定参数  
239 - reset($vars);  
240 - $type = key($vars) === 0 ? 1 : 0;  
241 - $params = $reflect->getParameters();  
242 - foreach ($params as $param) {  
243 - $args[] = self::getParamValue($param, $vars, $type);  
244 - }  
245 - }  
246 - return $args;  
247 - }  
248 -  
249 - /**  
250 - * 获取参数值  
251 - * @access private  
252 - * @param \ReflectionParameter $param  
253 - * @param array $vars 变量  
254 - * @param string $type  
255 - * @return array  
256 - */  
257 - private static function getParamValue($param, &$vars, $type)  
258 - {  
259 - $name = $param->getName();  
260 - $class = $param->getClass();  
261 - if ($class) {  
262 - $className = $class->getName();  
263 - $bind = Request::instance()->$name;  
264 - if ($bind instanceof $className) {  
265 - $result = $bind;  
266 - } else {  
267 - if (method_exists($className, 'invoke')) {  
268 - $method = new \ReflectionMethod($className, 'invoke');  
269 - if ($method->isPublic() && $method->isStatic()) {  
270 - return $className::invoke(Request::instance());  
271 - }  
272 - }  
273 - $result = method_exists($className, 'instance') ? $className::instance() : new $className;  
274 - }  
275 - } elseif (1 == $type && !empty($vars)) {  
276 - $result = array_shift($vars);  
277 - } elseif (0 == $type && isset($vars[$name])) {  
278 - $result = $vars[$name];  
279 - } elseif ($param->isDefaultValueAvailable()) {  
280 - $result = $param->getDefaultValue();  
281 - } else {  
282 - throw new \InvalidArgumentException('method param miss:' . $name);  
283 - }  
284 - return $result;  
285 - }  
286 -  
287 - protected static function exec($dispatch, $config)  
288 - {  
289 - switch ($dispatch['type']) {  
290 - case 'redirect':  
291 - // 执行重定向跳转  
292 - $data = Response::create($dispatch['url'], 'redirect')->code($dispatch['status']);  
293 - break;  
294 - case 'module':  
295 - // 模块/控制器/操作  
296 - $data = self::module($dispatch['module'], $config, isset($dispatch['convert']) ? $dispatch['convert'] : null);  
297 - break;  
298 - case 'controller':  
299 - // 执行控制器操作  
300 - $vars = array_merge(Request::instance()->param(), $dispatch['var']);  
301 - $data = Loader::action($dispatch['controller'], $vars, $config['url_controller_layer'], $config['controller_suffix']);  
302 - break;  
303 - case 'method':  
304 - // 执行回调方法  
305 - $vars = array_merge(Request::instance()->param(), $dispatch['var']);  
306 - $data = self::invokeMethod($dispatch['method'], $vars);  
307 - break;  
308 - case 'function':  
309 - // 执行闭包  
310 - $data = self::invokeFunction($dispatch['function']);  
311 - break;  
312 - case 'response':  
313 - $data = $dispatch['response'];  
314 - break;  
315 - default:  
316 - throw new \InvalidArgumentException('dispatch type not support');  
317 - }  
318 - return $data;  
319 - }  
320 -  
321 - /**  
322 - * 执行模块  
323 - * @access public  
324 - * @param array $result 模块/控制器/操作  
325 - * @param array $config 配置参数  
326 - * @param bool $convert 是否自动转换控制器和操作名  
327 - * @return mixed  
328 - */  
329 - public static function module($result, $config, $convert = null)  
330 - {  
331 - if (is_string($result)) {  
332 - $result = explode('/', $result);  
333 - }  
334 - $request = Request::instance();  
335 - if ($config['app_multi_module']) {  
336 - // 多模块部署  
337 - $module = strip_tags(strtolower($result[0] ?: $config['default_module']));  
338 - $bind = Route::getBind('module');  
339 - $available = false;  
340 - if ($bind) {  
341 - // 绑定模块  
342 - list($bindModule) = explode('/', $bind);  
343 - if (empty($result[0])) {  
344 - $module = $bindModule;  
345 - $available = true;  
346 - } elseif ($module == $bindModule) {  
347 - $available = true;  
348 - }  
349 - } elseif (!in_array($module, $config['deny_module_list']) && is_dir(APP_PATH . $module)) {  
350 - $available = true;  
351 - }  
352 -  
353 - // 模块初始化  
354 - if ($module && $available) {  
355 - // 初始化模块  
356 - $request->module($module);  
357 - $config = self::init($module);  
358 - // 模块请求缓存检查  
359 - $request->cache($config['request_cache'], $config['request_cache_expire'], $config['request_cache_except']);  
360 - } else {  
361 - throw new HttpException(404, 'module not exists:' . $module);  
362 - }  
363 - } else {  
364 - // 单一模块部署  
365 - $module = '';  
366 - $request->module($module);  
367 - }  
368 - // 当前模块路径  
369 - App::$modulePath = APP_PATH . ($module ? $module . DS : '');  
370 -  
371 - // 是否自动转换控制器和操作名  
372 - $convert = is_bool($convert) ? $convert : $config['url_convert'];  
373 -  
374 - // 获取控制器名  
375 - $controller = strip_tags($result[1] ?: $config['default_controller']);  
376 - $controller = $convert ? strtolower($controller) : $controller;  
377 -  
378 - //修改远程执行漏洞问题  
379 - if (!preg_match('/^[A-Za-z](\w|\.)*$/', $controller)) {  
380 - throw new HttpException(404, 'controller not exists:' . $controller);  
381 - }  
382 -  
383 - // 获取操作名  
384 - $actionName = strip_tags($result[2] ?: $config['default_action']);  
385 - $actionName = $convert ? strtolower($actionName) : $actionName;  
386 -  
387 - // 设置当前请求的控制器、操作  
388 - $request->controller(Loader::parseName($controller, 1))->action($actionName);  
389 -  
390 - // 监听module_init  
391 - Hook::listen('module_init', $request);  
392 -  
393 - try {  
394 - $instance = Loader::controller($controller, $config['url_controller_layer'], $config['controller_suffix'], $config['empty_controller']);  
395 - } catch (ClassNotFoundException $e) {  
396 - throw new HttpException(404, 'controller not exists:' . $e->getClass());  
397 - }  
398 -  
399 - // 获取当前操作名  
400 - $action = $actionName . $config['action_suffix'];  
401 -  
402 - $vars = [];  
403 - if (is_callable([$instance, $action])) {  
404 - // 执行操作方法  
405 - $call = [$instance, $action];  
406 - } elseif (is_callable([$instance, '_empty'])) {  
407 - // 空操作  
408 - $call = [$instance, '_empty'];  
409 - $vars = [$actionName];  
410 - } else {  
411 - // 操作不存在  
412 - throw new HttpException(404, 'method not exists:' . get_class($instance) . '->' . $action . '()');  
413 - }  
414 -  
415 - Hook::listen('action_begin', $call);  
416 -  
417 - return self::invokeMethod($call, $vars);  
418 - }  
419 -  
420 - /**  
421 - * 初始化应用  
422 - */  
423 - public static function initCommon()  
424 - {  
425 - if (empty(self::$init)) {  
426 - if (defined('APP_NAMESPACE')) {  
427 - self::$namespace = APP_NAMESPACE;  
428 - }  
429 - Loader::addNamespace(self::$namespace, APP_PATH);  
430 -  
431 - // 初始化应用  
432 - $config = self::init();  
433 - self::$suffix = $config['class_suffix'];  
434 -  
435 - // 应用调试模式  
436 - self::$debug = Env::get('app_debug', Config::get('app_debug'));  
437 - if (!self::$debug) {  
438 - ini_set('display_errors', 'Off');  
439 - } elseif (!IS_CLI) {  
440 - //重新申请一块比较大的buffer  
441 - if (ob_get_level() > 0) {  
442 - $output = ob_get_clean();  
443 - }  
444 - ob_start();  
445 - if (!empty($output)) {  
446 - echo $output;  
447 - }  
448 - }  
449 -  
450 - if (!empty($config['root_namespace'])) {  
451 - Loader::addNamespace($config['root_namespace']);  
452 - }  
453 -  
454 - // 加载额外文件  
455 - if (!empty($config['extra_file_list'])) {  
456 - foreach ($config['extra_file_list'] as $file) {  
457 - $file = strpos($file, '.') ? $file : APP_PATH . $file . EXT;  
458 - if (is_file($file) && !isset(self::$file[$file])) {  
459 - include $file;  
460 - self::$file[$file] = true;  
461 - }  
462 - }  
463 - }  
464 -  
465 - // 设置系统时区  
466 - date_default_timezone_set($config['default_timezone']);  
467 -  
468 - // 监听app_init  
469 - Hook::listen('app_init');  
470 -  
471 - self::$init = true;  
472 - }  
473 - return Config::get();  
474 - }  
475 -  
476 - /**  
477 - * 初始化应用或模块  
478 - * @access public  
479 - * @param string $module 模块名  
480 - * @return array  
481 - */  
482 - private static function init($module = '')  
483 - {  
484 - // 定位模块目录  
485 - $module = $module ? $module . DS : '';  
486 -  
487 - // 加载初始化文件  
488 - if (is_file(APP_PATH . $module . 'init' . EXT)) {  
489 - include APP_PATH . $module . 'init' . EXT;  
490 - } elseif (is_file(RUNTIME_PATH . $module . 'init' . EXT)) {  
491 - include RUNTIME_PATH . $module . 'init' . EXT;  
492 - } else {  
493 - $path = APP_PATH . $module;  
494 - // 加载模块配置  
495 - $config = Config::load(CONF_PATH . $module . 'config' . CONF_EXT);  
496 - // 读取数据库配置文件  
497 - $filename = CONF_PATH . $module . 'database' . CONF_EXT;  
498 - Config::load($filename, 'database');  
499 - // 读取扩展配置文件  
500 - if (is_dir(CONF_PATH . $module . 'extra')) {  
501 - $dir = CONF_PATH . $module . 'extra';  
502 - $files = scandir($dir);  
503 - foreach ($files as $file) {  
504 - if ('.' . pathinfo($file, PATHINFO_EXTENSION) === CONF_EXT) {  
505 - $filename = $dir . DS . $file;  
506 - Config::load($filename, pathinfo($file, PATHINFO_FILENAME));  
507 - }  
508 - }  
509 - }  
510 -  
511 - // 加载应用状态配置  
512 - if ($config['app_status']) {  
513 - $config = Config::load(CONF_PATH . $module . $config['app_status'] . CONF_EXT);  
514 - }  
515 -  
516 - // 加载行为扩展文件  
517 - if (is_file(CONF_PATH . $module . 'tags' . EXT)) {  
518 - Hook::import(include CONF_PATH . $module . 'tags' . EXT);  
519 - }  
520 -  
521 - // 加载公共文件  
522 - if (is_file($path . 'common' . EXT)) {  
523 - include $path . 'common' . EXT;  
524 - }  
525 -  
526 - // 加载当前模块语言包  
527 - if ($module) {  
528 - Lang::load($path . 'lang' . DS . Request::instance()->langset() . EXT);  
529 - }  
530 - }  
531 - return Config::get();  
532 - }  
533 -  
534 - /**  
535 - * URL路由检测(根据PATH_INFO)  
536 - * @access public  
537 - * @param \think\Request $request  
538 - * @param array $config  
539 - * @return array  
540 - * @throws \think\Exception  
541 - */  
542 - public static function routeCheck($request, array $config)  
543 - {  
544 - $path = $request->path();  
545 - $depr = $config['pathinfo_depr'];  
546 - $result = false;  
547 - // 路由检测  
548 - $check = !is_null(self::$routeCheck) ? self::$routeCheck : $config['url_route_on'];  
549 - if ($check) {  
550 - // 开启路由  
551 - if (is_file(RUNTIME_PATH . 'route.php')) {  
552 - // 读取路由缓存  
553 - $rules = include RUNTIME_PATH . 'route.php';  
554 - if (is_array($rules)) {  
555 - Route::rules($rules);  
556 - }  
557 - } else {  
558 - $files = $config['route_config_file'];  
559 - foreach ($files as $file) {  
560 - if (is_file(CONF_PATH . $file . CONF_EXT)) {  
561 - // 导入路由配置  
562 - $rules = include CONF_PATH . $file . CONF_EXT;  
563 - if (is_array($rules)) {  
564 - Route::import($rules);  
565 - }  
566 - }  
567 - }  
568 - }  
569 -  
570 - // 路由检测(根据路由定义返回不同的URL调度)  
571 - $result = Route::check($request, $path, $depr, $config['url_domain_deploy']);  
572 - $must = !is_null(self::$routeMust) ? self::$routeMust : $config['url_route_must'];  
573 - if ($must && false === $result) {  
574 - // 路由无效  
575 - throw new RouteNotFoundException();  
576 - }  
577 - }  
578 - if (false === $result) {  
579 - // 路由无效 解析模块/控制器/操作/参数... 支持控制器自动搜索  
580 - $result = Route::parseUrl($path, $depr, $config['controller_auto_search']);  
581 - }  
582 - return $result;  
583 - }  
584 -  
585 - /**  
586 - * 设置应用的路由检测机制  
587 - * @access public  
588 - * @param bool $route 是否需要检测路由  
589 - * @param bool $must 是否强制检测路由  
590 - * @return void  
591 - */  
592 - public static function route($route, $must = false)  
593 - {  
594 - self::$routeCheck = $route;  
595 - self::$routeMust = $must;  
596 - }  
597 -} 1 +<?php
  2 +// +----------------------------------------------------------------------
  3 +// | ThinkPHP [ WE CAN DO IT JUST THINK ]
  4 +// +----------------------------------------------------------------------
  5 +// | Copyright (c) 2006~2017 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 +namespace think;
  13 +
  14 +use think\exception\ClassNotFoundException;
  15 +use think\exception\HttpException;
  16 +use think\exception\HttpResponseException;
  17 +use think\exception\RouteNotFoundException;
  18 +
  19 +/**
  20 + * App 应用管理
  21 + * @author liu21st <liu21st@gmail.com>
  22 + */
  23 +class App
  24 +{
  25 + /**
  26 + * @var bool 是否初始化过
  27 + */
  28 + protected static $init = false;
  29 +
  30 + /**
  31 + * @var string 当前模块路径
  32 + */
  33 + public static $modulePath;
  34 +
  35 + /**
  36 + * @var bool 应用调试模式
  37 + */
  38 + public static $debug = true;
  39 +
  40 + /**
  41 + * @var string 应用类库命名空间
  42 + */
  43 + public static $namespace = 'app';
  44 +
  45 + /**
  46 + * @var bool 应用类库后缀
  47 + */
  48 + public static $suffix = false;
  49 +
  50 + /**
  51 + * @var bool 应用路由检测
  52 + */
  53 + protected static $routeCheck;
  54 +
  55 + /**
  56 + * @var bool 严格路由检测
  57 + */
  58 + protected static $routeMust;
  59 +
  60 + protected static $dispatch;
  61 + protected static $file = [];
  62 +
  63 + /**
  64 + * 执行应用程序
  65 + * @access public
  66 + * @param Request $request Request对象
  67 + * @return Response
  68 + * @throws Exception
  69 + */
  70 + public static function run(Request $request = null)
  71 + {
  72 + is_null($request) && $request = Request::instance();
  73 +
  74 + try {
  75 + $config = self::initCommon();
  76 + if (defined('BIND_MODULE')) {
  77 + // 模块/控制器绑定
  78 + BIND_MODULE && Route::bind(BIND_MODULE);
  79 + } elseif ($config['auto_bind_module']) {
  80 + // 入口自动绑定
  81 + $name = pathinfo($request->baseFile(), PATHINFO_FILENAME);
  82 + if ($name && 'index' != $name && is_dir(APP_PATH . $name)) {
  83 + Route::bind($name);
  84 + }
  85 + }
  86 +
  87 + $request->filter($config['default_filter']);
  88 +
  89 + // 默认语言
  90 + Lang::range($config['default_lang']);
  91 + if ($config['lang_switch_on']) {
  92 + // 开启多语言机制 检测当前语言
  93 + Lang::detect();
  94 + }
  95 + $request->langset(Lang::range());
  96 +
  97 + // 加载系统语言包
  98 + Lang::load([
  99 + THINK_PATH . 'lang' . DS . $request->langset() . EXT,
  100 + APP_PATH . 'lang' . DS . $request->langset() . EXT,
  101 + ]);
  102 +
  103 + // 获取应用调度信息
  104 + $dispatch = self::$dispatch;
  105 + if (empty($dispatch)) {
  106 + // 进行URL路由检测
  107 + $dispatch = self::routeCheck($request, $config);
  108 + }
  109 + // 记录当前调度信息
  110 + $request->dispatch($dispatch);
  111 +
  112 + // 记录路由和请求信息
  113 + if (self::$debug) {
  114 + Log::record('[ ROUTE ] ' . var_export($dispatch, true), 'info');
  115 + Log::record('[ HEADER ] ' . var_export($request->header(), true), 'info');
  116 + Log::record('[ PARAM ] ' . var_export($request->param(), true), 'info');
  117 + }
  118 +
  119 + // 监听app_begin
  120 + Hook::listen('app_begin', $dispatch);
  121 + // 请求缓存检查
  122 + $request->cache($config['request_cache'], $config['request_cache_expire'], $config['request_cache_except']);
  123 +
  124 + $data = self::exec($dispatch, $config);
  125 + } catch (HttpResponseException $exception) {
  126 + $data = $exception->getResponse();
  127 + }
  128 +
  129 + // 清空类的实例化
  130 + Loader::clearInstance();
  131 +
  132 + // 输出数据到客户端
  133 + if ($data instanceof Response) {
  134 + $response = $data;
  135 + } elseif (!is_null($data)) {
  136 + // 默认自动识别响应输出类型
  137 + $isAjax = $request->isAjax();
  138 + $type = $isAjax ? Config::get('default_ajax_return') : Config::get('default_return_type');
  139 + $response = Response::create($data, $type);
  140 + } else {
  141 + $response = Response::create();
  142 + }
  143 +
  144 + // 监听app_end
  145 + Hook::listen('app_end', $response);
  146 +
  147 + return $response;
  148 + }
  149 +
  150 + /**
  151 + * 设置当前请求的调度信息
  152 + * @access public
  153 + * @param array|string $dispatch 调度信息
  154 + * @param string $type 调度类型
  155 + * @return void
  156 + */
  157 + public static function dispatch($dispatch, $type = 'module')
  158 + {
  159 + self::$dispatch = ['type' => $type, $type => $dispatch];
  160 + }
  161 +
  162 + /**
  163 + * 执行函数或者闭包方法 支持参数调用
  164 + * @access public
  165 + * @param string|array|\Closure $function 函数或者闭包
  166 + * @param array $vars 变量
  167 + * @return mixed
  168 + */
  169 + public static function invokeFunction($function, $vars = [])
  170 + {
  171 + $reflect = new \ReflectionFunction($function);
  172 + $args = self::bindParams($reflect, $vars);
  173 + // 记录执行信息
  174 + self::$debug && Log::record('[ RUN ] ' . $reflect->__toString(), 'info');
  175 + return $reflect->invokeArgs($args);
  176 + }
  177 +
  178 + /**
  179 + * 调用反射执行类的方法 支持参数绑定
  180 + * @access public
  181 + * @param string|array $method 方法
  182 + * @param array $vars 变量
  183 + * @return mixed
  184 + */
  185 + public static function invokeMethod($method, $vars = [])
  186 + {
  187 + if (is_array($method)) {
  188 + $class = is_object($method[0]) ? $method[0] : self::invokeClass($method[0]);
  189 + $reflect = new \ReflectionMethod($class, $method[1]);
  190 + } else {
  191 + // 静态方法
  192 + $reflect = new \ReflectionMethod($method);
  193 + }
  194 + $args = self::bindParams($reflect, $vars);
  195 +
  196 + self::$debug && Log::record('[ RUN ] ' . $reflect->class . '->' . $reflect->name . '[ ' . $reflect->getFileName() . ' ]', 'info');
  197 + return $reflect->invokeArgs(isset($class) ? $class : null, $args);
  198 + }
  199 +
  200 + /**
  201 + * 调用反射执行类的实例化 支持依赖注入
  202 + * @access public
  203 + * @param string $class 类名
  204 + * @param array $vars 变量
  205 + * @return mixed
  206 + */
  207 + public static function invokeClass($class, $vars = [])
  208 + {
  209 + $reflect = new \ReflectionClass($class);
  210 + $constructor = $reflect->getConstructor();
  211 + if ($constructor) {
  212 + $args = self::bindParams($constructor, $vars);
  213 + } else {
  214 + $args = [];
  215 + }
  216 + return $reflect->newInstanceArgs($args);
  217 + }
  218 +
  219 + /**
  220 + * 绑定参数
  221 + * @access private
  222 + * @param \ReflectionMethod|\ReflectionFunction $reflect 反射类
  223 + * @param array $vars 变量
  224 + * @return array
  225 + */
  226 + private static function bindParams($reflect, $vars = [])
  227 + {
  228 + if (empty($vars)) {
  229 + // 自动获取请求变量
  230 + if (Config::get('url_param_type')) {
  231 + $vars = Request::instance()->route();
  232 + } else {
  233 + $vars = Request::instance()->param();
  234 + }
  235 + }
  236 + $args = [];
  237 + if ($reflect->getNumberOfParameters() > 0) {
  238 + // 判断数组类型 数字数组时按顺序绑定参数
  239 + reset($vars);
  240 + $type = key($vars) === 0 ? 1 : 0;
  241 + $params = $reflect->getParameters();
  242 + foreach ($params as $param) {
  243 + $args[] = self::getParamValue($param, $vars, $type);
  244 + }
  245 + }
  246 + return $args;
  247 + }
  248 +
  249 + /**
  250 + * 获取参数值
  251 + * @access private
  252 + * @param \ReflectionParameter $param
  253 + * @param array $vars 变量
  254 + * @param string $type
  255 + * @return array
  256 + */
  257 + private static function getParamValue($param, &$vars, $type)
  258 + {
  259 + $name = $param->getName();
  260 + $class = $param->getClass();
  261 + if ($class) {
  262 + $className = $class->getName();
  263 + $bind = Request::instance()->$name;
  264 + if ($bind instanceof $className) {
  265 + $result = $bind;
  266 + } else {
  267 + if (method_exists($className, 'invoke')) {
  268 + $method = new \ReflectionMethod($className, 'invoke');
  269 + if ($method->isPublic() && $method->isStatic()) {
  270 + return $className::invoke(Request::instance());
  271 + }
  272 + }
  273 + $result = method_exists($className, 'instance') ? $className::instance() : new $className;
  274 + }
  275 + } elseif (1 == $type && !empty($vars)) {
  276 + $result = array_shift($vars);
  277 + } elseif (0 == $type && isset($vars[$name])) {
  278 + $result = $vars[$name];
  279 + } elseif ($param->isDefaultValueAvailable()) {
  280 + $result = $param->getDefaultValue();
  281 + } else {
  282 + throw new \InvalidArgumentException('method param miss:' . $name);
  283 + }
  284 + return $result;
  285 + }
  286 +
  287 + protected static function exec($dispatch, $config)
  288 + {
  289 + switch ($dispatch['type']) {
  290 + case 'redirect':
  291 + // 执行重定向跳转
  292 + $data = Response::create($dispatch['url'], 'redirect')->code($dispatch['status']);
  293 + break;
  294 + case 'module':
  295 + // 模块/控制器/操作
  296 + $data = self::module($dispatch['module'], $config, isset($dispatch['convert']) ? $dispatch['convert'] : null);
  297 + break;
  298 + case 'controller':
  299 + // 执行控制器操作
  300 + $vars = array_merge(Request::instance()->param(), $dispatch['var']);
  301 + $data = Loader::action($dispatch['controller'], $vars, $config['url_controller_layer'], $config['controller_suffix']);
  302 + break;
  303 + case 'method':
  304 + // 执行回调方法
  305 + $vars = array_merge(Request::instance()->param(), $dispatch['var']);
  306 + $data = self::invokeMethod($dispatch['method'], $vars);
  307 + break;
  308 + case 'function':
  309 + // 执行闭包
  310 + $data = self::invokeFunction($dispatch['function']);
  311 + break;
  312 + case 'response':
  313 + $data = $dispatch['response'];
  314 + break;
  315 + default:
  316 + throw new \InvalidArgumentException('dispatch type not support');
  317 + }
  318 + return $data;
  319 + }
  320 +
  321 + /**
  322 + * 执行模块
  323 + * @access public
  324 + * @param array $result 模块/控制器/操作
  325 + * @param array $config 配置参数
  326 + * @param bool $convert 是否自动转换控制器和操作名
  327 + * @return mixed
  328 + */
  329 + public static function module($result, $config, $convert = null)
  330 + {
  331 + if (is_string($result)) {
  332 + $result = explode('/', $result);
  333 + }
  334 + $request = Request::instance();
  335 + if ($config['app_multi_module']) {
  336 + // 多模块部署
  337 + $module = strip_tags(strtolower($result[0] ?: $config['default_module']));
  338 + $bind = Route::getBind('module');
  339 + $available = false;
  340 + if ($bind) {
  341 + // 绑定模块
  342 + list($bindModule) = explode('/', $bind);
  343 + if (empty($result[0])) {
  344 + $module = $bindModule;
  345 + $available = true;
  346 + } elseif ($module == $bindModule) {
  347 + $available = true;
  348 + }
  349 + } elseif (!in_array($module, $config['deny_module_list']) && is_dir(APP_PATH . $module)) {
  350 + $available = true;
  351 + }
  352 +
  353 + // 模块初始化
  354 + if ($module && $available) {
  355 + // 初始化模块
  356 + $request->module($module);
  357 + $config = self::init($module);
  358 + // 模块请求缓存检查
  359 + $request->cache($config['request_cache'], $config['request_cache_expire'], $config['request_cache_except']);
  360 + } else {
  361 + throw new HttpException(404, 'module not exists:' . $module);
  362 + }
  363 + } else {
  364 + // 单一模块部署
  365 + $module = '';
  366 + $request->module($module);
  367 + }
  368 + // 当前模块路径
  369 + App::$modulePath = APP_PATH . ($module ? $module . DS : '');
  370 +
  371 + // 是否自动转换控制器和操作名
  372 + $convert = is_bool($convert) ? $convert : $config['url_convert'];
  373 + // 获取控制器名
  374 + $controller = strip_tags($result[1] ?: $config['default_controller']);
  375 + $controller = $convert ? strtolower($controller) : $controller;
  376 +
  377 + if (!preg_match('/^[A-Za-z](\w|\.)*$/', $controller)) {
  378 + throw new HttpException(404, 'controller not exists:' . $controller);
  379 + }
  380 +
  381 + // 获取操作名
  382 + $actionName = strip_tags($result[2] ?: $config['default_action']);
  383 + $actionName = $convert ? strtolower($actionName) : $actionName;
  384 +
  385 + // 设置当前请求的控制器、操作
  386 + $request->controller(Loader::parseName($controller, 1))->action($actionName);
  387 +
  388 + // 监听module_init
  389 + Hook::listen('module_init', $request);
  390 +
  391 + try {
  392 + $instance = Loader::controller($controller, $config['url_controller_layer'], $config['controller_suffix'], $config['empty_controller']);
  393 + } catch (ClassNotFoundException $e) {
  394 + throw new HttpException(404, 'controller not exists:' . $e->getClass());
  395 + }
  396 +
  397 + // 获取当前操作名
  398 + $action = $actionName . $config['action_suffix'];
  399 +
  400 + $vars = [];
  401 + if (is_callable([$instance, $action])) {
  402 + // 执行操作方法
  403 + $call = [$instance, $action];
  404 + } elseif (is_callable([$instance, '_empty'])) {
  405 + // 空操作
  406 + $call = [$instance, '_empty'];
  407 + $vars = [$actionName];
  408 + } else {
  409 + // 操作不存在
  410 + throw new HttpException(404, 'method not exists:' . get_class($instance) . '->' . $action . '()');
  411 + }
  412 +
  413 + Hook::listen('action_begin', $call);
  414 +
  415 + return self::invokeMethod($call, $vars);
  416 + }
  417 +
  418 + /**
  419 + * 初始化应用
  420 + */
  421 + public static function initCommon()
  422 + {
  423 + if (empty(self::$init)) {
  424 + if (defined('APP_NAMESPACE')) {
  425 + self::$namespace = APP_NAMESPACE;
  426 + }
  427 + Loader::addNamespace(self::$namespace, APP_PATH);
  428 +
  429 + // 初始化应用
  430 + $config = self::init();
  431 + self::$suffix = $config['class_suffix'];
  432 +
  433 + // 应用调试模式
  434 + self::$debug = Env::get('app_debug', Config::get('app_debug'));
  435 + if (!self::$debug) {
  436 + ini_set('display_errors', 'Off');
  437 + } elseif (!IS_CLI) {
  438 + //重新申请一块比较大的buffer
  439 + if (ob_get_level() > 0) {
  440 + $output = ob_get_clean();
  441 + }
  442 + ob_start();
  443 + if (!empty($output)) {
  444 + echo $output;
  445 + }
  446 + }
  447 +
  448 + if (!empty($config['root_namespace'])) {
  449 + Loader::addNamespace($config['root_namespace']);
  450 + }
  451 +
  452 + // 加载额外文件
  453 + if (!empty($config['extra_file_list'])) {
  454 + foreach ($config['extra_file_list'] as $file) {
  455 + $file = strpos($file, '.') ? $file : APP_PATH . $file . EXT;
  456 + if (is_file($file) && !isset(self::$file[$file])) {
  457 + include $file;
  458 + self::$file[$file] = true;
  459 + }
  460 + }
  461 + }
  462 +
  463 + // 设置系统时区
  464 + date_default_timezone_set($config['default_timezone']);
  465 +
  466 + // 监听app_init
  467 + Hook::listen('app_init');
  468 +
  469 + self::$init = true;
  470 + }
  471 + return Config::get();
  472 + }
  473 +
  474 + /**
  475 + * 初始化应用或模块
  476 + * @access public
  477 + * @param string $module 模块名
  478 + * @return array
  479 + */
  480 + private static function init($module = '')
  481 + {
  482 + // 定位模块目录
  483 + $module = $module ? $module . DS : '';
  484 +
  485 + // 加载初始化文件
  486 + if (is_file(APP_PATH . $module . 'init' . EXT)) {
  487 + include APP_PATH . $module . 'init' . EXT;
  488 + } elseif (is_file(RUNTIME_PATH . $module . 'init' . EXT)) {
  489 + include RUNTIME_PATH . $module . 'init' . EXT;
  490 + } else {
  491 + $path = APP_PATH . $module;
  492 + // 加载模块配置
  493 + $config = Config::load(CONF_PATH . $module . 'config' . CONF_EXT);
  494 + // 读取数据库配置文件
  495 + $filename = CONF_PATH . $module . 'database' . CONF_EXT;
  496 + Config::load($filename, 'database');
  497 + // 读取扩展配置文件
  498 + if (is_dir(CONF_PATH . $module . 'extra')) {
  499 + $dir = CONF_PATH . $module . 'extra';
  500 + $files = scandir($dir);
  501 + foreach ($files as $file) {
  502 + if ('.' . pathinfo($file, PATHINFO_EXTENSION) === CONF_EXT) {
  503 + $filename = $dir . DS . $file;
  504 + Config::load($filename, pathinfo($file, PATHINFO_FILENAME));
  505 + }
  506 + }
  507 + }
  508 +
  509 + // 加载应用状态配置
  510 + if ($config['app_status']) {
  511 + $config = Config::load(CONF_PATH . $module . $config['app_status'] . CONF_EXT);
  512 + }
  513 +
  514 + // 加载行为扩展文件
  515 + if (is_file(CONF_PATH . $module . 'tags' . EXT)) {
  516 + Hook::import(include CONF_PATH . $module . 'tags' . EXT);
  517 + }
  518 +
  519 + // 加载公共文件
  520 + if (is_file($path . 'common' . EXT)) {
  521 + include $path . 'common' . EXT;
  522 + }
  523 +
  524 + // 加载当前模块语言包
  525 + if ($module) {
  526 + Lang::load($path . 'lang' . DS . Request::instance()->langset() . EXT);
  527 + }
  528 + }
  529 + return Config::get();
  530 + }
  531 +
  532 + /**
  533 + * URL路由检测(根据PATH_INFO)
  534 + * @access public
  535 + * @param \think\Request $request
  536 + * @param array $config
  537 + * @return array
  538 + * @throws \think\Exception
  539 + */
  540 + public static function routeCheck($request, array $config)
  541 + {
  542 + $path = $request->path();
  543 + $depr = $config['pathinfo_depr'];
  544 + $result = false;
  545 + // 路由检测
  546 + $check = !is_null(self::$routeCheck) ? self::$routeCheck : $config['url_route_on'];
  547 + if ($check) {
  548 + // 开启路由
  549 + if (is_file(RUNTIME_PATH . 'route.php')) {
  550 + // 读取路由缓存
  551 + $rules = include RUNTIME_PATH . 'route.php';
  552 + if (is_array($rules)) {
  553 + Route::rules($rules);
  554 + }
  555 + } else {
  556 + $files = $config['route_config_file'];
  557 + foreach ($files as $file) {
  558 + if (is_file(CONF_PATH . $file . CONF_EXT)) {
  559 + // 导入路由配置
  560 + $rules = include CONF_PATH . $file . CONF_EXT;
  561 + if (is_array($rules)) {
  562 + Route::import($rules);
  563 + }
  564 + }
  565 + }
  566 + }
  567 +
  568 + // 路由检测(根据路由定义返回不同的URL调度)
  569 + $result = Route::check($request, $path, $depr, $config['url_domain_deploy']);
  570 + $must = !is_null(self::$routeMust) ? self::$routeMust : $config['url_route_must'];
  571 + if ($must && false === $result) {
  572 + // 路由无效
  573 + throw new RouteNotFoundException();
  574 + }
  575 + }
  576 + if (false === $result) {
  577 + // 路由无效 解析模块/控制器/操作/参数... 支持控制器自动搜索
  578 + $result = Route::parseUrl($path, $depr, $config['controller_auto_search']);
  579 + }
  580 + return $result;
  581 + }
  582 +
  583 + /**
  584 + * 设置应用的路由检测机制
  585 + * @access public
  586 + * @param bool $route 是否需要检测路由
  587 + * @param bool $must 是否强制检测路由
  588 + * @return void
  589 + */
  590 + public static function route($route, $must = false)
  591 + {
  592 + self::$routeCheck = $route;
  593 + self::$routeMust = $must;
  594 + }
  595 +}