Building a Static Web Server with Middleware
In the last lecture, we learned about the basics of Express: endpoints and controllers. Today, we'll learn about a new kind of controller that will expand our server's capabilities: middleware.
Table of Contents:
Terms
Middleware - a function in express that intercepts and processes incoming HTTP requests. It can perform server-side actions such as parsing the request, modifying the response, or executing additional logic before passing control to the next middleware in the chain."
path
module - a module for creating absolute paths to static assets__dirname
— an environment variable that returns the path to the parent directory of the current file.Static Assets - unchanging files delivered to the client exactly as they are stored on a server. These include HTML, CSS, JavaScript files, images, videos, fonts, and documents. For React projects, we need to "build" our project to generate static assets (convert
.jsx
files to.js
files).
Express Review
Remember how the Express app works?
A client sends a request to the server.
The server receives the request and routes it to the proper controller based on the specific endpoint.
The controller processes the request, interacts with any necessary data or services, and generates a response.
The server sends the response back to the client.
The client is now free to do what it likes with the response.
And here is how we can create a server with two endpoints: /api/hello
and /api/data
A controller is a callback function that parses a request and sends a response. It will be invoked asynchronously when the associated endpoint is sent a request.
Every controller is invoked with three values:
A
req
object which holds data related to the request, including query parameters.A
res
object which has methods to send a response.A
next
function, typically only used by "Middleware".
Now its time to learn about that next
method!
Middleware and next()
next()
When a server receives an HTTP request, it can do more than just send back a response. Often, the server will perform a number of server-side actions before a response is sent.
For example, suppose that you wanted the server to keep track of every request that is sent to it by printing out some information like:
the endpoint that was requested (
/api/hello
or/api/data
, etc...)the request method (
GET
orPOST
, etc...)the time the request was received
We can add this functionality to our serveHello
controller, before we send the response, we can just add a few lines of code:
However, now we also need to add this code to serveData
. If we had more controllers, this would quickly become very repetitive.
Instead, we can use a middleware. Middleware in Express is a controller that can be invoked for all incoming requests before the final controller sends a response.
In many ways, middleware is like a controller. It receives the req
, res
, and next
values. There are two key differences:
We use
app.use
to register the middleware which means the function is invoked for ALL endpointsWe use
next()
instead ofres.send()
(or other response methods).next()
invokes the next middleware / controller registered to handle the current request.
We first create the
logRoutes
function to print out information about the requestAt the end, we invoke
next()
, passing along the request to one of our controllers to send a response.We register
logRoutes
usingapp.use()
which causes it to be invoked for ALL endpoints.Order matters! Middleware should be defined before controllers to ensure that it is invoked before the response is sent to the client.
With this middleware, we do not need to add this logging logic to every controller. Instead, this middleware will automatically be invoked for every incoming request before the final controller sends a response.
Middleware can be custom-made like this logRoutes
. However, we can also utilize some of the out-of-the-box middleware controllers provided by Express.
Static Web Servers
When you visit a website, like https://google.com, you are immediately presented with a rendered website. What's happening there?
Now, imagine that the website is just the "static assets" of a React project deployed on GitHub pages! But instead of using GitHub pages, Google has its own servers to store those files and serve them to visiting users.
We call these static web servers because they store static assets (HTML, CSS, and JS files) and then provide a server application that serves those assets when requested.
Let's look at how we can serve the static assets of a React project from our server.
Serving React Static Assets
As we've learned, a React project's static assets are built into a folder called dist
. In the provided repo, build the dist
folder and note the file structure:
When a user visits the server's homepage /
, we want to send the index.html
file.
Back in the server, we could add the following endpoint and controller:
If the frontend that we wanted to serve only consisted of a single index.html
file, then this code above would suffice!
But the index.html
file needs access to /assets/index-ABC123.js
and /assets/index-ABC123.css
. So we need two more controllers:
Seems repetitive, no? Now, imagine that your application has hundreds of static assets!
express.static()
Middleware
express.static()
MiddlewareRather than defining endpoints for every single static asset that you wish to serve, we can use the express.static()
middleware generator included with Express.
express.static()
is not middleware itself. Instead, invoking this function will generate a middleware function that we can use. We just need to provide a filepath to the folder containing our static assets:
Explanation:
Now, we just make a filepath to the entire dist folder and pass the filepath to
express.static()
which returns a middleware function which we callserveStatic
app.use(serveStatic)
will checks all incoming requests to see if they match files in the provided folder. if they do, they will be sent to the clientOrder matters! Remember to add this before the rest of your controllers.
Like logRoutes
, this middleware intercepts incoming requests before they reach the controllers. Unlike logRoutes
, the middleware generated by express.static()
can send a response to the client if a matching file is found. If not, it will pass the request to the controllers.
Summary
Controllers: Callback functions that handle requests by parsing them and sending responses.
Middleware Functions: Functions similar to controllers but pass requests to the next middleware without sending a response. They can also be executed for all requests while controllers typically handle a single endpoint.
Static Assets: Unchanging files (e.g., HTML, CSS, JS) served by a web server. For React projects, we need to "build" the project to convert "dynamic"
.jsx
files to "static".js
filesServing Static Assets:
Construct an absolute file path to the static assets folder using
path.join()
and__dirname
.Use
express.static(filepath)
middleware to make static assets publicly available.Register the middleware with
app.use()
Last updated