mirror of
https://github.com/vmware-tanzu/velero.git
synced 2025-12-23 06:15:21 +00:00
Add volume types filter in resource policies (#6863)
Signed-off-by: Ming Qiu <mqiu@vmware.com>
This commit is contained in:
1
changelogs/unreleased/6863-qiuming-best
Normal file
1
changelogs/unreleased/6863-qiuming-best
Normal file
@@ -0,0 +1 @@
|
||||
Add volume types filter in resource policies
|
||||
@@ -85,6 +85,7 @@ func (p *Policies) buildPolicy(resPolicies *resourcePolicies) error {
|
||||
volP.conditions = append(volP.conditions, &storageClassCondition{storageClass: con.StorageClass})
|
||||
volP.conditions = append(volP.conditions, &nfsCondition{nfs: con.NFS})
|
||||
volP.conditions = append(volP.conditions, &csiCondition{csi: con.CSI})
|
||||
volP.conditions = append(volP.conditions, &volumeTypeCondition{volumeTypes: con.VolumeTypes})
|
||||
p.volumePolicies = append(p.volumePolicies, volP)
|
||||
}
|
||||
|
||||
|
||||
@@ -370,6 +370,51 @@ volumePolicies:
|
||||
},
|
||||
skip: false,
|
||||
},
|
||||
{
|
||||
name: "match volume by types",
|
||||
yamlData: `version: v1
|
||||
volumePolicies:
|
||||
- conditions:
|
||||
capacity: "0,100Gi"
|
||||
volumeTypes:
|
||||
- local
|
||||
- hostPath
|
||||
action:
|
||||
type: skip`,
|
||||
vol: &v1.PersistentVolume{
|
||||
Spec: v1.PersistentVolumeSpec{
|
||||
Capacity: v1.ResourceList{
|
||||
v1.ResourceStorage: resource.MustParse("1Gi"),
|
||||
},
|
||||
PersistentVolumeSource: v1.PersistentVolumeSource{
|
||||
HostPath: &v1.HostPathVolumeSource{Path: "/mnt/data"},
|
||||
},
|
||||
},
|
||||
},
|
||||
skip: true,
|
||||
},
|
||||
{
|
||||
name: "dismatch volume by types",
|
||||
yamlData: `version: v1
|
||||
volumePolicies:
|
||||
- conditions:
|
||||
capacity: "0,100Gi"
|
||||
volumeTypes:
|
||||
- local
|
||||
action:
|
||||
type: skip`,
|
||||
vol: &v1.PersistentVolume{
|
||||
Spec: v1.PersistentVolumeSpec{
|
||||
Capacity: v1.ResourceList{
|
||||
v1.ResourceStorage: resource.MustParse("1Gi"),
|
||||
},
|
||||
PersistentVolumeSource: v1.PersistentVolumeSource{
|
||||
HostPath: &v1.HostPathVolumeSource{Path: "/mnt/data"},
|
||||
},
|
||||
},
|
||||
},
|
||||
skip: false,
|
||||
},
|
||||
}
|
||||
for _, tc := range testCases {
|
||||
t.Run(tc.name, func(t *testing.T) {
|
||||
|
||||
@@ -47,6 +47,7 @@ type structuredVolume struct {
|
||||
storageClass string
|
||||
nfs *nFSVolumeSource
|
||||
csi *csiVolumeSource
|
||||
volumeType SupportedVolume
|
||||
}
|
||||
|
||||
func (s *structuredVolume) parsePV(pv *corev1api.PersistentVolume) {
|
||||
@@ -61,6 +62,8 @@ func (s *structuredVolume) parsePV(pv *corev1api.PersistentVolume) {
|
||||
if csi != nil {
|
||||
s.csi = &csiVolumeSource{Driver: csi.Driver}
|
||||
}
|
||||
|
||||
s.volumeType = getVolumeTypeFromPV(pv)
|
||||
}
|
||||
|
||||
func (s *structuredVolume) parsePodVolume(vol *corev1api.Volume) {
|
||||
@@ -73,6 +76,8 @@ func (s *structuredVolume) parsePodVolume(vol *corev1api.Volume) {
|
||||
if csi != nil {
|
||||
s.csi = &csiVolumeSource{Driver: csi.Driver}
|
||||
}
|
||||
|
||||
s.volumeType = getVolumeTypeFromVolume(vol)
|
||||
}
|
||||
|
||||
type capacityCondition struct {
|
||||
|
||||
@@ -38,10 +38,11 @@ type nFSVolumeSource struct {
|
||||
|
||||
// volumeConditions defined the current format of conditions we parsed
|
||||
type volumeConditions struct {
|
||||
Capacity string `yaml:"capacity,omitempty"`
|
||||
StorageClass []string `yaml:"storageClass,omitempty"`
|
||||
NFS *nFSVolumeSource `yaml:"nfs,omitempty"`
|
||||
CSI *csiVolumeSource `yaml:"csi,omitempty"`
|
||||
Capacity string `yaml:"capacity,omitempty"`
|
||||
StorageClass []string `yaml:"storageClass,omitempty"`
|
||||
NFS *nFSVolumeSource `yaml:"nfs,omitempty"`
|
||||
CSI *csiVolumeSource `yaml:"csi,omitempty"`
|
||||
VolumeTypes []SupportedVolume `yaml:"volumeTypes,omitempty"`
|
||||
}
|
||||
|
||||
func (c *capacityCondition) validate() error {
|
||||
|
||||
247
internal/resourcepolicies/volume_types_conditions.go
Normal file
247
internal/resourcepolicies/volume_types_conditions.go
Normal file
@@ -0,0 +1,247 @@
|
||||
/*
|
||||
Copyright 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 resourcepolicies
|
||||
|
||||
import (
|
||||
corev1api "k8s.io/api/core/v1"
|
||||
)
|
||||
|
||||
type volumeTypeCondition struct {
|
||||
volumeTypes []SupportedVolume
|
||||
}
|
||||
|
||||
type SupportedVolume string
|
||||
|
||||
const (
|
||||
AWSAzureDisk SupportedVolume = "awsAzureDisk"
|
||||
AWSElasticBlockStore SupportedVolume = "awsElasticBlockStore"
|
||||
AzureDisk SupportedVolume = "azureDisk"
|
||||
AzureFile SupportedVolume = "azureFile"
|
||||
Cinder SupportedVolume = "cinder"
|
||||
CephFS SupportedVolume = "cephfs"
|
||||
ConfigMap SupportedVolume = "configMap"
|
||||
CSI SupportedVolume = "csi"
|
||||
DownwardAPI SupportedVolume = "downwardAPI"
|
||||
EmptyDir SupportedVolume = "emptyDir"
|
||||
Ephemeral SupportedVolume = "ephemeral"
|
||||
FC SupportedVolume = "fc"
|
||||
Flocker SupportedVolume = "flocker"
|
||||
FlexVolume SupportedVolume = "flexVolume"
|
||||
GitRepo SupportedVolume = "gitRepo"
|
||||
Glusterfs SupportedVolume = "glusterfs"
|
||||
GCEPersistentDisk SupportedVolume = "gcePersistentDisk"
|
||||
HostPath SupportedVolume = "hostPath"
|
||||
ISCSI SupportedVolume = "iscsi"
|
||||
Local SupportedVolume = "local"
|
||||
NFS SupportedVolume = "nfs"
|
||||
PhotonPersistentDisk SupportedVolume = "photonPersistentDisk"
|
||||
PortworxVolume SupportedVolume = "portworxVolume"
|
||||
Projected SupportedVolume = "projected"
|
||||
Quobyte SupportedVolume = "quobyte"
|
||||
RBD SupportedVolume = "rbd"
|
||||
ScaleIO SupportedVolume = "scaleIO"
|
||||
Secret SupportedVolume = "secret"
|
||||
StorageOS SupportedVolume = "storageOS"
|
||||
VsphereVolume SupportedVolume = "vsphereVolume"
|
||||
)
|
||||
|
||||
func (v *volumeTypeCondition) match(s *structuredVolume) bool {
|
||||
if len(v.volumeTypes) == 0 {
|
||||
return true
|
||||
}
|
||||
|
||||
for _, vt := range v.volumeTypes {
|
||||
if vt == s.volumeType {
|
||||
return true
|
||||
}
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
func (v *volumeTypeCondition) validate() error {
|
||||
// validate by yamlv3
|
||||
return nil
|
||||
}
|
||||
|
||||
func getVolumeTypeFromPV(pv *corev1api.PersistentVolume) SupportedVolume {
|
||||
if pv == nil {
|
||||
return ""
|
||||
}
|
||||
|
||||
if pv.Spec.AWSElasticBlockStore != nil {
|
||||
return AWSElasticBlockStore
|
||||
}
|
||||
if pv.Spec.AzureDisk != nil {
|
||||
return AzureDisk
|
||||
}
|
||||
if pv.Spec.AzureFile != nil {
|
||||
return AzureFile
|
||||
}
|
||||
if pv.Spec.CephFS != nil {
|
||||
return CephFS
|
||||
}
|
||||
if pv.Spec.Cinder != nil {
|
||||
return Cinder
|
||||
}
|
||||
if pv.Spec.CSI != nil {
|
||||
return CSI
|
||||
}
|
||||
if pv.Spec.FC != nil {
|
||||
return FC
|
||||
}
|
||||
if pv.Spec.Flocker != nil {
|
||||
return Flocker
|
||||
}
|
||||
if pv.Spec.FlexVolume != nil {
|
||||
return FlexVolume
|
||||
}
|
||||
if pv.Spec.GCEPersistentDisk != nil {
|
||||
return GCEPersistentDisk
|
||||
}
|
||||
if pv.Spec.Glusterfs != nil {
|
||||
return Glusterfs
|
||||
}
|
||||
if pv.Spec.HostPath != nil {
|
||||
return HostPath
|
||||
}
|
||||
if pv.Spec.ISCSI != nil {
|
||||
return ISCSI
|
||||
}
|
||||
if pv.Spec.Local != nil {
|
||||
return Local
|
||||
}
|
||||
if pv.Spec.NFS != nil {
|
||||
return NFS
|
||||
}
|
||||
if pv.Spec.PhotonPersistentDisk != nil {
|
||||
return PhotonPersistentDisk
|
||||
}
|
||||
if pv.Spec.PortworxVolume != nil {
|
||||
return PortworxVolume
|
||||
}
|
||||
if pv.Spec.Quobyte != nil {
|
||||
return Quobyte
|
||||
}
|
||||
if pv.Spec.RBD != nil {
|
||||
return RBD
|
||||
}
|
||||
if pv.Spec.ScaleIO != nil {
|
||||
return ScaleIO
|
||||
}
|
||||
if pv.Spec.StorageOS != nil {
|
||||
return StorageOS
|
||||
}
|
||||
if pv.Spec.VsphereVolume != nil {
|
||||
return VsphereVolume
|
||||
}
|
||||
return ""
|
||||
}
|
||||
|
||||
func getVolumeTypeFromVolume(vol *corev1api.Volume) SupportedVolume {
|
||||
if vol == nil {
|
||||
return ""
|
||||
}
|
||||
|
||||
if vol.AWSElasticBlockStore != nil {
|
||||
return AWSElasticBlockStore
|
||||
}
|
||||
if vol.AzureDisk != nil {
|
||||
return AzureDisk
|
||||
}
|
||||
if vol.AzureFile != nil {
|
||||
return AzureFile
|
||||
}
|
||||
if vol.CephFS != nil {
|
||||
return CephFS
|
||||
}
|
||||
if vol.Cinder != nil {
|
||||
return Cinder
|
||||
}
|
||||
if vol.CSI != nil {
|
||||
return CSI
|
||||
}
|
||||
if vol.FC != nil {
|
||||
return FC
|
||||
}
|
||||
if vol.Flocker != nil {
|
||||
return Flocker
|
||||
}
|
||||
if vol.FlexVolume != nil {
|
||||
return FlexVolume
|
||||
}
|
||||
if vol.GCEPersistentDisk != nil {
|
||||
return GCEPersistentDisk
|
||||
}
|
||||
if vol.GitRepo != nil {
|
||||
return GitRepo
|
||||
}
|
||||
if vol.Glusterfs != nil {
|
||||
return Glusterfs
|
||||
}
|
||||
if vol.ISCSI != nil {
|
||||
return ISCSI
|
||||
}
|
||||
if vol.NFS != nil {
|
||||
return NFS
|
||||
}
|
||||
if vol.Secret != nil {
|
||||
return Secret
|
||||
}
|
||||
if vol.RBD != nil {
|
||||
return RBD
|
||||
}
|
||||
if vol.DownwardAPI != nil {
|
||||
return DownwardAPI
|
||||
}
|
||||
if vol.ConfigMap != nil {
|
||||
return ConfigMap
|
||||
}
|
||||
if vol.Projected != nil {
|
||||
return Projected
|
||||
}
|
||||
if vol.Ephemeral != nil {
|
||||
return Ephemeral
|
||||
}
|
||||
if vol.FC != nil {
|
||||
return FC
|
||||
}
|
||||
if vol.PhotonPersistentDisk != nil {
|
||||
return PhotonPersistentDisk
|
||||
}
|
||||
if vol.PortworxVolume != nil {
|
||||
return PortworxVolume
|
||||
}
|
||||
if vol.Quobyte != nil {
|
||||
return Quobyte
|
||||
}
|
||||
if vol.ScaleIO != nil {
|
||||
return ScaleIO
|
||||
}
|
||||
if vol.StorageOS != nil {
|
||||
return StorageOS
|
||||
}
|
||||
if vol.VsphereVolume != nil {
|
||||
return VsphereVolume
|
||||
}
|
||||
if vol.HostPath != nil {
|
||||
return HostPath
|
||||
}
|
||||
if vol.EmptyDir != nil {
|
||||
return EmptyDir
|
||||
}
|
||||
return ""
|
||||
}
|
||||
576
internal/resourcepolicies/volume_types_conditions_test.go
Normal file
576
internal/resourcepolicies/volume_types_conditions_test.go
Normal file
@@ -0,0 +1,576 @@
|
||||
/*
|
||||
Copyright 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 resourcepolicies
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
corev1api "k8s.io/api/core/v1"
|
||||
)
|
||||
|
||||
func TestGetVolumeTypeFromPV(t *testing.T) {
|
||||
testCases := []struct {
|
||||
name string
|
||||
inputPV *corev1api.PersistentVolume
|
||||
expected SupportedVolume
|
||||
}{
|
||||
{
|
||||
name: "nil PersistentVolume",
|
||||
inputPV: nil,
|
||||
expected: "",
|
||||
},
|
||||
{
|
||||
name: "Test GCEPersistentDisk",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
GCEPersistentDisk: &corev1api.GCEPersistentDiskVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: GCEPersistentDisk,
|
||||
},
|
||||
{
|
||||
name: "Test AWSElasticBlockStore",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
AWSElasticBlockStore: &corev1api.AWSElasticBlockStoreVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: AWSElasticBlockStore,
|
||||
},
|
||||
{
|
||||
name: "Test HostPath",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
HostPath: &corev1api.HostPathVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: HostPath,
|
||||
},
|
||||
{
|
||||
name: "Test Glusterfs",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
Glusterfs: &corev1api.GlusterfsPersistentVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: Glusterfs,
|
||||
},
|
||||
{
|
||||
name: "Test NFS",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
NFS: &corev1api.NFSVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: NFS,
|
||||
},
|
||||
{
|
||||
name: "Test RBD",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
RBD: &corev1api.RBDPersistentVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: RBD,
|
||||
},
|
||||
{
|
||||
name: "Test ISCSI",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
ISCSI: &corev1api.ISCSIPersistentVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: ISCSI,
|
||||
},
|
||||
{
|
||||
name: "Test Cinder",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
Cinder: &corev1api.CinderPersistentVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: Cinder,
|
||||
},
|
||||
{
|
||||
name: "Test CephFS",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
CephFS: &corev1api.CephFSPersistentVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: CephFS,
|
||||
},
|
||||
{
|
||||
name: "Test FC",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
FC: &corev1api.FCVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: FC,
|
||||
},
|
||||
{
|
||||
name: "Test Flocker",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
Flocker: &corev1api.FlockerVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: Flocker,
|
||||
},
|
||||
{
|
||||
name: "Test FlexVolume",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
FlexVolume: &corev1api.FlexPersistentVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: FlexVolume,
|
||||
},
|
||||
{
|
||||
name: "Test AzureFile",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
AzureFile: &corev1api.AzureFilePersistentVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: AzureFile,
|
||||
},
|
||||
{
|
||||
name: "Test VsphereVolume",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
VsphereVolume: &corev1api.VsphereVirtualDiskVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: VsphereVolume,
|
||||
},
|
||||
{
|
||||
name: "Test Quobyte",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
Quobyte: &corev1api.QuobyteVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: Quobyte,
|
||||
},
|
||||
{
|
||||
name: "Test AzureDisk",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
AzureDisk: &corev1api.AzureDiskVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: AzureDisk,
|
||||
},
|
||||
{
|
||||
name: "Test PhotonPersistentDisk",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
PhotonPersistentDisk: &corev1api.PhotonPersistentDiskVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: PhotonPersistentDisk,
|
||||
},
|
||||
{
|
||||
name: "Test PortworxVolume",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
PortworxVolume: &corev1api.PortworxVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: PortworxVolume,
|
||||
},
|
||||
{
|
||||
name: "Test ScaleIO",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
ScaleIO: &corev1api.ScaleIOPersistentVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: ScaleIO,
|
||||
},
|
||||
{
|
||||
name: "Test Local",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
Local: &corev1api.LocalVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: Local,
|
||||
},
|
||||
{
|
||||
name: "Test StorageOS",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
StorageOS: &corev1api.StorageOSPersistentVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: StorageOS,
|
||||
},
|
||||
{
|
||||
name: "Test CSI",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{
|
||||
PersistentVolumeSource: corev1api.PersistentVolumeSource{
|
||||
CSI: &corev1api.CSIPersistentVolumeSource{},
|
||||
},
|
||||
},
|
||||
},
|
||||
expected: CSI,
|
||||
},
|
||||
{
|
||||
name: "Test Unknown Source",
|
||||
inputPV: &corev1api.PersistentVolume{
|
||||
Spec: corev1api.PersistentVolumeSpec{},
|
||||
},
|
||||
expected: "",
|
||||
},
|
||||
}
|
||||
|
||||
for _, tc := range testCases {
|
||||
t.Run(tc.name, func(t *testing.T) {
|
||||
result := getVolumeTypeFromPV(tc.inputPV)
|
||||
if result != tc.expected {
|
||||
t.Errorf("Expected %s, but got %s", tc.expected, result)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func TestGetVolumeTypeFromVolume(t *testing.T) {
|
||||
testCases := []struct {
|
||||
name string
|
||||
inputVol *corev1api.Volume
|
||||
expected SupportedVolume
|
||||
}{
|
||||
{
|
||||
name: "nil Volume",
|
||||
inputVol: nil,
|
||||
expected: "",
|
||||
},
|
||||
{
|
||||
name: "Test Unknown Source",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{},
|
||||
},
|
||||
expected: "",
|
||||
},
|
||||
{
|
||||
name: "Test HostPath",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
HostPath: &corev1api.HostPathVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: HostPath,
|
||||
},
|
||||
{
|
||||
name: "Test EmptyDir",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
EmptyDir: &corev1api.EmptyDirVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: EmptyDir,
|
||||
},
|
||||
{
|
||||
name: "Test GCEPersistentDisk",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
GCEPersistentDisk: &corev1api.GCEPersistentDiskVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: GCEPersistentDisk,
|
||||
},
|
||||
{
|
||||
name: "Test AWSElasticBlockStore",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
AWSElasticBlockStore: &corev1api.AWSElasticBlockStoreVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: AWSElasticBlockStore,
|
||||
},
|
||||
{
|
||||
name: "Test GitRepo",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
GitRepo: &corev1api.GitRepoVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: GitRepo,
|
||||
},
|
||||
{
|
||||
name: "Test Secret",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
Secret: &corev1api.SecretVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: Secret,
|
||||
},
|
||||
{
|
||||
name: "Test NFS",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
NFS: &corev1api.NFSVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: NFS,
|
||||
},
|
||||
{
|
||||
name: "Test ISCSI",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
ISCSI: &corev1api.ISCSIVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: ISCSI,
|
||||
},
|
||||
{
|
||||
name: "Test Glusterfs",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
Glusterfs: &corev1api.GlusterfsVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: Glusterfs,
|
||||
},
|
||||
{
|
||||
name: "Test RBD",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
RBD: &corev1api.RBDVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: RBD,
|
||||
},
|
||||
{
|
||||
name: "Test FlexVolume",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
FlexVolume: &corev1api.FlexVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: FlexVolume,
|
||||
},
|
||||
{
|
||||
name: "Test Cinder",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
Cinder: &corev1api.CinderVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: Cinder,
|
||||
},
|
||||
{
|
||||
name: "Test CephFS",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
CephFS: &corev1api.CephFSVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: CephFS,
|
||||
},
|
||||
{
|
||||
name: "Test Flocker",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
Flocker: &corev1api.FlockerVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: Flocker,
|
||||
},
|
||||
{
|
||||
name: "Test DownwardAPI",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
DownwardAPI: &corev1api.DownwardAPIVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: DownwardAPI,
|
||||
},
|
||||
{
|
||||
name: "Test FC",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
FC: &corev1api.FCVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: FC,
|
||||
},
|
||||
{
|
||||
name: "Test AzureFile",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
AzureFile: &corev1api.AzureFileVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: AzureFile,
|
||||
},
|
||||
{
|
||||
name: "Test ConfigMap",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
ConfigMap: &corev1api.ConfigMapVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: ConfigMap,
|
||||
},
|
||||
{
|
||||
name: "Test VsphereVolume",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
VsphereVolume: &corev1api.VsphereVirtualDiskVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: VsphereVolume,
|
||||
},
|
||||
{
|
||||
name: "Test Quobyte",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
Quobyte: &corev1api.QuobyteVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: Quobyte,
|
||||
},
|
||||
{
|
||||
name: "Test AzureDisk",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
AzureDisk: &corev1api.AzureDiskVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: AzureDisk,
|
||||
},
|
||||
{
|
||||
name: "Test PhotonPersistentDisk",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
PhotonPersistentDisk: &corev1api.PhotonPersistentDiskVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: PhotonPersistentDisk,
|
||||
},
|
||||
{
|
||||
name: "Test Projected",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
Projected: &corev1api.ProjectedVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: Projected,
|
||||
},
|
||||
{
|
||||
name: "Test PortworxVolume",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
PortworxVolume: &corev1api.PortworxVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: PortworxVolume,
|
||||
},
|
||||
{
|
||||
name: "Test ScaleIO",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
ScaleIO: &corev1api.ScaleIOVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: ScaleIO,
|
||||
},
|
||||
{
|
||||
name: "Test StorageOS",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
StorageOS: &corev1api.StorageOSVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: StorageOS,
|
||||
},
|
||||
{
|
||||
name: "Test CSI",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
CSI: &corev1api.CSIVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: CSI,
|
||||
},
|
||||
{
|
||||
name: "Test Ephemeral",
|
||||
inputVol: &corev1api.Volume{
|
||||
VolumeSource: corev1api.VolumeSource{
|
||||
Ephemeral: &corev1api.EphemeralVolumeSource{},
|
||||
},
|
||||
},
|
||||
expected: Ephemeral,
|
||||
},
|
||||
}
|
||||
|
||||
for _, tc := range testCases {
|
||||
t.Run(tc.name, func(t *testing.T) {
|
||||
result := getVolumeTypeFromVolume(tc.inputVol)
|
||||
if result != tc.expected {
|
||||
t.Errorf("Expected %s, but got %s", tc.expected, result)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
@@ -262,8 +262,8 @@ Velero only support volume resource policies currently, other kinds of resource
|
||||
driver: aws.ebs.csi.driver
|
||||
# pv matches one of the storage class list
|
||||
storageClass:
|
||||
- gp2
|
||||
- standard
|
||||
- gp2
|
||||
- standard
|
||||
action:
|
||||
type: skip
|
||||
- conditions:
|
||||
@@ -289,6 +289,14 @@ Velero only support volume resource policies currently, other kinds of resource
|
||||
csi: {}
|
||||
action:
|
||||
type: skip
|
||||
- conditions:
|
||||
volumeTypes:
|
||||
- emptyDir
|
||||
- downwardAPI
|
||||
- configmap
|
||||
- cinder
|
||||
action:
|
||||
type: skip
|
||||
```
|
||||
|
||||
**Supported conditions**
|
||||
@@ -336,6 +344,19 @@ Velero supported conditions and format listed below:
|
||||
```
|
||||
For volume provisioned by [Persistent Volumes](https://kubernetes.io/docs/concepts/storage/persistent-volumes) support all above attributes, but for pod [Volume](https://kubernetes.io/docs/concepts/storage/volumes) only support filtered by volume source.
|
||||
|
||||
- volume types
|
||||
|
||||
Support filter volumes by types
|
||||
```yaml
|
||||
volumeTypes:
|
||||
# matches volumes listed below
|
||||
- emptyDir
|
||||
- downwardAPI
|
||||
- configmap
|
||||
- cinder
|
||||
```
|
||||
Volume types could be found in [Persistent Volumes](https://kubernetes.io/docs/concepts/storage/persistent-volumes) and pod [Volume](https://kubernetes.io/docs/concepts/storage/volumes)
|
||||
|
||||
**Resource policies rules**
|
||||
- Velero already has lots of include or exclude filters. the resource policies are the final filters after others include or exclude filters in one backup processing workflow. So if use a defined similar filter like the opt-in approach to backup one pod volume but skip backup of the same pod volume in resource policies, as resource policies are the final filters that are applied, the volume will not be backed up.
|
||||
- If volume resource policies conflict with themselves the first matched policy will be respected when many policies are defined.
|
||||
Reference in New Issue
Block a user