如何在Laravel 5+中获取客户端IP地址?

时间:2015-10-21 20:32:42

标签: php laravel laravel-5 laravel-5.1 laravel-5.7

我想在Laravel中获取客户端的IP地址。众所周知,使用$_SERVER["REMOTE_ADDR"]在PHP中获取客户端的IP要容易得多。

它在核心PHP中运行良好,但是当我在Laravel中使用相同的东西时,它会提供服务器IP而不是访问者IP。

19 个答案:

答案 0 :(得分:142)

查看Laravel API

Request::ip();

在内部,它使用Symfony Request Object中的getClientIps方法:

public function getClientIps()
{
    $clientIps = array();
    $ip = $this->server->get('REMOTE_ADDR');
    if (!$this->isFromTrustedProxy()) {
        return array($ip);
    }
    if (self::$trustedHeaders[self::HEADER_FORWARDED] && $this->headers->has(self::$trustedHeaders[self::HEADER_FORWARDED])) {
        $forwardedHeader = $this->headers->get(self::$trustedHeaders[self::HEADER_FORWARDED]);
        preg_match_all('{(for)=("?\[?)([a-z0-9\.:_\-/]*)}', $forwardedHeader, $matches);
        $clientIps = $matches[3];
    } elseif (self::$trustedHeaders[self::HEADER_CLIENT_IP] && $this->headers->has(self::$trustedHeaders[self::HEADER_CLIENT_IP])) {
        $clientIps = array_map('trim', explode(',', $this->headers->get(self::$trustedHeaders[self::HEADER_CLIENT_IP])));
    }
    $clientIps[] = $ip; // Complete the IP chain with the IP the request actually came from
    $ip = $clientIps[0]; // Fallback to this when the client IP falls into the range of trusted proxies
    foreach ($clientIps as $key => $clientIp) {
        // Remove port (unfortunately, it does happen)
        if (preg_match('{((?:\d+\.){3}\d+)\:\d+}', $clientIp, $match)) {
            $clientIps[$key] = $clientIp = $match[1];
        }
        if (IpUtils::checkIp($clientIp, self::$trustedProxies)) {
            unset($clientIps[$key]);
        }
    }
    // Now the IP chain contains only untrusted proxies and the client IP
    return $clientIps ? array_reverse($clientIps) : array($ip);
} 

答案 1 :(得分:55)

使用request()->ip()

从Laravel 5开始(根据我的理解)建议/良好实践使用全局函数,如:

response()->json($v);
view('path.to.blade');
redirect();
route();
cookie();

你明白了这一点:-)如果有的话,当使用这些函数(而不是静态的notarion)时,我的IDE就不像圣诞树一样了; - )

答案 2 :(得分:51)

如果您在负载均衡器

Laravel' \Request::ip() 总是返回平衡器的IP

            echo $request->ip();
            // server ip

            echo \Request::ip();
            // server ip

            echo \request()->ip();
            // server ip

            echo $this->getIp(); //see the method below
            // clent ip

此自定义方法返回真实客户端ip:

public function getIp(){
    foreach (array('HTTP_CLIENT_IP', 'HTTP_X_FORWARDED_FOR', 'HTTP_X_FORWARDED', 'HTTP_X_CLUSTER_CLIENT_IP', 'HTTP_FORWARDED_FOR', 'HTTP_FORWARDED', 'REMOTE_ADDR') as $key){
        if (array_key_exists($key, $_SERVER) === true){
            foreach (explode(',', $_SERVER[$key]) as $ip){
                $ip = trim($ip); // just to be safe
                if (filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_NO_PRIV_RANGE | FILTER_FLAG_NO_RES_RANGE) !== false){
                    return $ip;
                }
            }
        }
    }
}

更多:如果您使用Laravel的节流中间件

除此之外,我建议您使用Laravel的节流中间件时要非常小心:它也使用Laravel的Request::ip(),所以你的所有访问者将被识别为同一个用户,您将很快达到限制。在现场经历......这让我遇到了大问题......

解决此问题:

照亮\ HTTP \ Request.php

    public function ip()
    {
        //return $this->getClientIp(); //original method
        return $this->getIp(); // the above method
    }

您现在也可以使用Request::ip(),它应该返回生产中的真实IP

答案 3 :(得分:22)

添加名称空间

use Request;

然后调用函数

Request::ip();

答案 4 :(得分:13)

