Safari 服务器发送事件 (SSE) 无限循环

Bra*_*ski 5 javascript safari eventsource

我在 Safari 9 和 Safari 10 中遇到服务器发送事件 (SSE) 问题。 SSE 连接打开,立即关闭,然后无限循环重新连接。

这是客户端代码:

var events = new EventSource("/stream/events")
Run Code Online (Sandbox Code Playgroud)

这些是 http 响应标头:

> GET /stream/events HTTP/1.1
> User-Agent: curl/7.43.0
> Accept: */*
> 
< HTTP/1.1 200 OK
< Access-Control-Allow-Origin: *
< Cache-Control: no-cache
< Connection: keep-alive
< Content-Type: text/event-stream
< Expires: Thu, 01 Jan 1970 00:00:00 GMT
< Last-Modified: Tue, 19 Sep 2017 05:28:22 GMT
< Strict-Transport-Security: max-age=31536000
< X-Accel-Buffering: no
< X-Content-Type-Options: nosniff
< X-Frame-Options: DENY
< X-Xss-Protection: 1; mode=block
< Date: Tue, 19 Sep 2017 05:28:22 GMT
< Transfer-Encoding: chunked
Run Code Online (Sandbox Code Playgroud)

一些补充说明:

  • 我在 Chrome 和 Firefox 中测试过,不能重复
  • 我在没有 https 的Safari 中进行了测试,无法重复
  • 我在 Safari 中使用 https进行了测试,可以重复
  • https 证书是使用 Lets Encrypt 自动生成的
  • 后端服务器是用 Go 编写的,默认使用 http/2

我只能在 Safari 中使用 https 重复这一事实很有趣。因此,我想知道 SSE 和 https 是否存在任何已知问题,或者是否还有其他任何我可能在这里配置错误或遗漏的地方。

编辑

我已经隔离了问题并找到了与协议的相关性。启用 http2 协议后,我可以重现此问题。在服务器上禁用 http2 后,我无法再重现此问题。

我使用以下服务器补丁进行验证:

--- before.go   2017-09-19 13:31:45.668891000 -0400
+++ after.go    2017-09-19 13:31:55.100891000 -0400
@@ -2,6 +2,6 @@
            Addr: ":443",
            TLSConfig: &tls.Config{
                GetCertificate: manager.GetCertificate,
-               NextProtos:     []string{"h2", "http/1.1"},
+               NextProtos:     []string{"http/1.1"},
            },
        }
Run Code Online (Sandbox Code Playgroud)

Cla*_*son 2

我敢打赌 Safari javascript 控制台充满了:

[Error] Failed to load resource: The operation couldnt be completed. Protocol error
Run Code Online (Sandbox Code Playgroud)

这很可能是一个协议错误。http/2 的第8.1.2.2 节

端点不得生成包含特定于连接的标头字段的 HTTP/2 消息;任何包含特定于连接的头字段的消息都必须被视为格式错误。

我无法确保这是您的问题,因为我没有看到您的 http/2 尝试的 HTTP 标头,但是许多 http/2 实现选择仍然发送Connection: Keep-Alive标头,而包括 Safari 和curl 在内的多个用户代理决定对此事要严格。

更新

我应该补充一点,错误的标头(最有可能是标准的“Connection:keep-alive”标头)可以由任何服务器端组件(通常是应用程序的 SSE 服务器端组件)设置,而不是由容器本身设置。我打赌容器应该在 http/2 中过滤它。