From dfa7b89006656dd26e4ec575422853b97a014905 Mon Sep 17 00:00:00 2001 From: Suraj Deshmukh Date: Wed, 9 Sep 2020 20:07:20 +0530 Subject: [PATCH] metallb: Add tests to verify conversion This commit adds tests that check if the specified component attributes in hcl configs are correctly converted to corresponding Kubernetes yaml objects. Signed-off-by: Suraj Deshmukh --- pkg/components/metallb/component_test.go | 199 +++++++++++++++++++++++ 1 file changed, 199 insertions(+) diff --git a/pkg/components/metallb/component_test.go b/pkg/components/metallb/component_test.go index 4a90c9b59..c0a68898e 100644 --- a/pkg/components/metallb/component_test.go +++ b/pkg/components/metallb/component_test.go @@ -15,9 +15,13 @@ package metallb import ( + "reflect" "testing" "github.com/hashicorp/hcl/v2" + appsv1 "k8s.io/api/apps/v1" + corev1 "k8s.io/api/core/v1" + "sigs.k8s.io/yaml" "github.com/kinvolk/lokomotive/pkg/components/util" ) @@ -103,3 +107,198 @@ component "metallb" { ` testRenderManifest(t, configHCL) } + +func getSpeakerDaemonset(t *testing.T, m map[string]string) *appsv1.DaemonSet { + dsStr, ok := m["daemonset-speaker.yaml"] + if !ok { + t.Fatalf("speaker daemonset config not found") + } + + ds := &appsv1.DaemonSet{} + if err := yaml.Unmarshal([]byte(dsStr), ds); err != nil { + t.Fatalf("failed unmarshaling manifest: %v", err) + } + + return ds +} + +// nolint:unused +func getDeployController(t *testing.T, m map[string]string) *appsv1.Deployment { + deployStr, ok := m["deployment-controller.yaml"] + if !ok { + t.Fatalf("controller deployment config not found") + } + + deploy := &appsv1.Deployment{} + if err := yaml.Unmarshal([]byte(deployStr), deploy); err != nil { + t.Fatalf("failed unmarshaling manifest: %v", err) + } + + return deploy +} + +// nolint:funlen +func TestConversion(t *testing.T) { + configHCL := ` +component "metallb" { + address_pools = { + default = ["1.1.1.1/32", "2.2.2.2/32"] + } + + speaker_toleration { + key = "speaker_key1" + operator = "Equal" + value = "value1" + } + + speaker_toleration { + key = "speaker_key2" + operator = "Equal" + value = "value2" + } + + speaker_node_selectors = { + "speaker_node_key1" = "speaker_node_value1" + "speaker_node_key2" = "speaker_node_value2" + } + + controller_toleration { + key = "controller_key1" + operator = "Equal" + value = "value1" + } + + controller_toleration { + key = "controller_key2" + operator = "Equal" + value = "value2" + } + + controller_node_selectors = { + "controller_node_key1" = "controller_node_value1" + "controller_node_key2" = "controller_node_value2" + } + + service_monitor = true +}` + + testRenderManifest(t, configHCL) + m := renderManifest(t, configHCL) + + tcs := []struct { + Name string + Test func(*testing.T, map[string]string) + }{ + { + "SpeakerConversions", func(t *testing.T, m map[string]string) { + ds := getSpeakerDaemonset(t, m) + expected := []corev1.Toleration{ + {Key: "speaker_key1", Operator: "Equal", Value: "value1"}, + {Key: "speaker_key2", Operator: "Equal", Value: "value2"}, + } + if !reflect.DeepEqual(expected, ds.Spec.Template.Spec.Tolerations) { + t.Fatalf("expected: %#v, got: %#v", expected, ds.Spec.Template.Spec.Tolerations) + } + }, + }, + { + "SpeakerNodeSelectors", func(t *testing.T, m map[string]string) { + ds := getSpeakerDaemonset(t, m) + expected := map[string]string{ + "beta.kubernetes.io/os": "linux", + "speaker_node_key1": "speaker_node_value1", + "speaker_node_key2": "speaker_node_value2", + } + if !reflect.DeepEqual(expected, ds.Spec.Template.Spec.NodeSelector) { + t.Fatalf("expected: %v, got: %v", expected, ds.Spec.Template.Spec.NodeSelector) + } + }, + }, + { + "ControllerTolerations", func(t *testing.T, m map[string]string) { + deploy := getDeployController(t, m) + expected := []corev1.Toleration{ + {Key: "controller_key1", Operator: "Equal", Value: "value1"}, + {Key: "controller_key2", Operator: "Equal", Value: "value2"}, + {Key: "node-role.kubernetes.io/master", Effect: "NoSchedule"}, + } + if !reflect.DeepEqual(expected, deploy.Spec.Template.Spec.Tolerations) { + t.Fatalf("expected: %+v\ngot: %+v", expected, deploy.Spec.Template.Spec.Tolerations) + } + }, + }, + { + "ControllerNodeSelectors", func(t *testing.T, m map[string]string) { + deploy := getDeployController(t, m) + expected := map[string]string{ + "beta.kubernetes.io/os": "linux", + "controller_node_key1": "controller_node_value1", + "controller_node_key2": "controller_node_value2", + "node.kubernetes.io/master": "", + } + if !reflect.DeepEqual(expected, deploy.Spec.Template.Spec.NodeSelector) { + t.Fatalf("expected: %v\ngot: %v", expected, deploy.Spec.Template.Spec.NodeSelector) + } + }, + }, + { + "MonitoringConfig", func(t *testing.T, m map[string]string) { + expectedConfig := []string{ + "service.yaml", + "service-monitor.yaml", + "grafana-dashboard.yaml", + "grafana-alertmanager-rule.yaml", + } + + for _, ec := range expectedConfig { + if _, ok := m[ec]; !ok { + t.Fatalf("expected %s to be generated but it is not available", ec) + } + } + }, + }, + { + "EIPConfig", func(t *testing.T, m map[string]string) { + expectedCM := `peer-autodiscovery: + from-labels: + my-asn: metallb.lokomotive.io/my-asn + peer-asn: metallb.lokomotive.io/peer-asn + peer-address: metallb.lokomotive.io/peer-address +address-pools: +- name: default + protocol: bgp + addresses: + - 1.1.1.1/32 + - 2.2.2.2/32 +` + + cmStr, ok := m["configmap.yaml"] + if !ok { + t.Fatalf("metallb configmap not found") + } + + cm := &corev1.ConfigMap{} + if err := yaml.Unmarshal([]byte(cmStr), cm); err != nil { + t.Fatalf("failed unmarshalling manifest: %v", err) + } + + gotCM, ok := cm.Data["config"] + if !ok { + t.Fatalf("metallb configmap is missing 'config' key") + } + + if gotCM != expectedCM { + t.Fatalf("expected: %s, got: %s", expectedCM, gotCM) + } + }, + }, + } + + for _, tc := range tcs { + tc := tc + t.Run(tc.Name, func(t *testing.T) { + t.Parallel() + tc.Test(t, m) + }) + } +}