Nginx add_header Sec-WebSocket-Protocol 不起作用

Posted

技术标签:

【中文标题】Nginx add_header Sec-WebSocket-Protocol 不起作用【英文标题】:Nginx add_header Sec-WebSocket-Protocol does not work 【发布时间】:2015-11-17 10:06:21 【问题描述】:

我使用nginx 作为我的WebSocket 流量的代理,效果很好。我的一个应用程序需要使用STOMP,因为我选择了STOMP js 库,它尝试通过Sec-WebSocket-Protocol 标头协商协议,但我无法在nginx 响应中添加此标头。

nginx version: nginx/1.6.2

下面是我的服务器块:

server 
  listen 80;
  server_name mware.example.com;

  client_max_body_size 100M;
  keepalive_timeout 5;

  root /srv/applications;
  autoindex off;

  location / 
    # WebSocket
    location = /ws 
      proxy_read_timeout 10m;
      proxy_http_version 1.1;
      proxy_set_header Upgrade $http_upgrade;
      proxy_set_header Connection "upgrade";
      proxy_pass http://127.0.0.1:8000/;
      proxy_redirect  http://127.0.0.1:8000/ /;

      # Custom headers
      proxy_set_header  Host            $host;
      proxy_set_header  X-Real-IP       $remote_addr;
      proxy_set_header  REMOTE_PORT     $remote_port;
      proxy_set_header  X-Forwarded-For $proxy_add_x_forwarded_for;

      #add_header    Sec-WebSocket-Protocol  $http_sec_websocket_protocol;
      add_header    Sec-WebSocket-Protocol  'v10.stomp, v11.stomp';
    
  

因此尝试通过$http_sec_websocket_protocol 变量添加标题以及逐字指定它,但这些都不适合我。下面是localhost上的抓包。

发送的数据包:

T 10.3.2.100:36798 -> y.y.y.y:80 [AP]
GET /ws HTTP/1.1.
Host: mware.example.com.
Connection: Upgrade.
Pragma: no-cache.
Cache-Control: no-cache.
Upgrade: websocket.
Origin: file://.
Sec-WebSocket-Version: 13.
User-Agent: Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (Khtml, like Gecko) Chrome/46.0.2490.71 Safari/537.36.
Accept-Encoding: gzip, deflate, sdch.
Accept-Language: en-US,en;q=0.8.
Cookie: __utma=82131859.1920436741.1415282304.1421757291.1421760088.10; session_id_api=x.x.x.x-87eeca97-c6db-4010-bef8-f4875a6feb52; session_id_portal=x.x.x.x-ace62281-2446-4708-aa9d-4390eb41e02a.
Sec-WebSocket-Key: bGN7hB4fw7/JuF6A5vSf6A==.
Sec-WebSocket-Extensions: permessage-deflate; client_max_window_bits.
Sec-WebSocket-Protocol: v10.stomp, v11.stomp.

Nginx 响应:

T y.y.y.y:80 -> 10.3.2.100:36798 [AP]
HTTP/1.1 101 Switching Protocols.
Server: nginx/1.6.2.
Date: Tue, 17 Nov 2015 09:35:18 GMT.
Connection: upgrade.
Upgrade: websocket.
Sec-WebSocket-Accept: D34d7GBjjQkSC8jrjpixNmMxDMc=.

您可以看到Sec-WebSocket-Protocol 标头丢失了。

在应用程序级别,我收到 nginx 正确格式化的数据包,添加了自定义标头等。

Upgrade: websocket
Connection: upgrade
Host: mware.example.com
X-Real-IP: x.x.x.x
REMOTE_PORT: 36798
X-Forwarded-For: x.x.x.x
Pragma: no-cache
Cache-Control: no-cache
Origin: file://
Sec-WebSocket-Version: 13
User-Agent: Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/46.0.2490.71 Safari/537.36
Accept-Encoding: gzip, deflate, sdch
Accept-Language: en-US,en;q=0.8
Cookie: __utma=82131859.1920436741.1415282304.1421757291.1421760088.10; session_id_api=x.x.x.x-87eeca97-c6db-4010-bef8-f4875a6feb52; session_id_portal=x.x.x.x-ace62281-2446-4708-aa9d-4390eb41e02a
Sec-WebSocket-Key: bGN7hB4fw7/JuF6A5vSf6A==
Sec-WebSocket-Extensions: permessage-deflate; client_max_window_bits
Sec-WebSocket-Protocol: v10.stomp, v11.stomp

【问题讨论】:

【参考方案1】:

http://nginx.org/r/add_header

如果响应代码等于 200、201、204、206、301、302、303、304 或 307,则将指定字段添加到响应标头。值可以包含变量。

您使用 101 代码响应。

从 1.7.5 版开始有 always 标志,但您使用的是旧版本。您也可以考虑使用第三方模块,如headers_morelua

【讨论】:

以上是关于Nginx add_header Sec-WebSocket-Protocol 不起作用的主要内容,如果未能解决你的问题,请参考以下文章

Nginx指令add_header和proxy_set_header的区别

nginx add_header Set-Cookie 过期不起作用

Nginx add_header Sec-WebSocket-Protocol 不起作用

Nginx PHP API CORS

配置nginx指定文件格式下载

nginx 定义:响应头和请求头