Skip to content

Commit

Permalink
Merge pull request #203 from replicatedhq/diamonwiggins/destination-a…
Browse files Browse the repository at this point in the history
…ccess-mode-annotation

Allow PVC AccessMode change during migration
  • Loading branch information
laverya authored Sep 5, 2023
2 parents c99ee09 + 0d1b4d1 commit 654c95d
Show file tree
Hide file tree
Showing 5 changed files with 490 additions and 14 deletions.
8 changes: 7 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -43,14 +43,20 @@ pvmigrate --source-sc "source" --dest-sc "destination" --preflight-validation-on
| --pod-ready-timeout | Integer | | 60 | length of time to wait (in seconds) for validation pod(s) to go into Ready phase |
| --delete-pv-timeout | Integer | | 300 | length of time to wait (in seconds) for backing PV to be removed when the temporary PVC is deleted |

## Annotations

`kurl.sh/pvcmigrate-destinationaccessmode` - Modifies the access mode of the PVC during migration. Valid options are - `[ReadWriteOnce, ReadWriteMany, ReadOnlyMany]`

## Process

In order, it:

1. Validates that both the `source` and `dest` StorageClasses exist
2. Finds PVs using the `source` StorageClass
3. Finds PVCs corresponding to the above PVs
4. Creates new PVCs for each existing PVC, but using the `dest` StorageClass
4. Creates new PVCs for each existing PVC
* Uses the `dest` StorageClass for the new PVCs
* Uses the access mode set in the annotation: `kurl.sh/pvcmigrate-destinationaccessmode` if specified on a source PVC
5. For each PVC:
* Finds all pods mounting the existing PVC
* Finds all StatefulSets and Deployments controlling those pods and adds an annotation with the original scale
Expand Down
47 changes: 39 additions & 8 deletions pkg/migrate/migrate.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,12 +20,13 @@ import (
)

const (
baseAnnotation = "kurl.sh/pvcmigrate"
scaleAnnotation = baseAnnotation + "-scale"
kindAnnotation = baseAnnotation + "-kind"
sourceNsAnnotation = baseAnnotation + "-sourcens"
sourcePVCAnnotation = baseAnnotation + "-sourcepvc"
desiredReclaimAnnotation = baseAnnotation + "-reclaim"
baseAnnotation = "kurl.sh/pvcmigrate"
scaleAnnotation = baseAnnotation + "-scale"
kindAnnotation = baseAnnotation + "-kind"
sourceNsAnnotation = baseAnnotation + "-sourcens"
sourcePVCAnnotation = baseAnnotation + "-sourcepvc"
desiredReclaimAnnotation = baseAnnotation + "-reclaim"
DesiredAccessModeAnnotation = baseAnnotation + "-destinationaccessmode"
)

// IsDefaultStorageClassAnnotation - this is also exported by /~https://github.com/kubernetes/kubernetes/blob/v1.21.3/pkg/apis/storage/v1/util/helpers.go#L25
Expand Down Expand Up @@ -511,6 +512,12 @@ func getPVCs(ctx context.Context, w *log.Logger, clientset k8sclient.Interface,
}
}

// Set destination access mode based on annotations
destAccessModes, err := GetDestAccessModes(*nsPvc.claim)
if err != nil {
return nil, nil, fmt.Errorf("failed to get destination access mode for PVC %s in %s: %w", nsPvc.claim.Name, ns, err)
}

// if it doesn't already exist, create it
newPVC, err := clientset.CoreV1().PersistentVolumeClaims(ns).Create(ctx, &corev1.PersistentVolumeClaim{
TypeMeta: nsPvc.claim.TypeMeta,
Expand All @@ -529,7 +536,7 @@ func getPVCs(ctx context.Context, w *log.Logger, clientset k8sclient.Interface,
corev1.ResourceStorage: desiredPvStorage,
},
},
AccessModes: nsPvc.claim.Spec.AccessModes,
AccessModes: destAccessModes,
},
}, metav1.CreateOptions{})
if err != nil {
Expand Down Expand Up @@ -1031,6 +1038,12 @@ func swapPVs(ctx context.Context, w *log.Logger, clientset k8sclient.Interface,
return fmt.Errorf("failed to remove claimrefs from PV %s: %w", migratedPVC.Spec.VolumeName, err)
}

// Set destination access mode based on annotations
destAccessModes, err := GetDestAccessModes(*originalPVC)
if err != nil {
return fmt.Errorf("failed to get destination access mode for PVC %s in %s: %w", originalPVC.Name, ns, err)
}

// create new PVC with the old name/annotations/settings, and the new PV
w.Printf("Creating new PVC %s with migrated-to PV %s\n", originalPVC.Name, migratedPVC.Spec.VolumeName)
newPVC := corev1.PersistentVolumeClaim{
Expand All @@ -1044,7 +1057,7 @@ func swapPVs(ctx context.Context, w *log.Logger, clientset k8sclient.Interface,
Labels: originalPVC.Labels, // copy labels, don't copy annotations
},
Spec: corev1.PersistentVolumeClaimSpec{
AccessModes: originalPVC.Spec.AccessModes,
AccessModes: destAccessModes,
Resources: originalPVC.Spec.Resources,
VolumeMode: originalPVC.Spec.VolumeMode,

Expand Down Expand Up @@ -1160,3 +1173,21 @@ func readLineWithTimeout(reader LineReader, timeout time.Duration) ([]byte, erro
return message.line, message.err
}
}

func GetDestAccessModes(srcPVC corev1.PersistentVolumeClaim) ([]corev1.PersistentVolumeAccessMode, error) {
// default to the source PVCs access mode if DesiredAccessModeAnnotation is not set
destAccessMode := srcPVC.Spec.AccessModes
if accessMode := srcPVC.Annotations[DesiredAccessModeAnnotation]; accessMode != "" {
switch accessMode {
case "ReadWriteOnce":
destAccessMode = []corev1.PersistentVolumeAccessMode{corev1.ReadWriteOnce}
case "ReadOnlyMany":
destAccessMode = []corev1.PersistentVolumeAccessMode{corev1.ReadOnlyMany}
case "ReadWriteMany":
destAccessMode = []corev1.PersistentVolumeAccessMode{corev1.ReadWriteMany}
default:
return nil, fmt.Errorf("invalid access mode '%s' used in annotation '%s' for PVC '%s' in namespace '%s'", accessMode, DesiredAccessModeAnnotation, srcPVC.Name, srcPVC.Namespace)
}
}
return destAccessMode, nil
}
Loading

0 comments on commit 654c95d

Please sign in to comment.