对于Laravel 5,您可以使用Request对象。只需调用其ip()方法即可。类似的东西:

$request->ip();

答案 5 :(得分:9)

在Laravel 5中

public function index(Request $request) {
  $request->ip();
}

答案 6 :(得分:3)

如果仍然获得IP 127.0.0.1,则需要添加“代理”。

但是请注意,在生产之前必须更改它!

阅读此部分:https://laravel.com/docs/5.7/requests#configuring-trusted-proxies

现在只需添加:

class TrustProxies extends Middleware
{
    /**
     * The trusted proxies for this application.
     *
     * @var array
     */
    protected $proxies = '*';

现在request()-> ip()为您提供正确的ip

答案 7 :(得分:2)

在版本laravel 5.4中我们无法调用ip static这是获取ip user的正确方法

 use Illuminate\Http\Request;

public function contactUS(Request $request)
    {
        echo $request->ip();
        return view('page.contactUS');
    }

答案 8 :(得分:2)

如果您想要客户端IP并且您的服务器位于aws elb之后,请使用以下代码。测试laravel 5.3

EV_SEP

答案 9 :(得分:2)

  

如果调用此函数,则可以轻松获取客户端IP地址。   我已经在现有项目中使用了此有用的代码。

public function getUserIpAddr(){
       $ipaddress = '';
       if (isset($_SERVER['HTTP_CLIENT_IP']))
           $ipaddress = $_SERVER['HTTP_CLIENT_IP'];
       else if(isset($_SERVER['HTTP_X_FORWARDED_FOR']))
           $ipaddress = $_SERVER['HTTP_X_FORWARDED_FOR'];
       else if(isset($_SERVER['HTTP_X_FORWARDED']))
           $ipaddress = $_SERVER['HTTP_X_FORWARDED'];
       else if(isset($_SERVER['HTTP_FORWARDED_FOR']))
           $ipaddress = $_SERVER['HTTP_FORWARDED_FOR'];
       else if(isset($_SERVER['HTTP_FORWARDED']))
           $ipaddress = $_SERVER['HTTP_FORWARDED'];
       else if(isset($_SERVER['REMOTE_ADDR']))
           $ipaddress = $_SERVER['REMOTE_ADDR'];
       else
           $ipaddress = 'UNKNOWN';    
       return $ipaddress;
    }

答案 10 :(得分:1)

有两件事要照顾

1)获取帮助函数,该函数返回Illuminate\Http\Request并调用->ip()方法。

request()->ip();

2)考虑您的服务器配置,它可能使用proxyload balancer(尤其是在AWS ELB配置中)

如果您是这种情况,则需要Configuring Trusted Proxies或什至设置Trusting All Proxies选项。

为什么?

因为成为您的服务器将获得您的代理/天平加载器IP。

如何?

如果您不是AWS balance-loader

转到App\Http\Middleware\TrustProxies

并使$proxies声明如下:

protected $proxies = '*';

现在测试一下并庆祝一下,因为您刚刚摆脱了throttle middleware的麻烦。它还依靠request()->ip(),而无需设置TrustProxies,您可以阻止所有用户登录,而不仅仅是阻止罪魁祸首的IP。

由于文档中对throttle middleware的解释不正确,我建议您注意this video

在Laravel 5.7中进行了测试

答案 11 :(得分:1)

解决方案1:您可以使用这种类型的功能来获取客户端IP

public function getClientIPaddress(Request $request) {
    $clientIp = $request->ip();
    return $clientIp;
}

解决方案2:,如果解决方案1 ​​未提供准确的IP ,那么您可以使用此功能获取访问者的真实IP。

 public function getClientIPaddress(Request $request) {

    if (isset($_SERVER["HTTP_CF_CONNECTING_IP"])) {
        $_SERVER['REMOTE_ADDR'] = $_SERVER["HTTP_CF_CONNECTING_IP"];
        $_SERVER['HTTP_CLIENT_IP'] = $_SERVER["HTTP_CF_CONNECTING_IP"];
    }
    $client  = @$_SERVER['HTTP_CLIENT_IP'];
    $forward = @$_SERVER['HTTP_X_FORWARDED_FOR'];
    $remote  = $_SERVER['REMOTE_ADDR'];

    if(filter_var($client, FILTER_VALIDATE_IP)){
        $clientIp = $client;
    }
    elseif(filter_var($forward, FILTER_VALIDATE_IP)){
        $clientIp = $forward;
    }
    else{
        $clientIp = $remote;
    }

    return $clientIp;
 }

