Getting Started

Getting Started

Before deploying Spegel read the compatibility section to make sure that the Kubernetes flavor of your choosing is supported or requires specific configuration to work.

Compatibility

Currently, Spegel only works with Containerd, in the future other container runtime interfaces may be supported. Spegel relies on Containerd registry mirroring to route requests to the correct destination. This requires Containerd to be properly configured, if it is not Spegel will exit. First of all the registry config path needs to be set, this is not done by default in Containerd. Second of all discarding unpacked layers cannot be enabled. Some Kubernetes flavors come with this setting out of the box, while others do not. Spegel is not able to write this configuration for you as it requires a restart of Containerd to take effect.

version = 2

[plugins."io.containerd.grpc.v1.cri".registry]
   config_path = "/etc/containerd/certs.d"
[plugins."io.containerd.grpc.v1.cri".containerd]
   discard_unpacked_layers = false

Spegel has been tested on the following Kubernetes distributions for compatibility. Green status means Spegel will work out of the box, yellow will require additional configuration, and red means that Spegel will not work.

StatusDistribution
🟢AKS
🟢Minikube
🟡EKS
🟡K3S and RKE2
🟡Kind
🟡Talos
🔴GKE
🔴DigitalOcean

EKS

Discard unpacked layers is enabled by default, meaning that layers that are not required for the container runtime will be removed after consumed. This needs to be disabled as otherwise all of the required layers of an image would not be present on the node.

Amazon Linux 2

If your EKS AMI is based on AL2, the included containerd config imports overrides from /etc/containerd/config.d/*.toml by default. The best way to change containerd settings is to add a file to the import directory using a custom node bootstrap script in your launch template.

#!/bin/bash
set -ex

mkdir -p /etc/containerd/config.d
cat > /etc/containerd/config.d/spegel.toml << EOL
[plugins."io.containerd.grpc.v1.cri".registry]
   config_path = "/etc/containerd/certs.d"
[plugins."io.containerd.grpc.v1.cri".containerd]
   discard_unpacked_layers = false
EOL

/etc/eks/bootstrap.sh

Amazon Linux 2023

If you are using an AL2023-based EKS AMI, bootstrap involves nodeadm configuration. To change containerd settings, you should add a nodeadm configuration section.

...
--MIMEBOUNDARY
Content-Transfer-Encoding: 7bit
Content-Type: application/node.eks.aws
Mime-Version: 1.0

---
apiVersion: node.eks.aws/v1alpha1
kind: NodeConfig
spec:
  containerd:
    config: |
      [plugins."io.containerd.grpc.v1.cri".containerd]
      discard_unpacked_layers = false      

--MIMEBOUNDARY
...

Kind

Spegel uses Kind for its end-to-end tests.

Discard unpacked layers is enabled by default, meaning that layers that are not required for the container runtime will be removed after consumed. This needs to be disabled as otherwise all of the required layers of an image would not be present on the node.

In order to be more like a “real” Kubernetes cluster, you may want to set Containerd’s metadata sharing policy to isolated.

apiVersion: kind.x-k8s.io/v1alpha4
kind: Cluster
containerdConfigPatches:
- |-
  [plugins."io.containerd.grpc.v1.cri".registry]
    config_path = "/etc/containerd/certs.d"
  [plugins."io.containerd.grpc.v1.cri".containerd]
    discard_unpacked_layers = false
  [plugins."io.containerd.metadata.v1.bolt"]
    content_sharing_policy = "isolated"  

For a full example, see the end-to-end tests’ Kind configuration for a full example.

K3S and RKE2

K3S and RKE2 embeds Spegel, refer to their documentation for deployment information.

Talos

Talos will by default discard unpacked layers, which has to be disabled with a machine configuration.

machine:
  files:
    - path: /etc/cri/conf.d/20-customization.part
      op: create
      content: |
        [plugins."io.containerd.cri.v1.images"]
          discard_unpacked_layers = false        

Talos also uses a different path as its Containerd registry config path.

spegel:
  containerdRegistryConfigPath: /etc/cri/conf.d/hosts

Talos comes with Pod Security Admission pre-configured. The default profile is too restrictive and needs to be changed to privileged.

kubectl label namespace spegel pod-security.kubernetes.io/enforce=privileged

GKE

GKE does not set the registry config path in its Containerd configuration. On top of that it uses the old mirror configuration for the internal mirroring service.

DigitalOcean

DigitalOcean does not set the registry config path in its Containerd configuration.

Deploying

Use the Helm chart to deploy Spegel into your Kubernetes cluster. Refer to the Helm Chart for detailed configuration documentation.

CLI

To deploy Spegel with the Helm CLI run the command.

helm upgrade --create-namespace --namespace spegel --install --version ${SPEGEL_VERSION} spegel oci://ghcr.io/spegel-org/helm-charts/spegel

Flux

To deploy Spegel with Flux commit the configuration.

apiVersion: v1
kind: Namespace
metadata:
  name: spegel
---
apiVersion: source.toolkit.fluxcd.io/v1beta2
kind: HelmRepository
metadata:
  name: spegel
  namespace: spegel
spec:
  type: "oci"
  interval: 5m0s
  url: oci://ghcr.io/spegel-org/helm-charts
---
apiVersion: helm.toolkit.fluxcd.io/v2beta1
kind: HelmRelease
metadata:
  name: spegel
  namespace: spegel
spec:
  interval: 1m
  chart:
    spec:
      chart: spegel
      version: ${SPEGEL_VERSION}
      interval: 5m
      sourceRef:
        kind: HelmRepository
        name: spegel