瞭解php擷取用戶端ip的方法

來源:互聯網
上載者:User

本文講解了如何通過php擷取用戶端ip的方法

擷取ip函數如下:

function getIP() {    $realip = ''; //設定預設值    if (isset($_SERVER['HTTP_X_FORWARDED_FOR'])) {        $realip = $_SERVER['HTTP_X_FORWARDED_FOR'];    } elseif (isset($_SERVER['HTTP_CLIENT_IP'])) {        $realip = $_SERVER['HTTP_CLIENT_IP'];    } else {        $realip = $_SERVER['REMOTE_ADDR'];    }    preg_match('/^((?:\d{1,3}\.){3}\d{1,3})/',$realip,$match);        if($match && ipType($match[0]) == 'InterNet網地址'){                return  $match[0];        }else{                return  false;        }}

//互連網允許使用IP地址

function ipType($ip) {    $iplist = explode(".", $ip);    if ($iplist[0] >= 224 && $iplist[0] <= 239)                return '多播';        if ($iplist[0] >= 240 && $iplist[0] <= 255)        return '保留';    if (preg_match('/^198\.51\.100/', $ip))        return 'TEST-NET-2,文檔和樣本';    if (preg_match('/^203\.0\.113/', $ip))        return 'TEST-NET-3,文檔和樣本';    if (preg_match('/^192\.(18|19)\./', $ip))        return '網路基準測試';    if (preg_match('/^192\.168/', $ip))        return '專用網路[內部網]';    if (preg_match('/^192\.88\.99/', $ip))        return 'ipv6to4中繼';    if (preg_match('/^192\.0\.2\./', $ip))        return 'TEST-NET-1,文檔和樣本';    if (preg_match('/^192\.0\.0\./', $ip))        return '保留(IANA)';    if (preg_match('/^192\.0\.0\./', $ip))        return '保留(IANA)';    if ($iplist[0] == 172 && $iplist[1] <= 31 && $iplist[1] >= 16)        return '專用網路[內部網]';    if ($iplist[0] == 169 && $iplist[1] == 254)        return '鏈路本地';    if ($iplist[0] == 127)        return '迴路位址';    if ($iplist[0] == 10)        return '專用網路[內部網]';    if ($iplist[0] == 0)        return '本網路(僅作為源地址時合法)';    return 'InterNet網地址';}

網上常見擷取ip函數如下:

public function get_real_ip() {    static $realip;    if (isset($_SERVER)) {        if (isset($_SERVER['HTTP_X_FORWARDED_FOR'])) {            $realip = $_SERVER['HTTP_X_FORWARDED_FOR'];        } else if (isset($_SERVER['HTTP_CLIENT_IP'])) {            $realip = $_SERVER['HTTP_CLIENT_IP'];        } else {            $realip = $_SERVER['REMOTE_ADDR'];        }    } else {        if (getenv('HTTP_X_FORWARDED_FOR')) {            $realip = getenv('HTTP_X_FORWARDED_FOR');        } else if (getenv('HTTP_CLIENT_IP')) {            $realip = getenv('HTTP_CLIENT_IP');        } else {            $realip = getenv('REMOTE_ADDR');        }    }    return $realip;}

’REMOTE_ADDR’ ,’HTTP_X_FORWARDED_FOR’,’HTTP_CLIENT_IP’之間的區別?

1.’REMOTE_ADDR’ 是遠端IP,預設來自tcp 串連是,用戶端的Ip。可以說,它最準確,確定是,只會得到直接連伺服器用戶端IP。如果對方通過Proxy 伺服器上網,就發現。擷取到的是Proxy 伺服器IP了。

如:a->b(proxy)->c ,如果c 通過’REMOTE_ADDR’ ,只能擷取到b的IP,擷取不到a的IP了。

2.’HTTP_X_FORWARDED_FOR’,’HTTP_CLIENT_IP’ 為了能在大型網路中,擷取到最原始使用者IP,或者代理IP地址。對HTTp協議進行擴充。定義了實體頭。

HTTP_X_FORWARDED_FOR = clientip,proxy1,proxy2 所有IP用”,”分割。 HTTP_CLIENT_IP 在進階匿名代理中,這個代表了Proxy 伺服器IP。既然是http協議擴充一個實體頭,並且這個值對於傳入端是信任的,信任傳入方按照規則格式輸入的。以下以x_forword_for例子加以說明,正常情況下,這個值變化過程。

風險點所在:

這些變數,來自http請求的:x-forword-for欄位,以及client-ip欄位。 正常Proxy 伺服器,當然會按rfc規範來傳入這些值。但是,當一個使用者直接構造該x-forword-for值,發送給使用者,這樣就好比就直接有一個可以寫入任意值的欄位。並且伺服器直接讀取,或者寫入資料庫,或者做顯示。它將帶來危險性,跟一般對入輸入沒有做任何過濾檢測,之間操作資料源結果一樣。

對於上面getip函數:

除了用戶端可以任意偽造IP,並且可以傳入任意格式IP。 這樣結果會帶來2大問題,其一,如果你設定某個頁面,做IP限制。 對方可以容易修改IP不斷請求該頁面。 其二,這類資料你如果直接使用,將帶來SQL註冊,跨站攻擊等漏洞。至於其一,可以在業務上面做限制,最好不採用IP限制。 對於其二,這類可以帶來巨大網路風險。我們必須加以糾正。

本文講解了如何通過php擷取用戶端ip的方法,更多相關內容請關注php中文網。

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.