Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow users to take GCS-based backups #302

Merged
merged 23 commits into from
Sep 27, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
8e6f69d
Allow users to take GCS-based backups
gerlowskija Aug 5, 2021
93371f1
Reject GCS backups for Solr < 8.9.0
gerlowskija Aug 10, 2021
999b687
Address initial review feedback
gerlowskija Aug 13, 2021
998a311
Review feedback, rd 2
gerlowskija Sep 13, 2021
5d85c94
Fix persistence-status updating bug
gerlowskija Sep 14, 2021
ad48018
Address lint error, rd 1
gerlowskija Sep 14, 2021
e871fc9
Add unit tests for backup helpers
gerlowskija Sep 15, 2021
696e5d3
Add YAML examples for various backup scenarios
gerlowskija Sep 15, 2021
290b720
Refresh README for solrbackups
gerlowskija Sep 16, 2021
84c3eda
Remove some personal debug logging
gerlowskija Sep 16, 2021
dfc805e
Fix lint errors, rd 2
gerlowskija Sep 16, 2021
97616ff
Change syntax of backupRepos - first pass
HoustonPutman Sep 23, 2021
1dab4ec
Fix formatting issues.
HoustonPutman Sep 23, 2021
9a24015
Default the legacy-backup-repo and move it to the new location
HoustonPutman Sep 23, 2021
bb99f9c
Add documentation for the new backup features and upgrade.
HoustonPutman Sep 23, 2021
65d7fcb
Add a few more tests. Add messages for all tests. Refactor locaiton o…
HoustonPutman Sep 24, 2021
4cfaf80
Add changelog entry. Add Backup examples in helm chart metadata
HoustonPutman Sep 24, 2021
2665aa4
Add more to the deprecation warning. Add new option in Helm chart.
HoustonPutman Sep 24, 2021
668d1c3
Merge branch 'main' into 301_gcs_backup_support
HoustonPutman Sep 24, 2021
411373c
Fix manifests
HoustonPutman Sep 24, 2021
ac5e79a
Fix default permissions for volumes & backupIsReady status
HoustonPutman Sep 27, 2021
2b29fd9
Rename example backup files
HoustonPutman Sep 27, 2021
39e386c
Formatting
HoustonPutman Sep 27, 2021
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 10 additions & 2 deletions api/v1beta1/solrbackup_types.go
Original file line number Diff line number Diff line change
Expand Up @@ -38,16 +38,24 @@ type SolrBackupSpec struct {
// A reference to the SolrCloud to create a backup for
SolrCloud string `json:"solrCloud"`

// The name of the repository to use for the backup. Defaults to "legacy_local_repository" if not specified (the
// auto-configured repository for legacy singleton volumes).
// +optional
RepositoryName string `json:"repositoryName,omitempty"`
gerlowskija marked this conversation as resolved.
Show resolved Hide resolved

// The list of collections to backup. If empty, all collections in the cloud will be backed up.
// +optional
Collections []string `json:"collections,omitempty"`

// Persistence is the specification on how to persist the backup data.
Persistence PersistenceSource `json:"persistence"`
// +optional
Persistence *PersistenceSource `json:"persistence,omitempty"`
}

func (spec *SolrBackupSpec) withDefaults(backupName string) (changed bool) {
changed = spec.Persistence.withDefaults(backupName) || changed
if spec.Persistence != nil {
changed = spec.Persistence.withDefaults(backupName) || changed
}

return changed
}
Expand Down
80 changes: 74 additions & 6 deletions api/v1beta1/solrcloud_types.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,15 +19,13 @@ package v1beta1

import (
"fmt"
"strconv"
"strings"

"k8s.io/apimachinery/pkg/util/intstr"

zk "github.com/pravega/zookeeper-operator/pkg/apis/zookeeper/v1beta1"
corev1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/api/resource"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/util/intstr"
"strconv"
"strings"
)

