Published: Apr 6, 2021 by Dan Jones
Introduction to devcontainers
Devcontainers within Visual Studio Code (VS Code) is a powerful concept, being able to create a Docker container with all the tooling and runtime stacks a developer needs to work on a given project, without having to install anything but Docker, VS Code and the VS Code Remote Containers extension.
Essentially devcontainers are VS Code workspace instances that live within the Docker container, when first opening the container the workspace instance is installed and connected to the VS Code UI working in the browser or your local machine.
The above diagram is taken from the VS Code website
Probably you’ve heard a lot about GitHub Codespaces lately, an upcoming feature in GitHub, which essentially uses devcontainers to make its features work.
What has this got to do with continuous integration?
Devcontainers as a concept have little to do with continuous integration (CI), but the Docker container used for the devcontainer to run in does. Your dockerfile contains all the specific versions of tools and runtime stacks you need to do the work.
When moving to CI you then need to make your CI runner build and test the application, meaning you now need to ensure the runner has all the correct tooling with the same versions to do the job correctly. Though having slightly different versions can make little difference, for consistency we want to ensure what the developer is developing with the same tools used in CI.
What are we going to do?
We will create a simple application to run within VS Code devcontainers and set up testing, then move to the CI to run the build and test process there within the Docker container. We will be using GitHub actions for the example, but the concepts work with most modern CI systems.
Before continuing, ensure you have Docker and VS Code installed on to your machine.
Create a devcontainer workspace
We want to build a React web app, so we need a container that has NodeJS installed, we can create our own Dockerfile with all our tools and configurations setup, or pull an existing image from a Docker repo, for this app, we will use the Microsoft standard NodeJS devcontainer.
Firstly, lets create our workspace.
mkdir ci-sample-app
cd ci-sample-app
Next create a folder .devcontainer
and inside add a new file .devcontainer.json
and inside add the image we want to use as our devcontainer.
{
"image": "mcr.microsoft.com/vscode/devcontainers/typescript-node:0-12"
}
Next we will install the Remote - Containers
extension for VS Code. You can install this from the extension’s menu in VS Code or if you have the VS Code path reference set, you can run a command in the root of the project.
code --install-extension ms-vscode-remote.remote-containers
You may need to restart VS Code and once you do, you will be asked if you would like to open the project in a devcontainer.
Click reopen. The first time you do this it will pull the Docker image down from the remote repository, which may take some time.
When everything is complete, everything will still look the same as before, but the VS Code workspace is now running exclusively inside the Docker container.
Creating the React web app
Now we are running in the context of the container, we will create the sample React app and get it running using npm as our package manager.
Using the VS Code terminal, which can be opened from the top menu under Terminal
, New Terminal
, run the following to create a new React app.
npx create-react-app app
cd app
rm -rf .git # remove git binding as we want to initialize in the root directory
npm start
This will start the React app on port 3000, but if you try to access http://localhost:3000/, you will notice you can’t, this is because we first need to expose the port from within the devcontainer, which can be done inside devcontainer.json
.
{
"image": "mcr.microsoft.com/vscode/devcontainers/typescript-node:0-12",
"forwardPorts": [3000]
}
Inside the .devcontainer.json
you can also add VS Code extensions, so every time someone opens the devcontainer they are greeted with the same IDE setup, for example, you may want to add an ESLint extension.
{
"image": "mcr.microsoft.com/vscode/devcontainers/typescript-node:0-12",
"extensions": ["dbaeumer.vscode-eslint"],
"forwardPorts": [3000]
}
There are extensions to add Unit Test UI for easy of testing and many, many more.
When editing .devcontainer within the devcontainer, we need to rebuild the container to get the changes.
At the bottom left of VS Code you will see a section containing the words Dev Container
.
Clicking this will bring up a menu at the top, select “Remote-Containers: Rebuild Container”.
You should now be able to visit http://localhost:3000/ and see the basic React app up and running.
The sample contains tests, which can be run using npm.
npm test
You can create a release bundle by running npm run build
, which will output the content to a folder called build
.
CI with GitHub Actions
For this next bit, we will use GitHub Actions to build and test our app using the Docker image used within the devcontainer above.
Create folders in the root of the project as .github\workflows
, then create a file called build.yml
inside the workflows
folder, this will contain our CI code.
We will be using a Linux Hosted machine to do our build, which supports container jobs.
name: Build
on: push
jobs:
container:
runs-on: ubuntu-latest
container: mcr.microsoft.com/vscode/devcontainers/typescript-node:0-12
steps:
Our pipeline is now defined to pull down the same container we used inside our devcontainer and when it runs the steps
(which we will define next), these will be run in the context of the container.
We can run the predefined npm tasks with GitHub Actions, and it will run against our version of npm, or we can run the same commands we have already used to keep consistency.
name: Build
on: push
jobs:
container:
runs-on: ubuntu-latest
container: mcr.microsoft.com/vscode/devcontainers/typescript-node:0-12
steps:
- uses: actions/checkout@v2
- run: npm install
name: Install npm packages
working-directory: app
- run: CI=true && npm test
name: Test React application
working-directory: app
- run: CI=true && npm run build
name: Build React app bundle
working-directory: app
- uses: actions/upload-artifact@v2
with:
name: app
path: app/build
The last command, will take our build and publish it to the artifacts section of the pipeline run.
Now our code is written we will push it up to a GitHub repo.
git init
git add .
git commit -m "devcontainer CI sample"
Next we will publish to our remote repo (You will want to substitute the GitHub repo with your own).
git remote add origin https://github.com/skilledcookie/devcontainers-and-ci.git
git branch -M main
git push -u origin main
This should automatically run the CI pipeline, which you can see under “Actions” in the repository. See this in our GitHub repo now!
CI in Azure Pipelines example
Azure Pipelines is very similar to GitHub Actions syntax, if you would like to run this example using Azure DevOps, below is the equivalent to the script posted above.
pool:
vmImage: 'ubuntu-latest'
container: "mcr.microsoft.com/vscode/devcontainers/typescript-node:0-12"
steps:
- bash: npm install
displayName: Install npm packages
workingDirectory: app
- bash: npm test
displayName: Test React application
workingDirectory: app
env:
CI: true
- bash: npm run build
displayName: Build React app bundle
workingDirectory: app
env:
CI: true
- publish: app/build
artifact: app
displayName: Publish to Pipeline Artifacts
More patterns
Above you saw a basic example of how you can use the same Docker container you use in devcontainers in your CI pipelines, but there are other patterns you can use alongside this to add more flexibility to your development workflow.
- Use a Dockerfile to customize your container adding more tools specific to your development and CI workflow, for example adding Terraform to manage your infrastructure.
- Remote Docker repositories like Docker Hub or Azure Container Registry can store and share your private or public images across projects, without the need to build each time.
- Use Docker tags to version your Docker images, so devcontainers and CI can target a specific version, and you can upgrade when you choose, or force the use of the
latest
tag to ensure everyone is always working with the latest development container.
Exciting times ahead
Overall devcontainers are a very powerful concept and bringing the same container in to your CI pipeline ensures consistency in your development workflow, ensuring you aren’t using different tools or versions of tools in either place.
It will be interesting to see how the devcontainer landscape evolves once GitHub Codespaces is finally released to the public.
Related resources
- Git Hub actions
- Azure Pipelines
- Developing inside a container
- Getting started with Docker
- Getting started with VS Code
Source code
The source code for this post can be found at https://github.com/skilledcookie/devcontainers-and-ci
Comments
Comments can be made on the LinkedIn post here.