21
test/e2e/constants.go
Normal file
21
test/e2e/constants.go
Normal file
@@ -0,0 +1,21 @@
|
||||
/*
|
||||
Copyright 2020 KubeSphere Authors
|
||||
|
||||
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 e2e
|
||||
|
||||
const (
|
||||
DefaultWorkspaceRoleAdmin = "%v-admin"
|
||||
)
|
||||
36
test/e2e/e2e.go
Normal file
36
test/e2e/e2e.go
Normal file
@@ -0,0 +1,36 @@
|
||||
/*
|
||||
Copyright 2020 KubeSphere Authors
|
||||
|
||||
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 e2e
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/onsi/ginkgo"
|
||||
"github.com/onsi/ginkgo/config"
|
||||
"github.com/onsi/gomega"
|
||||
"k8s.io/klog"
|
||||
"kubesphere.io/kubesphere/test/e2e/framework/ginkgowrapper"
|
||||
)
|
||||
|
||||
// RunE2ETests checks configuration parameters (specified through flags) and then runs
|
||||
// E2E tests using the Ginkgo runner.
|
||||
// This function is called on each Ginkgo node in parallel mode.
|
||||
func RunE2ETests(t *testing.T) {
|
||||
gomega.RegisterFailHandler(ginkgowrapper.Fail)
|
||||
klog.Infof("Starting e2e run on Ginkgo node %d", config.GinkgoConfig.ParallelNode)
|
||||
ginkgo.RunSpecs(t, "KubeSphere e2e suite")
|
||||
}
|
||||
@@ -1,56 +0,0 @@
|
||||
/*
|
||||
Copyright 2020 KubeSphere Authors
|
||||
|
||||
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 e2e_test
|
||||
|
||||
import (
|
||||
"flag"
|
||||
"os"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
. "github.com/onsi/ginkgo"
|
||||
. "github.com/onsi/gomega"
|
||||
"k8s.io/klog"
|
||||
"kubesphere.io/kubesphere/pkg/apis/network/v1alpha1"
|
||||
"kubesphere.io/kubesphere/pkg/test"
|
||||
)
|
||||
|
||||
var ctx *test.TestCtx
|
||||
|
||||
func TestE2e(t *testing.T) {
|
||||
RegisterFailHandler(Fail)
|
||||
RunSpecs(t, "Networking E2e Suite")
|
||||
}
|
||||
|
||||
var _ = BeforeSuite(func() {
|
||||
klog.InitFlags(nil)
|
||||
flag.Set("logtostderr", "false")
|
||||
flag.Set("alsologtostderr", "false")
|
||||
flag.Set("v", "4")
|
||||
flag.Parse()
|
||||
klog.SetOutput(GinkgoWriter)
|
||||
|
||||
ctx = test.NewTestCtx(nil, os.Getenv("TEST_NAMESPACE"))
|
||||
Expect(ctx.Setup(os.Getenv("YAML_PATH"), "", v1alpha1.AddToScheme)).ShouldNot(HaveOccurred())
|
||||
deployName := os.Getenv("DEPLOY_NAME")
|
||||
Expect(test.WaitForController(ctx.Client, ctx.Namespace, deployName, 1, time.Second*5, time.Minute)).ShouldNot(HaveOccurred(), "Controlller failed to start")
|
||||
klog.Infoln("Controller is up, begin to test ")
|
||||
})
|
||||
|
||||
var _ = AfterSuite(func() {
|
||||
ctx.Cleanup(nil)
|
||||
})
|
||||
@@ -14,160 +14,20 @@ See the License for the specific language governing permissions and
|
||||
limitations under the License.
|
||||
*/
|
||||
|
||||
package e2e_test
|
||||
package e2e
|
||||
|
||||
import (
|
||||
"context"
|
||||
"time"
|
||||
"os"
|
||||
"testing"
|
||||
|
||||
"k8s.io/klog"
|
||||
|
||||
. "github.com/onsi/ginkgo"
|
||||
. "github.com/onsi/gomega"
|
||||
appsv1 "k8s.io/api/apps/v1"
|
||||
batchv1 "k8s.io/api/batch/v1"
|
||||
corev1 "k8s.io/api/core/v1"
|
||||
"k8s.io/apimachinery/pkg/labels"
|
||||
"k8s.io/client-go/kubernetes/scheme"
|
||||
"kubesphere.io/kubesphere/pkg/apis/network/v1alpha1"
|
||||
"kubesphere.io/kubesphere/pkg/test"
|
||||
"sigs.k8s.io/controller-runtime/pkg/client"
|
||||
"kubesphere.io/kubesphere/test/e2e/framework"
|
||||
)
|
||||
|
||||
var simpleDeployYaml = `apiVersion: apps/v1
|
||||
kind: Deployment
|
||||
metadata:
|
||||
name: nginx
|
||||
namespace: production
|
||||
labels:
|
||||
name: nginx
|
||||
app: nginx
|
||||
spec:
|
||||
replicas: 1
|
||||
selector:
|
||||
matchLabels:
|
||||
app: nginx
|
||||
strategy:
|
||||
rollingUpdate:
|
||||
maxSurge: 1
|
||||
maxUnavailable: 1
|
||||
type: RollingUpdate
|
||||
template:
|
||||
metadata:
|
||||
labels:
|
||||
name: nginx
|
||||
app: nginx
|
||||
color : red
|
||||
spec:
|
||||
containers:
|
||||
- image: nginx:alpine
|
||||
name: nginx
|
||||
imagePullPolicy: IfNotPresent
|
||||
resources:
|
||||
requests:
|
||||
cpu: "20m"
|
||||
memory: "55M"
|
||||
env:
|
||||
- name: ENVVARNAME
|
||||
value: ENVVARVALUE
|
||||
ports:
|
||||
- containerPort: 80
|
||||
name: http
|
||||
restartPolicy: Always`
|
||||
func TestMain(m *testing.M) {
|
||||
framework.ParseFlags()
|
||||
os.Exit(m.Run())
|
||||
}
|
||||
|
||||
var simpleNPYaml = `apiVersion: network.kubesphere.io/v1alpha1
|
||||
kind: NamespaceNetworkPolicy
|
||||
metadata:
|
||||
name: allow-icmp-only
|
||||
namespace: production
|
||||
spec:
|
||||
selector: color == 'red'
|
||||
ingress:
|
||||
- action: Allow
|
||||
protocol: ICMP
|
||||
source:
|
||||
selector: color == 'blue'
|
||||
namespaceSelector: all()`
|
||||
|
||||
var simpleJobYaml = `apiVersion: batch/v1
|
||||
kind: Job
|
||||
metadata:
|
||||
name: test-connect
|
||||
namespace: production
|
||||
spec:
|
||||
template:
|
||||
metadata:
|
||||
labels:
|
||||
color : blue
|
||||
spec:
|
||||
containers:
|
||||
- name: test-connect
|
||||
image: alpine
|
||||
command: ["ping", "1.1.1.1"]
|
||||
restartPolicy: Never
|
||||
backoffLimit: 1`
|
||||
|
||||
var testNs = "production"
|
||||
var _ = Describe("E2e for network policy", func() {
|
||||
BeforeEach(func() {
|
||||
Expect(test.EnsureNamespace(ctx.Client, testNs)).ShouldNot(HaveOccurred())
|
||||
})
|
||||
|
||||
AfterEach(func() {
|
||||
Expect(test.DeleteNamespace(ctx.Client, testNs)).ShouldNot(HaveOccurred())
|
||||
ns := &corev1.Namespace{}
|
||||
ns.Name = testNs
|
||||
Expect(test.WaitForDeletion(ctx.Client, ns, time.Second*5, time.Minute)).ShouldNot(HaveOccurred())
|
||||
})
|
||||
|
||||
It("Should work well in simple namespaceNetworkPolicy", func() {
|
||||
decode := scheme.Codecs.UniversalDeserializer().Decode
|
||||
obj, _, err := decode([]byte(simpleDeployYaml), nil, nil)
|
||||
Expect(err).ShouldNot(HaveOccurred(), "Failed to parse yaml")
|
||||
deploy := obj.(*appsv1.Deployment)
|
||||
Expect(ctx.Client.Create(context.TODO(), obj)).ShouldNot(HaveOccurred())
|
||||
Expect(test.WaitForController(ctx.Client, deploy.Namespace, deploy.Name, *deploy.Spec.Replicas, time.Second*2, time.Minute)).ShouldNot(HaveOccurred())
|
||||
defer func() {
|
||||
Expect(ctx.Client.Delete(context.TODO(), deploy)).ShouldNot(HaveOccurred())
|
||||
}()
|
||||
obj, _, err = decode([]byte(simpleNPYaml), nil, nil)
|
||||
Expect(err).ShouldNot(HaveOccurred(), "Failed to parse networkpolicy yaml")
|
||||
np := obj.(*v1alpha1.NamespaceNetworkPolicy)
|
||||
Expect(ctx.Client.Create(context.TODO(), np)).ShouldNot(HaveOccurred())
|
||||
defer func() {
|
||||
Expect(ctx.Client.Delete(context.TODO(), np)).ShouldNot(HaveOccurred())
|
||||
Expect(test.WaitForDeletion(ctx.Client, np, time.Second*2, time.Minute)).ShouldNot(HaveOccurred())
|
||||
}()
|
||||
obj, _, err = decode([]byte(simpleJobYaml), nil, nil)
|
||||
Expect(err).ShouldNot(HaveOccurred(), "Failed to parse job yaml")
|
||||
|
||||
//create a job to test
|
||||
job := obj.(*batchv1.Job)
|
||||
selector, _ := labels.Parse("app=nginx")
|
||||
podlist := &corev1.PodList{}
|
||||
Expect(ctx.Client.List(context.TODO(), podlist, &client.ListOptions{
|
||||
Namespace: deploy.Namespace,
|
||||
LabelSelector: selector,
|
||||
})).ShouldNot(HaveOccurred())
|
||||
Expect(podlist.Items).To(HaveLen(int(*deploy.Spec.Replicas)))
|
||||
podip := podlist.Items[0].Status.PodIP
|
||||
job.Spec.Template.Spec.Containers[0].Command = []string{"ping", "-c", "4", podip}
|
||||
job.Spec.Template.Labels["color"] = "yellow"
|
||||
orginalJob := job.DeepCopy()
|
||||
Expect(ctx.Client.Create(context.TODO(), job)).ShouldNot(HaveOccurred())
|
||||
defer func() {
|
||||
Expect(ctx.Client.Delete(context.TODO(), job)).ShouldNot(HaveOccurred())
|
||||
}()
|
||||
klog.Infoln("sleep 10s to wait for controller creating np")
|
||||
time.Sleep(time.Second * 10)
|
||||
Expect(test.WaitForJobFail(ctx.Client, job.Namespace, job.Name, time.Second*3, time.Minute)).ShouldNot(HaveOccurred(), "Failed to block connection")
|
||||
|
||||
//change job color
|
||||
job = orginalJob.DeepCopy()
|
||||
Expect(ctx.Client.Delete(context.TODO(), job)).ShouldNot(HaveOccurred())
|
||||
Expect(test.WaitForDeletion(ctx.Client, job, time.Second*2, time.Minute)).ShouldNot(HaveOccurred())
|
||||
job.Spec.Template.Labels["color"] = "blue"
|
||||
Expect(ctx.Client.Create(context.TODO(), job)).ShouldNot(HaveOccurred())
|
||||
Expect(test.WaitForJobSucceed(ctx.Client, job.Namespace, job.Name, time.Second*3, time.Minute)).ShouldNot(HaveOccurred(), "Connection failed")
|
||||
})
|
||||
})
|
||||
func TestE2E(t *testing.T) {
|
||||
RunE2ETests(t)
|
||||
}
|
||||
|
||||
62
test/e2e/framework/expect.go
Normal file
62
test/e2e/framework/expect.go
Normal file
@@ -0,0 +1,62 @@
|
||||
/*
|
||||
Copyright 2020 KubeSphere Authors
|
||||
|
||||
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 framework
|
||||
|
||||
import (
|
||||
"github.com/onsi/gomega"
|
||||
)
|
||||
|
||||
// ExpectEqual expects the specified two are the same, otherwise an exception raises
|
||||
func ExpectEqual(actual interface{}, extra interface{}, explain ...interface{}) {
|
||||
gomega.ExpectWithOffset(1, actual).To(gomega.Equal(extra), explain...)
|
||||
}
|
||||
|
||||
// ExpectNotEqual expects the specified two are not the same, otherwise an exception raises
|
||||
func ExpectNotEqual(actual interface{}, extra interface{}, explain ...interface{}) {
|
||||
gomega.ExpectWithOffset(1, actual).NotTo(gomega.Equal(extra), explain...)
|
||||
}
|
||||
|
||||
// ExpectError expects an error happens, otherwise an exception raises
|
||||
func ExpectError(err error, explain ...interface{}) {
|
||||
gomega.ExpectWithOffset(1, err).To(gomega.HaveOccurred(), explain...)
|
||||
}
|
||||
|
||||
// ExpectNoError checks if "err" is set, and if so, fails assertion while logging the error.
|
||||
func ExpectNoError(err error, explain ...interface{}) {
|
||||
ExpectNoErrorWithOffset(1, err, explain...)
|
||||
}
|
||||
|
||||
// ExpectNoErrorWithOffset checks if "err" is set, and if so, fails assertion while logging the error at "offset" levels above its caller
|
||||
// (for example, for call chain f -> g -> ExpectNoErrorWithOffset(1, ...) error would be logged for "f").
|
||||
func ExpectNoErrorWithOffset(offset int, err error, explain ...interface{}) {
|
||||
gomega.ExpectWithOffset(1+offset, err).NotTo(gomega.HaveOccurred(), explain...)
|
||||
}
|
||||
|
||||
// ExpectConsistOf expects actual contains precisely the extra elements. The ordering of the elements does not matter.
|
||||
func ExpectConsistOf(actual interface{}, extra interface{}, explain ...interface{}) {
|
||||
gomega.ExpectWithOffset(1, actual).To(gomega.ConsistOf(extra), explain...)
|
||||
}
|
||||
|
||||
// ExpectHaveKey expects the actual map has the key in the keyset
|
||||
func ExpectHaveKey(actual interface{}, key interface{}, explain ...interface{}) {
|
||||
gomega.ExpectWithOffset(1, actual).To(gomega.HaveKey(key), explain...)
|
||||
}
|
||||
|
||||
// ExpectEmpty expects actual is empty
|
||||
func ExpectEmpty(actual interface{}, explain ...interface{}) {
|
||||
gomega.ExpectWithOffset(1, actual).To(gomega.BeEmpty(), explain...)
|
||||
}
|
||||
151
test/e2e/framework/framework.go
Normal file
151
test/e2e/framework/framework.go
Normal file
@@ -0,0 +1,151 @@
|
||||
/*
|
||||
Copyright 2020 KubeSphere Authors
|
||||
|
||||
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 framework
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
|
||||
"github.com/onsi/ginkgo" //nolint:stylecheck
|
||||
"github.com/onsi/gomega"
|
||||
corev1 "k8s.io/api/core/v1"
|
||||
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||
"k8s.io/apimachinery/pkg/runtime"
|
||||
"k8s.io/client-go/kubernetes/scheme"
|
||||
"k8s.io/client-go/rest"
|
||||
"kubesphere.io/client-go/client"
|
||||
"kubesphere.io/client-go/client/generic"
|
||||
"kubesphere.io/kubesphere/pkg/apis"
|
||||
"kubesphere.io/kubesphere/pkg/constants"
|
||||
"kubesphere.io/kubesphere/test/e2e/framework/workspace"
|
||||
)
|
||||
|
||||
type Framework struct {
|
||||
BaseName string
|
||||
Workspace string
|
||||
Namespaces []string
|
||||
Scheme *runtime.Scheme
|
||||
}
|
||||
|
||||
// KubeSphereFramework provides an interface to a test control plane so
|
||||
// that the implementation can vary without affecting tests.
|
||||
type KubeSphereFramework interface {
|
||||
GenericClient(userAgent string) client.Client
|
||||
KubeSphereSystemNamespace() string
|
||||
|
||||
// Name of the workspace for the current test to target
|
||||
TestWorkSpaceName() string
|
||||
|
||||
// Create a Namespace under current Worksapce
|
||||
CreateNamespace(name string) string
|
||||
// Get Names of the namespaces for the current test to target
|
||||
GetNamespaceNames() []string
|
||||
}
|
||||
|
||||
func NewKubeSphereFramework(baseName string) KubeSphereFramework {
|
||||
|
||||
sch := runtime.NewScheme()
|
||||
if err := apis.AddToScheme(sch); err != nil {
|
||||
Failf("unable add KubeSphere APIs to scheme: %v", err)
|
||||
}
|
||||
if err := scheme.AddToScheme(sch); err != nil {
|
||||
Failf("unable add Kubernetes APIs to scheme: %v", err)
|
||||
}
|
||||
f := &Framework{
|
||||
BaseName: baseName,
|
||||
Scheme: sch,
|
||||
}
|
||||
|
||||
ginkgo.AfterEach(f.AfterEach)
|
||||
ginkgo.BeforeEach(f.BeforeEach)
|
||||
return f
|
||||
}
|
||||
|
||||
// BeforeEach
|
||||
func (f *Framework) BeforeEach() {
|
||||
|
||||
}
|
||||
|
||||
// AfterEach
|
||||
func (f *Framework) AfterEach() {
|
||||
}
|
||||
|
||||
func (f *Framework) TestWorkSpaceName() string {
|
||||
if f.Workspace == "" {
|
||||
f.Workspace = CreateTestWorkSpace(f.GenericClient(f.BaseName), f.BaseName)
|
||||
}
|
||||
return f.Workspace
|
||||
}
|
||||
|
||||
func CreateTestWorkSpace(client client.Client, baseName string) string {
|
||||
ginkgo.By("Creating a WorkSpace to execute the test in")
|
||||
wspt := workspace.NewWorkspaceTemplate("", "admin")
|
||||
wspt.GenerateName = fmt.Sprintf("e2e-tests-%v-", baseName)
|
||||
wspt, err := workspace.CreateWorkspace(client, wspt)
|
||||
gomega.Expect(err).NotTo(gomega.HaveOccurred())
|
||||
ginkgo.By(fmt.Sprintf("Created test namespace %s", wspt.Name))
|
||||
return wspt.Name
|
||||
}
|
||||
|
||||
func (f *Framework) GetNamespaceNames() []string {
|
||||
return f.Namespaces
|
||||
}
|
||||
|
||||
func (f *Framework) CreateNamespace(name string) string {
|
||||
ns := &corev1.Namespace{
|
||||
ObjectMeta: metav1.ObjectMeta{
|
||||
Name: name,
|
||||
Labels: map[string]string{
|
||||
constants.WorkspaceLabelKey: f.TestWorkSpaceName(),
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
opts := &client.URLOptions{
|
||||
Group: "tenant.kubesphere.io",
|
||||
Version: "v1alpha2",
|
||||
}
|
||||
|
||||
err := f.GenericClient(f.BaseName).Create(context.TODO(), ns, opts, &client.WorkspaceOptions{Name: f.TestWorkSpaceName()})
|
||||
gomega.Expect(err).NotTo(gomega.HaveOccurred())
|
||||
|
||||
return ns.Name
|
||||
}
|
||||
|
||||
func (f *Framework) KubeSphereSystemNamespace() string {
|
||||
return "Kubesphere-system"
|
||||
}
|
||||
|
||||
func (f *Framework) GenericClient(userAgent string) client.Client {
|
||||
|
||||
ctx := TestContext
|
||||
config := &rest.Config{
|
||||
Host: "127.0.0.1:9090",
|
||||
Username: "admin",
|
||||
Password: "P@88w0rd",
|
||||
}
|
||||
if ctx.Host != "" {
|
||||
config = &rest.Config{
|
||||
Host: ctx.Host,
|
||||
Username: ctx.Username,
|
||||
Password: ctx.Password,
|
||||
}
|
||||
}
|
||||
rest.AddUserAgent(config, userAgent)
|
||||
|
||||
return generic.NewForConfigOrDie(config, client.Options{Scheme: f.Scheme})
|
||||
}
|
||||
96
test/e2e/framework/ginkgowrapper/wrapper.go
Normal file
96
test/e2e/framework/ginkgowrapper/wrapper.go
Normal file
@@ -0,0 +1,96 @@
|
||||
/*
|
||||
Copyright 2020 KubeSphere Authors
|
||||
|
||||
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 ginkgowrapper wraps Ginkgo Fail and Skip functions to panic
|
||||
// with structured data instead of a constant string.
|
||||
package ginkgowrapper
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"bytes"
|
||||
"regexp"
|
||||
"runtime"
|
||||
"runtime/debug"
|
||||
"strings"
|
||||
|
||||
"github.com/onsi/ginkgo"
|
||||
)
|
||||
|
||||
// FailurePanic is the value that will be panicked from Fail.
|
||||
type FailurePanic struct {
|
||||
Message string // The failure message passed to Fail
|
||||
Filename string // The filename that is the source of the failure
|
||||
Line int // The line number of the filename that is the source of the failure
|
||||
FullStackTrace string // A full stack trace starting at the source of the failure
|
||||
}
|
||||
|
||||
// String makes FailurePanic look like the old Ginkgo panic when printed.
|
||||
func (FailurePanic) String() string { return ginkgo.GINKGO_PANIC }
|
||||
|
||||
// Fail wraps ginkgo.Fail so that it panics with more useful
|
||||
// information about the failure. This function will panic with a
|
||||
// FailurePanic.
|
||||
func Fail(message string, callerSkip ...int) {
|
||||
skip := 1
|
||||
if len(callerSkip) > 0 {
|
||||
skip += callerSkip[0]
|
||||
}
|
||||
|
||||
_, file, line, _ := runtime.Caller(skip)
|
||||
fp := FailurePanic{
|
||||
Message: message,
|
||||
Filename: file,
|
||||
Line: line,
|
||||
FullStackTrace: pruneStack(skip),
|
||||
}
|
||||
|
||||
defer func() {
|
||||
e := recover()
|
||||
if e != nil {
|
||||
panic(fp)
|
||||
}
|
||||
}()
|
||||
|
||||
ginkgo.Fail(message, skip)
|
||||
}
|
||||
|
||||
// ginkgo adds a lot of test running infrastructure to the stack, so
|
||||
// we filter those out
|
||||
var stackSkipPattern = regexp.MustCompile(`onsi/ginkgo`)
|
||||
|
||||
func pruneStack(skip int) string {
|
||||
skip += 2 // one for pruneStack and one for debug.Stack
|
||||
stack := debug.Stack()
|
||||
scanner := bufio.NewScanner(bytes.NewBuffer(stack))
|
||||
var prunedStack []string
|
||||
|
||||
// skip the top of the stack
|
||||
for i := 0; i < 2*skip+1; i++ {
|
||||
scanner.Scan()
|
||||
}
|
||||
|
||||
for scanner.Scan() {
|
||||
if stackSkipPattern.Match(scanner.Bytes()) {
|
||||
scanner.Scan() // these come in pairs
|
||||
} else {
|
||||
prunedStack = append(prunedStack, scanner.Text())
|
||||
scanner.Scan() // these come in pairs
|
||||
prunedStack = append(prunedStack, scanner.Text())
|
||||
}
|
||||
}
|
||||
|
||||
return strings.Join(prunedStack, "\n")
|
||||
}
|
||||
93
test/e2e/framework/iam/fixtures.go
Normal file
93
test/e2e/framework/iam/fixtures.go
Normal file
@@ -0,0 +1,93 @@
|
||||
/*
|
||||
Copyright 2020 KubeSphere Authors
|
||||
|
||||
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 iam
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
|
||||
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||
iamv1alpha2 "kubesphere.io/kubesphere/pkg/apis/iam/v1alpha2"
|
||||
|
||||
"kubesphere.io/client-go/client"
|
||||
)
|
||||
|
||||
var URLOptions = client.URLOptions{
|
||||
Group: "iam.kubesphere.io",
|
||||
Version: "v1alpha2",
|
||||
}
|
||||
|
||||
// NewUser returns a User spec with the specified argument.
|
||||
func NewUser(name, globelRole string) *iamv1alpha2.User {
|
||||
return &iamv1alpha2.User{
|
||||
ObjectMeta: metav1.ObjectMeta{
|
||||
Name: name,
|
||||
Annotations: map[string]string{
|
||||
"iam.kubesphere.io/globalrole": globelRole,
|
||||
},
|
||||
},
|
||||
Spec: iamv1alpha2.UserSpec{
|
||||
Email: fmt.Sprint("%v@kubesphere.io", name),
|
||||
EncryptedPassword: "P@88w0rd",
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
// CreateUser uses c to create User. If the returned error is nil, the returned User is valid and has
|
||||
// been created.
|
||||
func CreateUser(c client.Client, u *iamv1alpha2.User) (*iamv1alpha2.User, error) {
|
||||
err := c.Create(context.TODO(), u, &URLOptions)
|
||||
return u, err
|
||||
}
|
||||
|
||||
// GetUser uses c to get the User by name. If the returned error is nil, the returned User is valid.
|
||||
func GetUser(c client.Client, name string) (*iamv1alpha2.User, error) {
|
||||
u := &iamv1alpha2.User{}
|
||||
|
||||
err := c.Get(context.TODO(), client.ObjectKey{Name: name}, u, &URLOptions)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return u, nil
|
||||
}
|
||||
|
||||
// NewGroup returns a Group spec with the specified argument.
|
||||
func NewGroup(name, workspace string) *iamv1alpha2.Group {
|
||||
return &iamv1alpha2.Group{
|
||||
ObjectMeta: metav1.ObjectMeta{
|
||||
GenerateName: name,
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
// CreateGroup uses c to create Group. If the returned error is nil, the returned Group is valid and has
|
||||
// been created.
|
||||
func CreateGroup(c client.Client, u *iamv1alpha2.Group, workspace string) (*iamv1alpha2.Group, error) {
|
||||
err := c.Create(context.TODO(), u, &URLOptions, &client.WorkspaceOptions{Name: workspace})
|
||||
return u, err
|
||||
}
|
||||
|
||||
// GetGroup uses c to get the User by name. If the returned error is nil, the returned User is valid.
|
||||
func GetGroup(c client.Client, name, workspace string) (*iamv1alpha2.Group, error) {
|
||||
u := &iamv1alpha2.Group{}
|
||||
|
||||
err := c.Get(context.TODO(), client.ObjectKey{Name: name}, u, &URLOptions, &client.WorkspaceOptions{Name: workspace})
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return u, nil
|
||||
}
|
||||
110
test/e2e/framework/log.go
Normal file
110
test/e2e/framework/log.go
Normal file
@@ -0,0 +1,110 @@
|
||||
/*
|
||||
Copyright 2020 KubeSphere Authors
|
||||
|
||||
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 framework
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"fmt"
|
||||
"regexp"
|
||||
"runtime/debug"
|
||||
"time"
|
||||
|
||||
"github.com/onsi/ginkgo"
|
||||
|
||||
"kubesphere.io/kubesphere/test/e2e/framework/ginkgowrapper"
|
||||
)
|
||||
|
||||
func nowStamp() string {
|
||||
return time.Now().Format(time.StampMilli)
|
||||
}
|
||||
|
||||
func log(level string, format string, args ...interface{}) {
|
||||
fmt.Fprintf(ginkgo.GinkgoWriter, nowStamp()+": "+level+": "+format+"\n", args...)
|
||||
}
|
||||
|
||||
// Logf logs the info.
|
||||
func Logf(format string, args ...interface{}) {
|
||||
log("INFO", format, args...)
|
||||
}
|
||||
|
||||
// Failf logs the fail info, including a stack trace.
|
||||
func Failf(format string, args ...interface{}) {
|
||||
FailfWithOffset(1, format, args...)
|
||||
}
|
||||
|
||||
// FailfWithOffset calls "Fail" and logs the error with a stack trace that starts at "offset" levels above its caller
|
||||
// (for example, for call chain f -> g -> FailfWithOffset(1, ...) error would be logged for "f").
|
||||
func FailfWithOffset(offset int, format string, args ...interface{}) {
|
||||
msg := fmt.Sprintf(format, args...)
|
||||
skip := offset + 1
|
||||
log("FAIL", "%s\n\nFull Stack Trace\n%s", msg, PrunedStack(skip))
|
||||
ginkgowrapper.Fail(nowStamp()+": "+msg, skip)
|
||||
}
|
||||
|
||||
// Fail is a replacement for ginkgo.Fail which logs the problem as it occurs
|
||||
// together with a stack trace and then calls ginkgowrapper.Fail.
|
||||
func Fail(msg string, callerSkip ...int) {
|
||||
skip := 1
|
||||
if len(callerSkip) > 0 {
|
||||
skip += callerSkip[0]
|
||||
}
|
||||
log("FAIL", "%s\n\nFull Stack Trace\n%s", msg, PrunedStack(skip))
|
||||
ginkgowrapper.Fail(nowStamp()+": "+msg, skip)
|
||||
}
|
||||
|
||||
var codeFilterRE = regexp.MustCompile(`/github.com/onsi/ginkgo/`)
|
||||
|
||||
// PrunedStack is a wrapper around debug.Stack() that removes information
|
||||
// about the current goroutine and optionally skips some of the initial stack entries.
|
||||
// With skip == 0, the returned stack will start with the caller of PruneStack.
|
||||
// From the remaining entries it automatically filters out useless ones like
|
||||
// entries coming from Ginkgo.
|
||||
//
|
||||
// This is a modified copy of PruneStack in https://github.com/onsi/ginkgo/blob/f90f37d87fa6b1dd9625e2b1e83c23ffae3de228/internal/codelocation/code_location.go#L25:
|
||||
// - simplified API and thus renamed (calls debug.Stack() instead of taking a parameter)
|
||||
// - source code filtering updated to be specific to Kubernetes
|
||||
// - optimized to use bytes and in-place slice filtering from
|
||||
// https://github.com/golang/go/wiki/SliceTricks#filter-in-place
|
||||
func PrunedStack(skip int) []byte {
|
||||
fullStackTrace := debug.Stack()
|
||||
stack := bytes.Split(fullStackTrace, []byte("\n"))
|
||||
// Ensure that the even entries are the method names and the
|
||||
// the odd entries the source code information.
|
||||
if len(stack) > 0 && bytes.HasPrefix(stack[0], []byte("goroutine ")) {
|
||||
// Ignore "goroutine 29 [running]:" line.
|
||||
stack = stack[1:]
|
||||
}
|
||||
// The "+2" is for skipping over:
|
||||
// - runtime/debug.Stack()
|
||||
// - PrunedStack()
|
||||
skip += 2
|
||||
if len(stack) > 2*skip {
|
||||
stack = stack[2*skip:]
|
||||
}
|
||||
n := 0
|
||||
for i := 0; i < len(stack)/2; i++ {
|
||||
// We filter out based on the source code file name.
|
||||
if !codeFilterRE.Match([]byte(stack[i*2+1])) {
|
||||
stack[n] = stack[i*2]
|
||||
stack[n+1] = stack[i*2+1]
|
||||
n += 2
|
||||
}
|
||||
}
|
||||
stack = stack[:n]
|
||||
|
||||
return bytes.Join(stack, []byte("\n"))
|
||||
}
|
||||
47
test/e2e/framework/test_context.go
Normal file
47
test/e2e/framework/test_context.go
Normal file
@@ -0,0 +1,47 @@
|
||||
/*
|
||||
Copyright 2020 KubeSphere Authors
|
||||
|
||||
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 framework
|
||||
|
||||
import (
|
||||
"flag"
|
||||
"os"
|
||||
)
|
||||
|
||||
type TestContextType struct {
|
||||
Host string
|
||||
InMemoryTest bool
|
||||
Username string
|
||||
Password string
|
||||
}
|
||||
|
||||
func registerFlags(t *TestContextType) {
|
||||
flag.BoolVar(&t.InMemoryTest, "in-memory-test", false,
|
||||
"Whether KubeSphere controllers and APIServer be started in memory.")
|
||||
flag.StringVar(&t.Host, "ks-apiserver", os.Getenv("KS_APISERVER"),
|
||||
"KubeSphere API Server IP/DNS")
|
||||
flag.StringVar(&t.Username, "username", os.Getenv("KS_USERNAME"),
|
||||
"Username to login to KubeSphere API Server")
|
||||
flag.StringVar(&t.Password, "password", os.Getenv("KS_PASSWORD"),
|
||||
"Password to login to KubeSphere API Server")
|
||||
}
|
||||
|
||||
var TestContext *TestContextType = &TestContextType{}
|
||||
|
||||
func ParseFlags() {
|
||||
registerFlags(TestContext)
|
||||
flag.Parse()
|
||||
}
|
||||
95
test/e2e/framework/workspace/fixtures.go
Normal file
95
test/e2e/framework/workspace/fixtures.go
Normal file
@@ -0,0 +1,95 @@
|
||||
/*
|
||||
Copyright 2020 KubeSphere Authors
|
||||
|
||||
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 workspace
|
||||
|
||||
import (
|
||||
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
||||
tenantv1alpha1 "kubesphere.io/kubesphere/pkg/apis/tenant/v1alpha1"
|
||||
tenantv1alpha2 "kubesphere.io/kubesphere/pkg/apis/tenant/v1alpha2"
|
||||
fedb1 "kubesphere.io/kubesphere/pkg/apis/types/v1beta1"
|
||||
|
||||
"context"
|
||||
|
||||
"kubesphere.io/client-go/client"
|
||||
)
|
||||
|
||||
var URLOptions = client.URLOptions{
|
||||
Group: "tenant.kubesphere.io",
|
||||
Version: "v1alpha2",
|
||||
}
|
||||
|
||||
// NewWorkspaceTemplate returns a WorkspaceTemplate spec with the specified argument.
|
||||
func NewWorkspaceTemplate(name string, manager string, hosts ...string) *tenantv1alpha2.WorkspaceTemplate {
|
||||
|
||||
clusters := []fedb1.GenericClusterReference{}
|
||||
|
||||
if hosts != nil {
|
||||
for _, h := range hosts {
|
||||
clusters = append(clusters, fedb1.GenericClusterReference{Name: h})
|
||||
}
|
||||
}
|
||||
|
||||
return &tenantv1alpha2.WorkspaceTemplate{
|
||||
ObjectMeta: metav1.ObjectMeta{
|
||||
Name: name,
|
||||
},
|
||||
Spec: fedb1.FederatedWorkspaceSpec{
|
||||
Placement: fedb1.GenericPlacementFields{
|
||||
Clusters: clusters,
|
||||
},
|
||||
Template: fedb1.WorkspaceTemplate{
|
||||
Spec: tenantv1alpha1.WorkspaceSpec{
|
||||
Manager: manager,
|
||||
},
|
||||
},
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
// CreateWorkSpace uses c to create Workspace. If the returned error is nil, the returned Workspace is valid and has
|
||||
// been created.
|
||||
func CreateWorkspace(c client.Client, w *tenantv1alpha2.WorkspaceTemplate) (*tenantv1alpha2.WorkspaceTemplate, error) {
|
||||
opts := &client.URLOptions{
|
||||
AbsPath: "kapis/tenant.kubesphere.io/v1alpha2/workspaces",
|
||||
}
|
||||
err := c.Create(context.TODO(), w, opts)
|
||||
return w, err
|
||||
}
|
||||
|
||||
// GetJob uses c to get the Workspace by name. If the returned error is nil, the returned Workspace is valid.
|
||||
func GetWorkspace(c client.Client, name string) (*tenantv1alpha1.Workspace, error) {
|
||||
wsp := &tenantv1alpha1.Workspace{}
|
||||
|
||||
err := c.Get(context.TODO(), client.ObjectKey{Name: name}, wsp, &URLOptions)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return wsp, nil
|
||||
}
|
||||
|
||||
// DeleteWorkspace uses c to delete the Workspace by name. If the returned error is nil, the returned Workspace is valid.
|
||||
func DeleteWorkspace(c client.Client, name string, opts ...client.DeleteOption) (*tenantv1alpha1.Workspace, error) {
|
||||
wsp := &tenantv1alpha1.Workspace{
|
||||
ObjectMeta: metav1.ObjectMeta{Name: name},
|
||||
}
|
||||
opts = append(opts, &URLOptions)
|
||||
err := c.Delete(context.TODO(), wsp, opts...)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return wsp, nil
|
||||
}
|
||||
50
test/e2e/groups.go
Normal file
50
test/e2e/groups.go
Normal file
@@ -0,0 +1,50 @@
|
||||
/*
|
||||
Copyright 2020 KubeSphere Authors
|
||||
|
||||
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 e2e
|
||||
|
||||
import (
|
||||
. "github.com/onsi/ginkgo" //nolint:stylecheck
|
||||
. "github.com/onsi/gomega" //nolint:stylecheck
|
||||
"kubesphere.io/client-go/client"
|
||||
"kubesphere.io/kubesphere/test/e2e/framework"
|
||||
"kubesphere.io/kubesphere/test/e2e/framework/iam"
|
||||
)
|
||||
|
||||
var _ = Describe("Worksspace", func() {
|
||||
f := framework.NewKubeSphereFramework("group")
|
||||
|
||||
var wsName = ""
|
||||
var gclient client.Client
|
||||
|
||||
BeforeEach(func() {
|
||||
gclient = f.GenericClient("group")
|
||||
})
|
||||
|
||||
It("Should create group successfully", func() {
|
||||
|
||||
By("Expecting to create workspace thronght workspace template")
|
||||
wsName = f.TestWorkSpaceName()
|
||||
|
||||
group, err := iam.CreateGroup(gclient, iam.NewGroup("group1", wsName), wsName)
|
||||
framework.ExpectNoError(err)
|
||||
Eventually(func() bool {
|
||||
expGroup, err := iam.GetGroup(gclient, group.Name, wsName)
|
||||
return err == nil && expGroup.Name == group.Name
|
||||
}, timeout, interval).Should(BeTrue())
|
||||
})
|
||||
|
||||
})
|
||||
61
test/e2e/quick.go
Normal file
61
test/e2e/quick.go
Normal file
@@ -0,0 +1,61 @@
|
||||
package e2e
|
||||
|
||||
import (
|
||||
"context"
|
||||
|
||||
. "github.com/onsi/ginkgo" //nolint:stylecheck
|
||||
. "github.com/onsi/gomega" //nolint:stylecheck
|
||||
appsv1 "k8s.io/api/apps/v1"
|
||||
corev1 "k8s.io/api/core/v1"
|
||||
"kubesphere.io/client-go/client"
|
||||
"kubesphere.io/kubesphere/test/e2e/framework"
|
||||
"kubesphere.io/kubesphere/test/e2e/framework/workspace"
|
||||
)
|
||||
|
||||
var _ = Describe("API Test", func() {
|
||||
|
||||
f := framework.NewKubeSphereFramework("worksspace")
|
||||
var gclient client.Client
|
||||
|
||||
BeforeEach(func() {
|
||||
gclient = f.GenericClient("worksspace")
|
||||
})
|
||||
|
||||
It("Should list Kubernetes objects through ks proxy", func() {
|
||||
results := &corev1.NamespaceList{}
|
||||
|
||||
Expect(gclient.List(context.TODO(), results)).Should(Succeed())
|
||||
|
||||
if len(results.Items) < 1 {
|
||||
framework.Failf("Test Failed caused no Cluster role was found")
|
||||
}
|
||||
framework.Logf("Response is %v", results)
|
||||
})
|
||||
|
||||
It("Should get Kubernetes objects through ks proxy", func() {
|
||||
deploy := &appsv1.Deployment{}
|
||||
|
||||
Expect(gclient.Get(context.TODO(), client.ObjectKey{Namespace: "kubesphere-system", Name: "ks-apiserver"}, deploy)).Should(Succeed())
|
||||
Expect(deploy.Name).To(Equal("ks-apiserver"))
|
||||
})
|
||||
|
||||
It("Should list Kubernetes objects through ksapi", func() {
|
||||
results := &corev1.NamespaceList{}
|
||||
|
||||
opts := workspace.URLOptions
|
||||
|
||||
err := gclient.List(context.TODO(), results, &opts)
|
||||
framework.ExpectNoError(err)
|
||||
framework.Logf("Response is %v", results)
|
||||
})
|
||||
|
||||
It("Should list Kubernetes objects through ksapi by workspaces", func() {
|
||||
results := &corev1.NamespaceList{}
|
||||
|
||||
opts := workspace.URLOptions
|
||||
|
||||
err := gclient.List(context.TODO(), results, &opts, &client.WorkspaceOptions{Name: "kubesphere-system"})
|
||||
framework.ExpectNoError(err)
|
||||
framework.Logf("Response is %v", results)
|
||||
})
|
||||
})
|
||||
72
test/e2e/workspaces.go
Normal file
72
test/e2e/workspaces.go
Normal file
@@ -0,0 +1,72 @@
|
||||
/*
|
||||
Copyright 2020 KubeSphere Authors
|
||||
|
||||
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 e2e
|
||||
|
||||
import (
|
||||
"context"
|
||||
"time"
|
||||
|
||||
. "github.com/onsi/ginkgo" //nolint:stylecheck
|
||||
. "github.com/onsi/gomega"
|
||||
"kubesphere.io/client-go/client"
|
||||
iamv1alpha2 "kubesphere.io/kubesphere/pkg/apis/iam/v1alpha2"
|
||||
"kubesphere.io/kubesphere/test/e2e/framework"
|
||||
"kubesphere.io/kubesphere/test/e2e/framework/iam"
|
||||
"kubesphere.io/kubesphere/test/e2e/framework/workspace"
|
||||
)
|
||||
|
||||
const timeout = time.Second * 30
|
||||
const interval = time.Second * 1
|
||||
|
||||
var _ = Describe("Worksspace", func() {
|
||||
|
||||
f := framework.NewKubeSphereFramework("worksspace")
|
||||
|
||||
var wsName = ""
|
||||
var gclient client.Client
|
||||
|
||||
BeforeEach(func() {
|
||||
gclient = f.GenericClient("worksspace")
|
||||
})
|
||||
|
||||
It("Should create workspace and initial workspace settings", func() {
|
||||
|
||||
By("Expecting to create workspace thronght workspace template")
|
||||
wsName = f.TestWorkSpaceName()
|
||||
|
||||
By("Expecting to create workspace successfully")
|
||||
Eventually(func() bool {
|
||||
wsp, err := workspace.GetWorkspace(gclient, wsName)
|
||||
return err == nil && wsp.Name == wsName
|
||||
}, timeout, interval).Should(BeTrue())
|
||||
|
||||
By("Expecting initial workspace roles successfully")
|
||||
|
||||
wspr := &iamv1alpha2.WorkspaceRoleList{}
|
||||
Eventually(func() bool {
|
||||
opts := iam.URLOptions
|
||||
|
||||
err := gclient.List(context.TODO(), wspr, &opts)
|
||||
return err == nil && len(wspr.Items) > 0
|
||||
}, timeout, interval).Should(BeTrue())
|
||||
|
||||
_, err := workspace.DeleteWorkspace(gclient, wsName)
|
||||
framework.ExpectNoError(err)
|
||||
|
||||
framework.Logf("Workspace created successfully")
|
||||
})
|
||||
})
|
||||
Reference in New Issue
Block a user