# Use persistent storage Persistent storage in Kubernetes requires cloud-specific configuration. For abstraction of container storage, Kubernetes offers [volumes](https://kubernetes.io/docs/concepts/storage/volumes/), allowing users to mount storage solutions directly into containers. The [Container Storage Interface (CSI)](https://kubernetes-csi.github.io/docs/) is the standard interface for exposing arbitrary block and file storage systems into containers in Kubernetes. Cloud service providers (CSPs) offer their own CSI-based solutions for cloud storage. ### Confidential storage Most cloud storage solutions support encryption, such as [GCE Persistent Disks (PD)](https://cloud.google.com/kubernetes-engine/docs/how-to/using-cmek). Constellation supports the available CSI-based storage options for Kubernetes engines in Azure and GCP. However, their encryption takes place in the storage backend and is managed by the CSP. Thus, using the default CSI drivers for these storage types means trusting the CSP with your persistent data. To address this, Constellation provides CSI drivers for Azure Disk and GCE PD, offering [encryption on the node level](../architecture/keys.md#storage-encryption). They enable transparent encryption for persistent volumes without needing to trust the cloud backend. Plaintext data never leaves the confidential VM context, offering you confidential storage. For more details see [encrypted persistent storage](../architecture/encrypted-storage.md). ## CSI drivers Constellation supports the following drivers, which offer node-level encryption and optional integrity protection. <tabs groupId="csp"> <tabItem value="azure" label="Azure"> **Constellation CSI driver for Azure Disk**: Mount Azure [Disk Storage](https://azure.microsoft.com/en-us/services/storage/disks/#overview) into your Constellation cluster. See the instructions on how to [install the Constellation CSI driver](#installation) or check out the [repository](https://github.com/edgelesssys/constellation-azuredisk-csi-driver) for more information. Since Azure Disks are mounted as ReadWriteOnce, they're only available to a single pod. </tabItem> <tabItem value="gcp" label="GCP"> **Constellation CSI driver for GCP Persistent Disk**: Mount [Persistent Disk](https://cloud.google.com/persistent-disk) block storage into your Constellation cluster. This includes support for [volume snapshots](https://cloud.google.com/kubernetes-engine/docs/how-to/persistent-volumes/volume-snapshots), which let you create copies of your volume at a specific point in time. You can use them to bring a volume back to a prior state or provision new volumes. Follow the instructions on how to [install the Constellation CSI driver](#installation) or check out the [repository](https://github.com/edgelesssys/constellation-gcp-compute-persistent-disk-csi-driver) for information about the configuration. </tabItem> </tabs> Note that in case the options above aren't a suitable solution for you, Constellation is compatible with all other CSI-based storage options. For example, you can use [Azure Files](https://docs.microsoft.com/en-us/azure/storage/files/storage-files-introduction) or [GCP Filestore](https://cloud.google.com/filestore) with Constellation out of the box. Constellation is just not providing transparent encryption on the node level for these storage types yet. ## Installation The following installation guide gives an overview of how to securely use CSI-based cloud storage for persistent volumes in Constellation. <tabs groupId="csp"> <tabItem value="azure" label="Azure"> 1. Install the driver: ```bash git clone --branch v1.1.0 https://github.com/edgelesssys/constellation-azuredisk-csi-driver cd constellation-azuredisk-csi-driver helm install azuredisk-csi-driver charts/edgeless --namespace kube-system ``` </tabItem> <tabItem value="gcp" label="GCP"> 1. Install the driver: ```bash git clone --branch v1.1.0 https://github.com/edgelesssys/constellation-gcp-compute-persistent-disk-csi-driver helm install gcp-compute-persistent-disk-csi-driver charts/ --namespace kube-system ``` </tabItem> </tabs> :::info The default storage class installed by the driver is set to `encrypted-rwo` for performance reasons. If you want integrity-protected storage, set the `storageClassName` parameter of your persistent volume claim to `integrity-encrypted-rwo`. Alternatively, you can create your own storage class with integrity protection enabled by adding `csi.storage.k8s.io/fstype: ext4-integrity` to the class `parameters`. Or use another filesystem by specifying another file system type with the suffix `-integrity`, e.g., `csi.storage.k8s.io/fstype: xfs-integrity`. Note that volume expansion isn't supported for integrity-protected disks. ::: 2. Create a [persistent volume](https://kubernetes.io/docs/concepts/storage/persistent-volumes/) A [persistent volume claim](https://kubernetes.io/docs/concepts/storage/persistent-volumes/#persistentvolumeclaims) is a request for storage with certain properties. It can refer to a storage class. The following creates a persistent volume claim, requesting 20 GB of storage via the `encrypted-rwo` storage class: ```bash cat <<EOF | kubectl apply -f - kind: PersistentVolumeClaim apiVersion: v1 metadata: name: pvc-example namespace: default spec: accessModes: - ReadWriteOnce storageClassName: encrypted-rwo resources: requests: storage: 20Gi EOF ``` 3. Create a Pod with persistent storage You can assign a persistent volume claim to an application in need of persistent storage. The mounted volume will persist restarts. The following creates a pod that uses the previously created persistent volume claim: ```bash cat <<EOF | kubectl apply -f - apiVersion: v1 kind: Pod metadata: name: web-server namespace: default spec: containers: - name: web-server image: nginx volumeMounts: - mountPath: /var/lib/www/html name: mypvc volumes: - name: mypvc persistentVolumeClaim: claimName: pvc-example readOnly: false EOF ``` ### Change the default storage class The default storage class is responsible for all persistent volume claims that don't explicitly request `storageClassName`. The previous instructions create a storage class with encryption enabled and sets this as the default class. In case you wish to change it, follow the steps below: <tabs groupId="csp"> <tabItem value="azure" label="Azure"> 1. List the storage classes in your cluster: ```bash kubectl get storageclass ``` The output is similar to this: ```shell-session NAME PROVISIONER RECLAIMPOLICY VOLUMEBINDINGMODE ALLOWVOLUMEEXPANSION AGE encrypted-rwo (default) azuredisk.csi.confidential.cloud Delete Immediate true 1d integrity-encrypted-rwo azuredisk.csi.confidential.cloud Delete Immediate false 1d ``` The default storage class is marked by `(default)`. 2. Mark old default storage class as non default If you previously used another storage class as the default, you will have to remove that annotation: ```bash kubectl patch storageclass encrypted-rwo -p '{"metadata": {"annotations":{"storageclass.kubernetes.io/is-default-class":"false"}}}' ``` 3. Mark new class as the default ```bash kubectl patch storageclass integrity-encrypted-rwo -p '{"metadata": {"annotations":{"storageclass.kubernetes.io/is-default-class":"true"}}}' ``` 4. Verify that your chosen storage class is default: ```bash kubectl get storageclass ``` The output is similar to this: ```shell-session NAME PROVISIONER RECLAIMPOLICY VOLUMEBINDINGMODE ALLOWVOLUMEEXPANSION AGE encrypted-rwo azuredisk.csi.confidential.cloud Delete Immediate true 1d integrity-encrypted-rwo (default) azuredisk.csi.confidential.cloud Delete Immediate false 1d ``` </tabItem> <tabItem value="gcp" label="GCP"> 1. List the storage classes in your cluster: ```bash kubectl get storageclass ``` The output is similar to this: ```shell-session NAME PROVISIONER RECLAIMPOLICY VOLUMEBINDINGMODE ALLOWVOLUMEEXPANSION AGE encrypted-rwo (default) gcp.csi.confidential.cloud Delete Immediate true 1d integrity-encrypted-rwo gcp.csi.confidential.cloud Delete Immediate false 1d ``` The default storage class is marked by `(default)`. 2. Mark old default storage class as non default If you previously used another storage class as the default, you will have to remove that annotation: ```bash kubectl patch storageclass encrypted-rwo -p '{"metadata": {"annotations":{"storageclass.kubernetes.io/is-default-class":"false"}}}' ``` 3. Mark new class as the default ```bash kubectl patch storageclass integrity-encrypted-rwo -p '{"metadata": {"annotations":{"storageclass.kubernetes.io/is-default-class":"true"}}}' ``` 4. Verify that your chosen storage class is default: ```bash kubectl get storageclass ``` The output is similar to this: ```shell-session NAME PROVISIONER RECLAIMPOLICY VOLUMEBINDINGMODE ALLOWVOLUMEEXPANSION AGE encrypted-rwo gcp.csi.confidential.cloud Delete Immediate true 1d integrity-encrypted-rwo (default) gcp.csi.confidential.cloud Delete Immediate false 1d ``` </tabItem> </tabs>