ÓÅ·¢¹ú¼ÊÍøÕ¾¹ÙÍøADLab£ºThinkPHP5Ô¶³Ì´úÂëÖ´ÐÐÎó²îÆÊÎö

Ðû²¼Ê±¼ä 2018-12-11
Îó²îÏÈÈÝ

2018Äê12ÔÂ9ÈÕ£¬ThinkPHPÍŶÓÐû²¼ÁËÒ»¸ö²¹¶¡¸üУ¬ÐÞ¸´ÁËÒ»´¦ÓÉÓÚ·ÓÉÆÊÎöȱÏݵ¼ÖµĴúÂëÖ´ÐÐÎó²î¡£¸ÃÎó²îΣº¦Ë®Æ½ºÜÊǸߣ¬Ä¬ÈÏÇéÐÎÉèÖü´¿Éµ¼ÖÂÔ¶³Ì´úÂëÖ´ÐС£¾­ÓÉÓÅ·¢¹ú¼ÊÍøÕ¾¹ÙÍøADLabÇå¾²Ñо¿Ô±¶ÔThinkPHPµÄ56¸öС°æ±¾µÄÔ´ÂëÆÊÎöºÍÑéÖ¤£¬È·¶¨ÏêϸÊÜÓ°ÏìµÄ°æ±¾Îª:


ThinkPHP 5.0.5-5.0.22

ThinkPHP 5.1.0-5.1.30


Îó²î¸´ÏÖ


ÍâµØÇéÐνÓÄÉThinkPHP 5.1.29+PHP7+Apache¾ÙÐи´ÏÖ¡£×°ÖÃÇéÐκóÖ±½Ó»á¼ûPOC¸ø¶¨µÄURL¼´¿ÉÖ´ÐÐphpinfo()£¬ÈçͼËùʾ¡£

ÓÅ·¢¹ú¼Ê¡¤ËæÓŶø¶¯Ò»´¥¼´·¢


Îó²îÆÊÎö


ÒÔ5.1.29°æ±¾¾ÙÐÐÆÊÎö£¬Ê×ÏÈ¿´È¡Â·Óɵĺ¯Êýpathinfo£º

library/think/Request.php:678

public function pathinfo()
    {
        if (is_null($this->pathinfo)) {
            if (isset($_GET[$this->config['var_pathinfo']])) {
                // ÅжÏURLÄÚÀïÊÇ·ñÓмæÈÝģʽ²ÎÊý
                $pathinfo = $_GET[$this->config['var_pathinfo']];
                unset($_GET[$this->config['var_pathinfo']]);
            } elseif ($this->isCli()) {
                // CLIģʽÏ index.php module/controller/action/params/...
                $pathinfo = isset($_SERVER['argv'][1]) ? $_SERVER['argv'][1] : '';
            } elseif ('cli-server' == PHP_SAPI) {
                $pathinfo = strpos($this->server('REQUEST_URI'), '?') ? strstr($this->server('REQUEST_URI'), '?', true) : $this->server('REQUEST_URI');
            } elseif ($this->server('PATH_INFO')) {
                $pathinfo = $this->server('PATH_INFO');
            }
 
            // ÆÊÎöPATHINFOÐÅÏ¢
            if (!isset($pathinfo)) {
                foreach ($this->config['pathinfo_fetch'] as $type) {
                    if ($this->server($type)) {
                        $pathinfo = (0 === strpos($this->server($type), $this->server('SCRIPT_NAME'))) ?
                        substr($this->server($type), strlen($this->server('SCRIPT_NAME'))) : $this->server($type);
                        break;
                    }
                }
            }
 
            $this->pathinfo = empty($pathinfo) || '/' == $pathinfo ? '' : ltrim($pathinfo, '/');
        }
 
        return $this->pathinfo;
    }


¸Ã·Óɺ¯ÊýÖÐ$this->config['var_pathinfo']ÊÇÉèÖÃÎļþµÄĬÈÏÖµ£¬Æä³õʼ»¯´úÂëÈçÏ£¬ÖµÎª¡¯s¡¯:


