Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update deploy-nodejs-with-docker tutorial #936

Merged
merged 2 commits into from
Sep 5, 2024
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
170 changes: 96 additions & 74 deletions tutorials/deploy-nodejs-with-docker/01.en.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
SPDX-License-Identifier: MIT
path: "/tutorials/deploy-nodejs-with-docker"
slug: "deploy-nodejs-with-docker"
date: "2019-09-26"
date: "2024-09-05"
title: "How to deploy a Node.js application with Docker"
short_description: "Learn how to deploy a Node.js app to an Ubuntu server with Docker, Docker Hub and Docker Compose"
tags: ["Docker", "Deploy", "Cloud", "Lang:JS"]
Expand All @@ -24,53 +24,103 @@ This tutorial shows how to deploy a Node.js application to a cloud server throug

**Prerequisites**

- This tutorial assumes that you have Docker installed on your local system. If you don't have it, you can find the instructions for installing it in the [official documentation](https://docs.docker.com/install/)
- You should also have a cloud server with a Linux distribution, preferably Ubuntu 18.04. If you are using another distribution, you might have to look for specific instructions when it's time to install Docker on your server
- Some steps also require that you have Docker Hub (free) account to upload the Docker image for the application
- If you don't have any previous Docker experience, that's fine, this tutorial is pretty basic and explains the main concepts around what we're doing
- This tutorial assumes that you have Docker installed on your local system. If you don't have it, you can find the instructions for installing it in the [official documentation](https://docs.docker.com/install/).
- You should also have a cloud server with a Linux distribution, preferably Ubuntu 24.04. If you are using another distribution, you might have to look for specific instructions when it's time to install Docker on your server.
- Some steps also require that you have Docker Hub (free) account to upload the Docker image for the application.
- If you don't have any previous Docker experience, that's fine, this tutorial is pretty basic and explains the main concepts around what we're doing.
- You need a Node.js application that you can deploy.

<blockquote>
<details>
<summary>Click here to view an example</summary>

```
└── <project_name>
├── package.json
├── package-lock.json
└── src
└── index.js
```

* <kbd>package.json</kbd>
```json
{
"version": "1.1.0",
"name": "myproject",
"description": "Example package.json file.",
"main": "src/index.js",
"scripts": {
"build": "echo \"Building the application\""
}
}
```

* <kbd>src/index.js</kbd>
```js
const http = require('http');

const server = http.createServer((req, res) => {
res.statusCode = 200;
res.setHeader('Content-Type', 'text/plain');
res.end('Hello World!\n');
});

server.listen(8080, () => {
console.log('Server running at http://localhost:80/');
});
```

</details>
</blockquote>

**About Docker**

In case you're just getting started with Docker, here are some terms that are worth reviewing, to make sure that we're on the same track.

- **Images**: in Docker, images are "snapshots" or templates of a file system, and contain everything that is needed to launch an application
- **Containers**: these are the actual running instances of the application. They are created by taking a template (an image) and turning it into something that can be started and has a state
- **Layers** are the elements that compose a Docker image. Each layer is built on top of another one, allowing to provide a feature called layer caching. This means that you don't need to re-build or re-download all the layers of an image when only one of them changes
- **Images**: in Docker, images are "snapshots" or templates of a file system, and contain everything that is needed to launch an application.
- **Containers**: these are the actual running instances of the application. They are created by taking a template (an image) and turning it into something that can be started and has a state.
- **Layers** are the elements that compose a Docker image. Each layer is built on top of another one, allowing to provide a feature called layer caching. This means that you don't need to re-build or re-download all the layers of an image when only one of them changes.
- **Registries** are the place where you upload (*push*) images to make them available to the world, or to those that have the credentials to access it. In this tutorial, we're going to use Docker Hub, but there are also alternatives provided by GCP, AWS, Azure, GitHub, and others.

## Step 1 - Create a Dockerfile

