我想在Laravel中获取客户端的IP地址。众所周知,使用$_SERVER["REMOTE_ADDR"]
在PHP中获取客户端的IP要容易得多。
它在核心PHP中运行良好,但是当我在Laravel中使用相同的东西时,它会提供服务器IP而不是访问者IP。
答案 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
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()
,所以你的所有访问者将被识别为同一个用户,您将很快达到限制。在现场经历......这让我遇到了大问题......
解决此问题:
照亮\ 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)考虑您的服务器配置,它可能使用proxy
或load 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']