thinkphp 3.2.3 - Dispatcher.class.php 解析(转发器)

class Dispatcher {
    public static function dispatch() {

        $varPath        =   C(‘VAR_PATHINFO‘); // ‘s‘
        $varAddon       =   C(‘VAR_ADDON‘); // ‘addon‘
        $varModule      =   C(‘VAR_MODULE‘); // ‘m‘
        $varController  =   C(‘VAR_CONTROLLER‘); // ‘c‘
        $varAction      =   C(‘VAR_ACTION‘); // ‘a‘
        $urlCase        =   C(‘URL_CASE_INSENSITIVE‘); // true

        if(isset($_GET[$varPath])) { // 判断URL里面是否有兼容模式参数
            $_SERVER[‘PATH_INFO‘] = $_GET[$varPath];
            unset($_GET[$varPath]);
        }elseif(IS_CLI){ // CLI模式下 index.php module/controller/action/params/...
            $_SERVER[‘PATH_INFO‘] = isset($_SERVER[‘argv‘][1]) ? $_SERVER[‘argv‘][1] : ‘‘;
        }

        // 开启子域名部署
        if(C(‘APP_SUB_DOMAIN_DEPLOY‘)) {
            $rules      = C(‘APP_SUB_DOMAIN_RULES‘);
            if(isset($rules[$_SERVER[‘HTTP_HOST‘]])) { // 完整域名或者IP配置
                define(‘APP_DOMAIN‘,$_SERVER[‘HTTP_HOST‘]); // 当前完整域名
                $rule = $rules[APP_DOMAIN];
            }else{
                // $_SERVER[‘HTTP_HOST‘] == ‘www0.www1.domain.com.cn‘
                if(strpos(C(‘APP_DOMAIN_SUFFIX‘),‘.‘)){ // com.cn net.cn
                    $domain = array_slice(explode(‘.‘, $_SERVER[‘HTTP_HOST‘]), 0, -3); // $domain = array(‘www0‘,‘www1‘);
                }else{
                    $domain = array_slice(explode(‘.‘, $_SERVER[‘HTTP_HOST‘]), 0, -2);
                }
                if(!empty($domain)) {
                    $subDomain = implode(‘.‘, $domain); // $domain = array(‘www0‘,‘www1‘);
                    define(‘SUB_DOMAIN‘,$subDomain); // 当前完整子域名
                    $domain2   = array_pop($domain); // 二级域名 // www1
                    if($domain) { // 存在三级域名
                        $domain3 = array_pop($domain); // www0
                    }
                    if(isset($rules[$subDomain])) { // 子域名 "www0.www1"
                        $rule = $rules[$subDomain];
                    }elseif(isset($rules[‘*.‘ . $domain2]) && !empty($domain3)){ // 泛三级域名  $rules[‘*.www1‘]
                        $rule = $rules[‘*.‘ . $domain2];
                        $panDomain = $domain3;
                    }elseif(isset($rules[‘*‘]) && !empty($domain2) && ‘www‘ != $domain2 ){ // 泛二级域名
                        $rule      = $rules[‘*‘];
                        $panDomain = $domain2;
                    }
                }
            }

            if(!empty($rule)) {
                // 子域名部署规则 ‘子域名‘=>array(‘模块名[/控制器名]‘,‘var1=a&var2=b‘);
                if(is_array($rule)){
                    list($rule,$vars) = $rule;
                }
                $array      =   explode(‘/‘,$rule);
                // 模块绑定
                define(‘BIND_MODULE‘,array_shift($array));
                // 控制器绑定
                if(!empty($array)) {
                    $controller  =   array_shift($array);
                    if($controller){
                        define(‘BIND_CONTROLLER‘,$controller);
                    }
                }
                if(isset($vars)) { // 传入参数
                    parse_str($vars,$parms);
                    if(isset($panDomain)){
                        $pos = array_search(‘*‘, $parms);
                        if(false !== $pos) {
                            // 泛域名作为参数
                            $parms[$pos] = $panDomain;
                        }
                    }
                    $_GET   =  array_merge($_GET,$parms);
                }
            }
        }
        // 分析PATHINFO信息
        if(!isset($_SERVER[‘PATH_INFO‘])) {
            $types   =  explode(‘,‘,C(‘URL_PATHINFO_FETCH‘)); // ORIG_PATH_INFO,REDIRECT_PATH_INFO,REDIRECT_URL
            foreach ($types as $type){
                if(0===strpos($type,‘:‘)) {// 支持函数判断
                    $_SERVER[‘PATH_INFO‘] =   call_user_func(substr($type,1));
                    break;
                }elseif(!empty($_SERVER[$type])) {
                    $_SERVER[‘PATH_INFO‘] = (0 === strpos($_SERVER[$type],$_SERVER[‘SCRIPT_NAME‘]))?
                    substr($_SERVER[$type], strlen($_SERVER[‘SCRIPT_NAME‘]))   :  $_SERVER[$type];
                    break;
                }
            }
        }

        $depr = C(‘URL_PATHINFO_DEPR‘); // ‘/‘
        define(‘MODULE_PATHINFO_DEPR‘,  $depr);

        if(empty($_SERVER[‘PATH_INFO‘])) {
            $_SERVER[‘PATH_INFO‘] = ‘‘;
            define(‘__INFO__‘,‘‘);
            define(‘__EXT__‘,‘‘);
        }else{
            define(‘__INFO__‘,trim($_SERVER[‘PATH_INFO‘],‘/‘));
            // URL后缀
            define(‘__EXT__‘, strtolower(pathinfo($_SERVER[‘PATH_INFO‘],PATHINFO_EXTENSION)));
            $_SERVER[‘PATH_INFO‘] = __INFO__;
            if(!defined(‘BIND_MODULE‘) && (!C(‘URL_ROUTER_ON‘) || !Route::check())){  // 没有绑定模块 && (没有开启路由,默认false || 没有匹配到“静态路由/动态路由”)
                if (__INFO__ && C(‘MULTI_MODULE‘)){ // 获取模块名  ,是否允许多模块,true
                    $paths      =   explode($depr,__INFO__,2);
                    $allowList  =   C(‘MODULE_ALLOW_LIST‘); // 允许的模块列表
                    $module     =   preg_replace(‘/\.‘ . __EXT__ . ‘$/i‘, ‘‘,$paths[0]);
                    if( empty($allowList) || (is_array($allowList) && in_array_case($module, $allowList))){ // 没有配置“允许模块”,或者在“允许模块”列表中
                        $_GET[$varModule]       =   $module; // $_GET[‘m‘]  =   $module;
                        $_SERVER[‘PATH_INFO‘]   =   isset($paths[1])?$paths[1]:‘‘;
                    }
                }
            }
        }

        // URL常量
        define(‘__SELF__‘,strip_tags($_SERVER[C(‘URL_REQUEST_URI‘)])); // ‘REQUEST_URI‘

        // 获取模块名称
        define(‘MODULE_NAME‘, defined(‘BIND_MODULE‘)? BIND_MODULE : self::getModule($varModule)); // 获取模块名,如果是别名模块,要进行转换

        // 检测模块是否存在
        if( MODULE_NAME && (defined(‘BIND_MODULE‘) || !in_array_case(MODULE_NAME,C(‘MODULE_DENY_LIST‘)) ) && is_dir(APP_PATH.MODULE_NAME)){ // 匹配到模块 && (有定义绑定模块 || 不在拒绝列表中 && 模块目录存在)
            // 定义当前模块路径
            define(‘MODULE_PATH‘, APP_PATH.MODULE_NAME.‘/‘);
            // 定义当前模块的模版缓存路径
            C(‘CACHE_PATH‘,CACHE_PATH.MODULE_NAME.‘/‘);
            // 定义当前模块的日志目录
            C(‘LOG_PATH‘,  realpath(LOG_PATH).‘/‘.MODULE_NAME.‘/‘);

            // 模块检测
            Hook::listen(‘module_check‘);  // 执行处理 module_check标记的behavior

            // 加载模块配置文件
            if(is_file(MODULE_PATH.‘Conf/config‘.CONF_EXT))
                C(load_config(MODULE_PATH.‘Conf/config‘.CONF_EXT));
            // 加载应用模式对应的配置文件
            if(‘common‘ != APP_MODE && is_file(MODULE_PATH.‘Conf/config_‘.APP_MODE.CONF_EXT))
                C(load_config(MODULE_PATH.‘Conf/config_‘.APP_MODE.CONF_EXT));
            // 当前应用状态对应的配置文件
            if(APP_STATUS && is_file(MODULE_PATH.‘Conf/‘.APP_STATUS.CONF_EXT))
                C(load_config(MODULE_PATH.‘Conf/‘.APP_STATUS.CONF_EXT));

            // 加载模块别名定义
            if(is_file(MODULE_PATH.‘Conf/alias.php‘))
                Think::addMap(include MODULE_PATH.‘Conf/alias.php‘);
            // 加载模块tags文件定义
            if(is_file(MODULE_PATH.‘Conf/tags.php‘))
                Hook::import(include MODULE_PATH.‘Conf/tags.php‘);
            // 加载模块函数文件
            if(is_file(MODULE_PATH.‘Common/function.php‘))
                include MODULE_PATH.‘Common/function.php‘;

            $urlCase        =   C(‘URL_CASE_INSENSITIVE‘);
            // 加载模块的扩展配置文件
            load_ext_file(MODULE_PATH);
        }else{
            E(L(‘_MODULE_NOT_EXIST_‘).‘:‘.MODULE_NAME);
        }

        if(!defined(‘__APP__‘)){
            $urlMode        =   C(‘URL_MODEL‘);
            if($urlMode == URL_COMPAT ){// 兼容模式判断
                define(‘PHP_FILE‘,_PHP_FILE_.‘?‘.$varPath.‘=‘);
            }elseif($urlMode == URL_REWRITE ) {
                $url    =   dirname(_PHP_FILE_);
                if($url == ‘/‘ || $url == ‘\\‘)
                    $url    =   ‘‘;
                define(‘PHP_FILE‘,$url);
            }else {
                define(‘PHP_FILE‘,_PHP_FILE_);
            }
            // 当前应用地址
            define(‘__APP__‘,strip_tags(PHP_FILE));
        }
        // 模块URL地址
        $moduleName    =   defined(‘MODULE_ALIAS‘)? MODULE_ALIAS : MODULE_NAME;
        define(‘__MODULE__‘,(defined(‘BIND_MODULE‘) || !C(‘MULTI_MODULE‘))? __APP__ : __APP__.‘/‘.($urlCase ? strtolower($moduleName) : $moduleName));

        if(‘‘ != $_SERVER[‘PATH_INFO‘] && (!C(‘URL_ROUTER_ON‘) ||  !Route::check()) ){   // 检测路由规则 如果没有则按默认规则调度URL  // (没有开启路由,默认false || 没有匹配到“静态路由/动态路由”)
            Hook::listen(‘path_info‘);
            // 检查禁止访问的URL后缀
            if(C(‘URL_DENY_SUFFIX‘) && preg_match(‘/\.(‘.trim(C(‘URL_DENY_SUFFIX‘),‘.‘).‘)$/i‘, $_SERVER[‘PATH_INFO‘])){  // 不是拒绝的后缀
                send_http_status(404);
                exit;
            }

            // 去除URL后缀
            $_SERVER[‘PATH_INFO‘] = preg_replace(C(‘URL_HTML_SUFFIX‘)? ‘/\.(‘.trim(C(‘URL_HTML_SUFFIX‘),‘.‘).‘)$/i‘ : ‘/\.‘.__EXT__.‘$/i‘, ‘‘, $_SERVER[‘PATH_INFO‘]);

            $depr   =   C(‘URL_PATHINFO_DEPR‘);
            $paths  =   explode($depr,trim($_SERVER[‘PATH_INFO‘],$depr));

            if(!defined(‘BIND_CONTROLLER‘)) {// 获取控制器,没定义绑定控制器
                if(C(‘CONTROLLER_LEVEL‘)>1){// 控制器层次
                    $_GET[$varController]   =   implode(‘/‘,array_slice($paths,0,C(‘CONTROLLER_LEVEL‘)));
                    $paths  =   array_slice($paths, C(‘CONTROLLER_LEVEL‘));
                }else{
                    $_GET[$varController]   =   array_shift($paths); // $_GET[‘c‘]  =   array_shift($paths);
                }
            }
            // 获取操作
            if(!defined(‘BIND_ACTION‘)){ // 没有定义绑定action
                $_GET[$varAction]  =   array_shift($paths);  // $_GET[‘a‘]  =   array_shift($paths);
            }
            // 解析剩余的URL参数
            $var  =  array();
            if(C(‘URL_PARAMS_BIND‘) && 1 == C(‘URL_PARAMS_BIND_TYPE‘)){
                // URL参数按顺序绑定变量
                $var    =   $paths;
            }else{
                preg_replace_callback(‘/(\w+)\/([^\/]+)/‘, function($match) use(&$var){$var[$match[1]]=strip_tags($match[2]);}, implode(‘/‘,$paths));
            }
            $_GET   =  array_merge($var,$_GET);
        }
        // 获取控制器的命名空间(路径)
        define(‘CONTROLLER_PATH‘,   self::getSpace($varAddon,$urlCase)); // 插件  $_GET[‘addon‘]
        // 获取控制器和操作名
        define(‘CONTROLLER_NAME‘,   defined(‘BIND_CONTROLLER‘)? BIND_CONTROLLER : self::getController($varController,$urlCase)); //  $_GET[‘c‘] 获取控制器名称,如果有别名,要进行转换
        define(‘ACTION_NAME‘,       defined(‘BIND_ACTION‘)? BIND_ACTION : self::getAction($varAction,$urlCase)); //  $_GET[‘a‘] 获取action名称,如果有别名,要进行转换

        // 当前控制器的UR地址
        $controllerName    =   defined(‘CONTROLLER_ALIAS‘)? CONTROLLER_ALIAS : CONTROLLER_NAME;
        define(‘__CONTROLLER__‘,__MODULE__.$depr.(defined(‘BIND_CONTROLLER‘)? ‘‘: ( $urlCase ? parse_name($controllerName) : $controllerName )) );

        // 当前操作的URL地址
        define(‘__ACTION__‘,__CONTROLLER__.$depr.(defined(‘ACTION_ALIAS‘)?ACTION_ALIAS:ACTION_NAME));

        // 保证$_REQUEST正常取值
        $_REQUEST = array_merge($_POST,$_GET,$_COOKIE);    // -- 加了$_COOKIE.  保证哦..
    }

