More on HTTP

Just return a 200 here, it will be fine.
— Seconds before disaster

Status Codes

Responses can have status codes attached to them. These indicate the status of a request.

Each status code is a number that indicates some information about the response. For example, the status code 400 represents a Bad Request.

There are a lot of status codes and they're grouped in five classes:

  • the status codes 100-199 indicate informational responses (and you'll rarely see them)
  • the status codes 200-299 indicate successful responses
  • the status codes 300-399 indicate redirection responses
  • the status codes 400-499 indicate client errors
  • the status codes 500-599 indicate server errors

Quoting the famous HTTP status ranges in a nutshell, you could also rephrase this as:

  • 100-199: hold on
  • 200-299: here you go
  • 300-399: go away
  • 400-499: you fucked up
  • 500-599: I fucked up

Let's have a quick look at the most important status codes.

The status code 200 (OK) is the standard response for a successful HTTP request.

The status code 201 (Created) is often used for requests where a resource has been successfully created.

The status code 400 (Bad Request) means that the server can't process the request due to a client error. For example, a 400 should be returned if the client has provided the server with a malformed request.

The status code 401 (Unauthorized) usually means that the client couldn't be successfully authenticated.

The status code 403 (Forbidden) usually means that the client doesn't have the necessary rights to access the requested content. Note that there is an important difference between 401 and 403. The code 401 means that the client couldn't be authenticated at all, the code 403 means that the client could be authenticated, but doesn't have the correct access rights.

You've probably heard of the most famous status code 404 (Not Found). As the name already says, this means that the server can't find the requested resource (usually because the resource doesn't exist).

The status code 405 (Method Not Allowed) means that the resource exists, but the client is trying to access it using the wrong method. This occurs if, for example, a client is trying to send a GET request to a route that only supports POST requests.

The most important status codes for server errors are 500 (Internal Server Error) and 503 (Service Unavailable). The status code 500 basically indicates a server crash, while 503 indicates that the server is overloaded.

Here is how you can manually return a status code in Express:

app.post('/', (req, res) => {
  const data = req.body;

  // If the request body is empty, we return status code 400 (Bad Request)
  if (Object.keys(data).length === 0) {
    res.status(400).send('Bad Request: No data provided');
  } else {
    res.status(201).send(`Received data: ${JSON.stringify(data)}`);
  }
});

Try to curl the endpoint both with an empty and a non-empty object. For example, you can curl the endpoint with an empty object like this:

curl -X POST -H "Content-Type: application/json" -d '{}' -w '\nStatus: %{http_code}' http://localhost:3000

The output would be:

Bad Request: No data provided
Status: 400

Let's also curl the endpoint with a non-empty object like this:

curl -X POST -H "Content-Type: application/json" -d '{"key": "value"}' -w '\nStatus: %{http_code}' http://localhost:3000

The output would be:

Received data: {"key":"value"}
Status: 201

Headers

Both requests and responses can have headers which provide important information that is not part of the main request or response content. For example, request headers might contain the version of the browser that is making the request, what languages are accepted, what encodings are accepted etc.

Here is how you can read request headers in Express:

app.get('/request-header-example', (req, res) => {
  const customHeaderValue = req.get('Custom-Header');

  if (customHeaderValue) {
    const responseData = { 'Received-Custom-Header': customHeaderValue };
    res.send(`Received data: ${JSON.stringify(responseData)}`);
  } else {
    res.status(400).send('No Custom-Header provided');
  }
});

You can set a header in curl like this:

curl -H "Custom-Header: ExampleValue" http://localhost:3000/request-header-example

You will receive the following response:

Received data: {"Received-Custom-Header":"ExampleValue"}

Similarly, response headers might contain the type of the content, the date of the response etc.

Here is how you can set response headers in Express:

app.get('/response-header-example', (req, res) => {
  const data = req.body;

  // Add a custom header
  res.set('X-Custom-Header', 'MyHeaderValue');

  res.status(201).send(`Received data: ${JSON.stringify(data)}`);
});

You can see the response headers in curl by using the -i flag:

curl -i http://localhost:3000/response-header-example

This will output:

HTTP/1.1 201 Created
X-Powered-By: Express
X-Custom-Header: MyHeaderValue
Content-Type: text/html; charset=utf-8
Content-Length: 17
ETag: W/"11-10nwzUtYNUZcrMxeVU7fTdmCYaI"
Date: Thu, 14 Mar 2024 11:23:03 GMT
Connection: keep-alive
Keep-Alive: timeout=5

Received data: {}

Note that Express and other web frameworks already set quite a few response headers by default.

You can also see the response headers in the network tab of your browser if you click on an individual request.

Cookies

HTTP cookies are pieces of data that are created by the server, sent to the browser and then saved on the browser. Cookies can be used for things like authentication, personalization and tracking.

For example, the server might generate a cookie for a logged in user, send it to the browser and then the browser could send the cookie with each request (avoiding the need for the user to log in on every request).

Note that cookies are simply set as part of the headers. For example, a request might contain a Cookie header containing the cookies it has currently stored. A response might contain a Set-Cookie header which contains the cookies that the browser should set (or clear).

Let's have a look at a small example. First, we'll need to install the cookie-parser package in addition to the express package:

pnpm install -g cookie-parser

Next, we will create an Express application with three routes. The first route will be a /set-cookie route that will allow us to set a cookie. The second route will be a /read-cookie route that will allow us to read the cookie. Finally, we will add a third route /clear-cookie which will allows us to clear the cookie.

const express = require('express');
const cookieParser = require('cookie-parser');

const app = express();
const PORT = 3000;

app.use(cookieParser());

// Route to set a cookie
app.get('/set-cookie', (req, res) => {
  res.cookie('myCookie', 'test', { httpOnly: true });
  res.send('Cookie has been set!');
});

// Route to read the cookie
app.get('/read-cookie', (req, res) => {
  const myCookie = req.cookies.myCookie;
  if (myCookie) {
    res.send(`myCookie: ${myCookie}`);
  } else {
    res.status(404).send('No cookie found');
  }
});

// Route to clear the cookie
app.get('/clear-cookie', (req, res) => {
  res.clearCookie('myCookie');
  res.send('Cookie has been cleared');
});

app.listen(PORT, () => {
  console.log(`Server listening on port ${PORT}...`);
});

Let's launch the server, open a browser and go to http://localhost:3000/set-cookie. You should see a message Cookie has been set! and a new key-value pair appearing in your "cookies" tab.

If you now go to http://localhost:3000/read-cookie, you will see myCookie: test on the page.

Finally, if you go to http://localhost:3000/clear-cookie, you will see that the cookie has been cleared.

Redirects

The 301 (Moved Permanently) status code is used for URL redirection. This basically informs a client that the resource has been permanently moved. A response with a 301 status code should contain the URL where the resource has been moved to.

The 302 (Found) status code is also used for URL redirection, but it indicates to the client that the resource has been temporarily moved. Again, a response with a 302 status code should contain the URL where the resource has been moved to.

Let's have a look at an Express example:

const express = require('express');
const cookieParser = require('cookie-parser');

const app = express();
const PORT = 3000;

// 301 Redirect (Permanent)
app.get('/permanent-redirect', (req, res) => {
  res.redirect(301, '/permanent-target');
});

// 302 Redirect (Temporary)
app.get('/temporary-redirect', (req, res) => {
  res.redirect(302, '/temporary-target');
});

// Target Route for 301 Redirect
app.get('/permanent-target', (req, res) => {
  res.send('This is the permanent target route');
});

// Target Route for 302 Redirect
app.get('/temporary-target', (req, res) => {
  res.send('This is the temporary target route');
});

app.listen(PORT, () => {
  console.log(`Server listening on port ${PORT}...`);
});

If you open http://localhost:3000/permanent-redirect in your browser, you will see that the address in the browser address bar suddenly changes to http://localhost:3000/permanent-target and you see the content 'This is the permanent target route'. If you open the network tab, you will see that two requests are sent to accomplish this.

First, your browser sends a request to http://localhost:3000/permanent-redirect and receives a response with status code 301 and the response header Location set to /permanent-target. Your browser now sends another request to the URL that is indicated in the Location header and receives the actual content.

You will see similar behaviour with the /temporary-redirect and /temporary-target routes.