ÓÅ·¢¹ú¼Ê¡¤ËæÓŶø¶¯Ò»´¥¼´·¢



µ±ÇëÇó±¨ÎÄ°üÀ¨$_GET['s']£¬¾ÍÈ¡ÆäÖµ×÷Ϊpathinfo£¬²¢·µ»Øpathinfo¸øŲÓú¯Êý¡£

ÆÊÎö·¢Ã÷pathinfoº¯Êý±»library/think/Request.php:716ÖеÄpathº¯ÊýŲÓãº

public function path()
{
    if (is_null($this->path)) {
        $suffix   = $this->config['url_html_suffix'];
        $pathinfo = $this->pathinfo();


        if (false === $suffix) {
            // եȡα¾²Ì¬»á¼û
            $this->path = $pathinfo;
        } elseif ($suffix) {
            // È¥³ýÕý³£µÄURLºó׺
            $this->path = preg_replace('/\.(' . ltrim($suffix, '.') . ')$/i', '', $pathinfo);
        } else {
            // ÔÊÐíÈκκó׺»á¼û
            $this->path = preg_replace('/\.' . $this->ext() . '$/i', '', $pathinfo);
        }
    }
    return $this->path;
}

 

ÏÔÈ»£¬ÕâÀï$this->pathÔ´×Ôpathinfo£¬Òò´Ë¿ÉÒÔ±»¹¥»÷Õß¿ØÖÆ¡£¼ÌÐøÆÊÎö¸Ã±äÁ¿µÄת´ï£¬ÔÚlibrary/think/App.php:597Öб»ÒýÓãº

//public function routecheck()
$path = $this->request->path();
 
        // ÊÇ·ñÇ¿ÖÆ·ÓÉģʽ
        $must = !is_null($this->routeMust) ? $this->routeMust : $this->route->config('url_route_must');
 
        // ·Óɼì²â ·µ»ØÒ»¸öDispatch¹¤¾ß
        $dispatch = $this->route->check($path, $must);
 
        if (!empty($routeKey)) {
            try {
                if ($option) {
                    $this->cache->connect($option)->tag('route_cache')->set($routeKey, $dispatch);
                } else {
                    $this->cache->tag('route_cache')->set($routeKey, $dispatch);
                }
            } catch (\Exception $e) {
                // ±£´æ±Õ°üµÄʱ¼ä»º´æÎÞЧ
            }
        }
 

        return $dispatch;


ÕâÀïÊǾÙÐзÓɼì²â£¬¹¥»÷Õ߿ɿصÄ$path±»×ª´ï¸øÁËÈçϵÄcheckº¯Êý£º

public function check($url, $must = false)
    {
        // ×Ô¶¯¼ì²âÓòÃû·ÓÉ
        $domain = $this->checkDomain();
        $url    = str_replace($this->config['pathinfo_depr'], '|', $url);
 
        $completeMatch = $this->config['route_complete_match'];
 
        $result = $domain->check($this->request, $url, $completeMatch);
 
        if (false === $result && !empty($this->cross)) {
            // ¼ì²â¿çÓò·ÓÉ
            $result = $this->cross->check($this->request, $url, $completeMatch);
        }
 
        if (false !== $result) {
            // ·ÓÉÆ¥Åä
            return $result;
        } elseif ($must) {
            // Ç¿ÖÆ·Óɲ»Æ¥ÅäÔòÅ׳öÒì³£
            throw new RouteNotFoundException();
        }
        // ĬÈÏ·ÓÉÆÊÎö
        return new UrlDispatch($this->request, $this->group, $url, [
            'auto_search' => $this->autoSearchController,
        ]);
    }


ÆÊÎö´úÂë¿ÉÖª£¬ÈôÊÇ¿ªÆôÁËÇ¿ÖÆ·ÓÉÔò»áÅ׳öÒì³££¬Ò²¾ÍÊǹٷ½Ëù˵µÄ¸ÃÎó²îÔÚ¿ªÆôÇ¿ÖÆ·ÓɵÄÇéÐÎϲ»ÊÜÓ°Ï죨ĬÈϲ»¿ªÆô£©¡£