    /**
     * 获得控制器的命名空间路径 便于插件机制访问
     */
    static private function getSpace($var,$urlCase) {
        $space  =   !empty($_GET[$var])?strip_tags($_GET[$var]):‘‘;
        unset($_GET[$var]);
        return $space;
    }
}
时间: 2024-10-10 20:04:24

thinkphp 3.2.3 - Dispatcher.class.php 解析(转发器)的相关文章

thinkphp 3.2.3 - Think.class.php 解析

class Think { public static function start() { // 注册AUTOLOAD方法 spl_autoload_register('Think\Think::autoload'); // 设定错误和异常处理 register_shutdown_function('Think\Think::fatalError'); set_error_handler('Think\Think::appError'); set_exception_handler('Thin

thinkphp 3.2.3 - App.class.php 解析

class App { public static function init() { load_ext_file(COMMON_PATH); // { // /home/www/www.domain.com/myapp/Common/Common/<LOAD_EXT_FILE,field1>.php // /home/www/www.domain.com/myapp/Common/Conf/<LOAD_EXT_CONFIG,config1>.php // } // URL调度 D

thinkphp where语句中的变量不解析

解决办法:将where语句中的变量用单引号引起来就行了,亲测可用 举例:将$System->where("keywords=$keywords")->find();改为$System->where("keywords='$keywords'")->find();即可 原因:不明

利用asyncore.dispatcher写的端口转发器

创建三个继承类,PortForwarder用于监听本地,Receiver与PortForwarder套接字相连,用于接受本地请求,发给数据给远程主机,Sender与Receiver相互包含,用于接受远程主机数据和发送数据到本地.大致形成这样的关系:本地主机---PortForwarder---Receiver---Sender---远程主机 1 #!/usr/bin/env python 2 #-*- coding:utf-8 -*- 3 4 import argparse 5 import a

ThinkPHP框架系统源码解析——URL路由解析

1 一.ThinkPHP框架目录 2 /test/index.php //项目入口文件 3 /ThinkPHP/ThinkPHP.php //框架入口文件 4 5 Common 框架公共文件目录(函数库) 6 ThinkPHP/Common/runtime.php //框架初次运行文件 7 ThinkPHP/Common/common.php //框架基础函数库 8 ThinkPHP/Common/functions.php //标准模式公共函数库 9 10 Conf 框架配置文件目录 11 T

ThinkPHP 3.2.3 多模块应用的配置

多模块 在 ThinkPHP 3.2.3 中,默认的应用目录是 ./Application,下面的默认模块是 Home 模块,如果此时需要添加一个 Admin 模块用于后台应用,根据手册 http://www.kancloud.cn/manual/thinkphp/1696 中 “自动生成模块目录” 的部分的说明,在默认的入口文件 ./index.php 中添加: // 绑定Admin模块到当前入口文件 define('BIND_MODULE','Admin'); 此时运行 http://ser

Thinkphp 源码分析

ThinkPHP是国内非常火的一个轻量级框架,采用MVC模式,结构写的非常好,今天 大象 带大家走一下ThinkPHP框架系统运行流程,我将跟随ThinkPHP的执行进行代码讲解,第一课,先讲解URL路由解析 一.ThinkPHP框架目录 /test/index.php //项目入口文件 /ThinkPHP/ThinkPHP.php //框架入口文件 Common 框架公共文件目录(函数库) ThinkPHP/Common/runtime.php //框架初次运行文件 ThinkPHP/Comm

ThinkPHP 框架2.1,2.2和3.0版本URL命令执行漏洞

在开启了Lite模式后,在ThinkPHP/extend/Mode/Lite/Dispatcher.class.php中第73行: // 解析剩余的URL参数 $res = preg_replace('@(\w+)'.$depr.'([^'.$depr.'\/]+)@e', '$var[\'\\1\']="\\2\;"', implode($depr,$paths)); 有两点:一是加入了preg_replace使用了e修饰符,二是'$var[\'\\1\']="\\2\;&

lnmp支持thinkphp

lnmp环境配置好后,直接把thinkphp放到相应的目录里进行解析,是不行的,thinkphp默认是用apache的,相应目录下有个.htacess是关于apache重写的,lnmp是用nginx,不适用,需要修改nginx 配置文件里的server{}里面的 server { listen 80; server_name domain; error_page 500 502 503 504 /50x.html; location = /50x.html { root html; } loca