Create a file called `Dockerfile` with the following content in the root of your Node.js project directory:

```do
FROM node:10.16
FROM node:20.17

ENV NODE_ENV=production

WORKDIR /app

COPY package*.json .

RUN npm ci --only=production
RUN npm ci

COPY . .

EXPOSE 8080

CMD [ "node", "index.js" ]
CMD [ "node", "src/index.js" ]
```

The `Dockerfile` is the place where you put the instructions that allow Docker to *build* an image. Every instruction represents the creation of a layer, which is a modification of the image file system that is being created.

In this case, we're composing our image by starting from a template, sometimes called the *base image*, that in this case is `node:10.16`. This is an official image provided by the Docker company, and you can find more about it [here](https://hub.docker.com/_/node/).
In this case, we're composing our image by starting from a template, sometimes called the *base image*, that in this case is `node:20.17`. This is an official image provided by the Docker company, and you can find more about it [here](https://hub.docker.com/_/node).

The next step sets the `NODE_ENV` environment variable to `production`. The main effect here is to avoid installing development packages when running the npm installation below, but it can often lead to better optimizations in modules you might be relying on.

The next step moves the current working directory to `/app`, which is where the following instructions will be executed.
With the `WORKDIR` command we move the current working directory to `/app`, which therefore becomes the directory where the following instructions will be executed.

The line `COPY package*.json .` has the effect of copying the files `package.json` and `package-lock.json` into the `/app` directory of the Docker image file system. Note that the dot at the end is required to indicate the current directory.
The line `COPY package*.json .` copies the files `package.json` and `package-lock.json` into the `/app` directory of the Docker image file system. Note that the dot at the end is required to indicate the current directory.

