package client import ( "bytes" "errors" "io" "net/http" "testing" "github.com/edgelesssys/constellation/bootstrapper/internal/kubernetes/k8sapi/resources" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" "go.uber.org/goleak" "google.golang.org/protobuf/proto" apps "k8s.io/api/apps/v1" k8s "k8s.io/api/core/v1" "k8s.io/apimachinery/pkg/api/meta" "k8s.io/apimachinery/pkg/api/meta/testrestmapper" v1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/runtime" "k8s.io/apimachinery/pkg/runtime/schema" "k8s.io/apimachinery/pkg/runtime/serializer/json" "k8s.io/cli-runtime/pkg/resource" "k8s.io/client-go/kubernetes/fake" "k8s.io/client-go/kubernetes/scheme" restfake "k8s.io/client-go/rest/fake" "k8s.io/client-go/restmapper" ) func TestMain(m *testing.M) { goleak.VerifyTestMain(m) } var ( corev1GV = schema.GroupVersion{Version: "v1"} nginxDeployment = &apps.Deployment{ TypeMeta: v1.TypeMeta{ APIVersion: "apps/v1", Kind: "Deployment", }, ObjectMeta: v1.ObjectMeta{ Labels: map[string]string{ "app": "nginx", }, Name: "my-nginx", }, Spec: apps.DeploymentSpec{ Replicas: proto.Int32(3), Selector: &v1.LabelSelector{ MatchLabels: map[string]string{ "app": "nginx", }, }, Template: k8s.PodTemplateSpec{ ObjectMeta: v1.ObjectMeta{ Labels: map[string]string{ "app": "nginx", }, }, Spec: k8s.PodSpec{ Containers: []k8s.Container{ { Name: "nginx", Image: "nginx:1.14.2", Ports: []k8s.ContainerPort{ { ContainerPort: 80, }, }, }, }, }, }, }, } nginxDeplJSON, _ = marshalJSON(nginxDeployment) nginxDeplYAML, _ = marshalYAML(nginxDeployment) ) type unmarshableResource struct{} func (*unmarshableResource) Marshal() ([]byte, error) { return nil, errors.New("someErr") } func stringBody(body string) io.ReadCloser { return io.NopCloser(bytes.NewReader([]byte(body))) } func fakeClientWith(t *testing.T, testName string, data map[string]string) resource.FakeClientFunc { return func(version schema.GroupVersion) (resource.RESTClient, error) { return &restfake.RESTClient{ GroupVersion: corev1GV, NegotiatedSerializer: scheme.Codecs.WithoutConversion(), Client: restfake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) { p := req.URL.Path q := req.URL.RawQuery if len(q) != 0 { p = p + "?" + q } body, ok := data[p] if !ok { t.Fatalf("%s: unexpected request: %s (%s)\n%#v", testName, p, req.URL, req) } header := http.Header{} header.Set("Content-Type", runtime.ContentTypeJSON) return &http.Response{ StatusCode: http.StatusOK, Header: header, Body: stringBody(body), }, nil }), }, nil } } func newClientWithFakes(t *testing.T, data map[string]string, objects ...runtime.Object) Client { clientset := fake.NewSimpleClientset(objects...) builder := resource.NewFakeBuilder( fakeClientWith(t, "", data), func() (meta.RESTMapper, error) { return testrestmapper.TestOnlyStaticRESTMapper(scheme.Scheme), nil }, func() (restmapper.CategoryExpander, error) { return resource.FakeCategoryExpander, nil }). Unstructured() client := Client{ clientset: clientset, builder: builder, } return client } func failingClient() resource.FakeClientFunc { return func(version schema.GroupVersion) (resource.RESTClient, error) { return &restfake.RESTClient{ GroupVersion: corev1GV, NegotiatedSerializer: scheme.Codecs.WithoutConversion(), Resp: &http.Response{StatusCode: 501}, }, nil } } func newFailingClient(objects ...runtime.Object) Client { clientset := fake.NewSimpleClientset(objects...) builder := resource.NewFakeBuilder( failingClient(), func() (meta.RESTMapper, error) { return testrestmapper.TestOnlyStaticRESTMapper(scheme.Scheme), nil }, func() (restmapper.CategoryExpander, error) { return resource.FakeCategoryExpander, nil }). WithScheme(scheme.Scheme, scheme.Scheme.PrioritizedVersionsAllGroups()...) client := Client{ clientset: clientset, builder: builder, } return client } func marshalJSON(obj runtime.Object) ([]byte, error) { serializer := json.NewSerializer(json.DefaultMetaFactory, nil, nil, false) var buf bytes.Buffer if err := serializer.Encode(obj, &buf); err != nil { return nil, err } return buf.Bytes(), nil } func marshalYAML(obj runtime.Object) ([]byte, error) { serializer := json.NewYAMLSerializer(json.DefaultMetaFactory, nil, nil) var buf bytes.Buffer if err := serializer.Encode(obj, &buf); err != nil { return nil, err } return buf.Bytes(), nil } func TestApplyOneObject(t *testing.T) { testCases := map[string]struct { httpResponseData map[string]string wantObj runtime.Object resourcesYAML string failingClient bool wantErr bool }{ "apply works": { httpResponseData: map[string]string{ "/deployments/my-nginx?fieldManager=constellation-bootstrapper&force=true": string(nginxDeplJSON), }, wantObj: nginxDeployment, resourcesYAML: string(nginxDeplYAML), wantErr: false, }, "apply fails": { httpResponseData: map[string]string{}, wantObj: nginxDeployment, resourcesYAML: string(nginxDeplYAML), failingClient: true, wantErr: true, }, } for name, tc := range testCases { t.Run(name, func(t *testing.T) { assert := assert.New(t) require := require.New(t) var client Client if tc.failingClient { client = newFailingClient(tc.wantObj) } else { client = newClientWithFakes(t, tc.httpResponseData, tc.wantObj) } reader := bytes.NewReader([]byte(tc.resourcesYAML)) res := client.builder. ContinueOnError(). Stream(reader, "yaml"). Flatten(). Do() assert.NoError(res.Err()) infos, err := res.Infos() assert.NoError(err) require.Len(infos, 1) err = client.ApplyOneObject(infos[0], true) if tc.wantErr { assert.Error(err) return } require.NoError(err) }) } } func TestGetObjects(t *testing.T) { testCases := map[string]struct { wantResources resources.Marshaler httpResponseData map[string]string resourcesYAML string wantErr bool }{ "GetObjects works on cluster-autoscaler deployment": { wantResources: resources.NewDefaultAutoscalerDeployment(nil, nil, nil), resourcesYAML: string(nginxDeplYAML), wantErr: false, }, "GetObjects works on cloud-controller-manager deployment": { wantResources: resources.NewDefaultCloudControllerManagerDeployment("someProvider", "someImage", "somePath", "someCIDR", nil, nil, nil, nil), resourcesYAML: string(nginxDeplYAML), wantErr: false, }, "GetObjects Marshal failure detected": { wantResources: &unmarshableResource{}, resourcesYAML: string(nginxDeplYAML), wantErr: true, }, } for name, tc := range testCases { t.Run(name, func(t *testing.T) { assert := assert.New(t) require := require.New(t) client := newClientWithFakes(t, tc.httpResponseData) infos, err := client.GetObjects(tc.wantResources) if tc.wantErr { assert.Error(err) return } require.NoError(err) assert.NotNil(infos) }) } }