Express.js - Intro (in Javascript)
Throw together a quick Javascript-based web server.
- Overview
- Routing Requests
- Middleware
- Debugging Express
- Handling HTTP POST Requests
- Handling File Uploads
- Proxying Requests
- Environmental Variables
- Parameters
- Template Engines
- Authentication
- Database Integration
- Conclusions
Overview
Concept
Express.js is a web server framework that runs in the Node.js Javascript environment.
-
Recall that a web server responds to incoming HTTP(S) requests from a web browser, or other client app
-
The server sends an HTTP response to each request, including some content if the request is successful, or an HTTP error message if it fails.
Setting up an NPM project
Using express assumes you have initialized a Node project with the Node Package Manager (NPM).
- The
npm init
command will ask for some project details and auto-generate the NPM project’s settings file,package.json
.
cd back-end # or whichever directory you want to house the web server code
npm init # enter `server.js` as the entry point, when asked.
- Now you can install express and automatically add it into the list of dependencies outlined in the project’s
package.json
file.
npm install express --save
Set up a placeholder Express app
Place the following into a file named app.js
- this creates an express web server app that serves no purpose… for now.
// import and instantiate express
const express = require("express") // CommonJS import style!
const app = express() // instantiate an Express object
// we will put some server logic here later...
// export the express app we created to make it available to other modules
module.exports = app
- Do not fear, we will add some code to this file later to tell it how to handle incoming HTTP(S) requests.
Create code to launch the express app
Put the following into a file named server.js
, and enure the package.json
settings file mentions this as the project’s main
entry point.
#!/usr/bin/env node
const server = require("./app") // load up the web server
const port = 3000 // the port to listen to for incoming requests
// call express's listen function to start listening to the port
const listener = server.listen(port, function () {
console.log(`Server running on port: ${port}`)
})
// a function to stop listening to the port
const close = () => {
listener.close()
}
module.exports = {
close: close,
}
Launch it!
You are now ready to run the express app.
-
Use
npm start
, the default script to run an NPM project. -
npm start
is really just an alias fornode server
, assuming yourpackage.json
hasserver.js
listed as the main script. -
Every time you change the code, you must stop (
Control-C
) and restart the server. This can be tedious. -
A popular 3rd-party module called
nodemon
can do this for you. Install it and have it automatically saved into yourpackage.json
file’s list of development-specific dependencies.
npm install nodemon --save-dev
- Run
nodemon server
to let nodemon handle stopping and restarting the server with each code change…. the luxury of automation!
What now?
You now presumably have an Express server up-and-running on your local machine at port 3000. What to do with it?
-
Our goal is to be able to make HTTP(S) requests to this back-end from a front-end client, be that a web browser or a desktop or mobile native app of some kind.
-
To do that, we must instruct our app what to do when an HTTP(S) request is made at the port we’re listening to.
-
Doing such setup is called creating routes.
Follow along
The Express.js code examples in this slide deck are available on GitHub - the code is already set up and ready to run.
-
Fork and clone the companion git repository.
-
Install all dependencies by running
npm install
. -
Launch the app by running
nodemon server
.
Routing Requests
Concept
Routes instruct express what to do in response to incoming HTTP(S) requests.
-
Each route listens for requests to a particular URL, e.g.
/
,/animals
,/animals/narwhal
, etc. -
Each route can be set to listen for a specific type of HTTP request, most commonly GET or POST.
-
Each route then responds with an error message or some content.
-
Content in the response sent to the client is most commonly formatted as HTML or JSON.
-
For requests for images, videos, or other static files - express can simply send those static files in the response.
Example 1
A route that listens for any HTTP GET request for the /
path, and responds with the plain text, ‘Hello!’.
app.get("/", (req, res) => {
res.send("Hello!")
})
- This could be placed in the middle of the app.js file.
Example 2
A route that listens for any HTTP GET request for the /html
path, and responds with a static HTML file.
// respond to any GET requests for /html-example with an HTML document named some-page.html
app.get("/html-example", (req, res) => {
res.sendFile("/public/some-page.html", { root: __dirname })
})
-
A request for
/html-example
will simply respond with the contents of the HTML file located atpublic/some-page.html
. -
This is an example of serving up a static file - the content of the file does not change from one request to the next.
Example 3
Inform express that the public
directory contains all static files that it should just blindly serve up when requested under the /static
route.
// tell express that any requests for '/static' will map out to the 'public' directory, where we place static content
app.use("/static", express.static("public"))
-
A request for
/static/html-example.html
would serve up the file located atpublic/html-example.html
. -
A request for
/static/css/main.css
would serve up the file located atpublic/css/main.css
. -
A request for
/static/images/donkey.jpg
would serve up the file located atpublic/images/donkey.jpg
. -
This is useful for serving any static HTML, CSS, image, audio, video or other content.
Example 4
JSON is a common format for sending data between client and server in apps.
// route for HTTP GET requests to /json-example
app.get("/json-example", (req, res) => {
// assemble an object containing the data we want to send
const body = {
title: "Hello!",
heading: "Hello!",
message: "Welcome to this JSON document, served up by Express",
imagePath: "/static/images/donkey.jpg",
}
// send the response as JSON text to the client
res.json(body)
})
-
A request for
/json-example
would receive this object in JSON format in response. -
Client-side code would presumably use this data in some way.
Middleware
Concept
As you have seen, when a request is made, a route function executes that determines how to respond.
- These route functions are automatically passed two objects by express,
req
andres
, representing the HTTP request and the HTTP response.
app.get("/", (req, res) => {
//...
})
-
Sometimes, it is useful to modify the content of one or both of those objects prior to calling the route functions.
-
Other times, it’s useful to perform ‘side-effects’, such as maintain log files of all requests.
-
Middleware are functions executed prior to the routes functions that can handle these two sorts of tasks with any request.
Setting up a middleware function
We call express’s use
function and pass it our middleware function as an argument.
app.use((req, res, next) => {
//...
})
-
Like routes, middleware functions are passed
req
andres
objects, representing the HTTP request and response. -
Middleware functions receive third argument,
next
, which is a callback function to trigger the next middleware function to be run. -
Middleware functions can thus be chained together, with each one calling the next. Once they have all been called, any relevant route function is executed.
Example 1
Imagine two middleware functions …
First:
// custom middleware - first
app.use((req, res, next) => {
// make a modification to either the req or res objects
res.addedStuff = "First middleware function run!"
// run the next middleware function, if any
next()
})
Second:
// custom middleware - second
app.use((req, res, next) => {
// make a modification to either the req or res objects
res.addedStuff += " Second middleware function run!"
// run the next middleware function, if any
next()
})
Example 1 (continued)
… followed by a route:
// route for HTTP GET requests to /middleware-example
app.get("/middleware-example", (req, res) => {
// grab data passed along by the middleware, if available
const message = res.addedStuff
? res.addedStuff
: "Sorry, the middleware did not work!"
// use the data added by the middleware in some way
res.send(message)
})
-
When express detects an incoming request, it will automatically call the first middleware function.
-
The first middleware function adds a custom property named
addedStuff
to theres
object and then passes control to the next middleware function, which appends more text to this property. -
The route is then automatically called and detects the custom property and sends it blindly to the client, for simplicity of the example.
Debugging
Concept
As you work on your express-enabled web servers, you will inevitably want to be aware of a few useful debugging tools.
-
morgan - 3rd-party middleware for logging all incoming HTTP requests
-
Postman - a desktop app for making HTTP requests to a web server without requiring any front-end code.
-
mocha and chai - popular 3rd-party modules for unit testing code, i.e. making sure all functions behave as they are expected to in all circumstances.
-
ngrok - a tool to provide public HTTPS access to a server running on your local machine
Morgan
Morgan will log a bit of info about each incoming request to the server’s command shell output.
- Install the morgan module and save it to your project’s list of dependencies in the
package.json
settings file.
npm install morgan --save
- Import it into any code file where you define routes that handle incoming requests.
const morgan = require("morgan") // middleware for nice logging of incoming HTTP requests
- Set it as express middleware and specify which style of logs you desire - there are several output styles specified in their documentation.
app.use(morgan("dev")) // dev style gives a concise color-coded style of log output
Postman
Postman is a desktop app that allows you to trigger custom HTTP requests to your local development web server.
-
This relieves you from having to write (and debug) front-end code while your focus is on perfecting the back-end.
-
Once installed, get started by clicking the
Create a request
link on the Launchpad screen. Enter a route you have set up in express, such ashttp://localhost:3000/
, and click theSend
button. -
It is possible to tweak request settings to simulate any kind of request a front-end could make.
-
It is possible to save requests you want to run frequently so you can quickly test them as you work on the code.
-
The git repository accompanying these slides includes a file named
express-js-starter-app.postman_collection.json
that can be imported into Postman to easiy try out all these routes.
Mocha and chai
Mocha and chai are two 3rd-party Node.js modules often used together to perform unit testing of code.
-
a unit test is a test that can be written to validate a function and make sure it always works correctly regardless of context.
-
We are not concerned with unit testing for now, but will return to it in detail later. It never hurts to try it out now.
ngrok
ngrok is a tool that will provide a publically-accessible web address that forwards requests to a server you run on your local machine.
-
Install ngrok and make sure it is in your system’s PATH variable.
-
Start your server, for example by running
nodemon server
to launch an express.js server locally. -
Expose the port at which your server is running locally to the public, with
ngrok http 3000
, where3000
is the port your express server is running on. -
ngrok will output a public URL (both
http
andhttps
) that you and others can use to reach the web server running locally on your own machine. -
requests made to that ngrok URL will be forwarded to your local server via a secure tunneling connection.
Handling HTTP POST Requests
Concept
Many web sites and apps allow users to submit data to a server via forms. Usually these requests are submitted via HTTP POST, with data in the body (i.e. payload) of the request.
-
Express now includes middleware (formerly a separate project called
body-parser
) that is useful in parsing data in incoming HTTP POST request body. -
Tell express to
use
this request body parser middleware.
app.use(express.json()) // decode JSON-formatted incoming POST data
app.use(express.urlencoded({ extended: true })) // decode url-encoded incoming POST data
Example
Imagine an HTML form that sends data to the server via HTTP POST…
<form action="/post-example" method="POST">
<input type="text" name="your_name" placeholder="Your name" /> <br />
<input type="text" name="your_email" placeholder="Your email" /> <br />
<input type="checkbox" name="agree" /><label
>I agree to your onerous conditions</label
>
<br />
<input type="submit" value="Submit!!!" />
</form>
… and an Express route receives that data after the middleware has kindly added it to the req
object’s body
property.
app.post("/post-example", (req, res) => {
const name = req.body.your_name
const email = req.body.your_email
const agree = req.body.agree
// now do something amazing with this data...
// ... then send a response of some kind
})
Handling File Uploads
Concept
Clients can upload files to the server as part of their HTTP POST requests.
-
There exist 3rd-party middlewares, such as multer, that make extracting the files from the request easy.
-
Install it into your express app and save as a dependency to the
package.json
config file:
npm install multer --save
- Import it into the file that handles the upload routes:
const multer = require('multer')
Setup
multer
can save uploaded files to a number of different destinations. Here, we tell multer
to save uploaded files into a directory named public/uploads
, with a filename based on the current time.
// enable file uploads saved to disk in a directory named 'public/uploads'
const storage = multer.diskStorage({
destination: function (req, file, cb) {
cb(null, "public/uploads")
},
filename: function (req, file, cb) {
cb(
null,
`${file.fieldname}-${Date.now()}${path.extname(file.originalname)}`
)
},
})
Then instantiate a multer
object.
const upload = multer({ storage: storage })
Client-side code
Take the following HTML form that allows users to upload files to the server as part of the POST request.
<form action="/upload-example" method="POST" enctype="multipart/form-data">
<input name="my_files" type="file" multiple />
<input type="submit" value="Submit!!!" />
</form>
-
Note that the
form
tag has an addition attribute,enctype
that must be included for forms with file uploads. -
Note also the optional
multiple
attribute that allows the user to upload multiple files, if desired.
Server-side code
multer
middleware will automatically save any uploaded files in the request into the specified directory, rename them as instructed, and make a field named req.files
containing the paths to the files on the server.
// route for HTTP POST requests for /upload-example
app.post("/upload-example", upload.array("my_files", 3), (req, res, next) => {
// check whether anything was uploaded
if (req.files) {
// success! send data back to the client, e.g. some JSON data
const data = {
status: "all good",
message: "yup, the files were uploaded!!!",
files: req.files,
}
res.json(data) // send respose
}
})
- Note the code,
upload.array('my_files', 3)
, instructsmulter
to store no more than 3 files, coming from an HTML element namedmy_files
.
Proxying Requests
Concept
We often use our web server as a sort of proxy, relaying requests from the client to some other service, such as an external API or database, and relaying the responses from these other services back to the client.
-
For example, for an animal-relate app, the client might request data about one or more animals.
-
A route function on the web server will handle this request and make a request to a database or external API for the animal data.
-
The API or database will send the list of animals back to the express route function.
-
The express route function will pass the list of animals to the client as the web server response.
Example
Let’s see how to have an express route proxy a request from the client to an API and back, with no modification of the data.
- The 3rd-party middleware, axios, is useful for issuing requests to APIs. Install it and save it to the
package.json
list of dependencies:
npm install axios --save
- Create a route that issues a request to an API and forwards the results to the client
// proxy requests to/from an API
app.get("/proxy-example", (req, res, next) => {
// use axios to make a request to an API for animal data
axios
.get("https://my.api.mockaroo.com/users.json?key=d9ddfc40")
.then(apiResponse => res.json(apiResponse.data)) // pass data along directly to client
.catch(err => next(err)) // pass any errors to express
})
Environmental Variables
Concept
Hard-coding values like credentials for APIs and databases directly into production code is considered bad practice.
-
If the code is checked into a version control system, you may inadvertently disclose private credentials to the public.
-
If those credential settings are hard-coded in multiple places, code maintenance becomes an issue.
Setup
A 3rd-party middleware named, dotenv, makes it easy to store such values as environmental variables that are then imported into the app.
- Install it and save it to the
package.json
list of dependencies:
npm install dotenv --save
- Import it into any file where you have a need for the credentials or other private data.
require("dotenv").config({ silent: true })
- Now, environmental variables declared in a hidden file named
.env
will be available in your code asprocess.env.MY_VARIABLE_NAME
.
Note that as of version 20 (late 2023), Node natively supports the same capability.
Usage
Let’s place the credentials for the API we used earlier into a file named .env
:
API_BASE_URL=https://my.api.mockaroo.com/animals.json
API_SECRET_KEY=d9ddfc40
- And use those variables in a route to fetch from the API:
// same route as above, but using environmental variables for secret credentials
app.get("/dotenv-example", (req, res, next) => {
// insert the environmental variable into the URL we're requesting
axios
.get(`${process.env.API_BASE_URL}?key=${process.env.API_SECRET_KEY}&num=10`)
.then(apiResponse => res.json(apiResponse.data)) // pass data along directly to client
.catch(err => next(err)) // pass any errors to express
})
Parameters
Concept
Express routes can have parameters.
-
This means that routes can have variables passed to them by the client.
-
E.g., a route named
/parameter-example:animalId
contains a parameter namedanimalId
. -
If a client makes a request for
/parameter-example/22
, that22
value (or any other value in this spot in the URL) will be put into a variable namedanimalId
within the route function.
Example
The following example route accepts an ID into a parameter, makes a request to an API for a record with that ID, and returns the results to the client.
// this route is very similar to the dotenv-example route, but using async/await syntax for a change
app.get("/parameter-example/:animalId", async (req, res) => {
// use axios to make a request to an API to fetch a single animal's data
// we use a Mock API here, but imagine we passed the animalId to a real API and received back data about that animal
const apiResponse = await axios
.get(
`${process.env.API_BASE_URL}?key=${process.env.API_SECRET_KEY}&num=1&id=${req.params.animalId}`
)
.catch(err => next(err)) // pass any errors to express
// express places parameters into the req.params object
const responseData = {
status: "wonderful",
message: `Imagine we got the data from the API for animal #${req.params.animalId}`,
animalId: req.params.animalId,
animal: apiResponse.data,
}
// send the data in the response
res.json(responseData)
})
Template Engines
Concept
Sometimes, it is desireable to respond to incoming requests with content that is formatted in a way that is ready to be displayed by a web browser.
-
This usually meanse sending back HTML code to the web browser and the web browser then blindly renders this HTML content on the page.
-
In many cases, you will find that you will want to send the same basic HTML code in response to many different requests, with just a few bits of the content changed each time.
-
This is where a template engine can help - responding to requests with standardized templates of content where dynamic values are inserted into it as needed.
-
There are several popular template engines for use with Express - e.g. Pug, Mustache, and EJS.
Creating a Pug Template
For example let’s install and use the Pug template engine.
npm install pug --save
- It is now possible to set Express to use Pug as its template engine:
app.set("view engine", "pug")
- To create a template for your site’s home page, create a file named
index.pug
with two bits of dynamic content,title
andcontent
:
html
head
title= title
body
h1= content
Using a Pug Template
To use a Pug template, simply create a route that instructs Express to respond with the contents of the template by using the render
function.
- For example, to create a route for a site’s home page that points to the
index.pug
template we previously made and supplies the values to plug intotitle
andcontent
fields:
app.get("/", function (req, res) {
res.render("index", {
title: "My First Templated Site",
message: "Welcome to your first dynamic templated page!!",
})
})
-
Simply repeat this process for all the different sorts of routes and templates that your site might need.
-
The full set of template options are available in the official Pug language reference.
Single Page Applications
Template engines are less useful for web apps that are set up as single page applications.
-
In a single page application, an initial set of static HTML, CSS, and Javascript code is requested by and loaded into the web browser.
-
The static browser-based Javascript code then issues new requests to the server as necessary. The server responds to any subsequent requests with JSON data, not HTML documents, e.g.:
{
'title': 'My First Templated Site',
'message': 'Welcome to your first dynamic templated page!!'
}
- The static Javascript code initially loaded by the browser when the page was first loaded then updates relevant parts of the page in response to this JSON data.
Front-Ends Built With React.js
React.js-based front-ends are built as single page applications.
-
An initial set of HTML, CSS, and Javascript code is loaded into the browser.
-
The browser-based Javascript then makes requests to the server, as necessary.
-
The server responds with JSON data.
-
Browser-based Javascript code automatically then updates the components on the page based on this JSON data, as necessary.
-
Thus, there is no big need for a template engine to help generate repetitious HTML documents.
Authentication
Concept
Many web apps require users to sign up and log in.
-
Handling user account creation and authentication can be simple… or complicated.
-
Fortunately, there’s middleware to steamline this!
-
passport is a popular and very flexible authentication middleware solution.
-
We recommend you wait until you have learned to integrate a database into your project before using it.
Database Integration
Concept
Most classic technology stacks that rely on Express.js for the back-end use MongoDB as the database.
-
Most software relies on relational databases, such as Oracle, MS SQL, SQLite, MySQL, etc, which store data in a way similar to spreadsheets, with records stored in rows containing columns for each piece of data.
-
MongoDB is a document-oriented database that lends itself well to many common data needs of Javascript-based web and mobile apps, such as native support for geolocation, data stored internally as JSON objects, and easy scalability should your app take off to the moon.
-
The 3rd-party module named mongoose makes it easy to integrate express.js-based apps with MongoDB. Similar modules exist for most other database integrations as well.
-
We recommend you wait until you have mastered back-end development without a database before adding one into the mix.
Conclusions
You now have an baseline understanding of what express.js is, how it works, some useful middlewares, and various route patterns.
-
A simple Express.js project is available for you to fork, clone, and try out for yourself. Remember to run
npm install
to install all dependencies before running it. -
Thank you. Bye.