Checkº¯Êý×îºóʵÀý»¯Ò»¸öUrlDispatch¹¤¾ß£¬½«$urlת´ï¸øÁËÈ´¹¹º¯Êý¡£¼ÌÐøÆÊÎöUrlDispatchµÄ¸¸ÀàÒ²¾ÍÊÇDispatchÀàµÄ½á¹¹º¯Êý:

library/think/route/Dispatch.php:64


 public function __construct(Request $request, Rule $rule, $dispatch, $param = [], $code = null)
    {
        $this->request  = $request;
        $this->rule     = $rule;
        $this->app      = Container::get('app');
        $this->dispatch = $dispatch;
        $this->param    = $param;
        $this->code     = $code;
 
        if (isset($param['convert'])) {
            $this->convert = $param['convert'];
        }
    }


$dispatch±äÁ¿¿É¿Ø²¢¸³Öµ¸øÁË$this->dispatch£¬¾­Óɶà´Îº¯ÊýŲÓ÷µ»Ø£¬×îºóÈçϵÄUrlÀàµÄinit º¯Êý½«»á±»Å²ÓÃÀ´´¦Öóͷ£$this->dispatch¡£

class Url extends Dispatch
{
    public function init()
    {
        // ÆÊÎöĬÈϵÄURL¹æÔò
        $result = $this->parseUrl($this->dispatch);
 
        return (new Module($this->request, $this->rule, $result))->init();
    }
 
