前面有一個關于$_SERVER的文檔,今天,來看看怎么不同環境下其攜帶參數的差異與統一。
REQUEST_URI
由HTTP1.1協議定義,指向某個頁面的URI,去除開頭的協議、主機、端口等信息。如http://www.digpage.com:8080/index.php/foo/bar?queryParams,REQUEST_URI
為/index.php/foo/bar?queryParams
X-REWRITE-URL
當使用了以開啟ISAPI_Rewrite的IIS作為服務器時,ISAPI_Rewrite會在未對原始URI作任何修改前,將原始的REQUEST_URI以X-REWRITE0URL HTTP頭保存起來。PATH_INFO
CGI 1.1規范定義的環境變量。從形式上看。它是整個URI中,在腳本標識之后、查詢參數?之前的部分。對于Apache,需要設置AcceptPathinfo On,且在一個URL沒有</PATH_INFO>部分的時候 ,PATH_INFO無效。特殊情況,如http://www.digpage.com/index.php/
,PATH_INFO為/。而對于 Nginx,則需要設置:
fastcgi_split_path_info ^(.+?\.php)(/.*)$; fastcgi_param PATH_INFO $fastcgi_path_info;
ORIG_PATH_INFO
指未經 PHP 處理過的原始的 PATH_INFO”。 這個在 Apache 和 Nginx 需要配置一番才行,但一般用不到,已經有 PATH_INFO 可以用了嘛。而在 IIS 中則有點怪, 對于http://www.digpage.com/index.php/ ORIG_PATH_INFO
為/index.php/
;對于http://www.digapge.com/index.php ORIG_PATH_INFO
為/index.php 。
在yii2中的一個方法:用來獲取不同環境下統一的URI
<?php
//使用了ISAPI_Rewrite的IIS
if (isset($_SERVER['HTTP_X_REWRITE_URL'])) {
$requestUri = $_SERVER['HTTP_X_REWRITE_URL'];
//一般情況,需要去掉 URL 中的協議、主機、端口等內容
} elseif (isset($_SERVER['REQUEST_URI'])) {
$requestUri = $_SERVER['REQUEST_URI'];
if ($requestUri !== '' && $requestUri[0] !== '/') {
$requestUri = preg_replace('/^(http|https):\/\/[^\/]+/i', '', $requestUri);
}
// IIS 5.0, PHP 以 CGI 方式運行,需要把查詢參數接上
} elseif (isset($_SERVER['ORIG_PATH_INFO'])) {
$requestUri = $_SERVER['ORIG_PATH_INFO'];
if (!empty($SERVER['QUERY_STRING'])) {
$requestUri .= '?' . $SERVER['QUERY_STRING'];
}
} else {
throw new Exception('wrong');
}
echo $requestUri;
- SCRIPT_FILENAME
當前腳本的實際物理路徑,比如 /var/www/digpage.com/frontend/web/index.php , 或 WIN平臺的 D:\www\digpage.com\frontend\web\index.php 。 以 Nginx 為例,一般情況下,SCRIPT_FILENAME 有以下配置項:
fastcgi_split_path_info ^(.+?\.php)(/.*)$;
# 使用 document root 來得到物理路徑
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name
SCRIPT_NAME
CGI 1.1 規范所定義的環境變量,用于標識 CGI 腳本(而非腳本的輸出),如http://www.digapge.com/path/index.php
中的/path/index.php
。 仍以 Nginx 為例,SCRIPT_NAME 一般情況下有fastcgi_param SCRIPT_NAME $fastcgi_script_name
的設置。絕大多數情況下,使用 SCRIPT_NAME 即可獲取當前腳本。PHP_SELF
PHP_SELF
是 PHP 自己實現的一個 $_SERVER 變量,是相對于文檔根目錄(documentroot)而言的。 對于http://www.digpage.com/path/index.php?queryParams
,PHP_SELF為/path/index.php
。 一般 SCRIPT_NAME 與 PHP_SELF 無異。但是,在 PHP.INI 中,如cgi.fix_pathinfo=1
(默認即為 1)時, 對于形如http://www.digpage.com/path/index.php/post/view/123
, 則PHP_SELF為 /path/index.php/post/view/123
。 而根據 CGI 1.1 規范,SCRIPT_NAME 僅為/path/index.php
,至于剩余的/post/view/123
則為 PATH_INFO。ORIG_SCRIPT_NAME
當 PHP 以 CGI 模式運行時,默認會對一些環境變量進行調整。 首當其沖的,就是SCRIPT_NAME 的內容會變成 php.cgi 等二進制文件,而不再是 CGI 腳本文件。 當然,設置cgi.fix_pathinfo=0
可以關閉這一默認行為。但這導致的副作用比較大,影響范圍過大,不宜使用。 但天無絕人之路,九死之地總留一線生機,那就是 ORIG_SCRIPT_NAME,他保留了調整前 SCRIPT_NAME 的內容。 也就是說,在 CGI 模式下,可以使用ORIG_SCRIPT_NAME 來獲取想要的 SCRIPT_NAME。 請留意使用ORIG_SCRIPT_NAME 前一定要先確認它是否存在。
再來看看yii2中的相關方法:
<?php
$scriptFile = $_SERVER['SCRIPT_FILENAME'];
$scriptName = basename($scriptFile);
if (basename($_SERVER['SCRIPT_NAME']) === $scriptName){
$_scriptUrl = $_SERVER['SCRIPT_NAME'];
} elseif (basename($_SERVER['PHP_SELF']) === $scriptName) {
$_scriptUrl = $_SERVER['PHP_SELF'];
} elseif (isset($_SERVER['ORIG_SCRIPT_NAME']) && basename($_SERVER['ORIG_SCRIPT_NAME']) === $scriptName) {
$_scriptUrl = $_SERVER['ORIG_SCRIPT_NAME'];
} elseif (!empty($_SERVER['DOCUMENT_ROOT']) && strpos($scriptFile, $_SERVER['DOCUMENT_ROOT']) === 0) {
$_scriptUrl = str_replace("\\", '/', str_replace($_SERVER['DOCUMENT_ROOT'], '', $scriptFile));
} else {
throw new Exception ('wrong');
}
$baseUrl = retirm(dirname($_scriptUrl), '\\/');
在yii中通過這些的配合,也有一個獲取pathinfo的方法,如下是邏輯代碼:
$pathinfo = $requestUri;
if (($pos = strpos($pathinfo , '?')) !== false) {
$pathinfo = substr($pathinfo, 0, $pos);
}
if (!preg_match('%^(?:
[\x09\x0A\x0D\x20-\x7E] # ASCII
| [\xC2-\xDF][\x80-\xBF] # non-overlong 2-byte
| \xE0[\xA0-\xBF][\x80-\xBF] # excluding overlongs
| [\xE1-\xEC\xEE\xEF][\x80-\xBF]{2} # straight 3-byte
| \xED[\x80-\x9F][\x80-\xBF] # excluding surrogates
| \xF0[\x90-\xBF][\x80-\xBF]{2} # planes 1-3
| [\xF1-\xF3][\x80-\xBF]{3} # planes 4-15
| \xF4[\x80-\x8F][\x80-\xBF]{2} # plane 16
)*$%xs', $pathinfo)
) {
$pathInfo = utf8_encode($pathInfo);
}
if (strpos($pathinfo, $_scriptUrl) === 0) {
$pathinfo = substr($pathinfo, strlen($_scriptUrl));
} elseif ($baseUrl === "" || strpos($pathinfo, $baseUrl) === 0) {
$pathinfo = substr($pathinfo, strlen($baseUrl));
} elseif (isset($_SERVER['PHP_SELF']) && strpos($_SERVER['PHP_SELF'], $_scriptUrl) === 0) {
$pathinfo = substr($_SERVER['PHP_SELF'], str($_scriptUrl));
} else {
throw new Exception('wrong');
}
if ($pathinfo[0] === '/') {
$pathinfo = substr($pathinfo, 1);
}
這樣,就獲得了當前請求的URI($requestUri)、腳本路徑($_scriptUrl)、pathinfo($pathinfo),這些變量都是一個健壯的框架所需利用的元素。