const (
Expand Down Expand Up @@ -55,6 +53,8 @@ const (
ZookeeperTechnologyLabel = "zookeeper"

DefaultBasicAuthUsername = "k8s-oper"

LegacyBackupRepositoryName = "legacy_local_repository"
)

// SolrCloudSpec defines the desired state of SolrCloud
Expand Down Expand Up @@ -119,6 +119,12 @@ type SolrCloudSpec struct {
// Options to enable Solr security
// +optional
SolrSecurity *SolrSecurityOptions `json:"solrSecurity,omitempty"`

// Allows specification of multiple different "repositories" for Solr to use when backing up data.
//+optional
//+listType:=map
//+listMapKey:=name
BackupRepositories []SolrBackupRepository `json:"backupRepositories,omitempty"`
}

func (spec *SolrCloudSpec) withDefaults() (changed bool) {
Expand Down Expand Up @@ -170,6 +176,19 @@ func (spec *SolrCloudSpec) withDefaults() (changed bool) {
}
changed = spec.BusyBoxImage.withDefaults(DefaultBusyBoxImageRepo, DefaultBusyBoxImageVersion, DefaultPullPolicy) || changed

// TODO: Deprecated in v0.5.0 - remove in v0.6.0
if spec.StorageOptions.BackupRestoreOptions != nil {
spec.BackupRepositories = append(spec.BackupRepositories, SolrBackupRepository{
Name: LegacyBackupRepositoryName,
Managed: &ManagedRepository{
Volume: spec.StorageOptions.BackupRestoreOptions.Volume,
Directory: spec.StorageOptions.BackupRestoreOptions.Directory,
},
})
spec.StorageOptions.BackupRestoreOptions = nil
changed = true
}

return changed
}

Expand Down Expand Up @@ -221,7 +240,9 @@ type SolrDataStorageOptions struct {
// +optional
EphemeralStorage *SolrEphemeralDataStorageOptions `json:"ephemeral,omitempty"`

// Options required for backups & restores to be enabled for this solrCloud.
// Options required for backups to be enabled for this solrCloud.
// Deprecated: Use a SolrBackupRepository with a ManagedRepository instead
// TODO: Remove in v0.6.0
// +optional
BackupRestoreOptions *SolrBackupRestoreOptions `json:"backupRestoreOptions,omitempty"`
}
Expand Down Expand Up @@ -329,12 +350,59 @@ type SolrEphemeralDataStorageOptions struct {
EmptyDir *corev1.EmptyDirVolumeSource `json:"emptyDir,omitempty"`
}

// Deprecated: Use a SolrBackupRepository with a ManagedRepository instead
type SolrBackupRestoreOptions struct {
// This is a volumeSource for a volume that will be mounted to all solrNodes to store backups and load restores.
// The data within the volume will be namespaces for this instance, so feel free to use the same volume for multiple clouds.
// Since the volume will be mounted to all solrNodes, it must be able to be written from multiple pods.
// If a PVC reference is given, the PVC must have `accessModes: - ReadWriteMany`.
// Other options are to use a NFS volume.
// Deprecated: Create an explicit 'managedRepositories' entry instead.
HoustonPutman marked this conversation as resolved.
Show resolved Hide resolved
Volume corev1.VolumeSource `json:"volume"`

// Select a custom directory name to mount the backup/restore data from the given volume.
// If not specified, then the name of the solrcloud will be used by default.
// Deprecated: Create an explicit 'managedRepositories' entry instead.
// +optional
Directory string `json:"directory,omitempty"`
}

//+kubebuilder:validation:MinProperties:=2
//+kubebuilder:validation:MaxProperties:=2
type SolrBackupRepository struct {
// A name used to identify this local storage profile. Values should follow RFC-1123. (See here for more details:
// https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#dns-label-names)
Name string `json:"name"`

// A GCSRepository for Solr to use when backing up and restoring collections.
//+optional
GCS *GcsRepository `json:"gcs,omitempty"`

// Allows specification of a "repository" for Solr to use when backing up data "locally".
// Repositories defined here are considered "managed" and can take advantage of special operator features, such as
// post-backup compression.
//+optional
Managed *ManagedRepository `json:"managed,omitempty"`
}

type GcsRepository struct {
// The name of the GCS bucket that all backup data will be stored in
Bucket string `json:"bucket"`

// The name & key of a Kubernetes secret holding a Google cloud service account key
GcsCredentialSecret corev1.SecretKeySelector `json:"gcsCredentialSecret"`

// An already-created chroot within the bucket to store data in. Defaults to the root path "/" if not specified.
// +optional
BaseLocation string `json:"baseLocation,omitempty"`
}

type ManagedRepository struct {
// This is a volumeSource for a volume that will be mounted to all solrNodes to store backups and load restores.
// The data within the volume will be namespaced for this instance, so feel free to use the same volume for multiple clouds.
// Since the volume will be mounted to all solrNodes, it must be able to be written from multiple pods.
// If a PVC reference is given, the PVC must have `accessModes: - ReadWriteMany`.
// Other options are to use a NFS volume.
Volume corev1.VolumeSource `json:"volume"`

// Select a custom directory name to mount the backup/restore data from the given volume.
Expand Down
105 changes: 105 additions & 0 deletions api/v1beta1/solrcloud_with_defaults_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,105 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 v1beta1

import (
"github.com/stretchr/testify/assert"
corev1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"testing"
)

func TestDeprecatedBackupRepo(t *testing.T) {
volume := corev1.VolumeSource{
HostPath: &corev1.HostPathVolumeSource{Path: "temp"},
}

directory := "/another/dir"

solrCloud := &SolrCloud{
ObjectMeta: metav1.ObjectMeta{Name: "foo", Namespace: "default"},
Spec: SolrCloudSpec{},
}

// Set defaults for SolrCloud
solrCloud.WithDefaults()

//No deprecated repository to move, no existing repos
assert.Emptyf(t, solrCloud.Spec.BackupRepositories, "No backup repositories should exist when defaulting a SolrCloud without any")
assert.Nilf(t, solrCloud.Spec.StorageOptions.BackupRestoreOptions, "No deprecated BackupRestoreOptions should exist when defaulting a SolrCloud that didn't start with one")

var solrCloudTest *SolrCloud
backupRepos := []SolrBackupRepository{
{
Name: "managedrepository1",
Managed: &ManagedRepository{
Volume: corev1.VolumeSource{},
},
},
{
Name: "gcsrepository1",
GCS: &GcsRepository{
Bucket: "some-bucket-name1",
GcsCredentialSecret: corev1.SecretKeySelector{
LocalObjectReference: corev1.LocalObjectReference{Name: "some-secret-name1"},
Key: "some-secret-key",
},
},
},
}

//No deprecated repository to move, 2 existing repos
solrCloudTest = solrCloud.DeepCopy()
solrCloudTest.Spec.BackupRepositories = backupRepos
solrCloudTest = solrCloudTest.DeepCopy()
assert.False(t, solrCloudTest.WithDefaults(), "WithDefaults() returned true when nothing should have been changed (no legacy repo)")
assert.ElementsMatch(t, backupRepos, solrCloudTest.Spec.BackupRepositories, "Backup repos modified when they should be exactly the same")

//Has deprecated repository to move, no existing repos
solrCloudTest = solrCloud.DeepCopy()
solrCloudTest.Spec.StorageOptions.BackupRestoreOptions = &SolrBackupRestoreOptions{
Volume: volume,
Directory: directory,
}
assert.True(t, solrCloudTest.WithDefaults(), "WithDefaults() returned false when the provided legacy repo should have been moved to the new location")
assert.Nil(t, solrCloudTest.Spec.StorageOptions.BackupRestoreOptions, "Legacy BackupRestore location should be nil after defaulting")
assert.Len(t, solrCloudTest.Spec.BackupRepositories, 1, "Backup repos should have 1 entry, the legacy backup repo")
assertLegacyBackupRepo(t, solrCloudTest.Spec.BackupRepositories[0], volume, directory)

//Has deprecated repository to move, 2 existing repos
solrCloudTest = solrCloud.DeepCopy()
solrCloudTest.Spec.StorageOptions.BackupRestoreOptions = &SolrBackupRestoreOptions{
Volume: volume,
Directory: directory,
}
solrCloudTest.Spec.BackupRepositories = backupRepos
solrCloudTest = solrCloudTest.DeepCopy()
assert.True(t, solrCloudTest.WithDefaults(), "WithDefaults() returned false when the provided legacy repo should have been moved to the new location")
assert.Nil(t, solrCloudTest.Spec.StorageOptions.BackupRestoreOptions, "Legacy BackupRestore location should be nil after defaulting")
assert.Len(t, solrCloudTest.Spec.BackupRepositories, 3, "Backup repos should have 3 entries, the legacy backup repo and the 2 provided by the user")
assert.EqualValues(t, backupRepos, solrCloudTest.Spec.BackupRepositories[0:2], "The pre-existing repos should not have been modified during the defaulting")
assertLegacyBackupRepo(t, solrCloudTest.Spec.BackupRepositories[2], volume, directory)
}

func assertLegacyBackupRepo(t *testing.T, repository SolrBackupRepository, volume corev1.VolumeSource, dir string) {
assert.Equal(t, LegacyBackupRepositoryName, repository.Name, "Wrong name for the legacy backup repo")
assert.Nil(t, repository.GCS, "Legacy backup repo should not have GCS specs")
assert.NotNil(t, repository.Managed, "Legacy backup repo must have Managed specs")
assert.EqualValuesf(t, volume, repository.Managed.Volume, "Volume incorrectly copied over for legacy backup repo")
assert.Equal(t, dir, repository.Managed.Directory, "Directory incorrectly copied over for legacy backup repo")
}
70 changes: 69 additions & 1 deletion api/v1beta1/zz_generated.deepcopy.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 3 additions & 1 deletion config/crd/bases/solr.apache.org_solrbackups.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -1048,11 +1048,13 @@ spec:
- source
type: object
type: object
repositoryName:
description: The name of the repository to use for the backup. Defaults to "legacy_local_repository" if not specified (the auto-configured repository for legacy singleton volumes).
type: string
solrCloud:
description: A reference to the SolrCloud to create a backup for
type: string
required:
- persistence
- solrCloud
type: object
status:
Expand Down
Loading