java - 在 Websocket 上通过 Nginx 转发用户 IP

标签 java nginx websocket

我正在使用 this Websocket library来自 here 的示例基于 websockets 创建多个服务。

接受请求的主服务器是 Nginx 服务器,它根据来源(基于域)转发请求,因此我可以确定收到的请求是针对 websockets 的。

问题就在这里:

public void onOpen(WebSocket ws, ClientHandshake handshake) {
    System.out.println("-------------------------------");
    System.out.println(ws.getRemoteSocketAddress());
    System.out.println("-------------------------------");
}

getRemoteSocketAddress 将始终返回 nginx 服务器 IP,类似于 /192.168.1.100:43556

我的 nginx 配置是:

server{
    listen 80;
    server_name we.mydomain.com;
    access_log /var/log/nginx/access-ws-debug.log;
    error_log /var/log/nginx/error-ws-debug.log;

    set_real_ip_from  192.168.1.0/24;
    real_ip_header X-Real-IP;
    real_ip_recursive on;

    location / {
       proxy_pass http://192.168.1.101:54321;

       proxy_http_version 1.1;
       proxy_set_header X-Real-IP $remote_addr;
       proxy_set_header Host $http_host; #host
       proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
       proxy_set_header Y-Real-IP $realip_remote_addr;

       proxy_set_header Upgrade $http_upgrade;
       proxy_set_header Connection "Upgrade";

       proxy_connect_timeout 1d;
       proxy_send_timeout 1d;
       proxy_read_timeout 1d;
    }
}

nxing配置有问题吗?或者有没有其他方法可以获取 websocket 背后的用户的 IP?


感谢马克的回答解决了。以下代码与提到的库无关。它是 jsp 中的“经典”Web 实现,没有任何库。

我将握手修改为如下所示:

@Override
public void modifyHandshake(
        ServerEndpointConfig config,
        HandshakeRequest request,
        HandshakeResponse response
) {
    HttpSession httpSession = (HttpSession) request.getHttpSession();
    config.getUserProperties().put(HttpSession.class.getName(), httpSession);
    config.getUserProperties().put("Headers",request.getHeaders());
}

要获取标题,只需在 OnOpen 中调用它:

@OnOpen
public void onOpen(final Session session, EndpointConfig config){ //This happens when javascript opens socket session
    Object list = config.getUserProperties().get("Headers");
    System.out.println("-------------------------------------");
    System.out.println(list);
    System.out.println("-------------------------------------");
}

标题的返回格式如下(使用我的手机):

{upgrade=[websocket], 
accept-language=[en-US,en;q=0.9,es;q=0.8], 
cookie=[JSESSIONID=0BAE99DFDE5DD81D20CFF04E4257D7FE], 
sec-websocket-extensions=[permessage-deflate; client_max_window_bits], 
origin=[https://mywebsite.com], 
x-forwarded-for=[1.2.3.4], 
sec-websocket-version=[13], 
pragma=[no-cache], 
x-real-ip=[1.2.3.4], 
y-real-ip=[1.2.3.4], 
host=[mywebsite.com], 
connection=[Upgrade], 
sec-websocket-key=[ZroOewrBGrg660x7ocI+nog==], 
cache-control=[no-cache], 
accept-encoding=[gzip, deflate, br], 
user-agent=[Mozilla/5.0 (Linux; Android 8.0.0; SM-N950F) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/70.0.3538.80 Mobile Safari/537.36]}

而且 IP 可以在我所在的区域附近进行地理定位。

最佳答案

这不是一项简单的任务,但您的配置看起来不错,您只需从您设置的 header (X-Real-IP) 中获取原始 IP。

import javax.websocket.*;
import javax.websocket.server.HandshakeRequest;
import javax.websocket.server.ServerEndpoint;
import javax.websocket.server.ServerEndpointConfig;

@ServerEndpoint(value = "/test", configurator = Test.TestConfiguratior.class)
public class Test extends Endpoint {

    @Override
    public void onOpen(Session session, EndpointConfig endpointConfig) {
        Object o = endpointConfig.getUserProperties().get(TestConfiguratior.HEADER_NAME);
        //o should be the string object you are looking for
    }

    @OnMessage
    public void onMessage(Session session, String msg) {
        //do what u want
    }

    public static class TestConfiguratior extends ServerEndpointConfig.Configurator {
        static String HEADER_NAME = "X-Real-IP";

        @Override
        public void modifyHandshake(ServerEndpointConfig sec, HandshakeRequest request, HandshakeResponse response) {
            String firstFoundHeader = request.getHeaders().get(HEADER_NAME.toLowerCase()).get(0);
            sec.getUserProperties().put(HEADER_NAME, firstFoundHeader);
        }
    }
}

TL.DR.:您可以使用 javax.websocket.EndpointConfig#getUserProperties 映射将数据放入其中并稍后检索。最初我认为这可能会在使用这段确切的代码时导致一些并发问题,但我无法重现它。甚至文档也建议这样做: https://javaee.github.io/tutorial/websocket010.html

更新:这是一个 JSR 356 示例,并非特定于上述框架。

关于java - 在 Websocket 上通过 Nginx 转发用户 IP,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53186007/

相关文章:

java - 异常 : "org.codehaus.jackson.map.JsonMappingException: Can not deserialize instance of java.lang.String out of START_OBJECT token"

postgresql - nginx:在 PostgreSQL 中存储 POST 数据

python -/etc/nginx/sites-enabled/django 中的无效参数 server_name

javascript - Websocket 握手 Sec-WebSocket-Accept header 值不正确

java - 如何让我的 ContainerRequestFilter 在 @ServerEndpoint onOpen 上调用?

运行测试脚本时出现 java.util.concurrent.TimeoutException

java - 如果我从代码中排除 session.evict() 会发生什么?

java - 如何通过java程序获取exe运行的进程id

ubuntu - Varnish 503 后端获取失败

html - Chrome 16 websocket 二进制文件