WebSocket proxying
WebSocket代理
要将客户端和服务器之间的连接从HTTP / 1.1转换为WebSocket,将使用HTTP / 1.1中提供的协议切换机制。
然而,有一点微妙之处在于:由于“升级”是一个逐跳的标头,因此它不会从客户端传递到代理服务器。通过正向代理,客户可以使用该CONNECT
方法来绕过这个问题。但是,这不适用于反向代理,因为客户端不知道任何代理服务器,并且需要在代理服务器上进行特殊处理。
从版本1.3.13开始,nginx实现了特殊的操作模式,如果代理服务器返回了代码为101(交换协议)的响应,客户端和代理服务器之间可以建立隧道,客户端通过请求中的“升级”标题。
如上所述,包含“升级”和“连接”的逐跳标头不会从客户端传递到代理服务器,因此为了使代理服务器知道客户端将协议切换到WebSocket的意图,这些标头必须明确地通过:
location /chat/ {
proxy_pass http://backend;
proxy_http_version 1.1;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection "upgrade";
}
一个更复杂的例子,其中对代理服务器的请求中的“Connection”头字段的值取决于客户端请求头中“Upgrade”字段的存在:
http {
map $http_upgrade $connection_upgrade {
default upgrade;
'' close;
}
server {
...
location /chat/ {
proxy_pass http://backend;
proxy_http_version 1.1;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection $connection_upgrade;
}
}
默认情况下,如果代理服务器在60秒内未传输任何数据,则连接将被关闭。这个超时可以通过proxy_read_timeout指令来增加。或者,代理服务器可以配置为定期发送WebSocket ping帧以重置超时并检查连接是否仍然存在。