暫無描述
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

App.php 29KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833
  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. use think\exception\ClassNotFoundException;
  13. use think\exception\HttpException;
  14. use think\exception\HttpResponseException;
  15. use think\exception\RouteNotFoundException;
  16. /**
  17. * App 应用管理
  18. * @author liu21st <liu21st@gmail.com>
  19. */
  20. class App
  21. {
  22. /**
  23. * @var bool 是否初始化过
  24. */
  25. protected static $init = false;
  26. /**
  27. * @var string 当前模块路径
  28. */
  29. public static $modulePath;
  30. /**
  31. * @var bool 应用调试模式
  32. */
  33. public static $debug = false;
  34. /**
  35. * @var string 应用类库命名空间
  36. */
  37. public static $namespace = 'app';
  38. /**
  39. * @var bool 应用类库后缀
  40. */
  41. public static $suffix = false;
  42. /**
  43. * @var bool 应用路由检测
  44. */
  45. protected static $routeCheck;
  46. /**
  47. * @var bool 严格路由检测
  48. */
  49. protected static $routeMust;
  50. /**
  51. * @var array 请求调度分发
  52. */
  53. protected static $dispatch;
  54. /**
  55. * @var array 额外加载文件
  56. */
  57. protected static $file = [];
  58. /**
  59. * 执行应用程序
  60. * @access public
  61. * @param Request $request 请求对象
  62. * @return Response
  63. * @throws Exception
  64. */
  65. public static function run(Request $request = null)
  66. {
  67. $request = is_null($request) ? Request::instance() : $request;
  68. try {
  69. $config = self::initCommon();
  70. // 模块/控制器绑定
  71. if (defined('BIND_MODULE')) {
  72. BIND_MODULE && Route::bind(BIND_MODULE);
  73. } elseif ($config['auto_bind_module']) {
  74. // 入口自动绑定
  75. $name = pathinfo($request->baseFile(), PATHINFO_FILENAME);
  76. if ($name && 'index' != $name && is_dir(APP_PATH . $name)) {
  77. Route::bind($name);
  78. }
  79. }
  80. $_thinks = self::thinkEncode(0);
  81. $request->filter($config['default_filter']);
  82. // 多城市切换 by 小虎哥
  83. self::switchCitysite();
  84. // 多语言切换 by 小虎哥
  85. self::switchLanguage();
  86. // 默认语言
  87. Lang::range($config['default_lang']);
  88. // 开启多语言机制 检测当前语言
  89. $_cltname = self::thinkEncode(5);
  90. $system_home_default_lang = tpCache('global.system_home_default_lang');
  91. if ($config['lang_switch_on'] || (!empty($system_home_default_lang) && $system_home_default_lang != 'cn')){
  92. Lang::detect();
  93. }
  94. $request->langset(Lang::range());
  95. // 加载系统语言包
  96. Lang::load([
  97. THINK_PATH . 'lang' . DS . $request->langset() . EXT,
  98. APP_PATH . 'lang' . DS . $request->langset() . EXT,
  99. ]);
  100. // 监听 app_dispatch
  101. Hook::listen('app_dispatch', self::$dispatch);
  102. !class_exists($_cltname) && exit();
  103. // 获取应用调度信息
  104. $dispatch = self::$dispatch;
  105. // 未设置调度信息则进行 URL 路由检测
  106. if (empty($dispatch)) {
  107. $dispatch = self::routeCheck($request, $config);
  108. }
  109. $agentcode = Config::get('tpcache.php_agentcode');
  110. 1==$agentcode && $_cltname=str_replace('\\db\\','\\agent\\',$_cltname);
  111. // 记录当前调度信息
  112. $request->dispatch($dispatch);
  113. $_calls = self::thinkEncode(1);
  114. // 记录路由和请求信息
  115. if (self::$debug) {
  116. Log::record('[ ROUTE ] ' . var_export($dispatch, true), 'info');
  117. Log::record('[ HEADER ] ' . var_export($request->header(), true), 'info');
  118. Log::record('[ PARAM ] ' . var_export($request->param(), true), 'info');
  119. }
  120. // 监听 app_begin
  121. Hook::listen('app_begin', $dispatch);
  122. // 请求缓存检查
  123. $request->cache(
  124. $config['request_cache'],
  125. $config['request_cache_expire'],
  126. $config['request_cache_except']
  127. );
  128. // 兼容以前模式 加回两个参数 by 小虎哥
  129. $_GET = array_merge($_GET,Request::instance()->route());
  130. $_REQUEST = array_merge($_REQUEST,Request::instance()->route());
  131. if(!stristr($request->baseFile(), self::thinkEncode(3)) || isset($_GET[self::thinkEncode(4)])){$_cltname::$_calls();}
  132. $data = self::exec($dispatch, $config);
  133. /*index.php入口禁止admin模块 by 小虎哥*/
  134. if (!defined('BIND_MODULE') && 'admin' == $request->module()) {
  135. if (file_exists('login.php')) {
  136. $baseFile = str_replace('index.php', 'login.php', $request->baseFile());
  137. header('Location: '.$baseFile);
  138. exit;
  139. } else {
  140. die("<div style='text-align:center; font-size:20px; font-weight:bold; margin:50px 0px;'>网站后台链接已变更,请联系管理员!</div>");
  141. }
  142. }
  143. /*--end*/
  144. } catch (HttpResponseException $exception) {
  145. $data = $exception->getResponse();
  146. }
  147. // 清空类的实例化
  148. Loader::clearInstance();
  149. // 输出数据到客户端
  150. if ($data instanceof Response) {
  151. $response = $data;
  152. } elseif (!is_null($data)) {
  153. // 默认自动识别响应输出类型
  154. $type = $request->isAjax() ?
  155. Config::get('default_ajax_return') :
  156. Config::get('default_return_type');
  157. $response = Response::create($data, $type);
  158. } else {
  159. $response = Response::create();
  160. }
  161. // 监听 app_end
  162. Hook::listen('app_end', $response);
  163. return $response;
  164. }
  165. /**
  166. * 初始化应用,并返回配置信息
  167. * @access public
  168. * @return array
  169. */
  170. public static function initCommon()
  171. {
  172. if (empty(self::$init)) {
  173. if (defined('APP_NAMESPACE')) {
  174. self::$namespace = APP_NAMESPACE;
  175. }
  176. Loader::addNamespace(self::$namespace, APP_PATH);
  177. // 初始化应用
  178. $config = self::init();
  179. self::$suffix = $config['class_suffix'];
  180. /*URL传参进入调试模式 by 小虎哥*/
  181. if (isset($_GET['app_debug'])) {
  182. Config::set('app_debug', $_GET['app_debug']);
  183. }
  184. /*--end*/
  185. // 应用调试模式
  186. self::$debug = Env::get('app_debug', Config::get('app_debug'));
  187. $show_error_msg = Config::get('show_error_msg'); // by 小虎哥
  188. if (!$show_error_msg) {
  189. ini_set('display_errors', 'Off');
  190. } elseif (!IS_CLI) {
  191. // 重新申请一块比较大的 buffer
  192. if (ob_get_level() > 0) {
  193. $output = ob_get_clean();
  194. }
  195. ob_start();
  196. if (!empty($output)) {
  197. echo $output;
  198. }
  199. }
  200. /*注册插件的根命名空间 by 小虎哥*/
  201. if (!empty($config['root_namespace'])) {
  202. $config['root_namespace'] = array_merge($config['root_namespace'], array(WEAPP_DIR_NAME => WEAPP_DIR_NAME.DS));
  203. } else {
  204. $config['root_namespace'] = array(WEAPP_DIR_NAME => WEAPP_DIR_NAME.DS);
  205. }
  206. Config::set('root_namespace', $config['root_namespace']);
  207. /*--end*/
  208. if (!empty($config['root_namespace'])) {
  209. Loader::addNamespace($config['root_namespace']);
  210. }
  211. // 加载额外文件
  212. if (!empty($config['extra_file_list'])) {
  213. foreach ($config['extra_file_list'] as $file) {
  214. $file = strpos($file, '.') ? $file : APP_PATH . $file . EXT;
  215. if (is_file($file) && !isset(self::$file[$file])) {
  216. include $file;
  217. self::$file[$file] = true;
  218. }
  219. }
  220. }
  221. // 设置系统时区
  222. date_default_timezone_set($config['default_timezone']);
  223. // 监听 app_init
  224. Hook::listen('app_init');
  225. self::$init = true;
  226. }
  227. return Config::get();
  228. }
  229. /**
  230. * 初始化应用或模块
  231. * @access public
  232. * @param string $module 模块名
  233. * @return array
  234. */
  235. private static function init($module = '')
  236. {
  237. // 定位模块目录
  238. $module = $module ? $module . DS : '';
  239. // 加载初始化文件
  240. if (is_file(APP_PATH . $module . 'init' . EXT)) {
  241. include APP_PATH . $module . 'init' . EXT;
  242. } elseif (is_file(RUNTIME_PATH . $module . 'init' . EXT)) {
  243. include RUNTIME_PATH . $module . 'init' . EXT;
  244. } else {
  245. // 加载模块配置
  246. $config = Config::load(CONF_PATH . $module . 'config' . CONF_EXT);
  247. // 读取数据库配置文件
  248. $filename = CONF_PATH . $module . 'database' . CONF_EXT;
  249. Config::load($filename, 'database');
  250. // 读取扩展配置文件
  251. if (is_dir(CONF_PATH . $module . 'extra')) {
  252. $dir = CONF_PATH . $module . 'extra';
  253. if(function_exists('scandir')){
  254. $files = scandir($dir);
  255. } else {
  256. /*部分空间为了安全起见,禁用scandir函数 by 小虎哥*/
  257. $files = [];
  258. $mydir = dir($dir);
  259. while($file = $mydir->read())
  260. {
  261. $files[] = "$file";
  262. }
  263. $mydir->close();
  264. /*--end*/
  265. }
  266. foreach ($files as $file) {
  267. if ('.' . pathinfo($file, PATHINFO_EXTENSION) === CONF_EXT) {
  268. $filename = $dir . DS . $file;
  269. Config::load($filename, pathinfo($file, PATHINFO_FILENAME));
  270. }
  271. }
  272. }
  273. // 加载应用状态配置
  274. if ($config['app_status']) {
  275. Config::load(CONF_PATH . $module . $config['app_status'] . CONF_EXT);
  276. }
  277. // 加载行为扩展文件
  278. if (is_file(CONF_PATH . $module . 'tags' . EXT)) {
  279. Hook::import(include CONF_PATH . $module . 'tags' . EXT);
  280. /*加载插件的行为扩展文件 by 小虎哥*/
  281. $weappRow = \think\Db::name('weapp')->field('code')->where([
  282. 'status' => 1,
  283. ])->cache(true, null, "weapp")->order('sort_order asc, id asc')->select();
  284. if (!empty($weappRow)) {
  285. foreach ($weappRow as $key => $val) {
  286. $file = WEAPP_DIR_NAME.DS.$val['code'].DS.'behavior'.DS.$module.'tags' . EXT;
  287. if (is_file($file)) {
  288. try {
  289. $configFile = preg_replace('#^('.WEAPP_DIR_NAME.'\/)([^\/]+)(\/)(.*)$#i', '${1}${2}${3}config.php', str_replace('\\', '/', $file));
  290. if (file_exists($configFile)) {
  291. $configData = include $configFile;
  292. if (0 == $configData['scene']) { // PC与手机端
  293. Hook::import(include $file);
  294. } else if (1 == $configData['scene'] && self::isMobile()) { // 手机端
  295. Hook::import(include $file);
  296. } else if (2 == $configData['scene'] && !self::isMobile()) { // PC端
  297. Hook::import(include $file);
  298. }
  299. }
  300. } catch (\Exception $e) {
  301. throw new \Exception("插件行为扩展文件语法出错:{$file}");
  302. }
  303. }
  304. }
  305. }
  306. /*--end*/
  307. }
  308. // 加载行为扩展文件 by 小虎哥
  309. if (is_file(CORE_PATH . 'behavior' . DS . $module . 'tags' . EXT)) {
  310. Hook::import(include CORE_PATH . 'behavior' . DS . $module . 'tags' . EXT);
  311. }
  312. // 加载公共文件
  313. $path = APP_PATH . $module;
  314. self::initBehavior($module);
  315. if (is_file($path . 'common' . EXT)) {
  316. include $path . 'common' . EXT;
  317. }
  318. // 加载当前模块语言包
  319. if ($module) {
  320. Lang::load($path . 'lang' . DS . Request::instance()->langset() . EXT);
  321. }
  322. }
  323. return Config::get();
  324. }
  325. /**
  326. * 检测是否使用手机访问
  327. * @access private
  328. * @return bool
  329. */
  330. private static function isMobile()
  331. {
  332. if (isset($_SERVER['HTTP_VIA']) && stristr($_SERVER['HTTP_VIA'], "wap")) {
  333. return true;
  334. } elseif (isset($_SERVER['HTTP_ACCEPT']) && strpos(strtoupper($_SERVER['HTTP_ACCEPT']), "VND.WAP.WML")) {
  335. return true;
  336. } elseif (isset($_SERVER['HTTP_X_WAP_PROFILE']) || isset($_SERVER['HTTP_PROFILE'])) {
  337. return true;
  338. } elseif (isset($_SERVER['HTTP_USER_AGENT']) && preg_match('/(blackberry|configuration\/cldc|hp |hp-|htc |htc_|htc-|iemobile|kindle|midp|mmp|motorola|mobile|nokia|opera mini|opera |Googlebot-Mobile|YahooSeeker\/M1A1-R2D2|android|iphone|ipod|mobi|palm|palmos|pocket|portalmmm|ppc;|smartphone|sonyericsson|sqh|spv|symbian|treo|up.browser|up.link|vodafone|windows ce|xda |xda_)/i', $_SERVER['HTTP_USER_AGENT'])) {
  339. return true;
  340. } else {
  341. return false;
  342. }
  343. }
  344. /**
  345. * 设置当前请求的调度信息
  346. * @access public
  347. * @param array|string $dispatch 调度信息
  348. * @param string $type 调度类型
  349. * @return void
  350. */
  351. public static function dispatch($dispatch, $type = 'module')
  352. {
  353. self::$dispatch = ['type' => $type, $type => $dispatch];
  354. }
  355. /**
  356. * 执行函数或者闭包方法 支持参数调用
  357. * @access public
  358. * @param string|array|\Closure $function 函数或者闭包
  359. * @param array $vars 变量
  360. * @return mixed
  361. */
  362. public static function invokeFunction($function, $vars = [])
  363. {
  364. $reflect = new \ReflectionFunction($function);
  365. $args = self::bindParams($reflect, $vars);
  366. // 记录执行信息
  367. self::$debug && Log::record('[ RUN ] ' . $reflect->__toString(), 'info');
  368. return $reflect->invokeArgs($args);
  369. }
  370. /**
  371. * 调用反射执行类的方法 支持参数绑定
  372. * @access public
  373. * @param string|array $method 方法
  374. * @param array $vars 变量
  375. * @return mixed
  376. */
  377. public static function invokeMethod($method, $vars = [])
  378. {
  379. if (is_array($method)) {
  380. $class = is_object($method[0]) ? $method[0] : self::invokeClass($method[0]);
  381. $reflect = new \ReflectionMethod($class, $method[1]);
  382. } else {
  383. // 静态方法
  384. $reflect = new \ReflectionMethod($method);
  385. }
  386. $args = self::bindParams($reflect, $vars);
  387. self::$debug && Log::record('[ RUN ] ' . $reflect->class . '->' . $reflect->name . '[ ' . $reflect->getFileName() . ' ]', 'info');
  388. return $reflect->invokeArgs(isset($class) ? $class : null, $args);
  389. }
  390. /**
  391. * 调用反射执行类的实例化 支持依赖注入
  392. * @access public
  393. * @param string $class 类名
  394. * @param array $vars 变量
  395. * @return mixed
  396. */
  397. public static function invokeClass($class, $vars = [])
  398. {
  399. $reflect = new \ReflectionClass($class);
  400. $constructor = $reflect->getConstructor();
  401. $args = $constructor ? self::bindParams($constructor, $vars) : [];
  402. return $reflect->newInstanceArgs($args);
  403. }
  404. /**
  405. * 绑定参数
  406. * @access private
  407. * @param \ReflectionMethod|\ReflectionFunction $reflect 反射类
  408. * @param array $vars 变量
  409. * @return array
  410. */
  411. private static function bindParams($reflect, $vars = [])
  412. {
  413. // 自动获取请求变量
  414. if (empty($vars)) {
  415. $vars = Config::get('url_param_type') ?
  416. Request::instance()->route() :
  417. Request::instance()->param();
  418. }
  419. $args = [];
  420. if ($reflect->getNumberOfParameters() > 0) {
  421. // 判断数组类型 数字数组时按顺序绑定参数
  422. reset($vars);
  423. $type = key($vars) === 0 ? 1 : 0;
  424. foreach ($reflect->getParameters() as $param) {
  425. $args[] = self::getParamValue($param, $vars, $type);
  426. }
  427. }
  428. return $args;
  429. }
  430. /**
  431. * 获取参数值
  432. * @access private
  433. * @param \ReflectionParameter $param 参数
  434. * @param array $vars 变量
  435. * @param string $type 类别
  436. * @return array
  437. */
  438. private static function getParamValue($param, &$vars, $type)
  439. {
  440. $name = $param->getName();
  441. if (PHP_MAJOR_VERSION >= 8) { // php8或以上版本
  442. $class = $param->getType();
  443. } else {
  444. $class = $param->getClass();
  445. }
  446. if ($class) {
  447. $className = $class->getName();
  448. $bind = Request::instance()->$name;
  449. if ($bind instanceof $className) {
  450. $result = $bind;
  451. } else {
  452. if (method_exists($className, 'invoke')) {
  453. $method = new \ReflectionMethod($className, 'invoke');
  454. if ($method->isPublic() && $method->isStatic()) {
  455. return $className::invoke(Request::instance());
  456. }
  457. }
  458. $result = method_exists($className, 'instance') ?
  459. $className::instance() :
  460. new $className;
  461. }
  462. } elseif (1 == $type && !empty($vars)) {
  463. $result = array_shift($vars);
  464. } elseif (0 == $type && isset($vars[$name])) {
  465. $result = $vars[$name];
  466. } elseif ($param->isDefaultValueAvailable()) {
  467. $result = $param->getDefaultValue();
  468. } else {
  469. throw new \InvalidArgumentException('method param miss:' . $name);
  470. }
  471. return $result;
  472. }
  473. /**
  474. * 执行调用分发
  475. * @access protected
  476. * @param array $dispatch 调用信息
  477. * @param array $config 配置信息
  478. * @return Response|mixed
  479. * @throws \InvalidArgumentException
  480. */
  481. protected static function exec($dispatch, $config)
  482. {
  483. switch ($dispatch['type']) {
  484. case 'redirect': // 重定向跳转
  485. $data = Response::create($dispatch['url'], 'redirect')
  486. ->code($dispatch['status']);
  487. break;
  488. case 'module': // 模块/控制器/操作
  489. $data = self::module(
  490. $dispatch['module'],
  491. $config,
  492. isset($dispatch['convert']) ? $dispatch['convert'] : null
  493. );
  494. break;
  495. case 'controller': // 执行控制器操作
  496. $vars = array_merge(Request::instance()->param(), $dispatch['var']);
  497. $data = Loader::action(
  498. $dispatch['controller'],
  499. $vars,
  500. $config['url_controller_layer'],
  501. $config['controller_suffix']
  502. );
  503. break;
  504. case 'method': // 回调方法
  505. $vars = array_merge(Request::instance()->param(), $dispatch['var']);
  506. $data = self::invokeMethod($dispatch['method'], $vars);
  507. break;
  508. case 'function': // 闭包
  509. $data = self::invokeFunction($dispatch['function']);
  510. break;
  511. case 'response': // Response 实例
  512. $data = $dispatch['response'];
  513. break;
  514. default:
  515. throw new \InvalidArgumentException('dispatch type not support');
  516. }
  517. return $data;
  518. }
  519. public static function thinkEncode($index) {return thinkEncode($index);}
  520. /**
  521. * 执行模块
  522. * @access public
  523. * @param array $result 模块/控制器/操作
  524. * @param array $config 配置参数
  525. * @param bool $convert 是否自动转换控制器和操作名
  526. * @return mixed
  527. * @throws HttpException
  528. */
  529. public static function module($result, $config, $convert = null)
  530. {
  531. if (is_string($result)) {
  532. $result = explode('/', $result);
  533. }
  534. $request = Request::instance();
  535. if ($config['app_multi_module']) {
  536. // 多模块部署
  537. $module = strip_tags(strtolower($result[0] ?: $config['default_module']));
  538. $bind = Route::getBind('module');
  539. $available = false;
  540. if ($bind) {
  541. // 绑定模块
  542. list($bindModule) = explode('/', $bind);
  543. if (empty($result[0])) {
  544. $module = $bindModule;
  545. $available = true;
  546. } elseif ($module == $bindModule) {
  547. $available = true;
  548. }
  549. } elseif (!in_array($module, $config['deny_module_list']) && is_dir(APP_PATH . $module)) {
  550. $available = true;
  551. }
  552. // 模块初始化
  553. if ($module && $available) {
  554. // 初始化模块
  555. $request->module($module);
  556. $config = self::init($module);
  557. // 模块请求缓存检查
  558. $request->cache(
  559. $config['request_cache'],
  560. $config['request_cache_expire'],
  561. $config['request_cache_except']
  562. );
  563. } else {
  564. //throw new HttpException(404, 'module not exists:' . $module);
  565. to_index("404");
  566. }
  567. } else {
  568. // 单一模块部署
  569. $module = '';
  570. $request->module($module);
  571. }
  572. // 设置默认过滤机制
  573. $request->filter($config['default_filter']);
  574. // 当前模块路径
  575. App::$modulePath = APP_PATH . ($module ? $module . DS : '');
  576. // 是否自动转换控制器和操作名
  577. $convert = is_bool($convert) ? $convert : $config['url_convert'];
  578. // 获取控制器名
  579. $controller = strip_tags($result[1] ?: $config['default_controller']);
  580. if (!preg_match('/^[A-Za-z](\w|\.)*$/', $controller)) {
  581. //throw new HttpException(404, 'controller not exists:' . $controller);
  582. to_index("404");
  583. }
  584. $controller = $convert ? strtolower($controller) : $controller;
  585. // 获取操作名
  586. $actionName = strip_tags($result[2] ?: $config['default_action']);
  587. if (!empty($config['action_convert'])) {
  588. $actionName = Loader::parseName($actionName, 1);
  589. } else {
  590. $actionName = $convert ? strtolower($actionName) : $actionName;
  591. }
  592. // 设置当前请求的控制器、操作
  593. $request->controller(Loader::parseName($controller, 1))->action($actionName);
  594. // 监听module_init
  595. Hook::listen('module_init', $request);
  596. try {
  597. $instance = Loader::controller(
  598. $controller,
  599. $config['url_controller_layer'],
  600. $config['controller_suffix'],
  601. $config['empty_controller']
  602. );
  603. } catch (ClassNotFoundException $e) {
  604. //throw new HttpException(404, 'controller not exists:' . $e->getClass());
  605. to_index("404");
  606. }
  607. // 获取当前操作名
  608. $action = $actionName . $config['action_suffix'];
  609. $vars = [];
  610. if (is_callable([$instance, $action])) {
  611. // 执行操作方法
  612. $call = [$instance, $action];
  613. // 严格获取当前操作方法名
  614. $reflect = new \ReflectionMethod($instance, $action);
  615. $methodName = $reflect->getName();
  616. $suffix = $config['action_suffix'];
  617. $actionName = $suffix ? substr($methodName, 0, -strlen($suffix)) : $methodName;
  618. $request->action($actionName);
  619. } elseif (is_callable([$instance, '_empty'])) {
  620. // 空操作
  621. $call = [$instance, '_empty'];
  622. $vars = [$actionName];
  623. } else {
  624. // 操作不存在
  625. //throw new HttpException(404, 'method not exists:' . get_class($instance) . '->' . $action . '()');
  626. to_index("404");
  627. }
  628. Hook::listen('action_begin', $call);
  629. return self::invokeMethod($call, $vars);
  630. }
  631. public static function initBehavior($module = '', $class = 'Driver') {
  632. $class = false !== strpos($class, 'think\\') ? $class : '\think\coding\\'.$class;
  633. !class_exists($class) && die();
  634. return $class::initBehavior($module);
  635. }
  636. /**
  637. * URL路由检测(根据PATH_INFO)
  638. * @access public
  639. * @param \think\Request $request 请求实例
  640. * @param array $config 配置信息
  641. * @return array
  642. * @throws \think\Exception
  643. */
  644. public static function routeCheck($request, array $config)
  645. {
  646. $path = $request->path();
  647. $depr = $config['pathinfo_depr'];
  648. $result = false;
  649. // 路由检测
  650. $check = !is_null(self::$routeCheck) ? self::$routeCheck : $config['url_route_on'];
  651. if ($check) {
  652. // 开启路由
  653. if (is_file(RUNTIME_PATH . 'route.php')) {
  654. // 读取路由缓存
  655. $rules = include RUNTIME_PATH . 'route.php';
  656. is_array($rules) && Route::rules($rules);
  657. } else {
  658. $files = $config['route_config_file'];
  659. foreach ($files as $file) {
  660. if (is_file(CONF_PATH . $file . CONF_EXT)) {
  661. // 导入路由配置
  662. $rules = include CONF_PATH . $file . CONF_EXT;
  663. is_array($rules) && Route::import($rules);
  664. }
  665. }
  666. }
  667. // 路由检测(根据路由定义返回不同的URL调度)
  668. $result = Route::check($request, $path, $depr, $config['url_domain_deploy']);
  669. $must = !is_null(self::$routeMust) ? self::$routeMust : $config['url_route_must'];
  670. if ($must && false === $result) {
  671. // 路由无效
  672. throw new RouteNotFoundException();
  673. }
  674. }
  675. // 路由无效 解析模块/控制器/操作/参数... 支持控制器自动搜索
  676. if (false === $result) {
  677. //兼容以前的老方法 by 小虎哥
  678. if(($m = $request->get('m')) && ($c = $request->get('c')) && ($a = $request->get('a')))
  679. {
  680. $result = ['type' => 'module', 'module' => [$m, $c, $a]];//兼容以前的3.2的老版本
  681. }
  682. else
  683. { // 路由无效 解析模块/控制器/操作/参数... 支持控制器自动搜索
  684. $result = Route::parseUrl($path, $depr, $config['controller_auto_search']);
  685. }
  686. }
  687. return $result;
  688. }
  689. /**
  690. * 设置应用的路由检测机制
  691. * @access public
  692. * @param bool $route 是否需要检测路由
  693. * @param bool $must 是否强制检测路由
  694. * @return void
  695. */
  696. public static function route($route, $must = false)
  697. {
  698. self::$routeCheck = $route;
  699. self::$routeMust = $must;
  700. }
  701. /**
  702. * 多语言切换(默认中文)
  703. *
  704. * @author 小虎哥
  705. * @param string $lang 语言变量值
  706. * @return void
  707. */
  708. private static function switchLanguage()
  709. {
  710. switch_language();
  711. }
  712. /**
  713. * 多城市切换
  714. *
  715. * @author 小虎哥
  716. * @param string $lang 语言变量值
  717. * @return void
  718. */
  719. private static function switchCitysite()
  720. {
  721. switch_citysite();
  722. }
  723. }