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


136

我正在尝试在Laravel中获取客户端的IP地址。

通过使用,很容易在PHP中获得客户端的IP $_SERVER["REMOTE_ADDR"]。在核心PHP中它可以正常工作,但是当我在Laravel中使用相同的东西时,它将返回服务器IP而不是访问者的IP。

Answers:


194

查看Laravel API

Request::ip();

在内部,它使用Symfony Request对象中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);
} 

3
使用Request对象对我不起作用,它返回Homestead服务器的地址。192.168.10.10显然不是我的IP地址。
文斯·科隆

@VinceKronlein为您的案例检查此答案stackoverflow.com/a/41769505/3437790
Sebastien Horin

3
@VinceKronlein在您的情况下,这是非常正确的。因为您要访问的是Homestead,所以在您的本地网络中,您的IP地址为192。如果您正在通过互联网访问别人的家庭服务器,那么您的IP将通过您的ISP消失,而您的公共IP将被使用。
ied3vil

82

如果您在负载均衡器之下,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的油门中间件Request::ip(),因此,所有访问者都将被识别为同一用户,并且您将很快达到油门极限。我经历了现场直播,这引起了很多问题。

要解决此问题:

Illuminate \ Http \ Request.php

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

现在Request::ip(),您还可以使用,它将返回生产中的真实IP。


1
在第二个foreach中正确的if(filter_var ...)是正确的吗?此代码将永远不会执行。
Mistre83 '17

@ Mistre83是的,您是对的,我认为这是测试疏忽。我更新了!
Sebastien Horin

6
这实际上适用于laravel 5.4。请考虑在github上进行PR。我想这应该是默认的行为
水晶

1
当Laravel请求对象的ip()方法不断返回127.0.0.1
w5m

3
不能通过信任的代理解决此问题吗?- laravel.com/docs/master/requests#configuring-trusted-proxies
user2722667

74

使用request()->ip()

据我了解,从Laravel 5开始,建议/好的做法是使用如下全局函数:

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

而且,如果有的话,当使用函数而不是静态符号时,我的IDE不会像圣诞树一样亮起。


3
没错,这request是一个“全局”功能-它是laravel提供的全局帮助器功能之一。但是,请求门面,不是静态的(也不是该方法的ip) - request()->foo,和Reqest::foo$request->foo是相同的。看看这个要点作为示例:gist.github.com/cjke/026e3036c6a10c672dc5
Chris

1
足够公平-两者都是正确的。我只是想过您所说的话“ Request::ip可能不会引起误解
克里斯(Chris

3
问题在于这些全局函数不容易测试,无法模拟。外墙可以。我尝试避免使用全局函数,因为这意味着深入到全局函数源以模拟其调用,这是额外的工作,烦人的工作,不应该由我负责。
hackel

1
虽然request()->ip()是正确的,周围的文字实在是误导-特别是说“这不是Request::ip
克里斯-

1
@克里斯谢谢,你是完全正确的。为了清晰而编辑!
Stan Smulders,2016年

27

添加名称空间

use Request;

然后调用函数

Request::ip();

1
如果您使用命名空间:->使用Illuminate \ Http \ Request;
体重

原始答案是正确的。您需要导入,use Request因为您正在尝试使用Facade。您提供的名称空间用于基础类。如果导入,将由于ip()无法静态调用而出现错误,这就是外观的目的。
jfadich '16

如果您要麻烦导入类,则应使用实际的外观,而不是别名:use Illuminate\Support\Facades\Request。如果没有,请使用\Request::
hackel



12

有两件事要注意:

  1. 获取一个返回a的辅助函数Illuminate\Http\Request并调用该->ip()方法:

    request()->ip();
  2. 考虑您的服务器配置,它可能使用Proxy或load-balancer,尤其是在AWS ELB配置中。

如果是这种情况,则需要遵循“ 配置受信任的代理 ”,甚至可能要设置“信任所有代理”选项。

为什么?因为作为您的服务器将获得您的代理/ load-balancerIP。

如果您使用的是AWS天平,请转到App\Http\Middleware\TrustProxies并使$proxies声明如下所示:

protected $proxies = '*';

现在进行测试并庆祝,因为您刚刚摆脱了使用节流阀中间件的麻烦。它还依赖request()->ip()并且无需设置“ TrustProxies”,就可以阻止所有用户登录,而不仅仅是阻止罪魁祸首的IP。

并且由于在文档中没有正确解释节气门中间件,因此我建议观看“ 初学者的laravel 5.2教程,API速率限制

在Laravel 5.7中测试


7

在Laravel 5.4中,我们不能将ip称为static。这是获取用户IP的正确方法:

 use Illuminate\Http\Request;

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

7

如果调用此函数,则可以轻松获取客户端的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;
    }

5

如果仍然获得127.0.0.1作为IP,则需要添加“代理”,但要注意,在投入生产之前必须对其进行更改!

阅读“ 配置受信任的代理 ”。

并添加以下内容:

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

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




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;
            }
        }
    }
}

}


-2

当我们想要用户的ip_address

$_SERVER['REMOTE_ADDR']

并想要服务器地址:

$_SERVER['SERVER_ADDR']

-2
  $ip = $_SERVER['REMOTE_ADDR'];

1
如果您提供解释说明为什么这是首选解决方案并说明其工作原理,那么它会有所帮助。我们要教育,而不仅仅是提供代码。系统将其标记为低质量,因此请尝试对其进行改进。
锡人

谢谢您的建议。
rashedcs '19
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.