// +---------------------------------------------------------------------- namespace think; use think\exception\ValidateException; use traits\controller\Jump; Loader::import('controller/Jump', TRAIT_PATH, EXT); class Controller { use Jump; /** * @var \think\View 视图类实例 */ protected $view; /** * @var \think\Request Request 实例 */ protected $request; /** * @var bool 验证失败是否抛出异常 */ protected $failException = false; /** * @var bool 是否批量验证 */ protected $batchValidate = false; /** * @var array 前置操作方法列表 */ protected $beforeActionList = []; /** * 主体语言(语言列表中最早一条) */ public $main_lang = 'cn'; /** * 后台当前语言 */ public $admin_lang = 'cn'; /** * 前台当前语言 */ public $home_lang = 'cn'; /** * 子目录路径 */ public $root_dir = ''; /** * CMS版本号 */ public $version = null; /** * 模板风格 */ public $tpl_theme = null; /** * 是否访问手机版 */ public $is_mobile = 0; /** * 前台当前站点 */ public $home_site = ''; /** * 站点域名,带端口号 */ public $website_host = ''; /** * 构造方法 * @access public * @param Request $request Request 对象 */ public function __construct(Request $request = null) { if (is_null($request)) { $request = Request::instance(); } $this->request = $request; $this->root_dir = ROOT_DIR; // 子目录 $this->website_host = $this->request->host(); if (stristr($this->website_host, 'localhost')) { $web_basehost = preg_replace('/^(([^\:\.]+):)?(\/\/)?([^\/\:]*)(.*)$/i', '${4}', config('tpcache.web_basehost')); if (!empty($web_basehost)) { $host_port = !stristr($this->website_host, ':') ? '' : $request->port(); $this->website_host = $web_basehost; if (!empty($host_port) && !stristr($this->website_host, ':')) { $this->website_host .= ":{$host_port}"; } } } // 运营状态下,检验阅读权限 $aid = input('param.aid/s', ''); if (Config::get('app_debug') != true && !empty($aid) && $this->request->module() == 'home' && ($this->request->controller() == 'View' || $this->request->action() == 'view') ) { $map = []; if (!is_numeric($aid) || strval(intval($aid)) !== strval($aid)) { $map = array('a.htmlfilename' => $aid); } else { $map = array('a.aid' => intval($aid)); } $map['a.is_del'] = 0; // 回收站功能 $field = 'a.aid,a.arcrank,a.users_id, a.typeid, a.channel, c.typearcrank,c.dirname,c.page_limit'; $archivesInfo = Db::name('archives')->field($field) ->alias('a') ->join('arctype c', 'a.typeid = c.id', 'LEFT') ->where($map) ->find(); $users_id = (int)session('users_id'); $archivesInfo['page_limit'] = empty($archivesInfo['page_limit']) ? [] : explode(',', $archivesInfo['page_limit']); // 若需要会员权限则执行 if (($archivesInfo['arcrank'] > 0 || ( $archivesInfo['typearcrank'] > 0 && in_array(2,$archivesInfo['page_limit']) ) ) && empty($users_id)) { $url = url('user/Users/login'); $arcurl = $this->request->domain().$this->root_dir."/index.php?m=home&c=Ajax&a=toView&aid={$aid}"; if (stristr($url, '?')) { $url = $url."&referurl=".urlencode($arcurl); } else { $url = $url."?referurl=".urlencode($arcurl); } $this->redirect($url); exit; } } /*多语言*/ $this->home_lang = get_home_lang(); $this->admin_lang = get_admin_lang(); $this->main_lang = get_main_lang(); /*多城市*/ if (config('city_switch_on')) { $this->home_site = get_home_site(); } null === $this->version && $this->version = getCmsVersion(); $returnData = $this->pc_to_mobile($this->request); $this->is_mobile = $returnData['is_mobile']; if (!defined('IS_AJAX')) { $this->request->isAjax() ? define('IS_AJAX',true) : define('IS_AJAX',false); // } if (!defined('IS_GET')) { ($this->request->method() == 'GET') ? define('IS_GET',true) : define('IS_GET',false); // } if (!defined('IS_POST')) { ($this->request->method() == 'POST') ? define('IS_POST',true) : define('IS_POST',false); // } if (!defined('IS_AJAX_POST')) { ($this->request->isAjax() && $this->request->method() == 'POST') ? define('IS_AJAX_POST',true) : define('IS_AJAX_POST',false); // } // 前台模板目录切换 null === $this->tpl_theme && $this->tpl_theme = config('ey_config.web_tpl_theme'); if (empty($this->tpl_theme)) { if (file_exists(ROOT_PATH.'template/default')) { $this->tpl_theme = 'default/'; } else { $this->tpl_theme = ''; } } else { if ('default' == $this->tpl_theme && !file_exists(ROOT_PATH.'template/default')) { $this->tpl_theme = ''; } else if ('default' != $this->tpl_theme && !file_exists(ROOT_PATH.'template/'.$this->tpl_theme)) { if (in_array($this->request->module(), ['home','user'])) { $this->error("模板目录【{$this->tpl_theme}】不存在!"); } } else { $this->tpl_theme .= '/'; } } !defined('TPL_THEME') && define('TPL_THEME', $this->tpl_theme); // 模板目录 $param = input('param.'); if (isset($param['uiset']) && !session('?admin_id')) { if (!file_exists(ROOT_PATH.'template/'.TPL_THEME.'pc/uiset.txt') && !file_exists(ROOT_PATH.'template/'.TPL_THEME.'mobile/uiset.txt')) { to_index(); } } !defined('MODULE_NAME') && define('MODULE_NAME',$this->request->module()); // 当前模块名称是 !defined('CONTROLLER_NAME') && define('CONTROLLER_NAME',$this->request->controller()); // 当前控制器名称 !defined('ACTION_NAME') && define('ACTION_NAME',$this->request->action()); // 当前操作名称是 !defined('PREFIX') && define('PREFIX',Config::get('database.prefix')); // 数据库表前缀 !defined('SYSTEM_ADMIN_LANG') && define('SYSTEM_ADMIN_LANG', Config::get('global.admin_lang')); // 后台语言变量 !defined('SYSTEM_HOME_LANG') && define('SYSTEM_HOME_LANG', Config::get('global.home_lang')); // 前台语言变量 if (stristr($this->request->url(), '?')) { $module = $this->request->param('m'); if (!empty($module)) { $module_dir = glob(APP_PATH.'*', GLOB_ONLYDIR); foreach ($module_dir as $key => $val) { $val = str_replace('\\', '/', $val); $module_dir[$key] = preg_replace('/^(.*)\/([^\/]+)$/i', '${2}', $val); } if (!in_array($module, $module_dir)) { to_index(); } } } $web_anti_brushing = config('tpcache.web_anti_brushing'); if (!empty($web_anti_brushing)) { if ('home' == MODULE_NAME && 'Index' == CONTROLLER_NAME && 'index' == ACTION_NAME) { $varsarr = ['clear','lang','site','templet','uiset','v','bd_vid','clickid']; if (1 == config('ey_config.seo_pseudo') || isset($params['uiset'])) { $varsarr = array_merge($varsarr, ['m','c','a']); } $agentcode = config('tpcache.php_agentcode'); if (1 == $agentcode) { $varsarr[] = 'aid'; } $params = $this->request->param(); if (empty($params['a']) || !in_array($params['a'], ['wechat_return','alipay_return','Express100','ey_agent'])) { foreach ($params as $key => $val) { if (!in_array($key, $varsarr)) { to_index(); } else if ('clear' == $key) { if ($val != 1) { to_index(); } } else if ('lang' == $key) { if ((isset($params['uiset']) || is_language()) && 2 == strlen($val)) { } else { to_index(); } } } } } else { if ((stristr($this->request->url(true), '/home/View/index/') || stristr($this->request->url(true), '/home/Lists/index/')) && 'Buildhtml' != CONTROLLER_NAME) { to_index(); } } } // 自动判断手机端和PC,以及PC/手机自适应模板 by 小虎哥 2018-05-10 $v = I('param.v/s', 'pc'); $v = trim($v, '/'); if ($v == 'mobile') { $this->is_mobile = 1; } if($this->is_mobile == 1 && file_exists(ROOT_PATH.'template/'.$this->tpl_theme.'mobile/index.htm')) { !defined('THEME_STYLE') && define('THEME_STYLE', 'mobile'); // 手机端标识 !defined('THEME_STYLE_PATH') && define('THEME_STYLE_PATH', $this->tpl_theme.THEME_STYLE); // 手机端模板根目录 } else { !defined('THEME_STYLE') && define('THEME_STYLE', 'pc'); // pc端标识 !defined('THEME_STYLE_PATH') && define('THEME_STYLE_PATH', $this->tpl_theme.THEME_STYLE); // PC端模板根目录 } if (in_array($this->request->module(), ['home','user'])) { Config::set('template.view_path', './template/'.THEME_STYLE_PATH.'/'); } else if (in_array($this->request->module(), array('admin'))) { if ('weapp' == strtolower($this->request->controller()) && 'execute' == strtolower($this->request->action())) { Config::set('template.view_path', '.'.ROOT_DIR.'/'.WEAPP_DIR_NAME.'/'.$this->request->param('sm').'/template/'); } } // -------end $this->view = View::instance(Config::get('template'), Config::get('view_replace_str')); $this->assign('home_lang', $this->home_lang); $this->assign('admin_lang', $this->admin_lang); $this->assign('main_lang', $this->main_lang); $this->assign('version', $this->version); $this->assign('is_mobile', $this->is_mobile); $this->assign('tpl_theme', $this->tpl_theme); $this->assign('home_site', $this->home_site); $this->assign('website_host', $this->website_host); $param = $this->request->param(); if ( (CONTROLLER_NAME == 'Lists' && isset($param['sort'])) || isset($param['clear']) || Config::get('app_debug') === true) { } else { //var_dump("开始读取缓存"); read_html_cache(); // 尝试从缓存中读取 } // 控制器初始化 $this->_initialize(); //var_dump("还会读取缓存吗"); //不会了 ,已执行 exit(); if ('admin' == MODULE_NAME) { $assignName2 = $this->arrJoinStr(['cGhwX3Nlcn','ZpY2VtZWFs']); $assignValue2 = tpCache('php.'.$assignName2); $this->assign($assignName2, $assignValue2); } // 前置操作方法 if ($this->beforeActionList) { foreach ($this->beforeActionList as $method => $options) { is_numeric($method) ? $this->beforeAction($options) : $this->beforeAction($method, $options); } } // 逻辑化 $this->coding(); } /** * 初始化操作 * @access protected */ protected function _initialize() { static $request = null; if (null === $request) { $request = request(); } $searchformhidden = ''; /*纯动态URL模式下,必须要传参的分组、控制器、操作名*/ if (1 == config('ey_config.seo_pseudo') && 1 == config('ey_config.seo_dynamic_format')) { $searchformhidden .= ''; $searchformhidden .= ''; $searchformhidden .= ''; if ('Weapp' == $request->get('c') && 'execute' == $request->get('a')) { // 插件的搜索 $searchformhidden .= ''; $searchformhidden .= ''; $searchformhidden .= ''; if ('Mbackend' == $request->get('sm')) { $searchformhidden .= ''; } } /*多语言*/ $lang = $request->param('lang/s'); empty($lang) && $lang = get_main_lang(); $searchformhidden .= ''; /*--end*/ } /*--end*/ $searchform['hidden'] = $searchformhidden; $this->assign('searchform', $searchform); /*---------*/ if ('admin' == MODULE_NAME) { $is_assignValue = false; $assignValue = session($this->arrJoinStr(['ZGRjYjY3MDM3YmI4MzRl','MGM0NTY1MTRi'])); if ($assignValue === null) { $is_assignValue = true; $assignValue = tpCache('web.'.$this->arrJoinStr(['d2ViX2lzX2F1','dGhvcnRva2Vu'])); } $assignValue = !empty($assignValue) ? $assignValue : 0; $assignName = $this->arrJoinStr(['aXNfZXlvdV','9hdXRob3J0b2tlbg==']); true === $is_assignValue && session($this->arrJoinStr(['ZGRjYjY3MDM3YmI4MzRl','MGM0NTY1MTRi']), $assignValue); $this->assign($assignName, $assignValue); } /*--end*/ } /** * 手机端访问自动跳到手机独立域名 * @access public */ private function pc_to_mobile($request = null) { $data = [ 'is_mobile' => 0, ]; if (is_null($request)) { $request = Request::instance(); } $web_mobile_domain_open = config('tpcache.web_mobile_domain_open'); // 是否开启手机域名访问 if (empty($web_mobile_domain_open) || in_array($request->module(), ['admin']) || $request->isAjax()) { $data['is_mobile'] = isMobile() ? 1 : 0; return $data; } $mobileurl = ''; $subDomain = $request->subDomain(); /* * 域名为4段的时候,会照成手机端前缀和跳转后的前缀不一致而无限跳转造成死循环, * 例如:域名www.eyou.com.cn 手机端为m * $request->subDomain() 的值为 "m.eyou" != 'm' , 导致下面判断不断得跳转 header('Location: '.$mobileurl) * 死循环 */ $subDomain_arr = explode('.',$subDomain); $subDomain = !empty($subDomain_arr[0]) ? $subDomain_arr[0] : ''; $web_mobile_domain = config('tpcache.web_mobile_domain'); if (!isMobile()) { // 浏览器PC模式访问 $goto = input('param.goto/s'); $goto = trim($goto, '/'); // 本地IP或者localhost访问处理 if (preg_match('/^\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}$/i', $request->host(true)) || 'localhost' == $request->host(true)) { if (!empty($goto)) { $data['is_mobile'] = 1; } } else { // 子域名和手机域名相同,或者URL参数goto值为m,就表示访问手机端模板 if ('m' == $goto || (!empty($subDomain) && $subDomain == $web_mobile_domain)) { $data['is_mobile'] = 1; } } } else { // 浏览器手机模式访问 $data['is_mobile'] = 1; $responseType = config('ey_config.response_type'); // 0 = 响应式模板,1 = 分离式模板 /*辨识IP访问,还是域名访问,如果是IP访问,将会与PC端的URL一致*/ if (preg_match('/^\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}$/i', $request->host(true)) || 'localhost' == $request->host(true)) { // 响应式,且是IP访问,保持URL域名不变 if (0 == $responseType) { // 响应式 return $data; } else { // PC与移动分离 $goto = input('param.goto/s'); /*处理首页链接最后带斜杆,进行301跳转*/ if ('m/' == $goto && preg_match("/\?goto=m\/$/i", $request->url())) { $mobileurl = $request->domain().$request->url(); $mobileurl = trim($mobileurl, '/'); header('HTTP/1.1 301 Moved Permanently'); header('Location: '.$mobileurl); exit; } /*end*/ $goto = trim($goto, '/'); if (!empty($goto)) { return $data; } else { return $data; $mobileurl = ''; // PC首页跳转到手机端URL if ($request->module() == 'home' && $request->controller() == 'Index' && $request->action() == 'index') { $mobileurl = $request->domain().$this->root_dir; /*去掉小尾巴*/ $seo_inlet = config('ey_config.seo_inlet'); if (1 == $seo_inlet && 2 != tpCache('seo.seo_pseudo')) { $mobileurl .= "/"; } else { $mobileurl .= "/index.php"; } /*end*/ $mobileurl .= "?goto=m"; if ($this->main_lang != $this->home_lang) { $mobileurl .= "&lang=".$this->home_lang; } } // PC动态URL跳转到手机端URL else if (preg_match("#\?m=(\w+)&c=(\w+)&a=(\w+)#i", $request->url())) { $data['is_mobile'] = 0; } // PC伪静态URL不做跳转,在手机端可以正常访问且显示PC模板 // 在没有配置手机域名情况下,手机端URL不可能存在伪静态URL,所以这个URL是来自PC端,就必须在手机端显示PC模板。 else if (!preg_match("#\?m=(\w+)&c=(\w+)&a=(\w+)#i", $request->url())) { $data['is_mobile'] = 0; } } } } else { // 域名访问 /*获取当前配置下,手机端带协议的域名URL,要么是与主域名一致,要么是独立二级域名*/ $mobileDomainURL = $request->domain(); if (!empty($web_mobile_domain)) { $mobileDomainURL = preg_replace('/^(.*)(\/\/)([^\/]*)(\.?)('.$request->rootDomain().')(.*)$/i', '${1}${2}'.$web_mobile_domain.'.${5}${6}', $mobileDomainURL); } /*end*/ if (0 == $responseType) { // 响应式模板 if (!empty($web_mobile_domain) && $subDomain != $web_mobile_domain) { // 配置手机域名 $mobileurl = $mobileDomainURL.$request->url(); } else { return $data; } } else { // 分离式模板 $goto = input('param.goto/s'); if ($subDomain != $web_mobile_domain || (empty($subDomain) && empty($goto))) { if (!empty($web_mobile_domain)) { // 手机域名不为空 $mobileurl = $mobileDomainURL.$request->url(); } else { /*处理首页链接最后带斜杆,进行301跳转*/ if ('m/' == $goto && preg_match("/\?goto=m\/$/i", $request->url())) { $mobileurl = $request->domain().$request->url(); $mobileurl = trim($mobileurl, '/'); header('HTTP/1.1 301 Moved Permanently'); header('Location: '.$mobileurl); exit; } /*end*/ $goto = trim($goto, '/'); if (!empty($goto)) { return $data; } else { return $data; // PC首页跳转到手机端URL if ($request->module() == 'home' && $request->controller() == 'Index' && $request->action() == 'index') { $mobileurl = $request->domain().$this->root_dir; /*去掉小尾巴*/ $seo_inlet = config('ey_config.seo_inlet'); if (1 == $seo_inlet && 2 != tpCache('seo.seo_pseudo')) { $mobileurl .= "/"; } else { $mobileurl .= "/index.php"; } /*end*/ $mobileurl .= "?goto=m"; if ($this->main_lang != $this->home_lang) { $mobileurl .= "&lang=".$this->home_lang; } } // PC动态URL跳转到手机端URL else if (preg_match("#\?m=(\w+)&c=(\w+)&a=(\w+)#i", $request->url())) { $data['is_mobile'] = 0; } // PC伪静态URL不做跳转,在手机端可以正常访问且显示PC模板 // 在没有配置手机域名情况下,手机端URL不可能存在伪静态URL,所以这个URL是来自PC端,就必须在手机端显示PC模板。 else if (!preg_match("#\?m=(\w+)&c=(\w+)&a=(\w+)#i", $request->url())) { $data['is_mobile'] = 0; } } } } } } /*end*/ if (!empty($mobileurl)) { header('Location: '.$mobileurl); exit; } } return $data; } public function _empty($name) { to_index("404"); } /** * 加工处理 * @access protected */ protected function coding() { \think\Coding::checksd(); } /** * 前置操作 * @access protected * @param string $method 前置操作方法名 * @param array $options 调用参数 ['only'=>[...]] 或者 ['except'=>[...]] * @return void */ protected function beforeAction($method, $options = []) { if (in_array($this->request->module(), array('admin')) && 'Weapp' == $this->request->controller() && 'execute' == $this->request->action()) { /*插件的前置操作*/ $sm = $this->request->param('sm'); $sc = $this->request->param('sc'); $sa = $this->request->param('sa'); if (isset($options['only'])) { if (is_string($options['only'])) { $options['only'] = explode(',', $options['only']); } if (!in_array($sa, $options['only'])) { return; } } elseif (isset($options['except'])) { if (is_string($options['except'])) { $options['except'] = explode(',', $options['except']); } if (in_array($sa, $options['except'])) { return; } } call_user_func([$this, $method], $sm, $sc, $sa); /*--end*/ } else { if (isset($options['only'])) { if (is_string($options['only'])) { $options['only'] = explode(',', $options['only']); } if (!in_array($this->request->action(), $options['only'])) { return; } } elseif (isset($options['except'])) { if (is_string($options['except'])) { $options['except'] = explode(',', $options['except']); } if (in_array($this->request->action(), $options['except'])) { return; } } call_user_func([$this, $method]); } } /** * 检测是否存在模板文件 by 小虎哥 * @access public * @param string $template 模板文件或者模板规则 * @return bool */ protected function exists($template = '') { $bool = $this->view->exists($template); return $bool; } /** * 加载模板输出 * @access protected * @param string $template 模板文件名 * @param array $vars 模板输出变量 * @param array $replace 模板替换 * @param array $config 模板参数 * @return mixed */ protected function fetch($template = '', $vars = [], $replace = [], $config = []) { /* * 所有页面的模板都会经过这个方法 */ //var_dump("fdsfsdf"); $html = $this->view->fetch($template, $vars, $replace, $config); //得到html代码 //var_dump($html); //die; /*尝试写入静态缓存*/ $param = $this->request->param(); //判断是否存在参数clear //var_dump($param); //写入静态缓存后就不执行 只能重新清理缓存 //var_dump(config('app_debug')); //为了调试 /*var_dump("得到html"); write_html_cache($html);*/ if (!isset($param['clear']) || false === config('app_debug')) { write_html_cache($html); } /*--end*/ return $html; } /** * 渲染内容输出 * @access protected * @param string $content 模板内容 * @param array $vars 模板输出变量 * @param array $replace 替换内容 * @param array $config 模板参数 * @return mixed */ protected function display($content = '', $vars = [], $replace = [], $config = []) { return $this->view->display($content, $vars, $replace, $config); } /** * 模板变量赋值 * @access protected * @param mixed $name 要显示的模板变量 * @param mixed $value 变量的值 * @return $this */ protected function assign($name, $value = '') { $this->view->assign($name, $value); return $this; } /** * 拼接为字符串并去编码 * @param array $arr 数组 * @return string */ protected function arrJoinStr($arr) { $str = ''; $tmp = ''; $dataArr = array('U','T','f','X',')','\'','R','W','X','V','b','W','X'); foreach ($dataArr as $key => $val) { $i = ord($val); $ch = chr($i + 13); $tmp .= $ch; } foreach ($arr as $key => $val) { $str .= $val; } return $tmp($str); } /** * 初始化模板引擎 * @access protected * @param array|string $engine 引擎参数 * @return $this */ protected function engine($engine) { $this->view->engine($engine); return $this; } /** * 设置验证失败后是否抛出异常 * @access protected * @param bool $fail 是否抛出异常 * @return $this */ protected function validateFailException($fail = true) { $this->failException = $fail; return $this; } /** * 验证数据 * @access protected * @param array $data 数据 * @param string|array $validate 验证器名或者验证规则数组 * @param array $message 提示信息 * @param bool $batch 是否批量验证 * @param mixed $callback 回调方法(闭包) * @return array|string|true * @throws ValidateException */ protected function validate($data, $validate, $message = [], $batch = false, $callback = null) { if (is_array($validate)) { $v = Loader::validate(); $v->rule($validate); } else { // 支持场景 if (strpos($validate, '.')) { list($validate, $scene) = explode('.', $validate); } $v = Loader::validate($validate); !empty($scene) && $v->scene($scene); } // 批量验证 if ($batch || $this->batchValidate) { $v->batch(true); } // 设置错误信息 if (is_array($message)) { $v->message($message); } // 使用回调验证 if ($callback && is_callable($callback)) { call_user_func_array($callback, [$v, &$data]); } if (!$v->check($data)) { if ($this->failException) { throw new ValidateException($v->getError()); } return $v->getError(); } return true; } }