Docker for Web Devs: Part 1 - Getting Started

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