Volumes

    Docker has a concept of , though it is somewhat looser and less managed. A Docker volume is a directory on disk or in another container. Docker provides volume drivers, but the functionality is somewhat limited.

    Kubernetes supports many types of volumes. A Pod can use any number of volume types simultaneously. Ephemeral volume types have a lifetime of a pod, but persistent volumes exist beyond the lifetime of a pod. When a pod ceases to exist, Kubernetes destroys ephemeral volumes; however, Kubernetes does not destroy persistent volumes. For any kind of volume in a given pod, data is preserved across container restarts.

    At its core, a volume is a directory, possibly with some data in it, which is accessible to the containers in a pod. How that directory comes to be, the medium that backs it, and the contents of it are determined by the particular volume type used.

    To use a volume, specify the volumes to provide for the Pod in .spec.volumes and declare where to mount those volumes into containers in .spec.containers[*].volumeMounts. A process in a container sees a filesystem view composed from the initial contents of the , plus volumes (if defined) mounted inside the container. The process sees a root filesystem that initially matches the contents of the container image. Any writes to within that filesystem hierarchy, if allowed, affect what that process views when it performs a subsequent filesystem access. Volumes mount at the specified paths within the image. For each container defined within a Pod, you must independently specify where to mount each volume that the container uses.

    Volumes cannot mount within other volumes (but see for a related mechanism). Also, a volume cannot contain a hard link to anything in a different volume.

    Types of volumes

    Kubernetes supports several types of volumes.

    FEATURE STATE: Kubernetes v1.17 [deprecated]

    An awsElasticBlockStore volume mounts an Amazon Web Services (AWS) EBS volume into your pod. Unlike emptyDir, which is erased when a pod is removed, the contents of an EBS volume are persisted and the volume is unmounted. This means that an EBS volume can be pre-populated with data, and that data can be shared between pods.

    Note: You must create an EBS volume by using aws ec2 create-volume or the AWS API before you can use it.

    There are some restrictions when using an awsElasticBlockStore volume:

    • the nodes on which pods are running must be AWS EC2 instances
    • those instances need to be in the same region and availability zone as the EBS volume
    • EBS only supports a single EC2 instance mounting a volume

    Creating an AWS EBS volume

    Before you can use an EBS volume with a pod, you need to create it.

    Make sure the zone matches the zone you brought up your cluster in. Check that the size and EBS volume type are suitable for your use.

    AWS EBS configuration example

    1. apiVersion: v1
    2. kind: Pod
    3. metadata:
    4. name: test-ebs
    5. spec:
    6. containers:
    7. - image: registry.k8s.io/test-webserver
    8. name: test-container
    9. volumeMounts:
    10. - mountPath: /test-ebs
    11. name: test-volume
    12. volumes:
    13. - name: test-volume
    14. # This AWS EBS volume must already exist.
    15. awsElasticBlockStore:
    16. volumeID: "<volume id>"
    17. fsType: ext4

    If the EBS volume is partitioned, you can supply the optional field partition: "<partition number>" to specify which partition to mount on.

    AWS EBS CSI migration

    FEATURE STATE: Kubernetes v1.25 [stable]

    The CSIMigration feature for awsElasticBlockStore, when enabled, redirects all plugin operations from the existing in-tree plugin to the ebs.csi.aws.com Container Storage Interface (CSI) driver. In order to use this feature, the AWS EBS CSI driver must be installed on the cluster.

    AWS EBS CSI migration complete

    FEATURE STATE: Kubernetes v1.17 [alpha]

    To disable the awsElasticBlockStore storage plugin from being loaded by the controller manager and the kubelet, set the InTreePluginAWSUnregister flag to true.

    azureDisk (deprecated)

    FEATURE STATE: Kubernetes v1.19 [deprecated]

    The azureDisk volume type mounts a Microsoft Azure into a pod.

    For more details, see the azureDisk volume plugin.

    azureDisk CSI migration

    FEATURE STATE: Kubernetes v1.24 [stable]

    The CSIMigration feature for azureDisk, when enabled, redirects all plugin operations from the existing in-tree plugin to the disk.csi.azure.com Container Storage Interface (CSI) Driver. In order to use this feature, the Azure Disk CSI Driver must be installed on the cluster.

    azureDisk CSI migration complete

    FEATURE STATE: Kubernetes v1.21 [alpha]

    To disable the azureDisk storage plugin from being loaded by the controller manager and the kubelet, set the InTreePluginAzureDiskUnregister flag to true.

    azureFile (deprecated)

    FEATURE STATE: Kubernetes v1.21 [deprecated]

    The azureFile volume type mounts a Microsoft Azure File volume (SMB 2.1 and 3.0) into a pod.

    For more details, see the .

    azureFile CSI migration

    FEATURE STATE: Kubernetes v1.26 [stable]

    The CSIMigration feature for azureFile, when enabled, redirects all plugin operations from the existing in-tree plugin to the file.csi.azure.com Container Storage Interface (CSI) Driver. In order to use this feature, the must be installed on the cluster and the CSIMigrationAzureFile feature gates must be enabled.

    Azure File CSI driver does not support using same volume with different fsgroups. If CSIMigrationAzureFile is enabled, using same volume with different fsgroups won’t be supported at all.

    azureFile CSI migration complete

    FEATURE STATE: Kubernetes v1.21 [alpha]

    To disable the azureFile storage plugin from being loaded by the controller manager and the kubelet, set the InTreePluginAzureFileUnregister flag to true.

    cephfs

    A cephfs volume allows an existing CephFS volume to be mounted into your Pod. Unlike emptyDir, which is erased when a pod is removed, the contents of a cephfs volume are preserved and the volume is merely unmounted. This means that a cephfs volume can be pre-populated with data, and that data can be shared between pods. The cephfs volume can be mounted by multiple writers simultaneously.

    Note: You must have your own Ceph server running with the share exported before you can use it.

    See the for more details.

    cinder (deprecated)

    FEATURE STATE: Kubernetes v1.18 [deprecated]

    Note: Kubernetes must be configured with the OpenStack cloud provider.

    The cinder volume type is used to mount the OpenStack Cinder volume into your pod.

    Cinder volume configuration example

    1. apiVersion: v1
    2. kind: Pod
    3. metadata:
    4. name: test-cinder
    5. spec:
    6. containers:
    7. - image: registry.k8s.io/test-webserver
    8. name: test-cinder-container
    9. volumeMounts:
    10. - mountPath: /test-cinder
    11. name: test-volume
    12. volumes:
    13. - name: test-volume
    14. # This OpenStack volume must already exist.
    15. cinder:
    16. volumeID: "<volume id>"
    17. fsType: ext4

    OpenStack CSI migration

    FEATURE STATE: Kubernetes v1.24 [stable]

    The CSIMigration feature for Cinder is enabled by default since Kubernetes 1.21. It redirects all plugin operations from the existing in-tree plugin to the cinder.csi.openstack.org Container Storage Interface (CSI) Driver. must be installed on the cluster.

    To disable the in-tree Cinder plugin from being loaded by the controller manager and the kubelet, you can enable the InTreePluginOpenStackUnregister feature gate.

    configMap

    A ConfigMap provides a way to inject configuration data into pods. The data stored in a ConfigMap can be referenced in a volume of type configMap and then consumed by containerized applications running in a pod.

    When referencing a ConfigMap, you provide the name of the ConfigMap in the volume. You can customize the path to use for a specific entry in the ConfigMap. The following configuration shows how to mount the log-config ConfigMap onto a Pod called configmap-pod:

    1. apiVersion: v1
    2. kind: Pod
    3. metadata:
    4. name: configmap-pod
    5. spec:
    6. containers:
    7. - name: test
    8. image: busybox:1.28
    9. volumeMounts:
    10. - name: config-vol
    11. mountPath: /etc/config
    12. volumes:
    13. - name: config-vol
    14. configMap:
    15. name: log-config
    16. items:
    17. path: log_level

    The log-config ConfigMap is mounted as a volume, and all contents stored in its log_level entry are mounted into the Pod at path /etc/config/log_level. Note that this path is derived from the volume’s mountPath and the path keyed with log_level.

    Note:

    • You must create a before you can use it.

    • A container using a ConfigMap as a subPath volume mount will not receive ConfigMap updates.

    • Text data is exposed as files using the UTF-8 character encoding. For other character encodings, use binaryData.

    downwardAPI

    A downwardAPI volume makes downward API data available to applications. Within the volume, you can find the exposed data as read-only files in plain text format.

    Note: A container using the downward API as a volume mount does not receive updates when field values change.

    See Expose Pod Information to Containers Through Files to learn more.

    emptyDir

    An emptyDir volume is first created when a Pod is assigned to a node, and exists as long as that Pod is running on that node. As the name says, the emptyDir volume is initially empty. All containers in the Pod can read and write the same files in the emptyDir volume, though that volume can be mounted at the same or different paths in each container. When a Pod is removed from a node for any reason, the data in the emptyDir is deleted permanently.

    Note: A container crashing does not remove a Pod from a node. The data in an emptyDir volume is safe across container crashes.

    Some uses for an emptyDir are:

    • scratch space, such as for a disk-based merge sort
    • checkpointing a long computation for recovery from crashes
    • holding files that a content-manager container fetches while a webserver container serves the data

    The emptyDir.medium field controls where emptyDir volumes are stored. By default emptyDir volumes are stored on whatever medium that backs the node such as disk, SSD, or network storage, depending on your environment. If you set the emptyDir.medium field to "Memory", Kubernetes mounts a tmpfs (RAM-backed filesystem) for you instead. While tmpfs is very fast, be aware that unlike disks, tmpfs is cleared on node reboot and any files you write count against your container’s memory limit.

    A size limit can be specified for the default medium, which limits the capacity of the emptyDir volume. The storage is allocated from node ephemeral storage. If that is filled up from another source (for example, log files or image overlays), the emptyDir may run out of capacity before this limit.

    Note: If the SizeMemoryBackedVolumes is enabled, you can specify a size for memory backed volumes. If no size is specified, memory backed volumes are sized to 50% of the memory on a Linux host.

    emptyDir configuration example

    1. apiVersion: v1
    2. kind: Pod
    3. metadata:
    4. name: test-pd
    5. spec:
    6. containers:
    7. - image: registry.k8s.io/test-webserver
    8. name: test-container
    9. volumeMounts:
    10. - mountPath: /cache
    11. name: cache-volume
    12. volumes:
    13. - name: cache-volume
    14. emptyDir:
    15. sizeLimit: 500Mi

    An fc volume type allows an existing fibre channel block storage volume to mount in a Pod. You can specify single or multiple target world wide names (WWNs) using the parameter targetWWNs in your Volume configuration. If multiple WWNs are specified, targetWWNs expect that those WWNs are from multi-path connections.

    Note: You must configure FC SAN Zoning to allocate and mask those LUNs (volumes) to the target WWNs beforehand so that Kubernetes hosts can access them.

    See the fibre channel example for more details.

    gcePersistentDisk (deprecated)

    FEATURE STATE: Kubernetes v1.17 [deprecated]

    A gcePersistentDisk volume mounts a Google Compute Engine (GCE) persistent disk (PD) into your Pod. Unlike emptyDir, which is erased when a pod is removed, the contents of a PD are preserved and the volume is merely unmounted. This means that a PD can be pre-populated with data, and that data can be shared between pods.

    Note: You must create a PD using gcloud or the GCE API or UI before you can use it.

    There are some restrictions when using a gcePersistentDisk:

    • the nodes on which Pods are running must be GCE VMs
    • those VMs need to be in the same GCE project and zone as the persistent disk

    One feature of GCE persistent disk is concurrent read-only access to a persistent disk. A gcePersistentDisk volume permits multiple consumers to simultaneously mount a persistent disk as read-only. This means that you can pre-populate a PD with your dataset and then serve it in parallel from as many Pods as you need. Unfortunately, PDs can only be mounted by a single consumer in read-write mode. Simultaneous writers are not allowed.

    Creating a GCE persistent disk

    Before you can use a GCE persistent disk with a Pod, you need to create it.

    1. gcloud compute disks create --size=500GB --zone=us-central1-a my-data-disk

    GCE persistent disk configuration example

    Regional persistent disks

    The Regional persistent disks feature allows the creation of persistent disks that are available in two zones within the same region. In order to use this feature, the volume must be provisioned as a PersistentVolume; referencing the volume directly from a pod is not supported.

    Manually provisioning a Regional PD PersistentVolume

    Dynamic provisioning is possible using a StorageClass for GCE PD. Before creating a PersistentVolume, you must create the persistent disk:

    1. gcloud compute disks create --size=500GB my-data-disk
    2. --region us-central1
    3. --replica-zones us-central1-a,us-central1-b

    Regional persistent disk configuration example

    1. apiVersion: v1
    2. kind: PersistentVolume
    3. metadata:
    4. name: test-volume
    5. spec:
    6. capacity:
    7. storage: 400Gi
    8. accessModes:
    9. - ReadWriteOnce
    10. gcePersistentDisk:
    11. pdName: my-data-disk
    12. fsType: ext4
    13. nodeAffinity:
    14. required:
    15. nodeSelectorTerms:
    16. # failure-domain.beta.kubernetes.io/zone should be used prior to 1.21
    17. - key: topology.kubernetes.io/zone
    18. operator: In
    19. values:
    20. - us-central1-a
    21. - us-central1-b

    GCE CSI migration

    FEATURE STATE: Kubernetes v1.25 [stable]

    The CSIMigration feature for GCE PD, when enabled, redirects all plugin operations from the existing in-tree plugin to the pd.csi.storage.gke.io Container Storage Interface (CSI) Driver. In order to use this feature, the must be installed on the cluster.

    GCE CSI migration complete

    FEATURE STATE: Kubernetes v1.21 [alpha]

    To disable the gcePersistentDisk storage plugin from being loaded by the controller manager and the kubelet, set the InTreePluginGCEUnregister flag to true.

    gitRepo (deprecated)

    Warning: The gitRepo volume type is deprecated. To provision a container with a git repo, mount an EmptyDir into an InitContainer that clones the repo using git, then mount the into the Pod’s container.

    A gitRepo volume is an example of a volume plugin. This plugin mounts an empty directory and clones a git repository into this directory for your Pod to use.

    Here is an example of a gitRepo volume:

    1. apiVersion: v1
    2. kind: Pod
    3. metadata:
    4. name: server
    5. spec:
    6. containers:
    7. - image: nginx
    8. name: nginx
    9. volumeMounts:
    10. - mountPath: /mypath
    11. name: git-volume
    12. volumes:
    13. - name: git-volume
    14. gitRepo:
    15. repository: "git@somewhere:me/my-git-repository.git"
    16. revision: "22f1d8406d464b0c0874075539c1f2e96c253775"

    glusterfs (removed)

    Kubernetes 1.27 does not include a glusterfs volume type.

    The GlusterFS in-tree storage driver was deprecated in the Kubernetes v1.25 release and then removed entirely in the v1.26 release.

    hostPath

    Warning:

    HostPath volumes present many security risks, and it is a best practice to avoid the use of HostPaths when possible. When a HostPath volume must be used, it should be scoped to only the required file or directory, and mounted as ReadOnly.

    If restricting HostPath access to specific directories through AdmissionPolicy, volumeMounts MUST be required to use readOnly mounts for the policy to be effective.

    A hostPath volume mounts a file or directory from the host node’s filesystem into your Pod. This is not something that most Pods will need, but it offers a powerful escape hatch for some applications.

    For example, some uses for a hostPath are:

    • running a container that needs access to Docker internals; use a hostPath of /var/lib/docker
    • running cAdvisor in a container; use a hostPath of /sys
    • allowing a Pod to specify whether a given hostPath should exist prior to the Pod running, whether it should be created, and what it should exist as

    In addition to the required path property, you can optionally specify a type for a hostPath volume.

    The supported values for field type are:

    Watch out when using this type of volume, because:

    • HostPaths can expose privileged system credentials (such as for the Kubelet) or privileged APIs (such as container runtime socket), which can be used for container escape or to attack other parts of the cluster.
    • Pods with identical configuration (such as created from a PodTemplate) may behave differently on different nodes due to different files on the nodes
    • The files or directories created on the underlying hosts are only writable by root. You either need to run your process as root in a privileged Container or modify the file permissions on the host to be able to write to a hostPath volume

    hostPath configuration example

    1. apiVersion: v1
    2. kind: Pod
    3. metadata:
    4. name: test-pd
    5. spec:
    6. containers:
    7. - image: registry.k8s.io/test-webserver
    8. name: test-container
    9. volumeMounts:
    10. - mountPath: /test-pd
    11. name: test-volume
    12. volumes:
    13. - name: test-volume
    14. hostPath:
    15. # directory location on host
    16. path: /data
    17. # this field is optional
    18. type: Directory

    Caution: The FileOrCreate mode does not create the parent directory of the file. If the parent directory of the mounted file does not exist, the pod fails to start. To ensure that this mode works, you can try to mount directories and files separately, as shown in the FileOrCreateconfiguration.

    hostPath FileOrCreate configuration example

    1. apiVersion: v1
    2. kind: Pod
    3. metadata:
    4. name: test-webserver
    5. spec:
    6. containers:
    7. - name: test-webserver
    8. image: registry.k8s.io/test-webserver:latest
    9. volumeMounts:
    10. - mountPath: /var/local/aaa
    11. name: mydir
    12. - mountPath: /var/local/aaa/1.txt
    13. name: myfile
    14. volumes:
    15. - name: mydir
    16. hostPath:
    17. # Ensure the file directory is created.
    18. path: /var/local/aaa
    19. type: DirectoryOrCreate
    20. - name: myfile
    21. hostPath:
    22. path: /var/local/aaa/1.txt
    23. type: FileOrCreate

    iscsi

    An iscsi volume allows an existing iSCSI (SCSI over IP) volume to be mounted into your Pod. Unlike emptyDir, which is erased when a Pod is removed, the contents of an iscsi volume are preserved and the volume is merely unmounted. This means that an iscsi volume can be pre-populated with data, and that data can be shared between pods.

    Note: You must have your own iSCSI server running with the volume created before you can use it.

    A feature of iSCSI is that it can be mounted as read-only by multiple consumers simultaneously. This means that you can pre-populate a volume with your dataset and then serve it in parallel from as many Pods as you need. Unfortunately, iSCSI volumes can only be mounted by a single consumer in read-write mode. Simultaneous writers are not allowed.

    See the for more details.

    local

    A local volume represents a mounted local storage device such as a disk, partition or directory.

    Local volumes can only be used as a statically created PersistentVolume. Dynamic provisioning is not supported.

    Compared to hostPath volumes, local volumes are used in a durable and portable manner without manually scheduling pods to nodes. The system is aware of the volume’s node constraints by looking at the node affinity on the PersistentVolume.

    However, local volumes are subject to the availability of the underlying node and are not suitable for all applications. If a node becomes unhealthy, then the local volume becomes inaccessible by the pod. The pod using this volume is unable to run. Applications using local volumes must be able to tolerate this reduced availability, as well as potential data loss, depending on the durability characteristics of the underlying disk.

    The following example shows a PersistentVolume using a local volume and nodeAffinity:

    You must set a PersistentVolume nodeAffinity when using local volumes. The Kubernetes scheduler uses the PersistentVolume nodeAffinity to schedule these Pods to the correct node.

    PersistentVolume volumeMode can be set to “Block” (instead of the default value “Filesystem”) to expose the local volume as a raw block device.

    When using local volumes, it is recommended to create a StorageClass with volumeBindingMode set to WaitForFirstConsumer. For more details, see the local example. Delaying volume binding ensures that the PersistentVolumeClaim binding decision will also be evaluated with any other node constraints the Pod may have, such as node resource requirements, node selectors, Pod affinity, and Pod anti-affinity.

    An external static provisioner can be run separately for improved management of the local volume lifecycle. Note that this provisioner does not support dynamic provisioning yet. For an example on how to run an external local provisioner, see the local volume provisioner user guide.

    Note: The local PersistentVolume requires manual cleanup and deletion by the user if the external static provisioner is not used to manage the volume lifecycle.

    nfs

    An nfs volume allows an existing NFS (Network File System) share to be mounted into a Pod. Unlike emptyDir, which is erased when a Pod is removed, the contents of an nfs volume are preserved and the volume is merely unmounted. This means that an NFS volume can be pre-populated with data, and that data can be shared between pods. NFS can be mounted by multiple writers simultaneously.

    1. apiVersion: v1
    2. metadata:
    3. name: test-pd
    4. spec:
    5. containers:
    6. - image: registry.k8s.io/test-webserver
    7. name: test-container
    8. volumeMounts:
    9. - mountPath: /my-nfs-data
    10. name: test-volume
    11. volumes:
    12. - name: test-volume
    13. nfs:
    14. server: my-nfs-server.example.com
    15. path: /my-nfs-volume
    16. readOnly: true

    Note:

    You must have your own NFS server running with the share exported before you can use it.

    Also note that you can’t specify NFS mount options in a Pod spec. You can either set mount options server-side or use /etc/nfsmount.conf. You can also mount NFS volumes via PersistentVolumes which do allow you to set mount options.

    See the for an example of mounting NFS volumes with PersistentVolumes.

    A persistentVolumeClaim volume is used to mount a into a Pod. PersistentVolumeClaims are a way for users to “claim” durable storage (such as a GCE PersistentDisk or an iSCSI volume) without knowing the details of the particular cloud environment.

    See the information about PersistentVolumes for more details.

    portworxVolume (deprecated)

    FEATURE STATE: Kubernetes v1.25 [deprecated]

    A portworxVolume is an elastic block storage layer that runs hyperconverged with Kubernetes. Portworx fingerprints storage in a server, tiers based on capabilities, and aggregates capacity across multiple servers. Portworx runs in-guest in virtual machines or on bare metal Linux nodes.

    A portworxVolume can be dynamically created through Kubernetes or it can also be pre-provisioned and referenced inside a Pod. Here is an example Pod referencing a pre-provisioned Portworx volume:

    1. apiVersion: v1
    2. kind: Pod
    3. metadata:
    4. name: test-portworx-volume-pod
    5. spec:
    6. containers:
    7. - image: registry.k8s.io/test-webserver
    8. name: test-container
    9. volumeMounts:
    10. - mountPath: /mnt
    11. name: pxvol
    12. volumes:
    13. - name: pxvol
    14. # This Portworx volume must already exist.
    15. portworxVolume:
    16. volumeID: "pxvol"
    17. fsType: "<fs-type>"

    Note: Make sure you have an existing PortworxVolume with name pxvol before using it in the Pod.

    For more details, see the examples.

    Portworx CSI migration

    FEATURE STATE: Kubernetes v1.25 [beta]

    The CSIMigration feature for Portworx has been added but disabled by default in Kubernetes 1.23 since it’s in alpha state. It has been beta now since v1.25 but it is still turned off by default. It redirects all plugin operations from the existing in-tree plugin to the pxd.portworx.com Container Storage Interface (CSI) Driver. must be installed on the cluster. To enable the feature, set CSIMigrationPortworx=true in kube-controller-manager and kubelet.

    projected

    A projected volume maps several existing volume sources into the same directory. For more details, see .

    rbd

    An rbd volume allows a (RBD) volume to mount into your Pod. Unlike emptyDir, which is erased when a pod is removed, the contents of an rbd volume are preserved and the volume is unmounted. This means that a RBD volume can be pre-populated with data, and that data can be shared between pods.

    Note: You must have a Ceph installation running before you can use RBD.

    A feature of RBD is that it can be mounted as read-only by multiple consumers simultaneously. This means that you can pre-populate a volume with your dataset and then serve it in parallel from as many pods as you need. Unfortunately, RBD volumes can only be mounted by a single consumer in read-write mode. Simultaneous writers are not allowed.

    See the RBD example for more details.

    RBD CSI migration

    FEATURE STATE: Kubernetes v1.23 [alpha]

    The CSIMigration feature for RBD, when enabled, redirects all plugin operations from the existing in-tree plugin to the rbd.csi.ceph.com CSI driver. In order to use this feature, the must be installed on the cluster and the CSIMigrationRBD feature gate must be enabled. (Note that the csiMigrationRBD flag has been removed and replaced with in release v1.24)

    Note:

    As a Kubernetes cluster operator that administers storage, here are the prerequisites that you must complete before you attempt migration to the RBD CSI driver:

    • You must install the Ceph CSI driver (rbd.csi.ceph.com), v3.5.0 or above, into your Kubernetes cluster.
    • considering the clusterID field is a required parameter for CSI driver for its operations, but in-tree StorageClass has monitors field as a required parameter, a Kubernetes storage admin has to create a clusterID based on the monitors hash ( ex:#echo -n '<monitors_string>' | md5sum) in the CSI config map and keep the monitors under this clusterID configuration.
    • Also, if the value of adminId in the in-tree Storageclass is different from admin, the adminSecretName mentioned in the in-tree Storageclass has to be patched with the base64 value of the adminId parameter value, otherwise this step can be skipped.

    secret

    A secret volume is used to pass sensitive information, such as passwords, to Pods. You can store secrets in the Kubernetes API and mount them as files for use by pods without coupling to Kubernetes directly. secret volumes are backed by tmpfs (a RAM-backed filesystem) so they are never written to non-volatile storage.

    Note: You must create a Secret in the Kubernetes API before you can use it.

    Note: A container using a Secret as a subPath volume mount will not receive Secret updates.

    For more details, see .

    vsphereVolume (deprecated)

    Note: We recommend to use vSphere CSI out-of-tree driver instead.

    A vsphereVolume is used to mount a vSphere VMDK volume into your Pod. The contents of a volume are preserved when it is unmounted. It supports both VMFS and VSAN datastore.

    For more information, see the examples.

    vSphere CSI migration

    In Kubernetes 1.27, all operations for the in-tree vsphereVolume type are redirected to the csi.vsphere.vmware.com driver.

    vSphere CSI driver must be installed on the cluster. You can find additional advice on how to migrate in-tree vsphereVolume in VMware’s documentation page . If vSphere CSI Driver is not installed volume operations can not be performed on the PV created with the in-tree vsphereVolume type.

    You must run vSphere 7.0u2 or later in order to migrate to the vSphere CSI driver.

    If you are running a version of Kubernetes other than v1.27, consult the documentation for that version of Kubernetes.

    Note:

    The following StorageClass parameters from the built-in vsphereVolume plugin are not supported by the vSphere CSI driver:

    • diskformat
    • hostfailurestotolerate
    • forceprovisioning
    • cachereservation
    • diskstripes
    • objectspacereservation
    • iopslimit

    Existing volumes created using these parameters will be migrated to the vSphere CSI driver, but new volumes created by the vSphere CSI driver will not be honoring these parameters.

    vSphere CSI migration complete

    FEATURE STATE: Kubernetes v1.19 [beta]

    To turn off the vsphereVolume plugin from being loaded by the controller manager and the kubelet, you need to set InTreePluginvSphereUnregister feature flag to true. You must install a csi.vsphere.vmware.com driver on all worker nodes.

    Sometimes, it is useful to share one volume for multiple uses in a single pod. The volumeMounts.subPath property specifies a sub-path inside the referenced volume instead of its root.

    The following example shows how to configure a Pod with a LAMP stack (Linux Apache MySQL PHP) using a single, shared volume. This sample subPath configuration is not recommended for production use.

    The PHP application’s code and assets map to the volume’s html folder and the MySQL database is stored in the volume’s mysql folder. For example:

    1. apiVersion: v1
    2. kind: Pod
    3. metadata:
    4. name: my-lamp-site
    5. spec:
    6. containers:
    7. - name: mysql
    8. image: mysql
    9. env:
    10. - name: MYSQL_ROOT_PASSWORD
    11. value: "rootpasswd"
    12. volumeMounts:
    13. - mountPath: /var/lib/mysql
    14. name: site-data
    15. subPath: mysql
    16. - name: php
    17. image: php:7.0-apache
    18. volumeMounts:
    19. - mountPath: /var/www/html
    20. name: site-data
    21. subPath: html
    22. volumes:
    23. - name: site-data
    24. persistentVolumeClaim:
    25. claimName: my-lamp-site-data

    Using subPath with expanded environment variables

    FEATURE STATE: Kubernetes v1.17 [stable]

    Use the subPathExpr field to construct subPath directory names from downward API environment variables. The subPath and subPathExpr properties are mutually exclusive.

    In this example, a Pod uses subPathExpr to create a directory pod1 within the hostPath volume /var/log/pods. The hostPath volume takes the Pod name from the downwardAPI. The host directory /var/log/pods/pod1 is mounted at /logs in the container.

    1. apiVersion: v1
    2. kind: Pod
    3. metadata:
    4. name: pod1
    5. spec:
    6. containers:
    7. - name: container1
    8. env:
    9. - name: POD_NAME
    10. valueFrom:
    11. fieldRef:
    12. apiVersion: v1
    13. fieldPath: metadata.name
    14. image: busybox:1.28
    15. command: [ "sh", "-c", "while [ true ]; do echo 'Hello'; sleep 10; done | tee -a /logs/hello.txt" ]
    16. volumeMounts:
    17. - name: workdir1
    18. mountPath: /logs
    19. # The variable expansion uses round brackets (not curly brackets).
    20. subPathExpr: $(POD_NAME)
    21. restartPolicy: Never
    22. volumes:
    23. - name: workdir1
    24. hostPath:
    25. path: /var/log/pods

    Resources

    The storage media (such as Disk or SSD) of an emptyDir volume is determined by the medium of the filesystem holding the kubelet root dir (typically /var/lib/kubelet). There is no limit on how much space an emptyDir or hostPath volume can consume, and no isolation between containers or between pods.

    To learn about requesting space using a resource specification, see .

    The out-of-tree volume plugins include (CSI), and also FlexVolume (which is deprecated). These plugins enable storage vendors to create custom storage plugins without adding their plugin source code to the Kubernetes repository.

    Previously, all volume plugins were “in-tree”. The “in-tree” plugins were built, linked, compiled, and shipped with the core Kubernetes binaries. This meant that adding a new storage system to Kubernetes (a volume plugin) required checking code into the core Kubernetes code repository.

    Both CSI and FlexVolume allow volume plugins to be developed independent of the Kubernetes code base, and deployed (installed) on Kubernetes clusters as extensions.

    For storage vendors looking to create an out-of-tree volume plugin, please refer to the volume plugin FAQ.

    csi

    Container Storage Interface (CSI) defines a standard interface for container orchestration systems (like Kubernetes) to expose arbitrary storage systems to their container workloads.

    Please read the for more information.

    Note: Support for CSI spec versions 0.2 and 0.3 are deprecated in Kubernetes v1.13 and will be removed in a future release.

    Note: CSI drivers may not be compatible across all Kubernetes releases. Please check the specific CSI driver’s documentation for supported deployments steps for each Kubernetes release and a compatibility matrix.

    Once a CSI compatible volume driver is deployed on a Kubernetes cluster, users may use the csi volume type to attach or mount the volumes exposed by the CSI driver.

    A csi volume can be used in a Pod in three different ways:

    The following fields are available to storage administrators to configure a CSI persistent volume:

    • driver: A string value that specifies the name of the volume driver to use. This value must correspond to the value returned in the GetPluginInfoResponse by the CSI driver as defined in the . It is used by Kubernetes to identify which CSI driver to call out to, and by CSI driver components to identify which PV objects belong to the CSI driver.
    • volumeHandle: A string value that uniquely identifies the volume. This value must correspond to the value returned in the volume.id field of the CreateVolumeResponse by the CSI driver as defined in the CSI spec. The value is passed as volume_id on all calls to the CSI volume driver when referencing the volume.
    • readOnly: An optional boolean value indicating whether the volume is to be “ControllerPublished” (attached) as read only. Default is false. This value is passed to the CSI driver via the readonly field in the ControllerPublishVolumeRequest.
    • fsType: If the PV’s VolumeMode is Filesystem then this field may be used to specify the filesystem that should be used to mount the volume. If the volume has not been formatted and formatting is supported, this value will be used to format the volume. This value is passed to the CSI driver via the VolumeCapability field of ControllerPublishVolumeRequest, NodeStageVolumeRequest, and NodePublishVolumeRequest.
    • volumeAttributes: A map of string to string that specifies static properties of a volume. This map must correspond to the map returned in the volume.attributes field of the CreateVolumeResponse by the CSI driver as defined in the . The map is passed to the CSI driver via the volume_context field in the ControllerPublishVolumeRequest, NodeStageVolumeRequest, and NodePublishVolumeRequest.
    • controllerPublishSecretRef: A reference to the secret object containing sensitive information to pass to the CSI driver to complete the CSI ControllerPublishVolume and ControllerUnpublishVolume calls. This field is optional, and may be empty if no secret is required. If the Secret contains more than one secret, all secrets are passed. nodeExpandSecretRef: A reference to the secret containing sensitive information to pass to the CSI driver to complete the CSI NodeExpandVolume call. This field is optional, and may be empty if no secret is required. If the object contains more than one secret, all secrets are passed. When you have configured secret data for node-initiated volume expansion, the kubelet passes that data via the NodeExpandVolume() call to the CSI driver. In order to use the nodeExpandSecretRef field, your cluster should be running Kubernetes version 1.25 or later.
    • If you are running Kubernetes Version 1.25 or 1.26, you must enable the feature gate named CSINodeExpandSecret for each kube-apiserver and for the kubelet on every node. In Kubernetes version 1.27 this feature has been enabled by default and no explicit enablement of the feature gate is required. You must also be using a CSI driver that supports or requires secret data during node-initiated storage resize operations.
    • nodePublishSecretRef: A reference to the secret object containing sensitive information to pass to the CSI driver to complete the CSI NodePublishVolume call. This field is optional, and may be empty if no secret is required. If the secret object contains more than one secret, all secrets are passed.
    • nodeStageSecretRef: A reference to the secret object containing sensitive information to pass to the CSI driver to complete the CSI NodeStageVolume call. This field is optional, and may be empty if no secret is required. If the Secret contains more than one secret, all secrets are passed.

    CSI raw block volume support

    FEATURE STATE: Kubernetes v1.18 [stable]

    Vendors with external CSI drivers can implement raw block volume support in Kubernetes workloads.

    You can set up your PersistentVolume/PersistentVolumeClaim with raw block volume support as usual, without any CSI specific changes.

    CSI ephemeral volumes

    FEATURE STATE: Kubernetes v1.25 [stable]

    You can directly configure CSI volumes within the Pod specification. Volumes specified in this way are ephemeral and do not persist across pod restarts. See Ephemeral Volumes for more information.

    For more information on how to develop a CSI driver, refer to the

    Windows CSI proxy

    FEATURE STATE: Kubernetes v1.22 [stable]

    CSI node plugins need to perform various privileged operations like scanning of disk devices and mounting of file systems. These operations differ for each host operating system. For Linux worker nodes, containerized CSI node node plugins are typically deployed as privileged containers. For Windows worker nodes, privileged operations for containerized CSI node plugins is supported using , a community-managed, stand-alone binary that needs to be pre-installed on each Windows node.

    For more details, refer to the deployment guide of the CSI plugin you wish to deploy.

    Migrating to CSI drivers from in-tree plugins

    FEATURE STATE: Kubernetes v1.25 [stable]

    The CSIMigration feature directs operations against existing in-tree plugins to corresponding CSI plugins (which are expected to be installed and configured). As a result, operators do not have to make any configuration changes to existing Storage Classes, PersistentVolumes or PersistentVolumeClaims (referring to in-tree plugins) when transitioning to a CSI driver that supersedes an in-tree plugin.

    The operations and features that are supported include: provisioning/delete, attach/detach, mount/unmount and resizing of volumes.

    In-tree plugins that support CSIMigration and have a corresponding CSI driver implemented are listed in .

    The following in-tree plugins support persistent storage on Windows nodes:

    FEATURE STATE: Kubernetes v1.23 [deprecated]

    FlexVolume is an out-of-tree plugin interface that uses an exec-based model to interface with storage drivers. The FlexVolume driver binaries must be installed in a pre-defined volume plugin path on each node and in some cases the control plane nodes as well.

    Pods interact with FlexVolume drivers through the flexVolume in-tree volume plugin. For more details, see the FlexVolume README document.

    The following FlexVolume , deployed as PowerShell scripts on the host, support Windows nodes:

    Note:

    FlexVolume is deprecated. Using an out-of-tree CSI driver is the recommended way to integrate external storage with Kubernetes.

    Maintainers of FlexVolume driver should implement a CSI Driver and help to migrate users of FlexVolume drivers to CSI. Users of FlexVolume should move their workloads to use the equivalent CSI Driver.

    Mount propagation

    Mount propagation allows for sharing volumes mounted by a container to other containers in the same pod, or even to other pods on the same node.

    Mount propagation of a volume is controlled by the mountPropagation field in Container.volumeMounts. Its values are:

    • None - This volume mount will not receive any subsequent mounts that are mounted to this volume or any of its subdirectories by the host. In similar fashion, no mounts created by the container will be visible on the host. This is the default mode.

      This mode is equal to rprivate mount propagation as described in

      However, the CRI runtime may choose rslave mount propagation (i.e., HostToContainer) instead, when rprivate propagation is not applicable. cri-dockerd (Docker) is known to choose rslave mount propagation when the mount source contains the Docker daemon’s root directory (/var/lib/docker).

    • HostToContainer - This volume mount will receive all subsequent mounts that are mounted to this volume or any of its subdirectories.

      In other words, if the host mounts anything inside the volume mount, the container will see it mounted there.

      Similarly, if any Pod with Bidirectional mount propagation to the same volume mounts anything there, the container with HostToContainer mount propagation will see it.

      This mode is equal to rslave mount propagation as described in the mount(8)

    • Bidirectional - This volume mount behaves the same the HostToContainer mount. In addition, all volume mounts created by the container will be propagated back to the host and to all containers of all pods that use the same volume.

      A typical use case for this mode is a Pod with a FlexVolume or CSI driver or a Pod that needs to mount something on the host using a hostPath volume.

      This mode is equal to rshared mount propagation as described in the

      Warning: Bidirectional mount propagation can be dangerous. It can damage the host operating system and therefore it is allowed only in privileged containers. Familiarity with Linux kernel behavior is strongly recommended. In addition, any volume mounts created by containers in pods must be destroyed (unmounted) by the containers on termination.

    Configuration

    Before mount propagation can work properly on some deployments (CoreOS, RedHat/Centos, Ubuntu) mount share must be configured correctly in Docker as shown below.

    Edit your Docker’s systemd service file. Set MountFlags as follows:

      Or, remove if present. Then restart the Docker daemon: