วิธีรับที่อยู่ IP ของลูกค้าใน Laravel 5+


136

ฉันพยายามรับที่อยู่ IP ของลูกค้าใน Laravel

มันเป็นเรื่องง่ายที่จะได้รับ IP ของลูกค้าใน PHP $_SERVER["REMOTE_ADDR"]โดยใช้ มันทำงานได้ดีในคอร์ PHP แต่เมื่อฉันใช้สิ่งเดียวกันใน Laravel มันจะคืนค่า IP ของเซิร์ฟเวอร์แทนที่จะเป็น IP ของผู้เข้าชม

คำตอบ:


194

ดูที่Laravel API :

Request::ip();

ภายในจะใช้getClientIpsวิธีการจากวัตถุร้องขอ Symfony :

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
การใช้คำขอวัตถุไม่ทำงานสำหรับฉันมันส่งคืนที่อยู่ของเซิร์ฟเวอร์ Homestead ของฉัน 192.168.10.10 ซึ่งเห็นได้ชัดว่าไม่ใช่ที่อยู่ IP ของฉัน
Vince Kronlein

@VinceKronlein สำหรับกรณีของคุณตรวจสอบคำตอบนี้stackoverflow.com/a/41769505/3437790
Sebastien Horin

3
@VinceKronlein ในกรณีของคุณมันถูกต้องมาก เนื่องจากคุณเข้าถึง Homestead ในเครือข่าย LOCAL ของคุณคุณจึงมี IP 192 หากคุณเข้าถึงเซิร์ฟเวอร์ที่อยู่อาศัยของผู้อื่นผ่านทางอินเทอร์เน็ต IP ของคุณจะออกไปทาง ISP ของคุณและจะใช้งานสาธารณะของคุณ
ied3vil

83

หากคุณอยู่ภายใต้ load balancer, Laravel \Request::ip() จะส่งคืน IP ของ balancer เสมอ :

            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 : ใช้ของ LaravelRequest::ip()เช่นกันดังนั้นผู้เข้าชมทั้งหมดของคุณจะถูกระบุว่าเป็นผู้ใช้เดียวกันและคุณจะถึงขีด จำกัด เค้นอย่างรวดเร็ว ฉันมีชีวิตแบบสดๆและสิ่งนี้ทำให้เกิดปัญหาใหญ่

ในการแก้ไขปัญหานี้:

Illuminate \ Http \ Request.php

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

ตอนนี้คุณสามารถใช้Request::ip()ซึ่งควรส่งคืน IP จริงในการผลิต


1
ถูกต้อง if (filter_var ... ) ภายใน foreach ที่สองหรือไม่ รหัสนี้จะไม่ถูกดำเนินการ
Mistre83

@ Mistre83 ใช่คุณพูดถูกฉันคิดว่ามันเป็นการทดสอบกำกับ ฉันอัปเดต!
Sebastien Horin

6
ใช้งานได้จริงกับ laravel 5.4 โปรดพิจารณาการสร้าง PR บน GitHub ฉันคิดว่านี่น่าจะเป็นพฤติกรรมเริ่มต้น
Crystal

1
สิ่งนี้ทำงานได้ดีใน Laravel 5.3 เมื่อเมธอด ip () ของวัตถุ Laravel ร้องขอส่งคืน 127.0.0.1
w5m

3
คุณไม่สามารถแก้ไขปัญหานี้ด้วยพร็อกซีที่เชื่อถือได้? - laravel.com/docs/master/requests#configuring-trusted-proxies
2722667

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
ปัญหาคือว่าฟังก์ชั่นระดับโลกเหล่านี้ไม่สามารถทดสอบได้ง่าย - พวกเขาไม่สามารถเยาะเย้ย Façadesสามารถ ฉันพยายามหลีกเลี่ยงฟังก์ชั่นทั่วโลกเพราะมันหมายถึงการขุดผ่านไปยังแหล่งที่มาของฟังก์ชั่นทั่วโลกเพื่อเยาะเย้ยการโทรซึ่งเป็นงานพิเศษน่ารำคาญและไม่ควรเป็นความรับผิดชอบของฉัน
แฮ็ค

1
แม้ว่าrequest()->ip()จะถูกต้องแล้วข้อความโดยรอบนั้นทำให้เข้าใจผิดโดยเฉพาะอย่างยิ่งโดยเฉพาะการพูดว่า "ไม่ถูกต้อง"Request::ip .
คริส

