web-dev-qa-db-ja.com

サーブレットで実際のクライアントIPを取得する

単純な問題で問題が発生しています。 HTTPServlet内の実際のクライアントIPを取得します。

今から私は使用しました:

request.getRemoteAddr()

しかし、今では偽のIPを返します。例:xxx.xxx.xxx .5しかし、私のIPはxxx.xxx.xxx .159のようなものです。 ( http://whatismyipaddress.com/ で確認)。

今、私は使用しようとしました:

request.getHeader("X-Forwarded-For")

NULLを返します。

私はまた、次のクラスでプローブを取りました。

public class IpUtils {

public static final String _255 = "(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)";
public static final Pattern pattern = Pattern.compile("^(?:" + _255 + "\\.){3}" + _255 + "$");

public static String longToIpV4(long longIp) {
    int octet3 = (int) ((longIp >> 24) % 256);
    int octet2 = (int) ((longIp >> 16) % 256);
    int octet1 = (int) ((longIp >> 8) % 256);
    int octet0 = (int) ((longIp) % 256);

    return octet3 + "." + octet2 + "." + octet1 + "." + octet0;
}

public static long ipV4ToLong(String ip) {
    String[] octets = ip.split("\\.");
    return (Long.parseLong(octets[0]) << 24) + (Integer.parseInt(octets[1]) << 16)
            + (Integer.parseInt(octets[2]) << 8) + Integer.parseInt(octets[3]);
}

public static boolean isIPv4Private(String ip) {
    long longIp = ipV4ToLong(ip);
    return (longIp >= ipV4ToLong("10.0.0.0") && longIp <= ipV4ToLong("10.255.255.255"))
            || (longIp >= ipV4ToLong("172.16.0.0") && longIp <= ipV4ToLong("172.31.255.255"))
            || longIp >= ipV4ToLong("192.168.0.0") && longIp <= ipV4ToLong("192.168.255.255");
}

public static boolean isIPv4Valid(String ip) {
    return pattern.matcher(ip).matches();
}

public static String getIpFromRequest(HttpServletRequest request) {
    String ip;
    boolean found = false;
    if ((ip = request.getHeader("x-forwarded-for")) != null) {
        StringTokenizer tokenizer = new StringTokenizer(ip, ",");
        while (tokenizer.hasMoreTokens()) {
            ip = tokenizer.nextToken().trim();
            if (isIPv4Valid(ip) && !isIPv4Private(ip)) {
                found = true;
                break;
            }
        }
    }

    if (!found) {
        ip = request.getRemoteAddr();
    }

    return ip;
}
}

また、xxx.xxx.xxx.50 IPも返しました。 :(

現在、実際のクライアントIPを取得する方法がわかりません。誰かが解決策を知っているなら、答えてください。

33
TAR515

あなたの問題は、ローカルネットワークのどこかでサーバーを実行しているため、そのネットワークでIPを取得していることだと思います。ただし、IPアドレスを検出するオンラインサービスを使用しようとしている場合、IPはサービスプロバイダーのルーターのIPなどです。 request.getRemoteAddr()の使用は正しいです。これは、そのようなサービスが行うことであり、他の施設はありません。

31
AlexR

サーブレットコンテナrequest.getRemoteAddr()がアドレスを返す前にNAT(ネットワークアドレス変換))デバイスを持っている場合、NAT接続。

問題を解決するには、ヘッダーで元のクライアントのIPを送信するようにNATを設定します。その後、リクエストからヘッダーを取得します。NAT X-Forwarded-Forヘッダーは次のコードを使用します:request.getHeader("X-Forwarded-For")

機能は正しく、NATを構成するときに使用できます。

12
Michael

Java

String ip = request.getRemoteAddr();
if (ip.equalsIgnoreCase("0:0:0:0:0:0:0:1")) {
    InetAddress inetAddress = InetAddress.getLocalHost();
    String ipAddress = inetAddress.getHostAddress();
    ip = ipAddress;
}
model.addAttribute("IP", ip);

JSP <span><B>IP : <c:out value="${IP}" /></B></span> // Spring MAV

[〜#〜] jsp [〜#〜]

<%@ page language="Java" contentType="text/html; charset=ISO-8859-1"   pageEncoding="ISO-8859-1"%>
<%@ page import="Java.net.InetAddress" %>

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1">
<title>Displays Client IP Address</title>
</head>
<body>
<%
    // paste above Java code 
out.println("<br><B>Clients IP : </B>"+ip);
%>
<span><br>Local Host  : <%= request.getRemoteAddr() %></span>
<span><br>ServerPort  : <%= request.getServerPort() %></span>
<span><br>Path        : <%= request.getRequestURI()%></span>
</span>
</body>
</html>

Jspがローカルサーバーにデプロイされ、このjspにアクセスすると、0:0:0:0:0:0:0:1が返されます。 ipを使用してローカルにアプリケーションにアクセスすると、IPが返されます。

RESTful Jerseyを使用したWebサービス

@javax.ws.rs.Path("/ip")
public class GetIPInfo {

    @GET    @Produces(MediaType.TEXT_PLAIN) 
    public Response getIP(@Context javax.servlet.http.HttpServletRequest request) throws UnknownHostException { 

        String remoteHost = request.getRemoteHost();
        String remoteAddr = request.getRemoteAddr();
        if (remoteAddr.equals("0:0:0:0:0:0:0:1")) {
            InetAddress localip = Java.net.InetAddress.getLocalHost();
            remoteAddr = localip.getHostAddress();
            remoteHost = localip.getHostName();
        }
        int remotePort = request.getRemotePort();

        String msgoutput = remoteHost + " (" + remoteAddr + " : " + remotePort + ")";
        return Response.status(200).entity(msgoutput).build(); 
    }
}

JavaScriptを使用して検索[〜#〜] ip [〜#〜]

10
Yash