Skip to content

Commit

Permalink
require CSINode if topology enabled
Browse files Browse the repository at this point in the history
  • Loading branch information
msau42 committed Mar 15, 2019
1 parent ccd1dea commit 476d990
Show file tree
Hide file tree
Showing 4 changed files with 87 additions and 11 deletions.
57 changes: 57 additions & 0 deletions pkg/controller/controller_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -704,6 +704,13 @@ func createFakePVCWithVolumeMode(requestBytes int64, volumeMode v1.PersistentVol
return claim
}

// createFakePVCWithSelectedNode returns PVC with a selectedNode
func createFakePVCWithSelectedNode(requestBytes int64, selectedNode string) *v1.PersistentVolumeClaim {
claim := createFakePVC(requestBytes)
claim.Annotations = map[string]string{"volume.kubernetes.io/selected-node": selectedNode}
return claim
}

func TestGetSecretReference(t *testing.T) {
testcases := map[string]struct {
secretParams deprecatedSecretParamsMap
Expand Down Expand Up @@ -1880,6 +1887,56 @@ func TestProvisionWithTopology(t *testing.T) {
}
}

// TestProvisionWithTopologyDisabled checks that selected node is ignored and topology is not set on the PV
func TestProvisionWithTopologyDisabled(t *testing.T) {
defer utilfeaturetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.Topology, false)()

accessibleTopology := []*csi.Topology{
{
Segments: map[string]string{
"com.example.csi/zone": "zone1",
"com.example.csi/rack": "rack2",
},
},
}

const requestBytes = 100

tmpdir := tempDir(t)
defer os.RemoveAll(tmpdir)
mockController, driver, identityServer, controllerServer, csiConn, err := createMockServer(t, tmpdir)
if err != nil {
t.Fatal(err)
}
defer mockController.Finish()
defer driver.Stop()

clientSet := fakeclientset.NewSimpleClientset()
csiProvisioner := NewCSIProvisioner(clientSet, 5*time.Second, "test-provisioner", "test", 5, csiConn.conn, nil, driverName)

out := &csi.CreateVolumeResponse{
Volume: &csi.Volume{
CapacityBytes: requestBytes,
VolumeId: "test-volume-id",
AccessibleTopology: accessibleTopology,
},
}

provisionWithTopologyMockServerSetupExpectations(identityServer, controllerServer)
controllerServer.EXPECT().CreateVolume(gomock.Any(), gomock.Any()).Return(out, nil).Times(1)

pv, err := csiProvisioner.Provision(controller.VolumeOptions{
PVC: createFakePVCWithSelectedNode(requestBytes, "test-node"), // dummy PVC
})
if err != nil {
t.Errorf("got error from Provision call: %v", err)
}

if pv.Spec.NodeAffinity != nil {
t.Errorf("expected nil PV node affinity; got: %v", pv.Spec.NodeAffinity)
}
}

// TestProvisionWithMountOptions is a test of provisioner integration with mount options.
func TestProvisionWithMountOptions(t *testing.T) {
expectedOptions := []string{"foo=bar", "baz=qux"}
Expand Down
15 changes: 5 additions & 10 deletions pkg/controller/topology.go
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,6 @@ import (
storage "k8s.io/api/storage/v1beta1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/client-go/kubernetes"
"k8s.io/klog"
)

// topologyTerm represents a single term where its topology key value pairs are AND'd together.
Expand Down Expand Up @@ -108,7 +107,6 @@ func GenerateAccessibilityRequirements(
} else {
// selectedNode is set so use topology from that node to populate preferredTerms
// TODO (verult) reuse selected node info from aggregateTopologies
// TODO (verult) retry
nodeInfo, err := kubeClient.StorageV1beta1().CSINodes().Get(selectedNode.Name, metav1.GetOptions{})
if err != nil {
return nil, fmt.Errorf("error getting node info for selected node: %v", err)
Expand Down Expand Up @@ -141,12 +139,10 @@ func aggregateTopologies(

var topologyKeys []string
if selectedNode == nil {
// TODO (verult) retry
// TODO: use informers
nodeInfos, err := kubeClient.StorageV1beta1().CSINodes().List(metav1.ListOptions{})
if err != nil {
// We must support provisioning if CSINode is missing, for backward compatibility.
klog.Warningf("error listing CSINodes: %v; proceeding to provision without topology information", err)
return nil, nil
return nil, fmt.Errorf("error listing CSINodes: %v", err)
}

rand.Shuffle(len(nodeInfos.Items), func(i, j int) {
Expand All @@ -161,12 +157,10 @@ func aggregateTopologies(
}
}
} else {
// TODO (verult) retry
// TODO: use informers
selectedNodeInfo, err := kubeClient.StorageV1beta1().CSINodes().Get(selectedNode.Name, metav1.GetOptions{})
if err != nil {
// We must support provisioning if CSINode is missing, for backward compatibility.
klog.Warningf("error getting CSINode for selected node %q: %v; proceeding to provision without topology information", selectedNode.Name, err)
return nil, nil
return nil, fmt.Errorf("error getting CSINode for selected node %q: %v", selectedNode.Name, err)
}
topologyKeys = getTopologyKeys(selectedNodeInfo, driverName)
}
Expand All @@ -183,6 +177,7 @@ func aggregateTopologies(
if err != nil {
return nil, err
}
// TODO: use informers
nodes, err := kubeClient.CoreV1().Nodes().List(metav1.ListOptions{LabelSelector: selector})
if err != nil {
return nil, fmt.Errorf("error listing nodes: %v", err)
Expand Down
24 changes: 24 additions & 0 deletions pkg/controller/topology_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -918,6 +918,7 @@ func TestTopologyAggregation(t *testing.T) {
nodeLabels: []map[string]string{{}, {}, {}},
topologyKeys: nil,
expectedRequisite: nil,
expectError: true,
},
"random node: missing keys": {
nodeLabels: []map[string]string{{}, {}, {}},
Expand Down Expand Up @@ -1046,6 +1047,29 @@ func TestPreferredTopologies(t *testing.T) {
},
},
},
"allowedTopologies specified: no CSINode": {
allowedTopologies: []v1.TopologySelectorTerm{
{
MatchLabelExpressions: []v1.TopologySelectorLabelRequirement{
{
Key: "com.example.csi/zone",
Values: []string{"zone1", "zone2"},
},
{
Key: "com.example.csi/rack",
Values: []string{"rackA", "rackB"},
},
},
},
},
nodeLabels: []map[string]string{
{"com.example.csi/zone": "zone2", "com.example.csi/rack": "rackA"},
{"com.example.csi/zone": "zone1", "com.example.csi/rack": "rackA"},
{"com.example.csi/zone": "zone1", "com.example.csi/rack": "rackB"},
},
topologyKeys: nil,
expectError: true,
},
"topology aggregation": {
nodeLabels: []map[string]string{
{"com.example.csi/zone": "zone2", "com.example.csi/rack": "rackA"},
Expand Down
2 changes: 1 addition & 1 deletion pkg/features/features.go
Original file line number Diff line number Diff line change
Expand Up @@ -32,5 +32,5 @@ func init() {
// defaultKubernetesFeatureGates consists of all known feature keys specific to external-provisioner.
// To add a new feature, define a key for it above and add it here.
var defaultKubernetesFeatureGates = map[utilfeature.Feature]utilfeature.FeatureSpec{
Topology: {Default: true, PreRelease: utilfeature.Beta},
Topology: {Default: false, PreRelease: utilfeature.Beta},
}

0 comments on commit 476d990

Please sign in to comment.