How do I use Amazon EBS Multi-Attach to attach the same Volume to multiple workloads in Amazon EKS?
I want to use Amazon Elastic Block Store (Amazon EBS) Multi-Attach for multiple workloads across multiple clusters in Amazon Elastic Kubernetes Service (Amazon EKS).
Short description
When you create a persistent workload in Amazon EKS with Amazon EBS storage, the default volume type is gp2. Amazon EBS Multi-Attach allows you to attach a single Provisioned IOPS SSD (io1 or io2) volume to multiple instances in the same Availability Zone.
Amazon EBS Multi-Attach isn't supported in General Purpose SSD volumes such as gp2 and gp3. The Amazon EBS CSI driver doesn't support multi-attaching volumes to workloads that run on different nodes in the same cluster.
To multi-attach the same Amazon EBS persistent storage to multiple workloads across different clusters, use Provisioned IOPS SSD volumes. Make sure that the pods run on worker nodes that are in the same Availability Zone (AZ) across the clusters.
Note: If your workloads are in multiple Availability Zones in the same or different clusters, then use Amazon Elastic File System (Amazon EFS). For more information, see Create an Amazon EFS file system for Amazon EKS on the GitHub website.
Resolution
Before you begin, make sure that the Amazon EBS CSI driver is installed in the required Amazon EKS clusters.
Note: Multi-Attach enabled volumes can be attached to up to 16 Linux instances built on the Nitro System that are in the same Availability Zone.
To use Amazon EBS Multi-Attach to attach the same volume to multiple workloads across multiple clusters, complete the following steps.
Provision an Amazon EBS volume
To provision an Amazon EBS volume, do the following:
- Provision a volume statically:
Note: Replace <example-region> with your required AWS Region. Replace <example-az> with your required Availability Zone.aws ec2 create-volume --volume-type io2 --multi-attach-enabled --size 10 --iops 2000 --region <example-region> --availability-zone <example-az> --tag-specifications 'ResourceType=volume,Tags=[{Key=purpose,Value=prod},{Key=Name,Value=multi-attach-eks}]'
- If you have an existing workload with a gp2 or gp3 volume, first create a snapshot of the volume. Then, create an io2 volume from that snapshot.
Note: Amazon EBS Multi-Attach can't be turned on for io1 volumes after they're created. Amazon EBS Multi-Attach can be turned on for io2 volumes after they're created if they aren't attached to instances. For io2 dynamic storage provisioning in Amazon EKS, specify the Immediate mode to provision the volume without the pod to create the storage class. Make sure that you turn on Amazon EBS Multi-Attach before you create the pod.
Retrieve the volume ID
Retrieve the volume ID that was provisioned for the workload:
aws ec2 describe-volumes --filters "Name=tag:Name,Values=multi-attach-eks*" --query "Volumes[*].{ID:VolumeId}" --region <example-region>
Note: Replace example-region with the required AWS Region.
Provision a persistent workload in an existing cluster using the preceding volume ID
-
Create the following manifest named workloadA.yml:
apiVersion: v1 kind: PersistentVolumeClaim metadata: name: <example-pv-claim-name-a> spec: storageClassName: "" volumeName: <example-pv-name-a> accessModes: - ReadWriteOnce resources: requests: storage: 5Gi --- apiVersion: v1 kind: Pod metadata: name: <example-pod-a> spec: containers: - name: <example-pod-container-name> image: centos command: ["/bin/sh"] args: ["-c", "while true; do echo $(date -u) on pod A >> /data/out.txt; sleep 15; done"] volumeMounts: - name: <example-volume-mount-name> mountPath: /data volumes: - name: <example-volume-mount-name> persistentVolumeClaim: claimName: <example-pv-claim-name-a> --- apiVersion: v1 kind: PersistentVolume metadata: name: <example-pv-name-a> spec: accessModes: - ReadWriteOnce capacity: storage: 5Gi csi: driver: ebs.csi.aws.com fsType: ext4 volumeHandle: <example-preceding-volume-id> nodeAffinity: required: nodeSelectorTerms: - matchExpressions: - key: topology.ebs.csi.aws.com/zone operator: In values: - <example-az>
Note: Replace all example strings in the following command with your required values. Make sure that the storageClassName parameter value is set to empty "" strings.
-
Switch the kubectl context to cluster A, and then deploy the workload:
kubectl config use-context <example-clusterA-context> kubectl apply -f workloadA.yml
Use the preceding volume ID to create another workload in another cluster
-
Create and deploy the following manifest named workloadB.yml:
apiVersion: v1 kind: PersistentVolumeClaim metadata: name: <example-pv-claim-name-b> spec: storageClassName: "" volumeName: <example-pv-name-b> accessModes: - ReadWriteOnce resources: requests: storage: 5Gi --- apiVersion: v1 kind: Pod metadata: name: <example-pod-b> spec: containers: - name: <example-pod-container-name> image: centos command: ["/bin/sh"] args: ["-c", "tail -f /data/out.txt"] volumeMounts: - name: <example-volume-mount-name> mountPath: /data volumes: - name: <example-volume-mount-name> persistentVolumeClaim: claimName: <example-pv-claim-name-b> --- apiVersion: v1 kind: PersistentVolume metadata: name: <example-pv-name-b> spec: accessModes: - ReadWriteOnce capacity: storage: 5Gi csi: driver: ebs.csi.aws.com fsType: ext4 volumeHandle: <example-preceding-volume-id> nodeAffinity: required: nodeSelectorTerms: - matchExpressions: - key: topology.ebs.csi.aws.com/zone operator: In values: - example-az
Note: Replace all example strings with your required values. Make sure that the storageClassName parameter value is set to empty "" strings.
-
Switch the kubectl context to cluster B, and then deploy workload:
kubectl config use-context <example-clusterB-context> kubectl apply -f workloadB.ym
Note: Replace example-clusterB-context with your context.
Check that the pods are running and have the same content
-
Authenticate across the different clusters and run the following command:
kubectl get pods
Example output for cluster A:
NAME READY STATUS RESTARTS AGE example-pod-a 1/1 Running 0 18m
Example output for cluster B:
NAME READY STATUS RESTARTS AGE example-pod-b 1/1 Running 0 3m13s
-
For example-pod-a, run the following command to view the content written to the storage:
kubectl exec -it <example-pod-a> -- cat /data/out.txt
Example output:
Fri Sep 22 12:39:04 UTC 2024 on example-pod-a Fri Sep 22 12:39:19 UTC 2024 on example-pod-a Fri Sep 22 12:39:34 UTC 2024 on example-pod-a
-
For example-pod-b, run the following command to read the content written into the same storage as example-pod-a:
kubectl logs -f <example-pod-b>
Example output:
Fri Sep 22 12:39:04 UTC 2024 on example-pod-b Fri Sep 22 12:39:19 UTC 2024 on example-pod-b Fri Sep 22 12:39:34 UTC 2024 on example-pod-b
Note: Standard file systems such as XFS and EXT4 can't be accessed at the same time from multiple servers. For more information, see Considerations and limitations.
Related information
Relevant content
- Accepted Answerasked 2 years agolg...
- asked 2 years agolg...
- asked 9 months agolg...
- asked 2 years agolg...
- Accepted Answerasked 2 years agolg...
- AWS OFFICIALUpdated 2 years ago
- AWS OFFICIALUpdated 2 months ago
- AWS OFFICIALUpdated 5 months ago