WebSocket 是基于 HTTP 协议,为 Web 应用提供的实时双向通讯协议。自从发布以来,越来越多的浏览器和服务端软件都已经支持了 WebSocket 。
当我们使用 nginx 作为 HTTP 接入层时,却会发现默认情况下,WebSocket 通讯会失败。这是因为 nginx 的配置默认情况下不支持 WebSocket,需要额外的配置才能支持 WebSocket。
WebSocket 协议握手细节
当客户端发起 WebSocket 请求时,会首先尝试建议连接,此时使用的请求地址并不是普通的http://
或https://
开头的地址,而是以ws://
(未经TLS加密,与HTTP对应)或wss://
(经TLS加密,与HTTPS对应)开头的地址。
以ws://example.com/websocket
为例,请求头如下:
GET /websocket HTTP/1.1
Host: example.com
Upgrade: websocket
Connection: Upgrade
Sec-WebSocket-Key: dGhlIHNhbXBsZSBub25jZQ==
Origin: http://example.com
Sec-WebSocket-Version: 13
可见,该请求头与普通的 HTTP 请求头非常类似,除了多几个字段:
Upgrade
:必须为websocket
,表示需要升级协议为 WebSocket 进行通讯Connection
:必须为Upgrade
,表示需要升级连接Sec-WebSocket-Key
:必须为随机字符串,用于握手验证,服务器也会返回一个类似的字符串
响应头:
HTTP/1.1 101 Switching Protocols
Upgrade: websocket
Connection: Upgrade
Sec-WebSocket-Accept: s3pPLMBiTxaQ9kYGzzhZRbK+xOo=
经过这样的握手,双方就可以建立 WebSocket 连接,进行实时双向通讯了。
配置 WebSocket 反向代理
nginx 反向代理 WebSocket 的话,需要明确地添加Upgrade
和Connection
头:
# 如果没有Upgrade头,则$connection_upgrade为close,否则为upgrade
map $http_upgrade $connection_upgrade {
default upgrade;
'' close;
}
server {
…
location /websocket {
proxy_pass http://localhost:3000;
proxy_http_version 1.1;
proxy_set_header Host $host;
# 下面这两行是关键
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection $connection_upgrade;
}
}
通过以上配置,nginx 就可以正常代理 WebSocket 请求了。
如果有多个后端服务器,则可以使用 upstream
定义多个后端服务器,并在 location
中使用 proxy_pass
指定后端服务器即可:
upstream backend {
192.168.3.1:3000;
192.168.3.2:300;
}
map $http_upgrade $connection_upgrade {
default upgrade;
'' close;
}
server {
…
location /websocket {
proxy_pass http://upstream;
proxy_http_version 1.1;
proxy_set_header Host $host;
# 下面这两行是关键
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection $connection_upgrade;
}
}
小结
nginx 反向代理 WebSocket 还是比较简单的,最重要的就是在反向代理时配置好 Upgrade
和 Connection
头。
问题
Q: 如果有多个服务器, 是怎么建立 WebSocket 连接的呢?
A: Nginx 长连接负载均衡