如何在express.js中使用服务器发送事件

Ste*_*ich 5 javascript node.js express server-sent-events

我使用express.js设置了REST服务器。现在,我想向该服务器添加sse。实现 sse程序包后,出现错误。我知道我收到此错误,何时尝试使用res.send两次,但我没有。

ERROR: Error: Can't set headers after they are sent.                                            
    at ServerResponse.OutgoingMessage.setHeader (http.js:690:11)                            
    at ServerResponse.header (/home/root/node_modules/express/lib/response.js:718:10)         
    at ServerResponse.send (/home/root/node_modules/express/lib/response.js:163:12)            
    at app.get.str (/home/root/.node_app_slot/main.js:1330:25)                                 
    at Layer.handle [as handle_request] (/home/root/node_modules/express/lib/router/layer.js:95:5)    
    at next (/home/root/node_modules/express/lib/router/route.js:131:13)                       
    at sse (/home/root/node_modules/server-sent-events/index.js:35:2)                          
    at Layer.handle [as handle_request] (/home/root/node_modules/express/lib/router/layer.js:95:5)  
    at next (/home/root/node_modules/express/lib/router/route.js:131:13)                          
    at Route.dispatch (/home/root/node_modules/express/lib/router/route.js:112:3)
Run Code Online (Sandbox Code Playgroud)

是否有可能我无法在sse函数中使用express方法?例如:

app.get('/events', sse, function(req, res) {
    res.send('...');
});
Run Code Online (Sandbox Code Playgroud)

此外,我发现这个解决方案,这个。是否可以res.write在不使用其他软件包的情况下通过该函数或其他方式使用sse ?

Joh*_*alt 25

我不同意使用 Socket.IO 来实现基本的服务器发送事件。浏览器 API 非常简单,在 Express 中的实现只需要对正常响应路由进行一些更改:

app.get('/streaming', (req, res) => {

    res.setHeader('Cache-Control', 'no-cache');
    res.setHeader('Content-Type', 'text/event-stream');
    res.setHeader('Access-Control-Allow-Origin', '*');
    res.setHeader('Connection', 'keep-alive');
    res.flushHeaders(); // flush the headers to establish SSE with client

    let counter = 0;
    let interValID = setInterval(() => {
        counter++;
        if (counter >= 10) {
            clearInterval(interValID);
            res.end(); // terminates SSE session
            return;
        }
        res.write(`data: ${JSON.stringify({num: counter})}\n\n`); // res.write() instead of res.send()
    }, 1000);

    // If client closes connection, stop sending events
    res.on('close', () => {
        console.log('client dropped me');
        clearInterval(interValID);
        res.end();
    });
});
Run Code Online (Sandbox Code Playgroud)
  • 根据规范设置适当的标题
  • 使用 res.flushHeaders() 建立 SSE 连接
  • 使用 res.write() 而不是 res.send() 发送数据
  • 要结束来自服务器的流,请使用 res.end()

上面的代码片段使用 setInterval() 模拟向客户端发送数据 10 秒,然后结束连接。客户端将收到连接丢失的错误并自动尝试重新建立连接。为避免这种情况,您可以在出错时关闭客户端,或者让浏览器发送一个特定的事件消息,客户端理解该消息可以正常关闭。如果客户端关闭连接,我们可以捕获 'close' 事件以优雅地结束服务器上的连接并停止发送事件。

快递:4.17.1 节点:10.16.3

  • 很好的答案。我唯一的建议是,您通常还应该添加“Connection: keep-alive”标头(https://masteringjs.io/tutorials/express/server-sent-events)。绝对不要使用 socket.io,因为它使用 websockets,这与服务器发送的事件不同。SSE 基于 HTTP,websockets 是一个独特的协议。 (3认同)

bay*_*ezy 7

您绝对可以在没有其他软件包的情况下实现这一目标。

我写了一篇关于此的博客文章,第 1 部分列出了基础知识。

您不得关闭 SSE,因为这会破坏功能。重点是它是一个开放的 HTTP 连接。这允许随时将新事件推送到客户端。