注意事项:在实时服务器中使用负载均衡器 / 代理服务器后,您需要使用解决方案2 获取真实的访问者IP。

答案 12 :(得分:1)

您可以使用Request ip,Request getClientIp和请求合作伙伴工作来获得一些IP地址。在此模型中,我将告诉您在laravel 5.8中获取当前客户端ip地址的最佳方法。

$clientIP = request()->ip();

dd($clientIP);

您可以从here

开始

答案 13 :(得分:0)

如果您具有CDN +负载平衡器之类的多层代理。
使用Laravel Request :: ip()函数将获得最右边的代理IP,而不是客户端IP。
您可以尝试以下解决方案。

app / Http / Middleware / TrustProxies.php

protected $proxies = ['0.0.0.0/0'];

参考:https://github.com/fideloper/TrustedProxy/issues/107#issuecomment-373065215

答案 14 :(得分:0)

我使用了Sebastien Horin函数getIp和request()-> ip()(在全局请求下),因为对于localhost,getIp函数返回null:

$this->getIp() ?? request()->ip();

getIp函数:

public function getIp(){
foreach (array('HTTP_CLIENT_IP', 'HTTP_X_FORWARDED_FOR', 'HTTP_X_FORWARDED', 'HTTP_X_CLUSTER_CLIENT_IP', 'HTTP_FORWARDED_FOR', 'HTTP_FORWARDED', 'REMOTE_ADDR') as $key){
    if (array_key_exists($key, $_SERVER) === true){
        foreach (explode(',', $_SERVER[$key]) as $ip){
            $ip = trim($ip); // just to be safe
            if (filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_NO_PRIV_RANGE | FILTER_FLAG_NO_RES_RANGE) !== false){
                return $ip;
            }
        }
    }
}

}

答案 15 :(得分:0)

如果您担心获取IP地址但不需要或不想使用任何Laravel功能,则可以只使用php:

PHP <5.3.0
$ localIP = getHostByName(php_uname('n'));

PHP> = 5.3.0
$ localIP = getHostByName(getHostName());

在此线程中回答: PHP how to get local IP of system

答案 16 :(得分:0)

我在我的项目中使用的这个解决方案。我在这里找到了其他解决方案,要么不完整,要么太复杂而无法理解。

if (! function_exists('get_visitor_IP'))
{
    /**
     * Get the real IP address from visitors proxy. e.g. Cloudflare
     *
     * @return string IP
     */
    function get_visitor_IP()
    {
        // Get real visitor IP behind CloudFlare network
        if (isset($_SERVER["HTTP_CF_CONNECTING_IP"])) {
            $_SERVER['REMOTE_ADDR'] = $_SERVER["HTTP_CF_CONNECTING_IP"];
            $_SERVER['HTTP_CLIENT_IP'] = $_SERVER["HTTP_CF_CONNECTING_IP"];
        }

        // Sometimes the `HTTP_CLIENT_IP` can be used by proxy servers
        $ip = @$_SERVER['HTTP_CLIENT_IP'];
        if (filter_var($ip, FILTER_VALIDATE_IP)) {
           return $ip;
        }

        // Sometimes the `HTTP_X_FORWARDED_FOR` can contain more than IPs 
        $forward_ips = @$_SERVER['HTTP_X_FORWARDED_FOR'];
        if ($forward_ips) {
            $all_ips = explode(',', $forward_ips);

            foreach ($all_ips as $ip) {
                if (filter_var($ip, FILTER_VALIDATE_IP, FILTER_FLAG_NO_PRIV_RANGE | FILTER_FLAG_NO_RES_RANGE)){
                    return $ip;
                }
            }
        }

        return $_SERVER['REMOTE_ADDR'];
    }
}

答案 17 :(得分:-1)

//Sample Data
[
  {
    "_id": "Swimming Pool",
    "count": 1,
    "heat": 22,
    "humidity": 36
  },{
    "_id": "Flat",
    "count": 5,
    "heat": 160,
    "humidity": 190
  },{
     "_id": "Common Area",
     "count": 33,
     "heat": 756,
     "humidity": 1192
  }]

答案 18 :(得分:-3)

当我们想要用户的ip_address

$_SERVER['REMOTE_ADDR']

并想要服务器地址:

$_SERVER['SERVER_ADDR']