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

Add basic tests for cloud shell #90

Merged
merged 11 commits into from
Jun 5, 2020
Merged
Show file tree
Hide file tree
Changes from 9 commits
Commits
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
2 changes: 2 additions & 0 deletions go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,8 @@ require (
)

require (
github.com/onsi/ginkgo v1.11.0
github.com/onsi/gomega v1.8.1
github.com/operator-framework/operator-sdk v0.17.0
github.com/spf13/pflag v1.0.5
k8s.io/api v0.17.4
Expand Down
112 changes: 112 additions & 0 deletions test/e2e/cmd/workspaces_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,112 @@
//
// Copyright (c) 2019-2020 Red Hat, Inc.
// This program and the accompanying materials are made
// available under the terms of the Eclipse Public License 2.0
// which is available at https://www.eclipse.org/legal/epl-2.0/
//
// SPDX-License-Identifier: EPL-2.0
//
// Contributors:
// Red Hat, Inc. - initial API and implementation
//

package cmd

import (
"fmt"
"path/filepath"
"testing"

"github.com/che-incubator/che-workspace-operator/test/e2e/pkg/config"
"github.com/che-incubator/che-workspace-operator/test/e2e/pkg/deploy"

corev1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"

"github.com/che-incubator/che-workspace-operator/test/e2e/pkg/client"
_ "github.com/che-incubator/che-workspace-operator/test/e2e/pkg/tests"
"github.com/onsi/ginkgo"
"github.com/onsi/ginkgo/reporters"
"github.com/onsi/gomega"
)

//Create Constant file
const (
testResultsDirectory = "/tmp/artifacts"
jUnitOutputFilename = "junit-workspaces-operator.xml"
)

//SynchronizedBeforeSuite blocks is executed before run all test suites
var _ = ginkgo.SynchronizedBeforeSuite(func() []byte {
//!TODO: Try to create a specific function to call all <ginkgo suite> configuration.
fmt.Println("Starting to setup objects before run ginkgo suite")
config.Namespace = "che-workspace-controller"

workspaces, err := client.NewK8sClient()
flacatus marked this conversation as resolved.
Show resolved Hide resolved
if err != nil {
fmt.Println("Failed to create workspace client")
flacatus marked this conversation as resolved.
Show resolved Hide resolved
}

ns := newNamespace()
ns, err = workspaces.Kube().CoreV1().Namespaces().Create(ns)

if err != nil {
panic(err)
}

controller := deploy.NewDeployment(workspaces)

if err := controller.CreateAllOperatorRoles(); err != nil {
_ = fmt.Errorf("Failed to create roles in clusters %s", err)
}

if err := controller.CreateOperatorClusterRole(); err != nil {
_ = fmt.Errorf("Failed to create roles in clusters %s", err)
flacatus marked this conversation as resolved.
Show resolved Hide resolved
}

if err := controller.CustomResourceDefinitions(); err != nil {
_ = fmt.Errorf("Failed to add custom resources definitions to cluster %s", err)
flacatus marked this conversation as resolved.
Show resolved Hide resolved
}

if err := controller.DeployWorkspacesController(); err != nil {
_ = fmt.Errorf("Failed to deploy workspace controller %s", err)
}

return nil
}, func(data []byte) {})

var _ = ginkgo.SynchronizedAfterSuite(func() {
workspaces, err := client.NewK8sClient()

if err != nil {
_ = fmt.Errorf("Failed to uninstall workspace controller %s", err)
}

if err = workspaces.Kube().CoreV1().Namespaces().Delete(config.Namespace, &metav1.DeleteOptions{}); err != nil {
_ = fmt.Errorf("Failed to uninstall workspace controller %s", err)
}
}, func() {})

func TestWorkspaceController(t *testing.T) {
gomega.RegisterFailHandler(ginkgo.Fail)

fmt.Println("Creating ginkgo reporter for Test Harness: Junit and Debug Detail reporter")
var r []ginkgo.Reporter
r = append(r, reporters.NewJUnitReporter(filepath.Join(testResultsDirectory, jUnitOutputFilename)))

fmt.Println("Running Workspace Controller e2e tests...")
ginkgo.RunSpecsWithDefaultAndCustomReporters(t, "Workspaces Controller Operator Tests", r)

}

func newNamespace() (ns *corev1.Namespace) {
flacatus marked this conversation as resolved.
Show resolved Hide resolved
return &corev1.Namespace{
TypeMeta: metav1.TypeMeta{
Kind: "Namespace",
APIVersion: corev1.SchemeGroupVersion.Version,
},
ObjectMeta: metav1.ObjectMeta{
Name: config.Namespace,
},
}
}
42 changes: 42 additions & 0 deletions test/e2e/pkg/client/client.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
//
// Copyright (c) 2019-2020 Red Hat, Inc.
// This program and the accompanying materials are made
// available under the terms of the Eclipse Public License 2.0
// which is available at https://www.eclipse.org/legal/epl-2.0/
//
// SPDX-License-Identifier: EPL-2.0
//
// Contributors:
// Red Hat, Inc. - initial API and implementation
//

package client

import (
"k8s.io/client-go/kubernetes"
"sigs.k8s.io/controller-runtime/pkg/client/config"
)

type K8sClient struct {
kubeClient *kubernetes.Clientset
}

// NewCodeReady creates C, a workspaces used to expose common testing functions.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
// NewCodeReady creates C, a workspaces used to expose common testing functions.
// NewK8sClient creates kubernetes client wrapper with helper functions and direct access to k8s go client

func NewK8sClient() (*K8sClient, error) {
cfg, err := config.GetConfig()
if err != nil {
return nil, err
}
client, err := kubernetes.NewForConfig(cfg)
if err != nil {
return nil, err
}

h := &K8sClient{kubeClient: client}
return h, nil
}

// Kube returns the clientset for Kubernetes upstream.
func (c *K8sClient) Kube() kubernetes.Interface {
return c.kubeClient
}
30 changes: 30 additions & 0 deletions test/e2e/pkg/client/oc.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
//
// Copyright (c) 2019-2020 Red Hat, Inc.
// This program and the accompanying materials are made
// available under the terms of the Eclipse Public License 2.0
// which is available at https://www.eclipse.org/legal/epl-2.0/
//
// SPDX-License-Identifier: EPL-2.0
//
// Contributors:
// Red Hat, Inc. - initial API and implementation
//

package client

import (
"fmt"
"github.com/che-incubator/che-workspace-operator/test/e2e/pkg/config"
"os/exec"
"strings"
)

func (w *K8sClient) OcApply(filePath string) (err error) {
cmd := exec.Command("oc", "apply", "--namespace", config.Namespace, "-f", filePath)
output, err := cmd.CombinedOutput()
fmt.Println(string(output))
if err != nil && !strings.Contains(string(output), "AlreadyExists") {
fmt.Println(err)
}
return err
}
101 changes: 101 additions & 0 deletions test/e2e/pkg/client/pod.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,101 @@
//
// Copyright (c) 2019-2020 Red Hat, Inc.
// This program and the accompanying materials are made
// available under the terms of the Eclipse Public License 2.0
// which is available at https://www.eclipse.org/legal/epl-2.0/
//
// SPDX-License-Identifier: EPL-2.0
//
// Contributors:
// Red Hat, Inc. - initial API and implementation
//

package client

import (
"errors"
"fmt"
"time"

"github.com/che-incubator/che-workspace-operator/test/e2e/pkg/config"

v1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/util/wait"
)

func (w *K8sClient) PodDeployWaitUtil(label string) (deployed bool, err error) {
timeout := time.After(15 * time.Minute)
tick := time.Tick(1 * time.Second)

for {
select {
case <-timeout:
return false, errors.New("timed out")
case <-tick:
desc := w.WaitForPodBySelectorRunning(config.Namespace, label, 180)
flacatus marked this conversation as resolved.
Show resolved Hide resolved
if desc == nil {
return true, nil
}
}
}
}

// return a condition function that indicates whether the given pod is
// currently running
func (w *K8sClient) isPodRunning(podName, namespace string) wait.ConditionFunc {
return func() (bool, error) {
pod, _ := w.Kube().CoreV1().Pods(namespace).Get(podName, metav1.GetOptions{})
age := time.Since(pod.GetCreationTimestamp().Time).Seconds()

switch pod.Status.Phase {
case v1.PodRunning:
fmt.Println("Pod started after", age, "seconds")
return true, nil
case v1.PodFailed, v1.PodSucceeded:
return false, nil
}
return false, nil
}
}

// Poll up to timeout seconds for pod to enter running state.
// Returns an error if the pod never enters the running state.
func (w *K8sClient) waitForPodRunning(namespace, podName string, timeout time.Duration) error {
return wait.PollImmediate(time.Second, timeout, w.isPodRunning(podName, namespace))
}

// Returns the list of currently scheduled or running pods in `namespace` with the given selector
func (w *K8sClient) ListPods(namespace, selector string) (*v1.PodList, error) {
listOptions := metav1.ListOptions{LabelSelector: selector}
podList, err := w.Kube().CoreV1().Pods(namespace).List(listOptions)

if err != nil {
return nil, err
}
return podList, nil
}

// Wait up to timeout seconds for all pods in 'namespace' with given 'selector' to enter running state.
// Returns an error if no pods are found or not all discovered pods enter running state.
func (w *K8sClient) WaitForPodBySelectorRunning(namespace, selector string, timeout int) error {
podList, err := w.ListPods(namespace, selector)

if err != nil {
return err
}
if len(podList.Items) == 0 {
fmt.Println("Pod not created yet with selector " + selector + " in namespace " + namespace)

return fmt.Errorf("Pod not created yet in %s with label %s", namespace, selector)
}

for _, pod := range podList.Items {
fmt.Println("Pod " + pod.Name + " created in namespace " + namespace + "...Checking startup data.")
if err := w.waitForPodRunning(namespace, pod.Name, time.Duration(timeout)*time.Second); err != nil {
return err
}
}

return nil
}
15 changes: 15 additions & 0 deletions test/e2e/pkg/config/config.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
//
// Copyright (c) 2019-2020 Red Hat, Inc.
// This program and the accompanying materials are made
// available under the terms of the Eclipse Public License 2.0
// which is available at https://www.eclipse.org/legal/epl-2.0/
//
// SPDX-License-Identifier: EPL-2.0
//
// Contributors:
// Red Hat, Inc. - initial API and implementation
//

package config

var Namespace string
67 changes: 67 additions & 0 deletions test/e2e/pkg/deploy/controller.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
//
// Copyright (c) 2019-2020 Red Hat, Inc.
// This program and the accompanying materials are made
// available under the terms of the Eclipse Public License 2.0
// which is available at https://www.eclipse.org/legal/epl-2.0/
//
// SPDX-License-Identifier: EPL-2.0
//
// Contributors:
// Red Hat, Inc. - initial API and implementation
//

package deploy

import (
"fmt"
"os/exec"
"strings"

"github.com/che-incubator/che-workspace-operator/test/e2e/pkg/config"
)

func (w *Deployment) DeployWorkspacesController() (err error) {
label := "app=che-workspace-controller"
cmd := exec.Command("oc", "apply", "--namespace", config.Namespace, "-f", "deploy/os")
output, err := cmd.CombinedOutput()
fmt.Println(string(output))
if err != nil && !strings.Contains(string(output), "AlreadyExists") {
fmt.Println(err)
return err
}

deploy, err := w.kubeClient.PodDeployWaitUtil(label)
if !deploy {
fmt.Println("Che Workspaces Controller not deployed")
return err
}
return err
}

func (w *Deployment) CreateAllOperatorRoles() (err error) {
cmd := exec.Command("oc", "apply", "--namespace", config.Namespace, "-f", "deploy")
output, err := cmd.CombinedOutput()
fmt.Println(string(output))
if err != nil && !strings.Contains(string(output), "AlreadyExists") {
flacatus marked this conversation as resolved.
Show resolved Hide resolved
fmt.Println(err)
}
return err
}

func (w *Deployment) CustomResourceDefinitions() (err error) {
cmd := exec.Command("oc", "apply", "-f", "deploy/crds")
output, err := cmd.CombinedOutput()
if err != nil && !strings.Contains(string(output), "AlreadyExists") {
fmt.Println(err)
}
return err
}

func (w *Deployment) CreateOperatorClusterRole() (err error) {
cmd := exec.Command("oc", "apply", "--namespace", config.Namespace, "-f", "deploy/role.yaml")
output, err := cmd.CombinedOutput()
if err != nil && !strings.Contains(string(output), "AlreadyExists") {
fmt.Println(err)
}
return err
}
Loading