A fleet of remote development environments (with docker, fish shell, and even minikube) running in your kubernetes cluster, accessible through VS Code in the browser !
What & why
This setup is the v2 of a previous post on remote dev env using jupyterlab I made a year and a half ago. Thee OG setup was functionnal, but it had some issues, mainly around user management, container lifecycle and persistent data handling. As $dayjob has grown, so has the infrastructure, and so has the development needs. So a new solution was required.
A lot of them exists right now for remote environments, from providers like Github Codespaces, Gitpod, or even DevPod. But the folks at Coder released coder v2 a while back, and that’s what I’ve used for managing our team’s environments since late 2022.
The devs needs haven’t changed a lot since the first post. Our workflow is cluster-centric, based on skaffold to redeploy our built-on-the-fly-containers as pods directly onto the cluster.
How
The stack consists of these parts:
- a docker image that will be used as the base for our remote envs
- a kubernetes cluster, which will host everything,
- the coder platform deployed on said cluster,
- a custom kubernetes provider for running our docker image inside coder, handling some specific needs we have with our dev envs
The Base Image
The idea of the base image is to bake everything needed directly into it: vscode, git, fish shell, docker (running in a Docker-in-Docker fashion). I’ve already built this image, available at k0rventen/code, but if you want to tweak it, you’ll find the necessary files below:
We’re using an Ubuntu base, plus:
- we install basic dev utils and requirements for running dockerd
- we copy the dockerd stuff from their own image
- we install code-server
- we change the default shell to fish and copy over our base config files
- we start a bootstrap.fish script
# base image using ubuntu
FROM ubuntu:23.04
# install utils (fish shell, ssh)
RUN apt update && apt install -y --no-install-recommends curl ca-certificates git iptables fuse-overlayfs dnsutils less fish openssh-client && apt clean
#install code-server
RUN curl -fsSL https://code-server.dev/install.sh | sh -s -- --version 4.14.0 && rm -rvf /root/.cache
# copy dockerd binaries from the docker image
COPY --from=docker:20-dind /usr/local/bin/ /usr/local/bin/
# shell config
RUN chsh -s /usr/bin/fish
COPY config/ /tmp/code
# run our launch script
ENTRYPOINT ["fish", "/tmp/code/bootstrap.fish"]
The bootstrap.fish
has the following duties:
- make sure mandatory directories are here
- install Linuxbrew if not present (which will run only once, during the first startup of the env)
- start the dockerd daemon in the background
- then start code-server
Here is its content: bootstrap.fish
cd $home
mkdir -p projects .config/fish
if test ! -e .config/fish/config.fish
echo "copying fish config"
cp -v /tmp/code/config.fish .config/fish/config.fish
end
if test ! -e /home/linuxbrew/.linuxbrew/bin
echo "installing linuxbrew"
bash -c 'NONINTERACTIVE=1 /bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"' &
end
echo "starting dockerd.."
sh /usr/local/bin/dockerd-entrypoint.sh &
echo "starting code server"
exec code-server --bind-addr 0.0.0.0:9069 --auth none --disable-telemetry --disable-update-check projects
The config.fish is a very minimal config to get us started:
config.fish
# quiet fish
set fish_greeting
set -gx HOMEBREW_NO_ENV_HINTS 1
set -gx HOMEBREW_NO_INSTALL_CLEANUP 1
# brew fish
fish_add_path /home/linuxbrew/.linuxbrew/bin
# simple fish
function fish_prompt
printf '\n%s[%s]%s > ' (set_color cyan) (prompt_pwd) (set_color normal)
end
The Coder platform
Coder can be installed on a lot of platform, including docker, k8s and friends. Here we’ll concentrate on Kube. Requirements are a cluster with a storage class and an Ingress controller. You’ll need helm
as well.
From this point forward, I’ll assume that you have a cluster which can be accessed using the domain names coder.org
and *.coder.org
(You can add them in your local DNS server, or as entries in your /etc/hosts file wildcards aren’t supported, but for testing purposes you could write the required subdomains as needed).
Depending on the setup you’ll need to adjust some variables in the files below.
The configuration of the platform is done through the values.yaml
passed to helm. The important bit is the CODER_ACCESS_URL
and CODER_WILDCARD_ACCESS_URL
env vars and ingress config. They will define how clients can access the platform and their envs.
values.yaml
coder:
env:
- name: CODER_PG_CONNECTION_URL
valueFrom:
secretKeyRef:
name: coder-db-url
key: url
- name: CODER_ACCESS_URL
value: "https://coder.org"
- name: CODER_TELEMETRY
value: "false"
- name: CODER_WILDCARD_ACCESS_URL
value: "*.coder.org"
- name: CODER_AGENT_URL
value: "http://coder"
service:
type: ClusterIP
ingress:
enable: true
host: "coder.org"
wildcardHost: "*.coder.org"
annotations:
nginx.ingress.kubernetes.io/enable-cors: "true"
tls:
enable: true
Now we can deploy everything:
# create a namespace for our platform
kubectl create ns coder
kubectl config set-context --current --namespace coder
# postgres will be used by coder
helm repo add bitnami https://charts.bitnami.com/bitnami
helm install coder-db bitnami/postgresql --namespace coder --set auth.username=coder --set auth.password=coder --set auth.database=coder --set persistence.size=10Gi
# create a secret for coder holding the db creds
kubectl create secret generic coder-db-url --from-literal=url="postgres://coder:coder@coder-db-postgresql.coder.svc.cluster.local:5432/coder?sslmode=disable"
# deploy coder
helm repo add coder-v2 https://helm.coder.com/v2
helm upgrade --install coder coder-v2/coder --namespace coder --values values.yaml
You should be able to access the management interface at https://coder.org
. Create your admin user there and come back.
The custom Kubernetes provider
We now need to register a provider for our environments. The default Kubernetes
provider available is a good start, but we’ll tweak it a bit to our needs. It’s a single Terraform file defining the ressources to be created. It’s quite long but the gist is that each environment will be composed of:
- 1 Pod that will execute our dev env, with configurable ressources allocations (CPU & RAM)
- 3 PersistantVolumeClaims
- one for our home folder, mounted on
/root
- one for the dockerd daemon files, on
/var/lib/docker
- one for linuxbrew at
/home/linuxbrew
- one for our home folder, mounted on
The file is quite long:
main.tf
terraform {
required_providers {
coder = {
source = "coder/coder"
version = "~> 0.7.0"
}
kubernetes = {
source = "hashicorp/kubernetes"
version = "~> 2.18"
}
}
}
provider "coder" {
feature_use_managed_variables = true
}
variable "namespace" {
type = string
description = "The Kubernetes namespace to create workspaces in (must exist prior to creating workspaces)"
default = "coder"
}
data "coder_parameter" "cpu" {
name = "cpu"
display_name = "CPU"
description = "The number of CPU cores"
default = "2"
icon = "/icon/memory.svg"
mutable = true
option {
name = "Light machine (2 Cores)"
value = "2"
}
option {
name = "Heavy Machine (8 Cores)"
value = "8"
}
}
data "coder_parameter" "memory" {
name = "memory"
display_name = "Memory"
description = "The amount of memory in GB"
default = "2"
icon = "/icon/memory.svg"
mutable = true
option {
name = "2 GB"
value = "2"
}
option {
name = "8 GB"
value = "8"
}
}
data "coder_parameter" "image" {
name = "Image"
display_name = "Container Image"
description = "The base container image to use"
default = "k0rventen/code:0.1"
icon = "/icon/memory.svg"
mutable = true
type = "string"
}
provider "kubernetes" {
# Authenticate via ~/.kube/config or a Coder-specific ServiceAccount, depending on admin preferences
config_path = null
}
data "coder_workspace" "me" {}
resource "coder_agent" "main" {
os = "linux"
arch = "amd64"
startup_script_timeout = 180
startup_script = <<-EOT
set -e
fish /tmp/code/bootstrap.fish
EOT
}
# code-server
resource "coder_app" "code-server" {
agent_id = coder_agent.main.id
slug = "code-server"
display_name = "code-server"
icon = "/icon/code.svg"
url = "http://localhost:9069?folder=/root/projects"
subdomain = false
share = "owner"
healthcheck {
url = "http://localhost:9069/healthz"
interval = 3
threshold = 10
}
}
resource "kubernetes_persistent_volume_claim" "home" {
metadata {
name = "coder-${lower(data.coder_workspace.me.owner)}-${lower(data.coder_workspace.me.name)}-home"
namespace = var.namespace
labels = {
"app.kubernetes.io/name" = "coder-pvc"
"app.kubernetes.io/instance" = "coder-pvc-${lower(data.coder_workspace.me.owner)}-${lower(data.coder_workspace.me.name)}"
"app.kubernetes.io/part-of" = "coder"
// Coder specific labels.
"com.coder.resource" = "true"
"com.coder.workspace.id" = data.coder_workspace.me.id
"com.coder.workspace.name" = data.coder_workspace.me.name
"com.coder.user.id" = data.coder_workspace.me.owner_id
"com.coder.user.username" = data.coder_workspace.me.owner
}
annotations = {
"com.coder.user.email" = data.coder_workspace.me.owner_email
}
}
wait_until_bound = false
spec {
access_modes = ["ReadWriteOnce"]
resources {
requests = {
storage = "4Gi"
}
}
}
}
resource "kubernetes_persistent_volume_claim" "docker" {
metadata {
name = "coder-${lower(data.coder_workspace.me.owner)}-${lower(data.coder_workspace.me.name)}-docker"
namespace = var.namespace
labels = {
"app.kubernetes.io/name" = "coder-pvc"
"app.kubernetes.io/instance" = "coder-pvc-${lower(data.coder_workspace.me.owner)}-${lower(data.coder_workspace.me.name)}"
"app.kubernetes.io/part-of" = "coder"
// Coder specific labels.
"com.coder.resource" = "true"
"com.coder.workspace.id" = data.coder_workspace.me.id
"com.coder.workspace.name" = data.coder_workspace.me.name
"com.coder.user.id" = data.coder_workspace.me.owner_id
"com.coder.user.username" = data.coder_workspace.me.owner
}
annotations = {
"com.coder.user.email" = data.coder_workspace.me.owner_email
}
}
wait_until_bound = false
spec {
access_modes = ["ReadWriteOnce"]
resources {
requests = {
storage = "10Gi"
}
}
}
}
resource "kubernetes_persistent_volume_claim" "linuxbrew" {
metadata {
name = "coder-${lower(data.coder_workspace.me.owner)}-${lower(data.coder_workspace.me.name)}-linuxbrew"
namespace = var.namespace
labels = {
"app.kubernetes.io/name" = "coder-pvc"
"app.kubernetes.io/instance" = "coder-pvc-${lower(data.coder_workspace.me.owner)}-${lower(data.coder_workspace.me.name)}"
"app.kubernetes.io/part-of" = "coder"
// Coder specific labels.
"com.coder.resource" = "true"
"com.coder.workspace.id" = data.coder_workspace.me.id
"com.coder.workspace.name" = data.coder_workspace.me.name
"com.coder.user.id" = data.coder_workspace.me.owner_id
"com.coder.user.username" = data.coder_workspace.me.owner
}
annotations = {
"com.coder.user.email" = data.coder_workspace.me.owner_email
}
}
wait_until_bound = false
spec {
access_modes = ["ReadWriteOnce"]
resources {
requests = {
storage = "10Gi"
}
}
}
}
resource "kubernetes_pod" "main" {
count = data.coder_workspace.me.start_count
metadata {
name = "coder-${lower(data.coder_workspace.me.owner)}-${lower(data.coder_workspace.me.name)}"
namespace = var.namespace
labels = {
"app.kubernetes.io/name" = "coder-workspace"
"app.kubernetes.io/instance" = "coder-workspace-${lower(data.coder_workspace.me.owner)}-${lower(data.coder_workspace.me.name)}"
"app.kubernetes.io/part-of" = "coder"
// Coder specific labels.
"com.coder.resource" = "true"
"com.coder.workspace.id" = data.coder_workspace.me.id
"com.coder.workspace.name" = data.coder_workspace.me.name
"com.coder.user.id" = data.coder_workspace.me.owner_id
"com.coder.user.username" = data.coder_workspace.me.owner
}
annotations = {
"com.coder.user.email" = data.coder_workspace.me.owner_email
}
}
spec {
container {
name = "code-container"
image = "${data.coder_parameter.image.value}"
image_pull_policy = "Always"
command = ["sh", "-c", replace(coder_agent.main.init_script,"https://coder.org","http://coder")]
env {
name = "CODER_AGENT_TOKEN"
value = coder_agent.main.token
}
security_context {
privileged = "true"
}
resources {
requests = {
"cpu" = "250m"
"memory" = "512Mi"
}
limits = {
"cpu" = "${data.coder_parameter.cpu.value}"
"memory" = "${data.coder_parameter.memory.value}Gi"
}
}
volume_mount {
mount_path = "/root"
name = "home"
read_only = false
}
volume_mount {
mount_path = "/var/lib/docker"
name = "docker"
read_only = false
}
volume_mount {
mount_path = "/home/linuxbrew"
name = "linuxbrew"
read_only = false
}
}
volume {
name = "home"
persistent_volume_claim {
claim_name = kubernetes_persistent_volume_claim.home.metadata.0.name
read_only = false
}
}
volume {
name = "docker"
persistent_volume_claim {
claim_name = kubernetes_persistent_volume_claim.docker.metadata.0.name
read_only = false
}
}
volume {
name = "linuxbrew"
persistent_volume_claim {
claim_name = kubernetes_persistent_volume_claim.linuxbrew.metadata.0.name
read_only = false
}
}
}
}
Note: One quirk of this setup is that due to our environment using self signed certificate, we have to replace the external URL (the one used to access the envs) by the internal one (the coder service inside the ns) for our envs to start properly. In a more realistic scenario, trusted CA certs would be used instead.
To deploy this provider to our Coder instance, we’ll need the coder
cli, available here.
Depending on the exact setup (mainly due to self signed certificates), the login endpoint will vary, but the easiest is to port-forward the internal coder
service and login through this:
# in a tab
kubectl port-forward svc/coder 8080:80
# in another tab, then follow the login procedure
coder login http://127.0.0.1:8080
# once logged in, in the same dir as `main.tf`, reply yes to questions
coder template create kube
# You should preview the resources that will be created for each env:
┌──────────────────────────────────────────────────┐
│ Template Preview │
├──────────────────────────────────────────────────┤
│ RESOURCE │
├──────────────────────────────────────────────────┤
│ kubernetes_persistent_volume_claim.docker │
├──────────────────────────────────────────────────┤
│ kubernetes_persistent_volume_claim.home │
├──────────────────────────────────────────────────┤
│ kubernetes_persistent_volume_claim.linuxbrew │
├──────────────────────────────────────────────────┤
│ kubernetes_pod.main │
│ └─ main (linux, amd64) │
└──────────────────────────────────────────────────┘
Accessing our environment
Now everything required to work directly into our cluster is deployed. We can now create a Workspace based on the provider we defined earlier:
Wait for the pod to be created and the code-server
button to become available.
Now we can work using a web browser from our thin & light laptop (or even a Raspberry Pi) with the power of a cluster:
Disclaimer
This setup is loosely based on what is deployed and used daily at $job. The main upsides are that :
- onboarding/updating/offboarding the environments are dead-easy,
- laptops are running way cooler, batteries last longer and builds are way faster,
- and testing has very little friction due to the fact that we are right inside the cluster.
But there is some downsides:
- notably around security, our envs are running as privileged in the cluster (for dockerd), which might not be fine depending on the level of trust you want. One possible fix would be to switch to a rootless docker/podman, but as of time of writing, there are too many quirks when deploying such solutions.
- having the dev envs centralised means possibly a SPOF that could ruin everyone’s day if something goes wrong with the platform or the cluster.
- it might require a certain level of maturity around containers, mainly around how networking is affected by a remote setup.