Setting up Development Containers
What are Development Containers?
A development container (or dev container for short) allows you to use a container as a full-featured development environment. It can be used to run an application, to separate tools, libraries, or runtimes needed for working with a codebase, and to aid in continuous integration and testing. Dev containers can be run locally or remotely, in a private or public cloud, in a variety of supporting tools and editors.
Development Containers are useful for setting up a consistent development environment across different machines and sharing it with your team. They can even be used in your CI/CD pipeline to ensure that your code is built and tested in the same environment.
Prerequisites
You need to have docker installed on your machine. If you are on macOS or Windows, you can install Rancher Desktop, which includes all the tools you need to run docker
on your machine.
To achieve the best performance, we recommend to use VZ
as the default Virtual Machine Type and virtiofs
as the default File Sharing method.
To build and run development containers, you need to have a Supporting Tool installed on your machine. A supporting tool is a tool that can be used to build and run development containers. It can be a text editor, an IDE, a terminal, or a web-based tool.
This guide will show you how to set up and use development containers with two different supporting tools: Visual Studio Code and the Dev Container CLI.
Visual Studio Code
If you are using Visual Studio Code, you have to install the Dev Containers extension, which is the official extension to work with development containers.
Dev Container CLI
If you are not using Visual Studio Code, you can still use development containers from you terminal using the devcontainer
command line tool. You can install it using the following commands:
npm install -g @devcontainers/cli
# Check if the installation was successful
devcontainer --version
Create your first Development Container
The core of a development container is a devcontainer.json
file. This file contains the configuration of the development container, such as the base image, the features to install, the post-start command, and customizations.
To start create a devcontainer.json
file in the root in .devcontainer
directory of your project.
{
"name": "My Development Container",
"image": "mcr.microsoft.com/devcontainers/base:debian"
}
This file defines a development container named My Development Container
based on the mcr.microsoft.com/devcontainers/base:debian
image.
Start the Development Container
Visual Studio Code
- Open the command palette (
Cmd + Shift + P
on macOS orCtrl + Shift + P
on Windows/Linux). - Search for
Dev Containers: Reopen in Container
.
Visual Studio Code will build the development container and open a new window inside the container.
Dev Container CLI
- Open a terminal in the root of your project.
- Run the following command:
# build and start the development container
devcontainer up --workspace-folder .
# open a shell inside the container
devcontainer exec --workspace-folder . /bin/bash
The --workspace-folder
flag is used to specify the path to the folder that contains the .devcontainer
directory.
Adding the Node.js Feature
Once you have created the development container, you can add features to it. Features are pre-built configurations that can be added to a development container to install tools, libraries, or runtimes.
Let's install the Node.js feature to the development container. This feature will install node
, npm
and yarn
at the specified version.
{
"name": "My Development Container",
"image": "mcr.microsoft.com/devcontainers/base:debian",
"features": {
"ghcr.io/devcontainers/features/node:1": {
"version": "20.19.0"
}
}
}
You can find a list of available features, official and community, in the Dev Container Features page. Each feature has its own documentation that explains how to use it, check out the Node.js feature documentation here to learn more about it.
Test the Node.js Feature
Rebuild the development container and open a shell inside it. You can check if node
, npm
, and yarn
are installed and at the correct version.
node --version
# v20.19.0
npm --version
yarn --version
In Visual Studio Code you can rebuild the development container by running the Dev Containers: Rebuild Container
command from the command palette. In the terminal, you can run the devcontainer up --workspace-folder .
command.
Execute yarn install
on start
Once you have node
installed you may also want to run yarn install
when the development container starts.
You can do this by adding a postStartCommand
to the devcontainer.json
file.
{
"name": "My Development Container",
"image": "mcr.microsoft.com/devcontainers/base:debian",
"postStartCommand": "yarn install",
"features": {
"ghcr.io/devcontainers/features/node:1": {
"version": "20.19.0"
}
}
}
You can also run multiple commands by passing an object to postStartCommand
. These commands will be executed in parallel.
{
"name": "My Development Container",
"image": "mcr.microsoft.com/devcontainers/base:debian",
"postStartCommand": {
"yarn-install": "yarn install",
"greet": "echo 'Hello, from the development container!'"
},
"features": {
"ghcr.io/devcontainers/features/node:1": {
"version": "20.19.0"
}
}
}
The development containers also suppport others lifecycle hooks like preStartCommand
, postStartCommand
, preStopCommand
, and postStopCommand
. You can learn more about them in the Dev Container Lifecycle Hooks page.
Customize the Development Container
You can customize the development container by adding customizations to the devcontainer.json
file. Customizations can be applied to the editor, the terminal, the shell, and the environment variables.
One tipical customization is to set up extensions and settings for Visual Studio Code. Let's configure prettier
extension and set it as the default formatter.
{
"name": "My Development Container",
"image": "mcr.microsoft.com/devcontainers/base:debian",
"postStartCommand": "yarn install",
"features": {
"ghcr.io/devcontainers/features/node:1": {
"version": "20.19.0"
}
},
"customizations": {
"vscode": {
"extensions": [
"esbenp.prettier-vscode"
],
"settings": {
"editor.formatOnSave": true,
"editor.defaultFormatter": "esbenp.prettier-vscode"
}
}
}
}
Customizations are a great way to centralize the configuration of your development environment and share it with your team. You can learn more about customizations in the Supporting Tool page.
Some features includes customizations that are automatically applied to the development container. For example the Official Terraform Feature automatically installs the hashicorp.terraform
VSC extension and sets up the terraform
language server.
Use a Dockerfile
instead of a prebuilt image
Sometimes you may want to use a custom Dockerfile to build the development container instead of using a prebuilt image. You can do this by adding a build
object to the devcontainer.json
file.
Let's create a custom Dockerfile that adds an alias for yarn
and use it to build the development container.
-
Create a
Dockerfile
in the.devcontainer
directory:.devcontainer/DockerfileFROM mcr.microsoft.com/devcontainers/base:debian
# Add yarn alias
RUN echo 'alias y="yarn"' >> /etc/bash.bashrc
CMD ["sleep", "infinity"] -
Update the
devcontainer.json
file to use theDockerfile
:.devcontainer/devcontainer.json{
"name": "My Development Container",
"build": {
"dockerfile": "Dockerfile"
},
"postStartCommand": "yarn install",
"features": {
"ghcr.io/devcontainers/features/node:1": {
"version": "20.19.0"
}
},
"customizations": {
"vscode": {
"extensions": [
"esbenp.prettier-vscode"
],
"settings": {
"editor.formatOnSave": true,
"editor.defaultFormatter": "esbenp.prettier-vscode"
}
}
}
} -
Once you have updated the
devcontainer.json
file, you can rebuild the development container and open a shell inside it to test the newly addedy
alias.
If you need to run multiple containers alongside the development container, you can use Docker Compose to define the services that should be started with the development container. This is useful when you need to run backing services like databases, caches, or message brokers alongside the development container.
A real-world example of this setup can be found in the pagopa/io-messages GitHub repository where the development container is started alongside a redis
, azurite
and other services using Docker Compose.
Extra: Use a Development Container Template
You don't have to start from scratch when creating a development container. You can use a template to create a development container with pre-built configurations.
The PagoPA DX team maintain a Node.js template that you can use to create a development container for a Node.js project, that follow the best practices and conventions listed in our Tech Radar.
Since it is not published yet in the official registry, you need the Dev Container CLI to use it. To use it follow these steps:
cd path/to/your/project
devcontainer templates apply -t ghcr.io/pagopa/devcontainer-templates/node:1
This will scaffold a devcontainer.json
ready to work with node
, terraform
, azure
and the other tools we use in our projects.
The DX team is working to develop custom features and templates that are not available in the official registry. You can find them in the pagopa/dx GitHub repository.
Conclusion
You now know the basics of setting up a development container. Check out the official Dev Containers documentation to learn more about the features and customizations that you can add to your development container to fit the need of your project.