无法使用 Nginx 反向代理连接到 Websocket 服务器

Posted

技术标签:

【中文标题】无法使用 Nginx 反向代理连接到 Websocket 服务器【英文标题】:Unable to connect to Websocket Server with Nginx reverse proxy 【发布时间】:2021-01-27 08:07:11 【问题描述】:

我想设置一个带有反向代理的 websocket 服务器。为此,我使用 python 中的简单 websocket 服务器和 nginx 反向代理创建了一个 docker-compose。

设置:

docker-compose.yml:

version: '2.4'
services: 
    wsserver:
        restart: always
        ports: 
            - 8765:8765
        build: 
            context: ./server
            dockerfile: Dockerfile   
    ngproxy:
        image: nginx
        ports: 
            - 8020:80
            - 5000:5000
        restart: always
        depends_on:
            - wsserver
        volumes:
            - ./nginx/nginx.conf:/etc/nginx/conf.conf

nginx.conf:

http 
    map $http_upgrade $connection_upgrade 
        default upgrade;
        '' close;
    
 
    upstream websocket 
        server  wsserver:8765;
    
 
    server 
        listen 5000;

        location / 
            proxy_pass http://websocket;
            proxy_http_version 1.1;
            proxy_set_header Upgrade $http_upgrade;
            proxy_set_header Connection $connection_upgrade;
            proxy_set_header Host $host;
            proxy_set_header X-Real-IP $remote_addr;
            proxy_set_header X-Forwarded-For $remote_addr;
        
    

Websocket 服务器:

Dockerfile:

FROM python:3
RUN pip install websockets
RUN pip install asyncio
COPY server.py /
CMD [ "python", "./server.py" ]

server.py:

import asyncio
import websockets
import os


async def echo(websocket, path):
    async for message in websocket:
        print(message)
        await websocket.send(message)


asyncio.get_event_loop().run_until_complete(
    websockets.serve(echo, '0.0.0.0', 8765))
asyncio.get_event_loop().run_forever()

有助于运行测试的简单 ws 客户端(我也使用了 Chrome 扩展程序):

client.py:

import asyncio
import websockets


async def hello(uri):
    async with websockets.connect(uri) as websocket:
        await websocket.send("Hello world!")
        resp = await websocket.recv()
        print(resp)


asyncio.get_event_loop().run_until_complete(
    hello('ws://localhost:5000'))
# Without reverse proxy -> ws://localhost:8765

问题:

当我尝试连接(使用客户端或使用 Chrome 扩展程序)时,出现以下错误:

WebSocket connection to 'ws://localhost:5000/' failed: Connection closed before receiving a handshake response

我的步骤有什么问题吗?

更多信息:

当我尝试在没有反向代理的情况下连接到 websocket 时,它可以工作。 我的 nginx 错误日志中未显示任何消息。 所有配置均基于 nginx 文档:nginx websocket 我已经检查了这个线程:Docker NGINX Proxy not Forwarding Websockets、Configure NGINX reverse proxy with browser WebSocket and docker-compose 和 Docker nginx websocket proxy - client closed connection while waiting for request

谢谢!

【问题讨论】:

【参考方案1】:

经过一番研究,我终于明白了问题所在:我将本地 nginx 配置映射到容器上的错误文件。

所以要修复它,我改变了我的 docker-compose.yml 中的音量

发件人:

volumes:
            - ./nginx/nginx.conf:/etc/nginx/conf.conf

收件人:

volumes:
            - ./nginx/nginx.conf:/etc/nginx/conf.d/default.conf:ro

并且还从 nginx.conf 中删除了 de http:

nginx.conf

 map $http_upgrade $connection_upgrade 
        default upgrade;
        '' close;
    
 
    upstream websocket 
        server  wsserver:8765;
    
 
    server 
        listen 5000;

        location / 
            proxy_pass http://websocket;
            proxy_http_version 1.1;
            proxy_set_header Upgrade $http_upgrade;
            proxy_set_header Connection $connection_upgrade;
            proxy_set_header Host $host;
            proxy_set_header X-Real-IP $remote_addr;
            proxy_set_header X-Forwarded-For $remote_addr;
        
    

【讨论】:

以上是关于无法使用 Nginx 反向代理连接到 Websocket 服务器的主要内容,如果未能解决你的问题,请参考以下文章

是否可以使用 Java SDK 通过反向代理连接到 Azure Cosmos DB?

NGinX 无法连接到 CentOS 7 上的 Jenkins

通过反向代理服务器将socket.io android客户端连接到套接字服务器

nginx的tcp透明代理设置

【nginx】如何解决使用nginx作为反向代理端口耗尽问题?

Websockets nginx 反向代理 + Cloudflare