1
@ Chris ขอบคุณคุณพูดถูก แก้ไขเพื่อความชัดเจน!
Stan Smulders

27

เพิ่มเนมสเปซ

use Request;

จากนั้นเรียกใช้ฟังก์ชัน

Request::ip();

1
หากคุณใช้ namespace: -> ใช้ Illuminate \ Http \ Request; เปลี่ยนชื่อเนมสเปซที่เป็นตัวหนาสำหรับการร้องขอเนื่องจากทั้งคู่จะปะทะกัน
shalini

คำตอบเดิมถูกต้อง คุณต้องนำเข้าuse Requestเพราะคุณกำลังพยายามใช้ซุ้ม เนมสเปซที่คุณระบุมีไว้สำหรับคลาสพื้นฐาน หากคุณนำเข้าที่คุณจะได้รับข้อผิดพลาดเพราะip()ไม่สามารถเรียกได้ว่าเป็นแบบสแตติกนั่นคือสิ่งที่ใช้ทำซุ้ม
jfadich

use Illuminate\Support\Facades\Requestหากคุณกำลังจะไปรบกวนการนำเข้าชั้นเรียนคุณควรใช้façadeที่เกิดขึ้นจริงไม่ได้นามแฝง: \Request::ถ้าไม่ได้เพียงแค่ใช้
แฮ็ค

18

สำหรับ Laravel 5 คุณสามารถใช้ขอวัตถุ เพียงเรียกip()ใช้วิธีการบางอย่างเช่น:

$request->ip();


12

มีสองสิ่งที่ต้องดูแลคือ:

  1. รับฟังก์ชั่นตัวช่วยที่คืนค่า a Illuminate\Http\Requestและเรียก->ip()เมธอด:

    request()->ip();
  2. คิดถึงการกำหนดค่าเซิร์ฟเวอร์ของคุณอาจใช้พรอกซีหรือload-balancerโดยเฉพาะอย่างยิ่งในการกำหนดค่า AWS ELB

หากเป็นกรณีของคุณคุณต้องทำตาม "การกำหนดค่าพร็อกซีที่เชื่อถือได้ " หรืออาจตั้งค่าตัวเลือก "เชื่อใจพร็อกซีทั้งหมด"

ทำไม? เพราะเป็นเซิร์ฟเวอร์ของคุณจะได้รับ proxy / 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 คงที่ได้ นี่เป็นวิธีที่ถูกต้องในการรับ 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 ที่ถูกต้อง


4

หากคุณต้องการ IP ไคลเอ็นต์และเซิร์ฟเวอร์ของคุณอยู่หลัง aws elb ให้ใช้รหัสต่อไปนี้ ผ่านการทดสอบสำหรับ laravel 5.3

$elbSubnet = '172.31.0.0/16';
Request::setTrustedProxies([$elbSubnet]);
$clientIp = $request->ip();

1
ไม่ทำงานอีกต่อไปตอนนี้ "TrustedHeaderSet" ได้รับการแก้ไขแล้ว
Shadrix

สำหรับ laravel รุ่น "ล่าสุด" โปรดดูที่เอกสารlaravel.com/docs/5.5/requests#configuring-trusted-proxies
Sandra

0

หากคุณมีพร็อกซี่หลายชั้นเช่นเดียวกับ CDN + Load Balancer
การใช้ Laravel Request :: ip () ฟังก์ชั่นจะได้รับ IP proxy ที่ถูกต้องที่สุด แต่ไม่ใช่ IP ของไคลเอ็นต์
คุณอาจลองทำตามวิธีแก้ปัญหา

app / HTTP / มิดเดิ้ล / TrustProxies.php

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

การอ้างอิง: https://github.com/fideloper/TrustedProxy/issues/107#issuecomment-373065215


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
ช่วยได้มากขึ้นถ้าคุณให้คำอธิบายว่าทำไมนี่จึงเป็นโซลูชันที่ต้องการและอธิบายวิธีการทำงาน เราต้องการที่จะให้ความรู้ไม่เพียง แต่ให้รหัส เช่นเดียวกับระบบกำลังตั้งค่าสถานะเป็นคุณภาพต่ำดังนั้นพยายามปรับปรุง
Tin Man

ขอบคุณสำหรับคำแนะนำของคุณ
rashedcs
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.