diff --git a/Dockerfile b/Dockerfile index 25b314a11..3ce8f6003 100644 --- a/Dockerfile +++ b/Dockerfile @@ -42,6 +42,8 @@ RUN mkdir -p /output/usr/bin && \ export GOARM=$( echo "${GOARM}" | cut -c2-) && \ go build -o /output/${BIN} \ -ldflags "${LDFLAGS}" ${PKG}/cmd/${BIN} && \ + go build -o /output/velero-restore-helper \ + -ldflags "${LDFLAGS}" ${PKG}/cmd/velero-restore-helper && \ go build -o /output/velero-helper \ -ldflags "${LDFLAGS}" ${PKG}/cmd/velero-helper && \ go clean -modcache -cache diff --git a/Makefile b/Makefile index 67e528bf8..2c68bd947 100644 --- a/Makefile +++ b/Makefile @@ -148,17 +148,14 @@ GOBIN=$$(pwd)/.go/bin # If you want to build all containers, see the 'all-containers' rule. all: @$(MAKE) build - @$(MAKE) build BIN=velero-restore-helper build-%: @$(MAKE) --no-print-directory ARCH=$* build - @$(MAKE) --no-print-directory ARCH=$* build BIN=velero-restore-helper all-build: $(addprefix build-, $(CLI_PLATFORMS)) all-containers: @$(MAKE) --no-print-directory container - @$(MAKE) --no-print-directory container BIN=velero-restore-helper local: build-dirs # Add DEBUG=1 to enable debug locally diff --git a/changelogs/unreleased/8574-ywk253100 b/changelogs/unreleased/8574-ywk253100 new file mode 100644 index 000000000..c6f3b7bc4 --- /dev/null +++ b/changelogs/unreleased/8574-ywk253100 @@ -0,0 +1 @@ +Merge restore helper image into Velero server image \ No newline at end of file diff --git a/internal/velero/images.go b/internal/velero/images.go index b21d4aff8..cefbd1441 100644 --- a/internal/velero/images.go +++ b/internal/velero/images.go @@ -43,9 +43,3 @@ func ImageTag() string { func DefaultVeleroImage() string { return fmt.Sprintf("%s/%s:%s", imageRegistry(), "velero", ImageTag()) } - -// DefaultRestoreHelperImage returns the default container image to use for the restore helper -// for this version of Velero. -func DefaultRestoreHelperImage() string { - return fmt.Sprintf("%s/%s:%s", imageRegistry(), "velero-restore-helper", ImageTag()) -} diff --git a/internal/velero/images_test.go b/internal/velero/images_test.go index 4e63dc543..5a68a4fd1 100644 --- a/internal/velero/images_test.go +++ b/internal/velero/images_test.go @@ -134,7 +134,3 @@ func testDefaultImage(t *testing.T, defaultImageFn func() string, imageName stri func TestDefaultVeleroImage(t *testing.T) { testDefaultImage(t, DefaultVeleroImage, "velero") } - -func TestDefaultRestoreHelperImage(t *testing.T) { - testDefaultImage(t, DefaultRestoreHelperImage, "velero-restore-helper") -} diff --git a/pkg/cmd/server/plugin/plugin.go b/pkg/cmd/server/plugin/plugin.go index 1c5b06132..b0f8aae3f 100644 --- a/pkg/cmd/server/plugin/plugin.go +++ b/pkg/cmd/server/plugin/plugin.go @@ -295,7 +295,7 @@ func newPodVolumeRestoreItemAction(f client.Factory) plugincommon.HandlerInitial return nil, err } - return ria.NewPodVolumeRestoreAction(logger, client.CoreV1().ConfigMaps(f.Namespace()), crClient), nil + return ria.NewPodVolumeRestoreAction(logger, client.CoreV1().ConfigMaps(f.Namespace()), crClient, f.Namespace()) } } diff --git a/pkg/restore/actions/pod_volume_restore_action.go b/pkg/restore/actions/pod_volume_restore_action.go index 71af336ce..1970f54fa 100644 --- a/pkg/restore/actions/pod_volume_restore_action.go +++ b/pkg/restore/actions/pod_volume_restore_action.go @@ -25,9 +25,11 @@ import ( "github.com/pkg/errors" "github.com/sirupsen/logrus" + appsv1 "k8s.io/api/apps/v1" corev1 "k8s.io/api/core/v1" "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured" "k8s.io/apimachinery/pkg/runtime" + "k8s.io/apimachinery/pkg/types" corev1client "k8s.io/client-go/kubernetes/typed/core/v1" ctrlclient "sigs.k8s.io/controller-runtime/pkg/client" @@ -40,6 +42,7 @@ import ( "github.com/vmware-tanzu/velero/pkg/podvolume" "github.com/vmware-tanzu/velero/pkg/restorehelper" "github.com/vmware-tanzu/velero/pkg/util/kube" + veleroutil "github.com/vmware-tanzu/velero/pkg/util/velero" ) const ( @@ -50,17 +53,24 @@ const ( ) type PodVolumeRestoreAction struct { - logger logrus.FieldLogger - client corev1client.ConfigMapInterface - crClient ctrlclient.Client + logger logrus.FieldLogger + client corev1client.ConfigMapInterface + crClient ctrlclient.Client + veleroImage string } -func NewPodVolumeRestoreAction(logger logrus.FieldLogger, client corev1client.ConfigMapInterface, crClient ctrlclient.Client) *PodVolumeRestoreAction { - return &PodVolumeRestoreAction{ - logger: logger, - client: client, - crClient: crClient, +func NewPodVolumeRestoreAction(logger logrus.FieldLogger, client corev1client.ConfigMapInterface, crClient ctrlclient.Client, namespace string) (*PodVolumeRestoreAction, error) { + deployment := &appsv1.Deployment{} + if err := crClient.Get(context.TODO(), types.NamespacedName{Name: "velero", Namespace: namespace}, deployment); err != nil { + return nil, err } + image := veleroutil.GetVeleroServerImage(deployment) + return &PodVolumeRestoreAction{ + logger: logger, + client: client, + crClient: crClient, + veleroImage: image, + }, nil } func (a *PodVolumeRestoreAction) AppliesTo() (velero.ResourceSelector, error) { @@ -117,7 +127,7 @@ func (a *PodVolumeRestoreAction) Execute(input *velero.RestoreItemActionExecuteI return nil, err } - image := getImage(log, config) + image := getImage(log, config, a.veleroImage) log.Infof("Using image %q", image) cpuRequest, memRequest := getResourceRequests(log, config) @@ -200,16 +210,16 @@ func getCommand(log logrus.FieldLogger, config *corev1.ConfigMap) []string { return []string{config.Data["command"]} } -func getImage(log logrus.FieldLogger, config *corev1.ConfigMap) string { +func getImage(log logrus.FieldLogger, config *corev1.ConfigMap, defaultImage string) string { if config == nil { log.Debug("No config found for plugin") - return veleroimage.DefaultRestoreHelperImage() + return defaultImage } image := config.Data["image"] if image == "" { log.Debugf("No custom image configured") - return veleroimage.DefaultRestoreHelperImage() + return defaultImage } log = log.WithField("image", image) @@ -217,7 +227,6 @@ func getImage(log logrus.FieldLogger, config *corev1.ConfigMap) string { parts := strings.Split(image, "/") if len(parts) == 1 { - defaultImage := veleroimage.DefaultRestoreHelperImage() // Image supplied without registry part log.Infof("Plugin config contains image name without registry name. Using default init container image: %q", defaultImage) return defaultImage diff --git a/pkg/restore/actions/pod_volume_restore_action_test.go b/pkg/restore/actions/pod_volume_restore_action_test.go index 5b836cba1..4327214e6 100644 --- a/pkg/restore/actions/pod_volume_restore_action_test.go +++ b/pkg/restore/actions/pod_volume_restore_action_test.go @@ -25,12 +25,13 @@ import ( "github.com/sirupsen/logrus" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" + appsv1 "k8s.io/api/apps/v1" corev1api "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured" "k8s.io/apimachinery/pkg/runtime" "k8s.io/client-go/kubernetes/fake" - veleroimage "github.com/vmware-tanzu/velero/internal/velero" velerov1api "github.com/vmware-tanzu/velero/pkg/apis/velero/v1" "github.com/vmware-tanzu/velero/pkg/builder" "github.com/vmware-tanzu/velero/pkg/buildinfo" @@ -48,7 +49,7 @@ func TestGetImage(t *testing.T) { } } - defaultImage := veleroimage.DefaultRestoreHelperImage() + defaultImage := "velero/velero:v1.0" tests := []struct { name string @@ -104,7 +105,7 @@ func TestGetImage(t *testing.T) { buildinfo.Version = originalVersion }() } - assert.Equal(t, test.want, getImage(velerotest.NewLogger(), test.configMap)) + assert.Equal(t, test.want, getImage(velerotest.NewLogger(), test.configMap, defaultImage)) }) } } @@ -134,7 +135,7 @@ func TestPodVolumeRestoreActionExecute(t *testing.T) { veleroNs = "velero" ) - defaultRestoreHelperImage := veleroimage.DefaultRestoreHelperImage() + defaultRestoreHelperImage := "velero/velero:v1.0" tests := []struct { name string @@ -265,10 +266,34 @@ func TestPodVolumeRestoreActionExecute(t *testing.T) { }, } + veleroDeployment := &appsv1.Deployment{ + TypeMeta: metav1.TypeMeta{ + APIVersion: appsv1.SchemeGroupVersion.String(), + Kind: "Deployment", + }, + ObjectMeta: metav1.ObjectMeta{ + Namespace: "velero", + Name: "velero", + }, + Spec: appsv1.DeploymentSpec{ + Template: corev1api.PodTemplateSpec{ + Spec: corev1api.PodSpec{ + Containers: []corev1api.Container{ + { + Image: "velero/velero:v1.0", + }, + }, + }, + }, + }, + } for _, tc := range tests { t.Run(tc.name, func(t *testing.T) { clientset := fake.NewSimpleClientset() - crClient := velerotest.NewFakeControllerRuntimeClient(t, tc.podVolumeBackups...) + + objects := []runtime.Object{veleroDeployment} + objects = append(objects, tc.podVolumeBackups...) + crClient := velerotest.NewFakeControllerRuntimeClient(t, objects...) unstructuredPod, err := runtime.DefaultUnstructuredConverter.ToUnstructured(tc.pod) require.NoError(t, err) @@ -295,11 +320,13 @@ func TestPodVolumeRestoreActionExecute(t *testing.T) { Result(), } - a := NewPodVolumeRestoreAction( + a, err := NewPodVolumeRestoreAction( logrus.StandardLogger(), clientset.CoreV1().ConfigMaps(veleroNs), crClient, + "velero", ) + require.NoError(t, err) // method under test res, err := a.Execute(input) diff --git a/site/content/docs/main/file-system-backup.md b/site/content/docs/main/file-system-backup.md index 71e3c2d9e..a0f28f0dc 100644 --- a/site/content/docs/main/file-system-backup.md +++ b/site/content/docs/main/file-system-backup.md @@ -356,8 +356,7 @@ with an infinite sleep) to mount these PVC/PV pairs prior taking a Velero backup ## Customize Restore Helper Container -Velero uses a helper init container when performing a FSB restore. By default, the image for this container is -`velero/velero-restore-helper:`, where `VERSION` matches the version/tag of the main Velero image. +Velero uses a helper init container when performing a FSB restore. By default, the image for this container is same with the Velero server container. You can customize the image that is used for this helper by creating a ConfigMap in the Velero namespace with the alternate image. In addition, you can customize the resource requirements for the init container, should you need.