We now use the `RUN` instruction to install production dependencies, by using the `npm ci` command introduced with [npm 5.7.0](https://blog.npmjs.org/post/171556855892/introducing-npm-ci-for-faster-more-reliable).
We now use the `RUN` instruction to install production dependencies, by using the `npm ci` command (`ci` stands for clean install and is designed to be used in automated enviroments).

One thing that should be noted at this point is that until now we copied in the build only the `package*.json` files, instead of the whole project directory. This allows to leverage Docker layers caching, so that if the dependent packages are unchanged, the layers can be reused without rebuilding them.

The following line (`COPY . .`) copies the remaining files in the image. Optionally, we can tell Docker that we want to expose a specific network port of the container, so that a web application can be accessed through it.
The following line (`COPY . .`) copies the remaining files in the image.

Optionally, we can specify that we want to expose a specific network port of the container, so that a web application can be accessed through it. Note that the `EXPOSE` instruction doesn't actually expose the port: as the documentation [says](https://docs.docker.com/reference/dockerfile/#expose), «it functions as a type of documentation between the person who builds the image and the person who runs the container, about which ports are intended to be published».

Finally, the last instruction determines the command that should be used by Docker to run the application when the container starts. In this case, we're assuming that the entrypoint of the application is the `index.js` file.

Expand All @@ -94,9 +144,11 @@ The basic command to do that looks like the following one, and must be executed
docker build -t myproject .
```

> If you get the error "process '/bin/sh -c npm ci' did not complete successfully", replace `npm ci` in the Dockerfile with `npm install` and try again.

The `-t` option specifies the name of the image, in this case `myproject`. The `.` at the end of the line is required to tell Docker to look for a `Dockerfile` in the current directory.

**NOTE**: the first time that you run the build it will take a while, because Docker has to download all the layers of the base image (Node.js 10.16 in this case).
**NOTE**: the first time that you run the build, it will take a while because Docker has to download all the layers of the base image (Node.js 20.17 in this case).

Since we're going to upload this image to the Docker Hub online registry (to make it accessible from our server), we need to name the image by using a specific convention.

Expand All @@ -111,10 +163,10 @@ Where `username` is your Docker Hub username, and `latest` is the *tag* of the i
```sh
docker build -t myproject .
docker tag myproject username/myproject:latest
docker tag myproject username/myproject:20190926
docker tag myproject username/myproject:20240905
```

These commands build an image and then tag it with the tags `latest` and `20190926` (the date when this tutorial was written).
These commands build an image and then tag it with the tags `latest` and `20240904` (the date this tutorial was last updated).

Docker Hub doesn't remove old images by default, so this allows to have an history of all the images that you pushed to the registry. The image with tag `latest` will always be the one that was most recently built, while the older ones will be tagged with a date.

Expand All @@ -136,45 +188,38 @@ If your application is small, this command should be fast to complete, because i

When you have a new version of the image, you should run the push command again to make sure it's uploaded on Docker Hub.

## Step 4 - Install Docker on Ubuntu 18.04
## Step 4 - Install Docker on Ubuntu 24.04

We can now move to the server to install Docker and Docker Compose. As mentioned in the prerequisites, the assumption here is that you have an Ubuntu 18.04 server that is already up and running.
We can now move to the server to install Docker and Docker Compose. As mentioned in the prerequisites, the assumption here is that you have an Ubuntu 24.04 server that is already up and running.

First of all, installing Docker requires some system dependencies, which can be installed with the following commands:

```sh
sudo apt-get update
sudo apt-get install \
apt-transport-https \
ca-certificates \
curl \
gnupg-agent \
software-properties-common
sudo apt-get install ca-certificates curl
```

Now add the official Docker GPG key and configure a custom apt repository:

```sh
curl -fsSL https://download.docker.com/linux/ubuntu/gpg | sudo apt-key add -
sudo add-apt-repository \
"deb [arch=amd64] https://download.docker.com/linux/ubuntu \
$(lsb_release -cs) \
stable"
sudo install -m 0755 -d /etc/apt/keyrings
sudo curl -fsSL https://download.docker.com/linux/ubuntu/gpg -o /etc/apt/keyrings/docker.asc
sudo chmod a+r /etc/apt/keyrings/docker.asc

echo \
"deb [arch=$(dpkg --print-architecture) signed-by=/etc/apt/keyrings/docker.asc] https://download.docker.com/linux/ubuntu \
$(. /etc/os-release && echo "$VERSION_CODENAME") stable" | \
sudo tee /etc/apt/sources.list.d/docker.list > /dev/null
```

Finally, update the apt index again and install Docker Community Edition:

```sh
sudo apt-get update
sudo apt-get install docker-ce
sudo apt-get install docker-ce docker-ce-cli containerd.io docker-buildx-plugin docker-compose-plugin
```

We're also going to install Docker Compose 1.24.1, by simply downloading a binary file. Docker Compose is a tool that greatly simplifies the management of containers and their lifecycle.

```sh
sudo curl -L "https://github.com/docker/compose/releases/download/1.24.1/docker-compose-$(uname -s)-$(uname -m)" -o /usr/local/bin/docker-compose
sudo chmod +x /usr/local/bin/docker-compose
```
The above command also installs Docker Compose, a tool that greatly simplifies the management of containers and their lifecycle.

One last useful step consists of adding the current Ubuntu user to the `docker` group, so that we can run Docker commands directly from it.

Expand All @@ -189,7 +234,7 @@ Verify that everything went fine by executing the following commands:
```sh
docker --version
docker ps
docker-compose --version
docker compose version
```

If you're not seeing any errors or warnings, you're good to go.
Expand All @@ -199,8 +244,6 @@ If you're not seeing any errors or warnings, you're good to go.
Create a file called `docker-compose.yml` with the following content on your server:

```yaml
version: '3'

services:
myproject:
container_name: 'myproject'
Expand All @@ -211,12 +254,10 @@ services:
This is a very basic Docker Compose file that configures a single container called `myproject`, based on the `username/myproject` image from Docker Hub. If you don't specify a tag, it will default to `latest`, but you can also set a specific one, if you want:

```yaml
version: '3'

services:
myproject:
container_name: 'myproject'
image: 'username/myproject:20190901'
image: 'username/myproject:20240904'
restart: unless-stopped
```

Expand All @@ -225,69 +266,50 @@ Finally, the `restart` property indicates that the container should be automatic
If you now run this `up` Compose command, the Docker image will be pulled from the registry and your application will hopefully run:

```sh
docker-compose -f docker-compose.yml up
docker compose -f docker-compose.yml up
```

This command creates a container and executes it. The output of the container is captured by Docker and presented to you in the console. Press CTRL + C (or CMD + C) and wait some seconds for the container to stop.

If everything went fine, you're now ready to launch the container as a daemon, so that it will keep running in the background, until stopped. This can be achieved by adding the `-d` option to the command:

```sh
docker-compose -f docker-compose.yml up -d
docker compose -f docker-compose.yml up -d
```

Boom, node! (oops, I meant done)

Make sure you take a quick look at the Compose file [reference documentation](https://docs.docker.com/compose/compose-file/), where you can find useful features like mapping network ports between the server and the container. Here's a quick example that maps the external port 80 to the internal port 8080:
Make sure you take a quick look at the Compose file [reference documentation](https://docs.docker.com/reference/compose-file/), where you can find useful features like mapping network ports between the server and the container. Here's a quick example that maps the external port 80 to the internal port 8080:

```yaml
version: '3'

services:
myproject:
container_name: 'myproject'
image: 'username/myproject'
restart: unless-stopped
ports: '80:8080'
ports:
- '80:8080'
```

## Step 6 - Enable automated builds (optional)

Fantastic, you have now deployed the first release of your *containerized* Node.js application.

If your application code is hosted in a git repository on GitHub or BitBucket, I have great news for you: the Docker build process can be automated in the cloud for free by using Docker Hub.

If you go in the "Builds" tab of your image on Docker Hub, you'll be prompted with a similar page:

![Docker Hub automated build pages](images/docker-hub-automated-builds-1.png)

After linking your GitHub or BitBucket account, select the repository where your code and `Dockerfile` lie. The default configuration that Docker Hub suggests is fine for a first build, so click the "Save and Build" button and wait.

![Docker Hub automated builds configuration page](images/docker-hub-automated-builds-2.png)

A Docker build will be started in the cloud in a few minutes, and you'll then find a new image with the tag `latest` automatically pushed in the *Tags* section. Every new commit (push) on your git repository will now trigger a new build.

![Docker Hub automated builds status](images/docker-hub-automated-builds-3.png)

## Step 7 - Deploy a new version
## Step 6 - Deploy a new version

Let's say that you need to publish a change to your application. Unless you enabled automated builds, you need to repeat steps 2 and 3, so that a new image will appear on Docker Hub.
Let's say that you need to publish a change to your application. Unless you enabled [automated builds](https://docs.docker.com/docker-hub/builds/), you need to repeat steps 2 and 3, so that a new image will appear on Docker Hub.

Then, on your server, you must manually pull the new image, like this:

```sh
docker-compose -f docker-compose.yml pull
docker compose -f docker-compose.yml pull
```

And restart the container with the new image:

```sh
docker-compose -f docker-compose.yml up -d --force-recreate
docker compose -f docker-compose.yml up -d --force-recreate
```

## Conclusion

Great, you did it! This was a basic introduction to deploying a Node.js application to Ubuntu 18.04 by using Docker, Docker Hub and Docker Compose.
Great, you did it! This was a basic introduction to deploying a Node.js application to Ubuntu 24.04 by using Docker, Docker Hub and Docker Compose.

We've seen how to write a simple `Dockerfile`, how to build the image, push it, and deploy it on a server. Setting up automated builds is useful and a first step into the Continuous Integration and Delivery world (CI/CD).

Expand Down