
By Sebastián Alvarez

Server-Sent Events (SSE) is a technology based on HTTP. On the client-side, it provides an API called EventSource (part of the HTML5 standard) that allows us to connect to the server and receive updates from it.
Before making the decision to use server-sent events, we must take into account two very important aspects:
If your project only receives something like stock prices or text information about something in progress it is a candidate for using Server-Sent Events instead of an alternative like WebSockets.
In this article, you will build a complete solution for both the backend and frontend to handle real-time information flowing from server to client. The server will be in charge of dispatching new updates to all connected clients and the web app will connect to the server, receive these updates and display them.
To follow through this tutorial, you’ll need:
This tutorial was verified with cURL v7.64.1, Node v15.3.0, npm v7.4.0, express v4.17.1, body-parser v1.19.0, cors v2.8.5, and react v17.0.1.
In this section, you will create a new project directory. Inside of the project directory will be a subdirectory for the server. Later, you will also create a subdirectory for the client.
First, open your terminal and create a new project directory:
- mkdir node-sse-example
Navigate to the newly created project directory:
- cd node-sse-example
Next, create a new server directory:
- mkdir sse-server
Navigate to the newly created server directory:
- cd sse-server
Initialize a new npm project:
- npm init -y
Install express, body-parser, and cors:
- npm install express@4.17.1 body-parser@1.19.0 cors@2.8.5 --save
This completes setting up dependencies for the backend.
In this section, you will develop the backend of the application. It will need to support these features:
GET /events endpoint to register for updatesPOST /facts endpoint for new factsGET /status endpoint to know how many clients have connectedcors middleware to allow connections from the frontend appUse the first terminal session that is in the sse-server directory. Create a new server.js file:
Open the server.js file in your code editor. Require the needed modules and initialize Express app:
const express = require('express');
const bodyParser = require('body-parser');
const cors = require('cors');
const app = express();
app.use(cors());
app.use(bodyParser.json());
app.use(bodyParser.urlencoded({extended: false}));
app.get('/status', (request, response) => response.json({clients: clients.length}));
const PORT = 3000;
let clients = [];
let facts = [];
app.listen(PORT, () => {
  console.log(`Facts Events service listening at http://localhost:${PORT}`)
})
Then, build the middleware for GET requests to the /events endpoint. Add the following lines of the code to server.js:
// ...
function eventsHandler(request, response, next) {
  const headers = {
    'Content-Type': 'text/event-stream',
    'Connection': 'keep-alive',
    'Cache-Control': 'no-cache'
  };
  response.writeHead(200, headers);
  const data = `data: ${JSON.stringify(facts)}\n\n`;
  response.write(data);
  const clientId = Date.now();
  const newClient = {
    id: clientId,
    response
  };
  clients.push(newClient);
  request.on('close', () => {
    console.log(`${clientId} Connection closed`);
    clients = clients.filter(client => client.id !== clientId);
  });
}
app.get('/events', eventsHandler);
The eventsHandler middleware receives the request and response objects that Express provides.
Headers are required to keep the connection open. The Content-Type header is set to 'text/event-stream' and the Connection header is set to 'keep-alive'. The Cache-Control header is optional, set to 'no-cache'. Additionally, the HTTP Status is set to 200 - the status code for a successful request.
After a client opens a connection, the facts are turned into a string. Because this is a text-based transport you must stringify the array, also to fulfill the standard the message needs a specific format. This code declares a field called data and sets to it the stringified array. The last detail of note is the double trailing newline \n\n is mandatory to indicate the end of an event.
A clientId is generated based on the timestamp and the response Express object. These are saved to the clients array. When a client closes a connection, the array of clients is updated to filter out that client.
Then, build the middleware for POST requests to the /fact endpoint. Add the following lines of the code to server.js:
// ...
function sendEventsToAll(newFact) {
  clients.forEach(client => client.response.write(`data: ${JSON.stringify(newFact)}\n\n`))
}
async function addFact(request, respsonse, next) {
  const newFact = request.body;
  facts.push(newFact);
  respsonse.json(newFact)
  return sendEventsToAll(newFact);
}
app.post('/fact', addFact);
The main goal of the server is to keep all clients connected and informed when new facts are added. The addNest middleware saves the fact, returns it to the client which made POST request, and invokes the sendEventsToAll function.
sendEventsToAll iterates the clients array and uses the write method of each Express response object to send the update.
Before the web app implementation, you can test your server using cURL:
In a terminal window, navigate to the sse-server directory in your project directory. And run the following command:
- node server.js
It will display the following message:
- OutputFacts Events service listening at http://localhost:3001
In a second terminal window, open a connection waiting for updates with the following command:
- curl -H Accept:text/event-stream http://localhost:3001/events
This will generate the following response:
- Outputdata: []
An empty array.
In a third terminal window, create a post POST request to add a new fact with the following command:
- curl -X POST \
-  -H "Content-Type: application/json" \
-  -d '{"info": "Shark teeth are embedded in the gums rather than directly affixed to the jaw, and are constantly replaced throughout life.", "source": "https://en.wikipedia.org/wiki/Shark"}'\
-  -s http://localhost:3001/fact
After the POST request, the second terminal window should update with the new fact:
- Outputdata: {"info": "Shark teeth are embedded in the gums rather than directly affixed to the jaw, and are constantly replaced throughout life.", "source": "https://en.wikipedia.org/wiki/Shark"}
Now the facts array is populated with one item if you close the communication on the second tab and open it again:
- curl -H Accept:text/event-stream http://localhost:3001/events
Instead of an empty array, you should now receive a message with this new item:
- Outputdata: [{"info": "Shark teeth are embedded in the gums rather than directly affixed to the jaw, and are constantly replaced throughout life.", "source": "https://en.wikipedia.org/wiki/Shark"}]
At this point, the backend is fully functional. It is now time to implement the EventSource API on the frontend.
In this part of our project, you will write a React app that uses the EventSource API.
The web app will have the following set of features:
Now, open a new terminal window and navigate to the project directory. Use create-react-app to generate a React App.
- npx create-react-app sse-client
Navigate to the newly created client directory:
- cd sse-client
Run the client application:
- npm start
This should open a new browser window with your new React application. This completes setting up dependencies for the frontend.
For styling, open the App.css file in your code editor. And modify the contents with the following lines of code:
body {
  color: #555;
  font-size: 25px;
  line-height: 1.5;
  margin: 0 auto;
  max-width: 50em;
  padding: 4em 1em;
}
.stats-table {
  border-collapse: collapse;
  text-align: center;
  width: 100%;
}
.stats-table tbody tr:hover {
  background-color: #f5f5f5;
}
Then, open the App.js file in your code editor. And modify the contents with the following lines of code:
import React, { useState, useEffect } from 'react';
import './App.css';
function App() {
  const [ facts, setFacts ] = useState([]);
  const [ listening, setListening ] = useState(false);
  useEffect( () => {
    if (!listening) {
      const events = new EventSource('http://localhost:3001/events');
      events.onmessage = (event) => {
        const parsedData = JSON.parse(event.data);
        setFacts((facts) => facts.concat(parsedData));
      };
      setListening(true);
    }
  }, [listening, facts]);
  return (
    <table className="stats-table">
      <thead>
        <tr>
          <th>Fact</th>
          <th>Source</th>
        </tr>
      </thead>
      <tbody>
        {
          facts.map((fact, i) =>
            <tr key={i}>
              <td>{fact.info}</td>
              <td>{fact.source}</td>
            </tr>
          )
        }
      </tbody>
    </table>
  );
}
export default App;
The useEffect function argument contains the important parts: an EventSource object with the /events endpoint and an onmessage method where the data property of the event is parsed.
Unlike the cURL response, you now have the event as an object. You can now take the data property and parse it giving, as a result, a valid JSON object.
Finally, this code pushes the new fact to the list of facts, and the table gets re-rendered.
Now, try adding a new fact.
In a terminal window, run the following command:
- curl -X POST \
-  -H "Content-Type: application/json" \
-  -d '{"info": "Shark teeth are embedded in the gums rather than directly affixed to the jaw, and are constantly replaced throughout life.", "source": "https://en.wikipedia.org/wiki/Shark"}'\
-  -s http://localhost:3001/fact
The POST request added a new fact and all the connected clients should have received it. If you check the application in the browser you will have a new row with this information.
This article served as an introduction to server-sent events. In this article, you built a complete solution for both the backend and frontend to handle real-time information flowing from server to client.
SSE were designed for text-based and unidirectional transport. Here’s the current support for EventSource in browsers.
Continue your learning by exploring all of the features available to EventSource like retry.
Thanks for learning with the DigitalOcean Community. Check out our offerings for compute, storage, networking, and managed databases.
This textbox defaults to using Markdown to format your answer.
You can type !ref in this text area to quickly search our full set of tutorials, documentation & marketplace offerings and insert the link!
This comment has been deleted
bodyParser is now deprecated and its functionality is now included in Express, so the require(‘body-parser’) line needs to be deleted and these lines:
app.use(bodyParser.json());
app.use(bodyParser.urlencoded({extended: false}));
Change to:
app.use(express.json());
app.use(express.urlencoded({extended: false}));
Also the listen port is set to 3000, but all the port references after that are 3001.
And there’s a typo re: respsonse in function addFact.
Other than those minor issues, great tutorial saving me time in my SSE app development. Thanks!
Correct me if I’m missing something, but it seems that async and next should both be removed from the addFact() declaration because there is no await nor next() within the function.
How this exact example can be done if node application has more than 1 process. Let’s say that application works in pm2 cluster mode.
This comment has been deleted
This comment has been deleted
If you’re seeing double connections from the client, that’s useEffect running twice because of React Strict mode, which is on in dev by default
StrictMode renders components twice (on dev but not production) in order to detect any problems with your code and warn you about them (which can be quite useful).
You can remove the strict mode tags from index.js.
The provided example is just too trivial and can’t be directly applied to any real production application. What if you have multiple server instances and a load-balancer? What if you need to send events from server-side workers?
“After the POST request, the second terminal window should update with the new fact”
It should but it doesn’t. The list remain empty
Get paid to write technical tutorials and select a tech-focused charity to receive a matching donation.
Full documentation for every DigitalOcean product.
The Wave has everything you need to know about building a business, from raising funding to marketing your product.
Stay up to date by signing up for DigitalOcean’s Infrastructure as a Newsletter.
New accounts only. By submitting your email you agree to our Privacy Policy
Scale up as you grow — whether you're running one virtual machine or ten thousand.
Sign up and get $200 in credit for your first 60 days with DigitalOcean.*
*This promotional offer applies to new accounts only.