Docker
In very few words, it allows you to package your app with its dependencies included so it can run on any computer, as long as Docker is installed in it.
This simplifies a lot for us. But, what do we have to do to package our app?
Containers
Docker introduces the concept of “container” to refer to the environment where your app runs.
It is an isolated environment that runs on your computer, sharing the host’s operating system kernel but providing its own filesystem, processes, and network interfaces.
This allows you to run a Linux instance on a MacOS laptop, for example, without needing a full virtual machine.
graph BT subgraph containers[" "] direction LR subgraph c1[Container] app1[Application] end subgraph c2[Container] app2[Application] end subgraph c3[Container] app3[Application] end end subgraph os[Host OS] k[Kernel] end c1 -.-> k c2 -.-> k c3 -.-> k style containers fill:none,stroke:none style c1 fill:#93c5fd,stroke:#3b82f6 style c2 fill:#93c5fd,stroke:#3b82f6 style c3 fill:#93c5fd,stroke:#3b82f6 style os fill:#bfdbfe,stroke:#3b82f6 style app1 fill:white,stroke:#3b82f6 style app2 fill:white,stroke:#3b82f6 style app3 fill:white,stroke:#3b82f6 style k fill:white,stroke:#3b82f6
Figure 1: Container isolation from host system
Images
To be able to spawn these isolated environments in our computer, we need to create a Docker “image”.
An image in Docker is what we call the result of packaging everything you need to run your app: the code, runtime, libraries, and dependencies.
To create a docker image, we must define what’s gonna be included in it. We do it by creating a Dockerfile.
graph TB subgraph image[Docker Image] direction TB code[Application Code] runtime[Runtime Environment] libs[Libraries] deps[Dependencies] end dockerfile[Dockerfile] --> image style image fill:#dbeafe,stroke:#3b82f6 style dockerfile fill:#fef3c7,stroke:#d97706 style code fill:white,stroke:#3b82f6 style runtime fill:white,stroke:#3b82f6 style libs fill:white,stroke:#3b82f6 style deps fill:white,stroke:#3b82f6
Figure 2: From Dockerfile to Docker Image
But we first need to install Docker. Since this depends on what OS you use for development, I’ll leave a link to the Docker docs.
https://docs.docker.com/get-started/get-docker/
Anatomy of a Dockerfile
Let’s assume we have a simple Node.js server with this structure:
node-server/
├── node_modules/
├── src/
│ └── index.js
├── Dockerfile
├── package.json
└── package-lock.json
The image would look like this:
# Base image selection (the OS where your app will run inside the container)
FROM debian:stable-slim
# Install system dependencies - components needed at the OS level
# to provide the Node.js runtime environment
RUN apt-get update && apt-get install -y \
nodejs \
npm \
--no-install-recommends \
&& rm -rf /var/lib/apt/lists/*
# This is the directory where we will put our app inside our debian OS.
WORKDIR /app
# In this case, we want to copy all of our app files
# The left side is host's path - where the Docker images is being built
# The right side, defaults the WORKDIR directory (/app in this case)
# The dot indicates ALL files from given path
COPY . .
# Install application dependencies needed for building the application
RUN npm ci
# Build the application - compiles source code and prepares it for production
# This image assumes the output dir is `dist/` and no external deps.
RUN npm run build
# This is the port the application will listen on (not mandatory)
EXPOSE 3000
# Define the command to execute when the container starts
# this command runs on WORKDIR directory (/app)
CMD ["node", "dist/index.js"]
.dockerignore
Before building our image, we need to make sure only desired files are included in it.
So besides our Dockerfile, we’ll need a .dockerignore
file. You can think of it as .gitignore
but for Docker.
In our example, this is what it would look like:
node_modules
In our .dockerignore
, we exclude node_modules
because we install dependencies during the build process with npm ci. You might also exclude files like .git
, logs, or temporary build files to keep the image clean and small.
Building our Docker image
To be able to create the image so we can distribute/deploy our app, we run the following command:
docker build -t "node-server" .
This will effectively build your image, and it will be saved in your computer.
You can check it by running:
$ docker image list
REPOSITORY TAG IMAGE ID CREATED SIZE
node-server latest 603a98bd2bd9 19 seconds ago 327MB
Seasoned Docker images
For the sake of understanding what Docker does, I intentially did not use an image that came with Node pre-installed.
However, it is an option to use an official image that comes with the preinstalled OS level dependencies you need for your app; which results in a simpler Dockerfile.
Some images are based on very minimal OS like Alpine Linux which could reduce the image size from 327MB to about 162MB (~50% less).
I will leave some examples below:
Node.js
# Official node image using Alpine Linux and pre-installed Node v22
FROM node:22-alpine
WORKDIR /app
COPY . .
RUN npm ci
RUN npm run build
EXPOSE 3000
CMD ["node", "dist/index.js"]
Bun
# Official node image using Alpine Linux and pre-installed Bun v1
FROM oven/bun:1-alpine
WORKDIR /usr/src/app
COPY . .
RUN bun install
RUN bun build
EXPOSE 3000
CMD ["bun", "run", "dist/index.js"]
Want to learn how to make your Docker builds faster? Check out Part 2 where we dive into layer caching and build optimization.
➡️ Docker for Web Devs: Part 2 - Optimizing Builds with Layer Caching