server-sent-events

How do server-sent events actually work?


So I understand the concept of server-sent events (EventSource):

The thing I'm confused about is how it works on the server. I've had a look at different examples, but the one that comes to mind is Mozilla's: http://hacks.mozilla.org/2011/06/a-wall-powered-by-eventsource-and-server-sent-events/

Now this may be just a bad example, but it kinda makes sense how the server side would work, as I understand it:

Does that make sense? Is that really how it works from a barebones perspective?


Solution

  • The HTML5 doctor site has a great write-up on server-sent events, but I'll try to provide a (reasonably) short summary here as well.

    Server-sent events are, at its core, a long running http connection, a special mime type (text/event-stream) and a user agent that provides the EventSource API. Together, these make the foundation of a unidirectional connection between a server and a client, where messages can be sent from server to client.

    On the server side, it's rather simple. All you really need to do is set the following http headers:

    Content-Type: text/event-stream
    Cache-Control: no-cache
    Connection: keep-alive
    

    Be sure to respond with the code 200 and not 204 or any other code, as this will cause compliant user agents to disconnect. Also, make sure to not end the connection on the server side. You are now free to start pushing messages down that connection. In nodejs (using express), this might look something like the following:

    app.get("/my-stream", function(req, res) {
        res.status(200)
           .set({ "content-type"  : "text/event-stream"
                , "cache-control" : "no-cache"
                , "connection"    : "keep-alive"
                })
    
        res.write("data: Hello, world!\n\n")
    })
    

    On the client, you just use the EventSource API, as you noted:

    var source = new EventSource("/my-stream")
    source.addEventListener("message", function(message) {
        console.log(message.data)
    })
    

    And that's it, basically.

    Now, in practice, what actually happens here is that the connection is kept alive by the server and the client by means of a mutual contract. The server will keep the connection alive for as long as it sees fit. Should it want to, it may terminate the connection and respond with a 204 No Content next time the client tries to connect. This will cause the client to stop trying to reconnect. I'm not sure if there's a way to end the connection in a way that the client is told not to reconnect at all, thereby skipping the client trying to reconnect once.

    As mentioned, the client will keep the connection alive as well, and try to reconnect if it is dropped. The algorithm to reconnect is specified in the spec, and is fairly straight forward.

    One super important bit that I've so far barely touched on however is the mime type. The mime type defines the format of the message coming down the connection. Note however that it doesn't dictate the format of the contents of the messages, just the structure of the messages themselves. The mime type is extremely straight forward. Messages are essentially key/value pairs of information. The key must be one of a predefined set:

    Any other keys should be ignored. Messages are then delimited by the use of two newline characters: \n\n

    The following is a valid message: (last new line characters added for verbosity)

    data: Hello, world!
    \n
    

    The client will see this as: Hello, world!.

    As is this:

    data: Hello,
    data: world!
    \n
    

    The client will see this as: Hello,\nworld!.

    That pretty much sums up what server-sent events are: a long running non-cached http connection, a mime type and a simple javascript API.

    For more information, I strongly suggest reading the specification. It's small and describes things very well (although the requirements of the server side could possibly be summarized a bit better.) I highly suggest reading it for the expected behavior with certain http status codes, for instance.