    public function exec()
    {}


ÕâÀïŲÓÃparseUrl¶Ô$this->dispatch¾ÙÐÐÆÊÎö£¬ÕâÊǸÃÎó²îµÄ½¹µãµãÖ®Ò»£º

protected function parseUrl($url)
{
    $depr = $this->rule->getConfig('pathinfo_depr');
    $bind = $this->rule->getRouter()->getBind();
 
    if (!empty($bind) && preg_match('/^[a-z]/is', $bind)) {
        $bind = str_replace('/', $depr, $bind);
        // ÈôÊÇÓÐÄ £¿é/¿ØÖÆÆ÷°ó¶¨
        $url = $bind . ('.' != substr($bind, -1) ? $depr : '') . ltrim($url, $depr);
    }
 
    list($path, $var) = $this->rule->parseUrlPath($url);
    if (empty($path)) {


ÕâÀïŲÓÃparseUrlPathº¯Êý¶Ô$url¾ÙÐÐÆÊÎö£¬¼ÌÐøÆÊÎö¸Ãº¯Êý£º


public function parseUrlPath($url)

    {
       ....
    ....
        } elseif (strpos($url, '/')) {
            // [Ä £¿é/¿ØÖÆÆ÷/²Ù×÷]
            $path = explode('/', $url);
        } elseif (false !== strpos($url, '=')) {
            // ²ÎÊý1=Öµ1&²ÎÊý2=Öµ2...
            $path = [];
            parse_str($url, $var);
        } else {
            $path = [$url];
        }
 
        return [$path, $var];
    }


ÏÔÈ»£¬$urlµÄÃûÌÃΪ¡°Ä £¿é/¿ØÖÆÆ÷/²Ù×÷¡±£¬$urlÖ§½âÐγÉÒ»¸öÊý×é´æµ½$path±äÁ¿Öв¢·µ»Øµ½Å²ÓÃÕß¡£


¼ÌÐøÆÊÎö·âװ·ÓɵĴúÂë:


library/think/route/dispatch/Url.php:48

 list($path, $var) = $this->rule->parseUrlPath($url);
    ...
    ...
        // ÆÊÎöÄ £¿é
        $module = $this->rule->getConfig('app_multi_module') ? array_shift($path) : null;
 
        if ($this->param['auto_search']) {
            $controller = $this->autoFindController($module, $path);
        } else {
            // ÆÊÎö¿ØÖÆÆ÷
            $controller = !empty($path) ? array_shift($path) : null;
        }
 
        // ÆÊÎö²Ù×÷
        $action = !empty($path) ? array_shift($path) : null;
        ...
        ...
        // ÉèÖÃÄ¿½ñÇëÇóµÄ²ÎÊý
        $this->request->setRouteVars($var);
 
        // ·âװ·ÓÉ
        $route = [$module, $controller, $action];
        return $route;


·ÓÉ·â×°·µ»Øµ½library/think/route/dispatch/Url.php:20


class Url extends Dispatch

{
    public function init()
    {
        // ÆÊÎöĬÈϵÄURL¹æÔò
        $result = $this->parseUrl($this->dispatch);
 
        return (new Module($this->request, $this->rule, $result))->init();
    }


$result¾ÍÊÇ·â×°ºÃµÄ·ÓÉÊý×飬ת´ï¸øÁËModuleµÄ½á¹¹º¯Êý¡£
ÓÉÓÚModuleÒ²ÊǼÌÐø×ÔDispatchÀֱ࣬½Ó¿´DispatchµÄ½á¹¹º¯Êý:


public function __construct(Request $request, Rule $rule, $dispatch, $param = [], $code = null)
    {
        $this->request  = $request;
        $this->rule     = $rule;
        $this->app      = Container::get('app');
        $this->dispatch = $dispatch;
        $this->param    = $param;
        $this->code     = $code;
 
        if (isset($param['convert'])) {
            $this->convert = $param['convert'];
        }
    }


$result¸³Öµ¸øÁË$this->dispatch¡£È»ºóŲÓÃModuleÀàµÄinitº¯Êý£º


public function init()
    {
        parent::init();
        $result = $this->dispatch;
  
        if ($this->rule->getConfig('app_multi_module')) {
            // ¶àÄ £¿é°²ÅÅ
            $module    = strip_tags(strtolower($result[0] ?: $this->rule->getConfig('default_module')));
            ...
            ...
            } elseif (!in_array($module, $this->rule->getConfig('deny_module_list')) && is_dir($this->app->getAppPath() . $module)) {
                $available = true;
            } 
           ...
           ...
            // Ä £¿é³õʼ»¯
            if ($module && $available) {
                // ³õʼ»¯Ä £¿é
                $this->request->setModule($module);
                $this->app->init($module);
            } else {
                throw new HttpException(404, 'module not exists:' . $module);
            }
        }
        // »ñÈ¡¿ØÖÆÆ÷Ãû
        $controller       = strip_tags($result[1] ?: $this->rule->getConfig('default_controller'));
        $this->controller = $convert ? strtolower($controller) : $controller;
        // »ñÈ¡²Ù×÷Ãû
        $this->actionName = strip_tags($result[2] ?: $this->rule->getConfig('default_action'));
        // ÉèÖÃÄ¿½ñÇëÇóµÄ¿ØÖÆÆ÷¡¢²Ù×÷
        $this->request
            ->setController(Loader::parseName($this->controller, 1))
            ->setAction($this->actionName);
 
        return $this;
    }


ÕâÀï±£´æµÚÒ»¸ö¶Ô$moduleµÄÅжÏ£¬ÐèÒªÈÃ$available¼´ÊÇtrue£¬Õâ¾ÍÐèÒªis_dir($this->app->getAppPath() . $module)½¨Éè¡£¹Ù·½demo¸ø³öµÄÄ £¿éÊÇindex£¬¶øÏÖʵ¿ª·¢³ÌÐò·×Æ綨±£´æ¸ÃÄ £¿éÃû£¬ÒÔÊǽṹpayloadʱÕâÀïÊÇÒ»¸ö×¢Öصã¡£


Öª×ãÕâ¸öÅжÏÌõ¼þºó£¬¼ÌÐøÆÊÎöºóÐøµÄ¿ØÖÆÁ÷»á½øÈëÈçÏÂmoduleµÄexecº¯Êý£ºlibrary/think/route/dispatch/Module.php:80


public function exec()

    {
        // ¼àÌýmodule_init
        $this->app['hook']->listen('module_init');
 
        try {
            // ʵÀý»¯¿ØÖÆÆ÷
            $instance = $this->app->controller($this->controller,
                $this->rule->getConfig('url_controller_layer'),
                $this->rule->getConfig('controller_suffix'),
                $this->rule->getConfig('empty_controller'));
 
            if ($instance instanceof Controller) {
                $instance->registerMiddleware();
            }
        } catch (ClassNotFoundException $e) {
            throw new HttpException(404, 'controller not exists:' . $e->getClass());
        }


ÆÊÎö·¢Ã÷£¬$this->controllerÊǹ¥»÷Õ߿ɿصÄ£¬²¢×ª´ï¸øÁËÈçϵÄcontrollerº¯Êý£¬¼ÌÐøÆÊÎö¸Ãº¯Êý£º


  public function controller($name, $layer = 'controller', $appendSuffix = false, $empty = '')
    {
        list($module, $class) = $this->parseModuleAndClass($name, $layer, $appendSuffix);
 
        if (class_exists($class)) {
            return $this->__get($class);
        } elseif ($empty && class_exists($emptyClass = $this->parseClass($module, $layer, $empty, $appendSuffix))) {
            return $this->__get($emptyClass);
        }
        throw new ClassNotFoundException('class not exists:' . $class, $class);
    }


ÔÚÕâÀnameÊǹ¥»÷Õ߿ɿصÄ£¬²¢×ª´ï¸øÁËÈçϵÄparseModuleAndClassº¯Êý£º


protected function parseModuleAndClass($name, $layer, $appendSuffix)

    {
        if (false !== strpos($name, '\\')) {
            $class  = $name;
            $module = $this->request->module();
        } else {
            if (strpos($name, '/')) {
                list($module, $name) = explode('/', $name, 2);
            } else {
                $module = $this->request->module();
            }
 
            $class = $this->parseClass($module, $layer, $name, $appendSuffix);
        }
 
        return [$module, $class];
    }


ÆÊÎö·¢Ã÷£¬µ±$name±£´æ·´Ð±¸Üʱ¾ÍÖ±½Ó½«$name¸³Öµ¸ø$class²¢·µ»Ø¡£ÏÔÈ»£¬¹¥»÷Õßͨ¹ý¿ØÖÆÊäÈë¾Í¿ÉÒÔ²Ù¿ØÀàµÄʵÀý»¯Àú³Ì£¬´Ó¶øÔì³É´úÂëÖ´ÐÐÎó²î¡£



²¹¶¡ÆÊÎö

ÔÚThinkPHP5.0.23ÒÔ¼°5.1.31°æ±¾ÖУ¬ÔöÌíÁ˶Ô$controllerµÄÕýÔò¹ýÂË£º


µ¼ÖÂÎÞ·¨ÔÙ´«Èë\think\appÕâÖÖÐÎʽµÄ¿ØÖÆÆ÷¡£


½á ÂÛ

´ËÎó²îÊÇÓÉÓÚ¿ò¼Ü¶Ô´«ÈëµÄ·ÓɲÎÊý¹ýÂ˲»ÑϿᣬµ¼Ö¹¥»÷Õß¿ÉÒÔ²Ù×÷·ÇÔ¤ÆڵĿØÖÆÆ÷ÀàÀ´Ô¶³ÌÖ´ÐдúÂë¡£½øÒ»²½ÆÊÎö·¢Ã÷£¬Ä³Ð©ThinkPHP°æ±¾²»ÊÜÒѹûÕæµÄPOCµÄÓ°Ï죬ÕâÊÇÓÉÓÚ¸ÃPOCȱ·¦ÍêÕûÐÔ˼Á¿¡£Òò´Ë£¬Ç¿ÁÒ½¨ÒéÓû§ÊµÊ±½«5.0.x°æ±¾Éý¼¶µ½5.0.23£¬½«5.1.x°æ±¾Éý¼¶µ½5.1.31£¬ÒÔÃâÔâÊܹ¥»÷¡£