Learn how to use Node and Express to create and test HTTP servers
Node is often used to create HTTP servers for the web. It's a bit fiddly to do this with just the built-in modules, so we're going to use the Express library to help create our server.
HyperText Transfer Protocol (HTTP) is a way for computers to exchange messages over the internet. The "client" computer will send a "request" (often via a web browser). E.g. if you visit https://google.com your browser sends a request like this:
A "server" computer receives this request and sends a "response". E.g. Google's server would send a response like this:
We're going to learn how to use Node to create an HTTP server that can respond to requests.
Create a new directory
Move into that directory
Create an empty package.json
Install the Express library
Open your editor, then create a server.js
file
Follow along with each example in your own editor.
Before we start creating a server we need to learn how Node manages code in different files. Modules are used to isolate code. In the browser by default all JS has access to the same global scope, even if loaded via separate script tags.
Although browsers do now have Modules Node was created before they were added to JavaScript. This means it has its own system (called "CommonJS").
Confusingly Node has recently added support for these standard JS modules (ESM), however most server code has not yet been updated, so we must learn the older system.
By default all files are self-contained in Node. Code in one file cannot access anything in another file. To use something in another file you must "export" it. You can do so by assigning the value to module.exports
:
To access code that is exported from another file you must "import" it. You can do so by calling the require
function with the path to the file:
Note: the file extension is optional—if you leave it off Node will assume it's a .js
file. This is quite common in the Node ecosystem.
You can export multiple values by assigning an object to module.exports
:
Node provides some built-in modules. You can import these just like your own modules, only without the path (just using their name). Recent versions of Node allow you to add the node:
prefix to make it more explicit that this is a built-in. For example to use the built-in "fs" (filesystem) module:
The same applies to modules you installed via npm. Node will check in your node_modules
directory to find any imports that aren't relative file paths. For example to import the Express library:
Now we know how to manage our modules let's take a look at using Express to build a server.
We can create a new server object using the express
library:
You can run this file from your terminal with:
Unfortunately nothing much will happen yet, since we haven't told our server to do anything.
Our server isn't currently listening for requests. Servers need to connect to the network and listen for incoming HTTP requests via a "port".
It's a good idea to start the server in a different file: this will help with testing later on. We first need to export the server object we just created:
Then we create a new file named index.js
file. This is just the "entrypoint" for our app—its job is to start the server. All the actual server logic will stay in server.js
.
This tells the server to listen for any requests sent to port 3000. Now we can run the program in our terminal again:
The server will start, but you won't see anything happen. Your terminal will be "stuck" as the Node process is still running (the server will keep listening indefinitely). This means you can't run any more commands.
It would be nice see a log so we know our server started correctly. Luckily the .listen
method takes a second argument—a function to run once the server is listening.
We can use this to log a message:
Stop the previous process, re-run your file, and you should see "Listening on http://localhost:3000" logged.
Since your server is now running you can send some HTTP requests to it to see what happens. The easiest way is to visit http://localhost:3000/ in your browser.
You can also use the curl
CLI program to send requests right from your terminal. Open a new tab and run this command to send a GET request:
You should see the response logged. Whether your used your browser or curl
you should get a 404 Not found
response from your server.
This is Express' default "missing" page, which it will return when you haven't defined a response for a particular request. Let's fix that by defining our first route.
Our server currently does nothing. We need to add a "route". This is a function that will be run whenever the server receives a request for a specific path.
The server
object has methods representing all the HTTP verbs (GET
, POST
etc). These methods take two arguments: the path to match and a handler function.
Here we tell the server to call our function whenever it receives a HTTP GET request to the home path. This is similar to an event listener in the DOM.
The handler function will be passed two arguments: an object representing the incoming request, and an object representing the response that will eventually be sent. Here we've named them request
and response
respectively. These are often abbreviated to req
and res
.
We can use the send
method of the response object to tell Express to send the response. Whatever argument we pass will be sent as the response body.
Open http://localhost:3000/ in your browser. This will send a GET
request to your server. You should see the "hello" response on the page. It's helpful to open the network tab of the dev tools so you can see all the details of the request and response.
Making requests manually in the browser or terminal is fine for quick checks, but to be responsible developers we should write automated tests to verify our server is working correctly.
As of version 18 Node has a built-in test runner. It works in a similar way to the popular Tape testing library (most testing libraries are quite similar).
Let's try an example test. Create a new directory called test/
, and then a new file inside called server.test.js
. We can create a simple test like this:
Since this is a normal Node JS file we can run it from our terminal like any other JS file:
You should see some logs showing the status of your test.
Node also has a special way to run all the tests in your project:
It will find and run any files in a folder named test
, and also any file ending in .test.js
. This is handy when you want to divide your tests up into different files but run them all in one go.
Let's write a test that verifies the route we just added. Our test should:
Start the server
Send a request to the server
Verify the request was successful
Retrieve the response body
Verify that the response is what we expect
Stop the server
Note that this is exactly how we manually checked our server was working before. We just want to automate it.
First we need a test function. We will use an async
function here so that Node knows to wait for any promises to resolve:
Start the server
Next we need to start the server. We'll use a different port to before, just so we don't clash with any already running instances in your terminal:
We import the server object directly and start it listening ourselves. This is why splitting up our app into index.js
and server.js
helps testing: we have control over when the server starts.
Send a request to the server
Now we need to send a request to the server. As of Node 18 we can use fetch
to make HTTP requests, just like in the browser. We can use it to send a request to our server:
Here we're sending a request, waiting for the server to respond, closing the server, then checking that the response was successful. We need to close the server before any assertions, since the test will stop executing as soon as an assertion fails. If that happened before we closed it the server would never stop listening.
Check the response body
Finally we need to get the response body and check that it is correct:
Here we're using the .text()
method to get the raw text content of the response body, then checking it matches what we expect.
Now that we know how to test our server we can get back to learning a bit more about Express.
We are currently hard-coding the port our server listens on to 3000
. This works fine, but we could make it more flexible. When you deploy this code to a hosting provider like Heroku they will run your code on their computer. This computer may have lots of different programs running that all need to use ports. Ideally the host can tell your server which port to listen to.
This is usually achieved with environment variables. These are like global variables that are set before your program runs. For example we can set a variable named TEST
in our terminal like this:
Remember if you are using Windows (without WSL) you will have to use the SET command and ampersands between commands like this:
Node makes environment variables available via the global process.env
object. So we could read that TEST
variable using this JS:
We can tweak our server code in index.js
to use a PORT
environment variable if it's set, otherwise fallback to 3000
:
Now we can control what port our server listens on without editing the code. E.g. to start it using a different port:
When you deploy to a hosting provider like Heroku they will use this to start your server with a random available port.
HTTP responses need a few different things:
A status code (e.g. 200
for success or 404
for not found)
Headers to provide info about the response
A body (the response data itself)
Our home route currently only provides the body, as Express will set the status code to 200
by default when you call response.send()
. To set a different code use the response.status
method. Let's add a new route that returns a different status code:
You can chain .status
together with .send
to make it shorter:
Try creating a test for this new route in tests/server.test.js
: it should check that the status is 500
and the body is "something went wrong". It will look very similar to the first test we wrote.
Express will automatically set some HTTP headers describing the response. For example since we called send
with a string it will set the content-type
to text/html
and the content-length
to the size of the string.
You can set your own headers using the response.set
method. This can take two strings to set a single header:
Or it can take an object of string values to set multiple headers:
We aren't limited to plaintext in our body. The browser will parse any HTML tags and render them on the page. Change your home route to return some HTML instead:
Run your tests again and you should see the first one fail, since it is still expecting the response body to be "hello". Amend the test so that it matches the new body.
Visit http://localhost:3000/ again and you should see an h1
rendered.
Since we're rendering HTML using JS strings we can insert dynamic values using template literals. Let's add the current year to the response:
You'll need to amend your test again to keep it passing.
Now let's look at the HTTP request. This object include lots of information that's useful to us.
Incoming request URLs can contain extra information known as "search parameters" (or sometimes "query parameters"). These are key/value pairs listed after a ?
character in the URL.
They are usually added to the URL automatically as the result of a form submission. For example this form:
will send a GET request to /search?keyword=what-the-user-typed
. This allows your server to receive user-submitted information.
Let's write a test first before we try to implement this. Our route is going to be simple: it will return a message telling the user what keyword they submitted. Our new test should check that a request to /search?keyword=bananas
returns a body including "You searched for bananas".
This test will currently fail when you run it, as we now need to implement the new route in server.js
.
Express provides the search parameters from the URL on the request.query
object. Each key=value
pair will be parsed and represented as an object property.
Your test should now pass when you run it. If you visit http://localhost:3000/search?keyword=css you should see "You searched for css" on the page.
Sometimes you can't know in advance all the routes you need. For example if you wanted a page for each user profile in your app: /users/oli
, /users/dan
etc. You can't statically list every possible route here. Instead you can use a placeholder value in the path to indicate that part of it is variable:
We use a colon (:
) to indicate to Express that any value can match a part of the path. It will put any matched values on the request.params
object so you can use them. For example a request to /users/oli
would result in a request.params
object like: { name: "oli" }
.
If you visit http://localhost:3000/users/oli you should see "Hello oli". If you visit http://localhost:3000/users/knadkmnaf you should see "Hello knadkmnaf".
Try visiting http://localhost:3000/asdfg in your browser. You should see Cannot GET /asdfg
. This is Express' default response for when no handler matches a path.
You can customise this by putting a "catch-all" handler after all your other routes. If no other route matches then this will be used (since Express matches them in the order they are defined).
We can use the server.use
method to create a handler that will match any method/route:
Reload http://localhost:3000/asdfg and you should now see your custom response. Try writing a test that checks whether requests to different URLs correctly return this 404 response.
Express route handlers don't have to send a response. They actually receive a third argument: the next
function. Calling this function tells Express to move on to the next handler registered for the route, without sending a response to the browser.
Let's add another handler for the home route. It will just log the request, then move on to the next handler:
If you run this code and refresh the home page you should see GET /
logged in your terminal.
The route methods like .get
accept multiple handler functions, so you can actually pass several all in one go. I.e. this does the same thing as above:
Express calls handlers that don't send a response "middleware". Our example here isn't that useful, but we could change it to run before all requests. We can do this with server.use
:
Now we'll get a helpful log like GET /
in our terminal when we load any page. Without middleware we would have to copy this into every route we wrote.
We'll be making more use of middleware in later topics where we have shared logic that must run on many different routes.
It's common to have some static files that don't change for each request. E.g. CSS, images, maybe some basic HTML pages. For convenience Express includes a built-in middleware for serving a directory of files: express.static
.
Create a new directory named public
. This is where we'll keep all the files sent to the client. Create a public/style.css
file with some example CSS:
Finally configure the middleware to serve this directory:
The server will now handle requests to http://localhost:3000/style.css and respond with the file contents. Note that there is no public
in the final URL: Express serves the files from the root of the site.
We can link this CSS from our homepage to apply the styles:
Visit http://localhost:3000/ in your browser again and you should see the styles applied.
So far we've only created GET
handlers. Let's add a POST
handler to see how we'd deal with forms submitting user data to our server:
We can't make a POST
request as easily in our browser, since that would require a form. Instead let's write a test to check this worked.
A POST
request that doesn't send any data isn't very useful. Usually a form would be submitting some user input. Let's imagine we have a form that asks a user's name. We want to get the name they submit and send back a message like "thanks for submitting, oli".
Since bodies can be large they are sent in small chunks—this means there's no simple way to read the body. Instead we must use a "body parser" middleware. This will wait for all the chunks to be received, then make the final body available to our route handler via the request object.
For convenience Express includes parsers for different formats. Request bodies can come in different formats (JSON, form submission etc), so we must use the right middleware. We want express.urlencoded
, which is what HTML forms submit by default. It will add a request.body
property, which is an object containing the submitted data:
Now we can update our test to send a URL-encoded body as part of the POST request. We need to make sure we add the right content-type header to the request, as this is how Express knows how to decode the body. We'll also change the assertion to expect the right response:
This test simulates how a real form in the browser would send a POST request.
It's not good practice to send a response directly from a POST request like this. It can cause issues with duplicate requests—the POST will be re-submitted if the user refreshes the resulting page. This is why some retailers ask you to not refresh the page after buying something, to avoid being charged twice.
A safer pattern is to always redirect to the next page after a POST request. That will tell the browser to send a new GET request to the next page, avoiding any resubmission problems.
We can use the response.redirect
method to return a redirect response:
The next page needs to know the name that was submitted—we can pass that in the URL using the search parameters (in a "real" app this might be stored in a cookie or database, but the URL is fine for now). We then need a new route to show the "success" page:
This route reads the name from the URL search parameters, then returns the same HTML response as before.