2019-07-15 18:19:38 +00:00
|
|
|
/*
|
|
|
|
Copyright 2019 the Velero contributors.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package restore
|
|
|
|
|
|
|
|
import (
|
k8s 1.18 import (#2651)
* k8s 1.18 import wip
backup, cmd, controller, generated, restic, restore, serverstatusrequest, test and util
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* go mod tidy
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* add changelog file
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* go fmt
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* update code-generator and controller-gen in CI
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* checkout proper code-generator version, regen
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* fix remaining calls
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* regenerate CRDs with ./hack/update-generated-crd-code.sh
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* use existing context in restic and server
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* fix test cases by resetting resource version
also use main library go context, not golang.org/x/net/context, in pkg/restore/restore.go
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* clarify changelog message
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* use github.com/kubernetes-csi/external-snapshotter/v2@v2.2.0-rc1
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* run 'go mod tidy' to remove old external-snapshotter version
Signed-off-by: Andrew Lavery <laverya@umich.edu>
2020-07-16 16:21:37 +00:00
|
|
|
"context"
|
|
|
|
|
2019-07-15 18:19:38 +00:00
|
|
|
"github.com/pkg/errors"
|
|
|
|
"github.com/sirupsen/logrus"
|
2021-12-16 06:50:56 +00:00
|
|
|
appsv1 "k8s.io/api/apps/v1"
|
|
|
|
corev1 "k8s.io/api/core/v1"
|
2019-07-15 18:19:38 +00:00
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
|
|
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
|
2021-12-16 06:50:56 +00:00
|
|
|
"k8s.io/apimachinery/pkg/runtime"
|
2019-07-15 18:19:38 +00:00
|
|
|
corev1client "k8s.io/client-go/kubernetes/typed/core/v1"
|
|
|
|
storagev1client "k8s.io/client-go/kubernetes/typed/storage/v1"
|
|
|
|
|
2022-09-01 18:30:07 +00:00
|
|
|
"github.com/vmware-tanzu/velero/pkg/plugin/framework/common"
|
2019-09-30 21:26:56 +00:00
|
|
|
"github.com/vmware-tanzu/velero/pkg/plugin/velero"
|
2019-07-15 18:19:38 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// ChangeStorageClassAction updates a PV or PVC's storage class name
|
|
|
|
// if a mapping is found in the plugin's config map.
|
|
|
|
type ChangeStorageClassAction struct {
|
|
|
|
logger logrus.FieldLogger
|
|
|
|
configMapClient corev1client.ConfigMapInterface
|
|
|
|
storageClassClient storagev1client.StorageClassInterface
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewChangeStorageClassAction is the constructor for ChangeStorageClassAction.
|
|
|
|
func NewChangeStorageClassAction(
|
|
|
|
logger logrus.FieldLogger,
|
|
|
|
configMapClient corev1client.ConfigMapInterface,
|
|
|
|
storageClassClient storagev1client.StorageClassInterface,
|
|
|
|
) *ChangeStorageClassAction {
|
|
|
|
return &ChangeStorageClassAction{
|
|
|
|
logger: logger,
|
|
|
|
configMapClient: configMapClient,
|
|
|
|
storageClassClient: storageClassClient,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// AppliesTo returns the resources that ChangeStorageClassAction should
|
|
|
|
// be run for.
|
|
|
|
func (a *ChangeStorageClassAction) AppliesTo() (velero.ResourceSelector, error) {
|
|
|
|
return velero.ResourceSelector{
|
2021-12-16 06:50:56 +00:00
|
|
|
IncludedResources: []string{"persistentvolumeclaims", "persistentvolumes", "statefulsets"},
|
2019-07-15 18:19:38 +00:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Execute updates the item's spec.storageClassName if a mapping is found
|
|
|
|
// in the config map for the plugin.
|
2022-10-12 16:41:50 +00:00
|
|
|
func (a *ChangeStorageClassAction) Execute(input *velero.RestoreItemActionExecuteInput) (*velero.RestoreItemActionExecuteOutput, error) {
|
2019-07-15 18:19:38 +00:00
|
|
|
a.logger.Info("Executing ChangeStorageClassAction")
|
|
|
|
defer a.logger.Info("Done executing ChangeStorageClassAction")
|
|
|
|
|
|
|
|
a.logger.Debug("Getting plugin config")
|
2023-04-15 06:38:28 +00:00
|
|
|
config, err := common.GetPluginConfig(common.PluginKindRestoreItemAction, "velero.io/change-storage-class", a.configMapClient)
|
2019-07-15 18:19:38 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if config == nil || len(config.Data) == 0 {
|
|
|
|
a.logger.Debug("No storage class mappings found")
|
2022-10-12 16:41:50 +00:00
|
|
|
return velero.NewRestoreItemActionExecuteOutput(input.Item), nil
|
2019-07-15 18:19:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
obj, ok := input.Item.(*unstructured.Unstructured)
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.Errorf("object was of unexpected type %T", input.Item)
|
|
|
|
}
|
|
|
|
|
|
|
|
log := a.logger.WithFields(map[string]interface{}{
|
|
|
|
"kind": obj.GetKind(),
|
|
|
|
"namespace": obj.GetNamespace(),
|
|
|
|
"name": obj.GetName(),
|
|
|
|
})
|
|
|
|
|
2021-12-16 06:50:56 +00:00
|
|
|
// change StatefulSet volumeClaimTemplates storageClassName
|
|
|
|
if obj.GetKind() == "StatefulSet" {
|
|
|
|
sts := new(appsv1.StatefulSet)
|
|
|
|
if err := runtime.DefaultUnstructuredConverter.FromUnstructured(obj.UnstructuredContent(), sts); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(sts.Spec.VolumeClaimTemplates) > 0 {
|
|
|
|
for index, pvc := range sts.Spec.VolumeClaimTemplates {
|
2022-08-25 01:06:59 +00:00
|
|
|
exists, newStorageClass, err := a.isStorageClassExist(log, pvc.Spec.StorageClassName, config)
|
2021-12-16 06:50:56 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
} else if !exists {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Infof("Updating item's storage class name to %s", newStorageClass)
|
|
|
|
sts.Spec.VolumeClaimTemplates[index].Spec.StorageClassName = &newStorageClass
|
|
|
|
}
|
|
|
|
|
|
|
|
newObj, err := runtime.DefaultUnstructuredConverter.ToUnstructured(sts)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "convert obj to StatefulSet failed")
|
|
|
|
}
|
|
|
|
obj.Object = newObj
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// use the unstructured helpers here since this code is for both PVs and PVCs, and the
|
|
|
|
// field names are the same for both types.
|
|
|
|
storageClass, _, err := unstructured.NestedString(obj.UnstructuredContent(), "spec", "storageClassName")
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "error getting item's spec.storageClassName")
|
|
|
|
}
|
|
|
|
|
2022-08-25 01:06:59 +00:00
|
|
|
exists, newStorageClass, err := a.isStorageClassExist(log, &storageClass, config)
|
2021-12-16 06:50:56 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
} else if !exists {
|
2022-10-12 16:41:50 +00:00
|
|
|
return velero.NewRestoreItemActionExecuteOutput(input.Item), nil
|
2021-12-16 06:50:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
log.Infof("Updating item's storage class name to %s", newStorageClass)
|
|
|
|
|
|
|
|
if err := unstructured.SetNestedField(obj.UnstructuredContent(), newStorageClass, "spec", "storageClassName"); err != nil {
|
|
|
|
return nil, errors.Wrap(err, "unable to set item's spec.storageClassName")
|
|
|
|
}
|
2019-07-15 18:19:38 +00:00
|
|
|
}
|
2022-10-12 16:41:50 +00:00
|
|
|
return velero.NewRestoreItemActionExecuteOutput(obj), nil
|
2021-12-16 06:50:56 +00:00
|
|
|
}
|
|
|
|
|
2022-08-25 01:06:59 +00:00
|
|
|
func (a *ChangeStorageClassAction) isStorageClassExist(log *logrus.Entry, storageClass *string, cm *corev1.ConfigMap) (exists bool, newStorageClass string, err error) {
|
|
|
|
if storageClass == nil || *storageClass == "" {
|
2019-07-15 18:19:38 +00:00
|
|
|
log.Debug("Item has no storage class specified")
|
2021-12-16 06:50:56 +00:00
|
|
|
return false, "", nil
|
2019-07-15 18:19:38 +00:00
|
|
|
}
|
|
|
|
|
2022-08-25 01:06:59 +00:00
|
|
|
newStorageClass, ok := cm.Data[*storageClass]
|
2019-07-15 18:19:38 +00:00
|
|
|
if !ok {
|
2022-08-25 01:06:59 +00:00
|
|
|
log.Debugf("No mapping found for storage class %s", *storageClass)
|
2021-12-16 06:50:56 +00:00
|
|
|
return false, "", nil
|
2019-07-15 18:19:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// validate that new storage class exists
|
k8s 1.18 import (#2651)
* k8s 1.18 import wip
backup, cmd, controller, generated, restic, restore, serverstatusrequest, test and util
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* go mod tidy
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* add changelog file
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* go fmt
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* update code-generator and controller-gen in CI
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* checkout proper code-generator version, regen
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* fix remaining calls
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* regenerate CRDs with ./hack/update-generated-crd-code.sh
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* use existing context in restic and server
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* fix test cases by resetting resource version
also use main library go context, not golang.org/x/net/context, in pkg/restore/restore.go
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* clarify changelog message
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* use github.com/kubernetes-csi/external-snapshotter/v2@v2.2.0-rc1
Signed-off-by: Andrew Lavery <laverya@umich.edu>
* run 'go mod tidy' to remove old external-snapshotter version
Signed-off-by: Andrew Lavery <laverya@umich.edu>
2020-07-16 16:21:37 +00:00
|
|
|
if _, err := a.storageClassClient.Get(context.TODO(), newStorageClass, metav1.GetOptions{}); err != nil {
|
2021-12-16 06:50:56 +00:00
|
|
|
return false, "", errors.Wrapf(err, "error getting storage class %s from API", newStorageClass)
|
2019-07-15 18:19:38 +00:00
|
|
|
}
|
|
|
|
|
2021-12-16 06:50:56 +00:00
|
|
|
return true, newStorageClass, nil
|
2019-07-15 18:19:38 +00:00
|
|
|
}
|