If at any point something is unclear, please refer to Docker's Dockerfile reference for a more thorough explanation. Use the Contents on the right side to navigate to specific sections or Dockerfile instructions.
Developers always disagree about how to best write Dockerfiles. To avoid arguments, try to always follow Docker's Best practices for writing Dockerfiles . Bonus points if you send the link to teammates that don't.
A dockerfile is a file that describes how the
Docker engine will build an image of your
application. It simply lists some instructions
that either is executed by Docker engine while
building the image or that will have effect when
running a container from the image. The file is
often located in the root of the project you are
containerizing, but it can in theory be located
anywhere as long as the references to the file is
taken in account both when writing the content of
the file and when running
the docker build <path_to_dockerfile>
command.
FROM
is used to specify an existing container
image on which you build upon -
the starting point for your image. It is typically
an operating system (e.g
debian
or alpine
), a programming language or
collection of build tools (e.g.
node
, python
, or some sdk), or some sort of
runtime or program (e.g.
dotnet
or nginx
). It may be wise to also
specify a specific tag (e.g.
python:3.10
) so that the base don't suddenly
change when some third party
pushes a newer image.
FROM python:3
You may rarely encounter FROM scratch
in the
wild, which indicates that the
container image is build from scratch - a totally
empty starting point.
WORKDIR
sets the working directory within the
file-system of the descending instructions. The
folder will be
created if it does not already exist. If a
relative path is provided (not
starting with /
), the resulting workdir will be
relative to one previously
specified.
WORKDIR /app
COPY <path_on_local_machine> <path_in_container>
copies files from your local filesystem
into the container image. Only
files within the build context (a folder
specified on build) may be copied. It
is quite similar to cp
on unix systems. Both
single files and directories may
be copied, and the target path will automatically
be created if absent. If the
destination is relative (not staring with /
), it
will be relative to the
WORKDIR
.
# Copy all files within the build context into the working directory of the
# container image
COPY . .
# Copy only the src folder within the build context into src within the root
# of the container filesystem
COPY src /src
# Copy the file index.html into /var/www/html/index.html, even though the
# folder /var/www/html/ does not already exist
COPY html/index.html /var/www/html/index.html
RUN
runs a command within the shell of the
container being built. It is often
used to update or install packages, and compile or
build the project. Commands
are often written on one line each, with \
at
the end of all lines but the
last.
RUN npm install && \
npm build
ENV
is used to define environment variables
within the container. They are
variables that may be used by any process within
the container, such as PATH
or JAVA_HOME
. Note that you may define
additional environment variables when
running a container based on the image.
ENV LOG_LEVEL=warning
Here, one may override the log level when building
the application, but by
default it is info
.
CMD
is used to specify the command or executable
that runs on container startup.
# It can either be run with exec-syntax, like this
CMD [ "python3", "-m" , "flask", "run", "--host=0.0.0.0"]
# or the equivalent written in shell-syntax:
CMD python3 -m flask run --host=0.0.0.0
Similar to CMD
. If you want, see Docker's
description
, as
it is a bit tricky to explain.
Remember the Next/React workshop that you may have participated in a while back? Wouldn't it be cool to containerize that?! 🤔 Let's do it!
First, you need to move or copy the folder, next-workshop, that you
cloned/used in the earlier workshop. Use the following command to do so, but
remember to change the template-path to the path where you
cloned the repository to. Make sure you
have container-workshop
as your working
directory or adapt the destination path below.
If you're in windows CMD, use copy
and move
instead of cp
and mv
.
# Copy - recommended:
cp <path/to/the/next-workshop/on/your/machine> 02-run-app-in-container
# Or move:
mv <path/to/the/next-workshop/on/your/machine> 02-run-app-in-container
If you have not participated in the workshop or you were not able to finish, there is a bare minimum version of the project in next-workshop-cheat that you can use.
If you are using your own version of the next-workshop and have moved/copied it to 02-run-app-in-container
we will have to dumb down the ChatGPT backend a bit to reduce complexity. Navigate to next-workshop/app/api/chat/route.ts
and replace the contents with the following:
import { StreamingTextResponse } from 'ai';
import { NextRequest } from 'next/server';
export const runtime = 'edge';
const furiousChatGPTResponse = `
No no no no no!
These ingredients are NOT going to cut it!
What am i cooking for, a low-budget funeral??
Here is a REAL recipe, I present to you the Moonlit Dreamcatcher Delight!
ingredients:
1 cup of cloud fluff (alternatively, use the lightest whipped cream you can find)
3 slices of moonbeam (a thin slice of pearlescent edible glitter gelatin will do)
A pinch of starlight sparkle (edible glitter or sugar crystals work well)
5 drops of essence of night breeze (a mix of lavender and mint extract)
A handful of dreamberries (blueberries dipped in a shimmering edible silver dust)
2 spoons of solar syrup (a golden honey and saffron blend)
1 piece of the rainbow’s end (a slice of multicolored layer cake)
A whisper of unicorn laughter (a pinch of pop rocks for a surprising fizz)
Equipment:
A mixing bowl as light as a feather
A spoon made from the wishes of a shooting star
Serving plates that have caught the first light of dawn
Preparation:
Begin by laying your cloud fluff at the base of your serving plate, spreading it evenly to form a cushion of dreams.
Carefully place the moonbeam slices atop the cloud fluff, allowing them to catch the light and shimmer.
Sprinkle the starlight sparkle gently over everything, ensuring a delicate twinkle is visible with every glance.
In a feather-light mixing bowl, combine the essence of night breeze with the dreamberries. Stir gently with your shooting star spoon until the berries are coated in a mist of flavor.
Drizzle the solar syrup in a spiral, starting from the center and moving outward, to mimic the galaxy’s spin.
Artfully arrange the slice of the rainbow’s end on the side, providing a burst of color and a gateway to flavors unknown.
For the final touch, whisper your happiest thought over the dish and sprinkle it with the unicorn laughter. The pop rocks add not just a surprise but also the magic of joy and laughter to every bite.
Serving Instructions:
Serve this dish under a canopy of twinkling lights or by the gentle luminescence of a full moon. It's best enjoyed with an open heart and a vivid imagination, allowing each bite to transport you to a world of dreams and wonders.
`;
export async function POST(req: NextRequest) {
await req.json();
const stream = new ReadableStream({
start(controller) {
controller.enqueue(furiousChatGPTResponse);
controller.close();
}
});
return new StreamingTextResponse(stream);
}
The content above is also identical to next-workshop-cheat/app/api/chat/route.ts
if you would rather like to copy the contents from there
Before you continue on the tasks make sure that your application (or the cheat project) is running locally.
- Make sure your current working directory
is
/container-workshop/02-run-app-in-container/next-workshop
. Runpwd
if unsure. - Start up the frontend by building it:
npm install
npm run build -- --no-lint
npm start
- Open a web browser and go to
http://localhost:3000
and try to fetch some recipes!
🐳 Time for containerizing the frontend!
Stop the running process in the terminal where you started the frontend:
On MacOS: control^ + c
Open the dockerfile
located
in 02-run-app-in-container/next-workshop-cheat
and fill
inn the remaining.
If using your own project you will need to move the
dockerfile
to02-run-app-in-container/next-workshop
. The root of the project is recommended.
When you have filled in all the instructions, change your working
directory to 02-run-app-in-container/<next_workshop_name>
and run
the following command
to build the image:
docker build -t next-frontend-image .
The -t
flag provides the image with a tag which is essentially a
name for the image. The .
in the end describes the path
to where Docker Engine should find the dockerfile to build the image
upon.
To ensure that an image was actually made go to Docker Desktop and you should find your newly created image under Images. Or you may get equal information by typing the following command in your terminal:
docker images
✅ Solution 2.1
# ==== DOCKERFILE FOR FRONTEND =====
# Use a Node 20 base image
FROM node:20-alpine
# Set the working directory to /app inside the container
WORKDIR /app
# Copy necessary source code folders/files
COPY app app
COPY [ "package.json", "package-lock.json", "tsconfig.json", "./"]
# ==== BUILD =====
# Install dependencies
RUN npm install
RUN npm run build -- --no-lint
# ==== RUN =======
# Start the app
CMD [ "npm", "start" ]
We can now run a new container with the frontend code based on the docker image we just built. Run the following command to run a container:
docker run -d --name next-frontend-container next-frontend-image:latest
To ensure that a container was actually made go to Docker Desktop and
you
should find your newly created container under Containers. Or you
may get
equal information by typing the following command in your terminal (
adding -a
lists all not-running containers as well):
docker ps
docker ps -a
You may also see the logs (in realtime by adding the flag --follow
):
docker logs next-frontend-container
docker logs next-frontend-container --follow
Or you may inspect the containers configurations with the command:
docker inspect next-frontend-container
🤫 Want a life hack to show all your friends?
Instead of entering the name of the container, which in our case is next-frontend-container, we can also use a selection (from the beginning) of the container ID which is enough for Docker Engine to distinct the container from other containers running on your machine.
docker inspect <FIRST_CHAR_OF_CONTAINER_ID>
💡 Interested in more details on `docker run` command?
The docker run command has a lot of flags that configure how the
container is
run. These are placed in between docker run
and the the name of the
image.
E.g. docker run --rm -it -p 8080:80 -v $PWD/05-*/nginx/html:/usr/share/nginx/html nginx
--rm
may be specified to automatically delete a container once it is
stopped.
Comes in handy when you start a lot of containers you know you will no
longer
need once stopped.
-d
is short for --detach
. For long-running containers such as a
web server,
you might want to run the container in the background instead of it
occupying a
terminal.
-it
is a combination of both --interactive
and --tty
. -t
essentially
creates a virtual terminal session and -i
forwards what you write to
the
container. These are most often used together, in the case when you
want to
interact with the container.
Run for example:
docker run --rm -it ubuntu
You are now in a terminal of an ubuntu distribution of Linux and may play around without causing harm to you actual system. Such an image may be extended with a custom Dockerfile, to create a playground with a lot of tooling installed - that may be totally reset at will.
Type exit
to quit, and destroy (because of --rm
), the container.
-v
is short for --volume
. In the case with ubuntu above, you may
want to
make files from your own system accessible inside the container. -v
lets you
mount an (absolute) path from your host into a path of the
container. By
additionally specifying --workdir
, that path may be chosen as the
starting
directory.
Run the following:
docker run --rm -it -v $HOME:/hostuser --workdir /hostuser ubuntu
If you now run ls
inside the container, you should see the content
of your
computers home directory. Use with care though, you can delete stuff
as well.
Sidenote: docker volume
may be used to create additional volumes
that are
initially empty and may be mounted into one or more containers.
-p
is short for --publish
(aka port). It is used to forward a port
on the
local machine to another port of the container. The syntax
is -p <local port>:<container port>
.
Run the nginx container and port-forward any local port to 80
inside the
container. Then open localhost:
followed by the chosen port in a
browser and
see if you are able to communicate with the container.
Inspect the container logs. If you ran the container with -it
you
will see
them instantly. If in detached mode (-d
), use docker ps
and docker logs
.
-e
is short for --env
. It is often used to pass the container
secrets that
one does not want to build into the image (e.g --env KEY=123
).
Environment
values are not yet determined at build time or the values vary between
deployments. It may be the url to the api, that differ in dev and
production, or
the logging format to use.
docker network
is used to create and manage networks. Containers may
be
entered into a network when started, and can then communicate with
other
containers by names rather than IP addresses (e.g. the url to backend
is
http://backend:3000
because it was started with --name backend
).
Open Docker Desktop and see that the container is running. Click on the container and you should see something similar to the following logs:
2024-04-10 09:02:57
2024-04-10 09:02:57 > [email protected] start
2024-04-10 09:02:57 > next start
2024-04-10 09:02:57
2024-04-10 09:02:57 ▲ Next.js 14.1.0
2024-04-10 09:02:57 - Local: http://localhost:3000
2024-04-10 09:02:57
2024-04-10 09:02:57 ✓ Ready in 245ms
However, the application will not be reachable
at http://localhost:3000
🧐. This is because the container does not
expose a port to your local machine.
Stop and remove the container either in Docker Desktop or in the
terminal:
docker stop next-frontend-container
docker rm next-frontend-container
And run the container again with an additional flag:
docker run -d -p 3001:3000 --name next-frontend-container next-frontend-image:latest
The -p
flag exposes a port on your local machine and maps it to a
port on the docker container. With the flag values 3001:3000
the
port 3001
on your local machine is mapped to port 3000
on the
container. Go to http://localhost:3001
and now try to fetch some
recipes.
The default port a react application is running on is port 3000
, but
let's say we need to run the application on port 5050
.
Creating an ENV variable is normally done by running the following
command in
your terminal; PORT=5050
, before running the npm start
command to
start the application. However, when doing this with docker we can
inject the ENV variable as an instruction in the dockerfile.
Try to edit the dockerfile to include the ENV variable PORT=5050
. To
see if it worked stop and remove the running container and remove the
image:
docker stop next-frontend-container
docker rm next-frontend-container
docker rmi next-frontend-image
Rebuild the image:
docker build -t next-frontend-image .
And run a new container:
NB: do you see what you need to change in the below command for this to work?
docker run -d -p 3001:3000 --name next-frontend-container next-frontend-image:latest
🤫Hint
When you inject the image with an ENV variable that changes the port,
the react application inside the container will now run on the new
port instead of the default port 3000
. In the port mapping described
after the -p
flag when running the container change the last port
to 5050
.
The full command should be like this:
docker run -d -p 3001:5050 --name next-frontend-container next-frontend-image:latest
✅ Solution 2.4
# ==== DOCKERFILE FOR FRONTEND =====
# Use a Node 20 base image
FROM node:20-alpine
# Set the working directory to /app inside the container
WORKDIR /app
# Copy necessary source code folders/files
COPY app app
COPY [ "package.json", "package-lock.json", "tsconfig.json", "./"]
# Set ENV variable
ENV PORT=5050
# ==== BUILD =====
# Install dependencies
RUN npm install
RUN npm run build -- --no-lint
# ==== RUN =======
# Start the app
CMD [ "npm", "start" ]