阻止干净的URL影响页面HTML中的相对路径

时间:2012-06-18 17:38:26

标签: php relative-path clean-urls

我已在我的.htaccess

中使用以下内容实现了干净的网址
RewriteEngine on
RewriteCond %{SCRIPT_FILENAME} !-f
RewriteCond %{SCRIPT_FILENAME} !-d
RewriteRule ^(.*)$ index.php/$1 [L]

在我的index.php中,$_SERVER['PATH_INFO']被解析以确定所请求的页面和任何参数。

这一切都运行正常,但是对具有多个段的URL的任何请求,即

www.example.com/page/foo/bar

导致页面上的所有链接资产(css,js,images)和超链接中断,因为它们使用相对路径。具有隐含域的绝对路径(也称为使用“/”前置所有路径)是不可行的,因为除了自己的根URL之外,站点必须是可访问的并且作为父网站的子目录正常工作。我也无法硬编码所有链接的完全限定的绝对路径,因为我无法控制页面上的某些内容。也不可能使用<base>标签。

我还有其他选择吗?

3 个答案:

答案 0 :(得分:1)

执行创建php路由器时通常执行的操作...为资产和URL创建链接生成工具。

由于您使用前端控制器,您可以根据配置预先定义所有目录,然后生成以下内容的路径:

function asset_url($filename, $type = 'img') {
   global $config;
   if(isset($config['paths'][$type])) {
      return $config['paths'][$type] . '/' . $filename;
   }
}

通常id使用注册表模式实现注册此配置,而不是使用global,但我不知道你的设置,所以这只是一个简单的例子。

答案 1 :(得分:1)

对PHP的每个请求都属于同一个index.php文件吗?在这种情况下,文件本身就知道它在哪里,并且可以计算它所在的目录。

这不是最好的方法,但它可能会给你一些想法。

define('DR', $_SERVER['DOCUMENT_ROOT');
define('BD', str_replace(DR, '', dirname($_SERVER['SCRIPT_FILENAME'] . '/'); // base dir

然后将所有链接添加到BD

答案 2 :(得分:1)

这是一个棘手的解决方案。最简单的方法是在应用程序中引入baseUrl参数。如果站点在根目录下运行,它将为空,并且/corp/site之类的内容将耗尽domain.com/corp/site

这样,当您显示网址时,您会使用<?php echo $app_base_url . '/css/file.css' ?>

我经常使用Zend Framework,他们有一个名为Zend_Controller_Request_Http的类,它有一个名为getBaseUrl的方法,它通过一些复杂的逻辑来检查有关脚本路径的各种$_SERVER变量,请求文件名,最终确定您的基本URL实际上是什么。

如果您使用Zend Framework的特定部分,您可以执行以下操作:

$request = new Zend_Controller_Request_Http();
$baseUrl = $request->getBaseUrl();

由于Zend Framework是模块化的,你不需要包含整个库,我之前只使用过这个类(及其依赖项),所以我可以轻松地在一个小应用程序中计算基本URL,而不需要整个ZF库。

为了让您了解确定此内容所涉及的逻辑,以下是完成大部分工作的代码(注意,不要尝试直接使用它,而是使用如上所示的Zend Framework类)。

/**
 * Set the base URL of the request; i.e., the segment leading to the script name
 *
 * E.g.:
 * - /admin
 * - /myapp
 * - /subdir/index.php
 *
 * Do not use the full URI when providing the base. The following are
 * examples of what not to use:
 * - http://example.com/admin (should be just /admin)
 * - http://example.com/subdir/index.php (should be just /subdir/index.php)
 *
 * If no $baseUrl is provided, attempts to determine the base URL from the
 * environment, using SCRIPT_FILENAME, SCRIPT_NAME, PHP_SELF, and
 * ORIG_SCRIPT_NAME in its determination.
 *
 * @param mixed $baseUrl
 * @return Zend_Controller_Request_Http
 */
public function setBaseUrl($baseUrl = null)
{
    if ((null !== $baseUrl) && !is_string($baseUrl)) {
        return $this;
    }

    if ($baseUrl === null) {
        $filename = (isset($_SERVER['SCRIPT_FILENAME'])) ? basename($_SERVER['SCRIPT_FILENAME']) : '';

        if (isset($_SERVER['SCRIPT_NAME']) && basename($_SERVER['SCRIPT_NAME']) === $filename) {
            $baseUrl = $_SERVER['SCRIPT_NAME'];
        } elseif (isset($_SERVER['PHP_SELF']) && basename($_SERVER['PHP_SELF']) === $filename) {
            $baseUrl = $_SERVER['PHP_SELF'];
        } elseif (isset($_SERVER['ORIG_SCRIPT_NAME']) && basename($_SERVER['ORIG_SCRIPT_NAME']) === $filename) {
            $baseUrl = $_SERVER['ORIG_SCRIPT_NAME']; // 1and1 shared hosting compatibility
        } else {
            // Backtrack up the script_filename to find the portion matching
            // php_self
            $path    = isset($_SERVER['PHP_SELF']) ? $_SERVER['PHP_SELF'] : '';
            $file    = isset($_SERVER['SCRIPT_FILENAME']) ? $_SERVER['SCRIPT_FILENAME'] : '';
            $segs    = explode('/', trim($file, '/'));
            $segs    = array_reverse($segs);
            $index   = 0;
            $last    = count($segs);
            $baseUrl = '';
            do {
                $seg     = $segs[$index];
                $baseUrl = '/' . $seg . $baseUrl;
                ++$index;
            } while (($last > $index) && (false !== ($pos = strpos($path, $baseUrl))) && (0 != $pos));
        }

        // Does the baseUrl have anything in common with the request_uri?
        $requestUri = $this->getRequestUri();

        if (0 === strpos($requestUri, $baseUrl)) {
            // full $baseUrl matches
            $this->_baseUrl = $baseUrl;
            return $this;
        }

        if (0 === strpos($requestUri, dirname($baseUrl))) {
            // directory portion of $baseUrl matches
            $this->_baseUrl = rtrim(dirname($baseUrl), '/');
            return $this;
        }

        $truncatedRequestUri = $requestUri;
        if (($pos = strpos($requestUri, '?')) !== false) {
            $truncatedRequestUri = substr($requestUri, 0, $pos);
        }

        $basename = basename($baseUrl);
        if (empty($basename) || !strpos($truncatedRequestUri, $basename)) {
            // no match whatsoever; set it blank
            $this->_baseUrl = '';
            return $this;
        }

        // If using mod_rewrite or ISAPI_Rewrite strip the script filename
        // out of baseUrl. $pos !== 0 makes sure it is not matching a value
        // from PATH_INFO or QUERY_STRING
        if ((strlen($requestUri) >= strlen($baseUrl))
            && ((false !== ($pos = strpos($requestUri, $baseUrl))) && ($pos !== 0)))
        {
            $baseUrl = substr($requestUri, 0, $pos + strlen($baseUrl));
        }
    }

    $this->_baseUrl = rtrim($baseUrl, '/');
    return $this;
}