Skip to content

Commit c8267c5

Browse files
committed
K8SPXC-1341: Resize PVCs on user request
1 parent d6fd6f3 commit c8267c5

File tree

2 files changed

+125
-0
lines changed

2 files changed

+125
-0
lines changed

pkg/controller/pxc/controller.go

+5
Original file line numberDiff line numberDiff line change
@@ -294,6 +294,11 @@ func (r *ReconcilePerconaXtraDBCluster) Reconcile(ctx context.Context, request r
294294
log.Info("failed to ensure version, running with default", "error", err)
295295
}
296296
}
297+
298+
err = r.reconcilePersistentVolumes(ctx, o)
299+
if err != nil {
300+
return reconcile.Result{}, errors.Wrap(err, "reconcile persistent volumes")
301+
}
297302

298303
err = r.deploy(ctx, o)
299304
if err != nil {

pkg/controller/pxc/volumes.go

+120
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,120 @@
1+
package pxc
2+
3+
import (
4+
"context"
5+
"strings"
6+
7+
"github.com/pkg/errors"
8+
corev1 "k8s.io/api/core/v1"
9+
storagev1 "k8s.io/api/storage/v1"
10+
k8serrors "k8s.io/apimachinery/pkg/api/errors"
11+
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
12+
"sigs.k8s.io/controller-runtime/pkg/client"
13+
logf "sigs.k8s.io/controller-runtime/pkg/log"
14+
15+
api "github.com/percona/percona-xtradb-cluster-operator/pkg/apis/pxc/v1"
16+
"github.com/percona/percona-xtradb-cluster-operator/pkg/pxc/app/statefulset"
17+
)
18+
19+
func (r *ReconcilePerconaXtraDBCluster) reconcilePersistentVolumes(ctx context.Context, cr *api.PerconaXtraDBCluster) error {
20+
log := logf.FromContext(ctx)
21+
22+
pxcSet := statefulset.NewNode(cr)
23+
sts := pxcSet.StatefulSet()
24+
25+
err := r.client.Get(ctx, client.ObjectKeyFromObject(sts), sts)
26+
if err != nil {
27+
if k8serrors.IsNotFound(err) {
28+
return nil
29+
}
30+
return errors.Wrapf(err, "get statefulset/%s", sts.Name)
31+
}
32+
33+
if cr.Spec.PXC.VolumeSpec.PersistentVolumeClaim == nil {
34+
return nil
35+
}
36+
37+
var volumeTemplate corev1.PersistentVolumeClaim
38+
for _, vct := range sts.Spec.VolumeClaimTemplates {
39+
if vct.Name == "datadir" {
40+
volumeTemplate = vct
41+
}
42+
}
43+
44+
requested := cr.Spec.PXC.VolumeSpec.PersistentVolumeClaim.Resources.Requests[corev1.ResourceStorage]
45+
actual := volumeTemplate.Spec.Resources.Requests[corev1.ResourceStorage]
46+
47+
if requested.Cmp(actual) < 0 {
48+
return errors.Wrap(err, "requested storage is less than actual")
49+
}
50+
51+
if requested.Cmp(actual) == 0 {
52+
return nil
53+
}
54+
55+
// Check storage class for allowVolumeExpansion field
56+
57+
labels := map[string]string{
58+
"app.kubernetes.io/component": "pxc",
59+
"app.kubernetes.io/instance": cr.Name,
60+
"app.kubernetes.io/managed-by": "percona-xtradb-cluster-operator",
61+
"app.kubernetes.io/name": "percona-xtradb-cluster",
62+
"app.kubernetes.io/part-of": "percona-xtradb-cluster",
63+
}
64+
65+
pvcList := corev1.PersistentVolumeClaimList{}
66+
if err := r.client.List(ctx, &pvcList, client.InNamespace(cr.Namespace), client.MatchingLabels(labels)); err != nil {
67+
return errors.Wrap(err, "list persistentvolumeclaims")
68+
}
69+
70+
pvcNames := make([]string, 0, len(pvcList.Items))
71+
for _, pvc := range pvcList.Items {
72+
pvcNames = append(pvcNames, pvc.Name)
73+
}
74+
75+
log.Info("Resizing PVCs", "requested", requested, "actual", actual, "pvcList", strings.Join(pvcNames, ","))
76+
77+
// check storage class for allowVolumeExpansion field
78+
// we check only the first PVC, because all PVCs should
79+
// have the same storage class since they're created from
80+
// the same template
81+
firstPVC := pvcList.Items[0]
82+
83+
if firstPVC.Spec.StorageClassName == nil {
84+
return errors.Errorf("storageClassName is not set in pvc/%s", firstPVC.Name)
85+
}
86+
87+
storageClass := &storagev1.StorageClass{
88+
ObjectMeta: metav1.ObjectMeta{
89+
Name: *firstPVC.Spec.StorageClassName,
90+
},
91+
}
92+
if err := r.client.Get(ctx, client.ObjectKeyFromObject(storageClass), storageClass); err != nil {
93+
return errors.Wrapf(err, "get storageclass/%s", storageClass.Name)
94+
}
95+
96+
if storageClass.AllowVolumeExpansion == nil || !*storageClass.AllowVolumeExpansion {
97+
return errors.Errorf("storageclass/%s does not allow volume expansion", storageClass.Name)
98+
}
99+
100+
log.Info("Deleting statefulset", "name", sts.Name)
101+
102+
if err := r.client.Delete(ctx, sts, client.PropagationPolicy("Orphan")); err != nil {
103+
return errors.Wrapf(err, "delete statefulset/%s", sts.Name)
104+
}
105+
106+
for _, pvc := range pvcList.Items {
107+
if !strings.HasPrefix(pvc.Name, "datadir-"+sts.Name) {
108+
continue
109+
}
110+
111+
log.Info("Resizing PVC", "name", pvc.Name)
112+
pvc.Spec.Resources.Requests[corev1.ResourceStorage] = requested
113+
114+
if err := r.client.Update(ctx, &pvc); err != nil {
115+
return errors.Wrapf(err, "update persistentvolumeclaim/%s", pvc.Name)
116+
}
117+
}
118+
119+
return nil
120+
}

0 commit comments

Comments
 (0)