This is a CSI driver for mounting images as PVs or ephemeral volumes.
It pulls images via CRI and shares the image store with the container runtime, then mounts images via the snapshot/storage service of the runtime. Read-Only volumes of the same image share the same snapshot. Read-Write volumes keep their own snapshot and changes until pod deletion.
The driver requires to mount various host paths for different container runtimes.
So, I build a binary utility, warm-metal-csi-image-install
, to reduce the installation complexity.
It supports kubernetes, microk8s and k3s clusters with container runtime cri-o, containerd or docker.
Users can run this utility on any nodes in their clusters to generate proper manifests to install the driver.
The download link is available on the release page.
# To print manifests
warm-metal-csi-image-install
# To show the detected configuration
warm-metal-csi-image-install --print-detected-instead
# To change the default namespace that the driver to be installed in
warm-metal-csi-image-install --namespace=foo
# To set a Secret as the imagepullsecret
warm-metal-csi-image-install --pull-image-secret-for-daemonset=foo
# To disable the memroy cache for imagepullsecrets if Secrets are short-lived.
warm-metal-csi-image-install --pull-image-secret-for-daemonset=foo --enable-daemon-image-credential-cache=false
You can found some installation manifests as samples in examples.
Until Docker migrates its image and snapshot store to containerd, I recommend you use containerd instead. Otherwise, the driver can't use images managed by Docker daemon.
If your container runtime can't be migrated, you can enable the CRI plugin by clearing
the containerd config file /etc/containerd/config.toml
,
then restarting the containerd.
Users can mount images as either pre-provisioned PVs or ephemeral volumes. PVs can only be mounted in access mode ReadOnlyMany, while ephemeral volumes will be writable. Any changes in ephemeral volumes will be discarded after unmounting.
For ephemeral volumes, volumeAttributes
contains image(required), secret, secretNamespace, and pullAlways.
apiVersion: batch/v1
kind: Job
metadata:
name: ephemeral-volume
spec:
template:
metadata:
name: ephemeral-volume
spec:
containers:
- name: ephemeral-volume
image: docker.io/warmmetal/csi-image-test:check-fs
env:
- name: TARGET
value: /target
volumeMounts:
- mountPath: /target
name: target
restartPolicy: Never
volumes:
- name: target
csi:
driver: csi-image.warm-metal.tech
# nodePublishSecretRef:
# name: "ImagePullSecret name in the same namespace"
volumeAttributes:
image: "docker.io/warmmetal/csi-image-test:simple-fs"
# # set pullAlways if you want to ignore local images
# pullAlways: "true"
backoffLimit: 0
For pre-provisioned PVs, volumeHandle
instead of the attribute image, specify the target image.
apiVersion: v1
kind: PersistentVolume
metadata:
name: pv-test-csi-image-test-simple-fs
spec:
storageClassName: csi-image.warm-metal.tech
capacity:
storage: 5Gi
accessModes:
- ReadOnlyMany
persistentVolumeReclaimPolicy: Retain
csi:
driver: csi-image.warm-metal.tech
volumeHandle: "docker.io/warmmetal/csi-image-test:simple-fs"
# nodePublishSecretRef:
# name: "name of the ImagePullSecret"
# namespace: "namespace of the secret"
# volumeAttributes:
# # set pullAlways if you want to ignore local images
# pullAlways: "true"
See all examples.
There are several ways to configure credentials for private image pulling.
If your clusters are in cloud, the credential provider are enabled automatically.
If your cloud provider provides a credential provider plugin instead, you can enable it by adding
both --image-credential-provider-config
and --image-credential-provider-bin-dir
flags to the driver.
See credential provider plugin.
Otherwise, you need ImagePullSecrets to store your credential. The following links may help.
- https://kubernetes.io/docs/tasks/configure-pod-container/pull-image-private-registry/.
- https://kubernetes.io/docs/tasks/configure-pod-container/configure-service-account/#add-imagepullsecrets-to-a-service-account
If the secret is desired to be shared by all volumes, you can add it to the ServiceAccount of the driver,
csi-image-warm-metal
by default, and update the Role csi-image-warm-metal
to make sure the daemon has permissions to fetch the secret,
then restart the driver daemon pod. Users can run warm-metal-csi-image-install
to generate new manifests and apply them to update.
# use secrets foo and bar
warm-metal-csi-image-install --pull-image-secret-for-daemonset=foo --pull-image-secret-for-daemonset=bar
If the secret works only for particular workloads, you can set via the nodePublishSecretRef
attribute of ephemeral volumes.
See the above sample manifest, and notice that secrets and workloads must in the same namespace.
(Since version v0.5.1, pulling private images using the ImagePullSecrets which attached to workload service accounts is no longer supported for security reasons.)
You can also set the secret to a PV, then share the PV with multiple workloads. See the sample above.
See test/sanity.
See test/e2e.