2021-01-29 19:22:43 +08:00
|
|
|
/*
|
|
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
|
|
* contributor license agreements. See the NOTICE file distributed with
|
|
|
|
* this work for additional information regarding copyright ownership.
|
|
|
|
* The ASF licenses this file to You 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 chaos
|
|
|
|
|
|
|
|
import (
|
2021-02-08 10:33:06 +08:00
|
|
|
"context"
|
2021-01-29 19:22:43 +08:00
|
|
|
"net/http"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2021-02-08 10:33:06 +08:00
|
|
|
"github.com/chaos-mesh/chaos-mesh/api/v1alpha1"
|
2021-01-29 19:22:43 +08:00
|
|
|
"github.com/gavv/httpexpect/v2"
|
|
|
|
. "github.com/onsi/gomega"
|
2021-02-08 10:33:06 +08:00
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
|
|
"sigs.k8s.io/controller-runtime/pkg/client"
|
2021-01-29 19:22:43 +08:00
|
|
|
)
|
|
|
|
|
2021-02-08 10:33:06 +08:00
|
|
|
func createEtcdKillChaos(g *WithT, cli client.Client) {
|
|
|
|
chaos := &v1alpha1.PodChaos{
|
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
|
|
Name: "kill-etcd",
|
|
|
|
Namespace: metav1.NamespaceDefault,
|
|
|
|
},
|
|
|
|
Spec: v1alpha1.PodChaosSpec{
|
|
|
|
Selector: v1alpha1.SelectorSpec{
|
|
|
|
LabelSelectors: map[string]string{"app": "etcd"},
|
2021-01-29 19:22:43 +08:00
|
|
|
},
|
2021-02-08 10:33:06 +08:00
|
|
|
Action: v1alpha1.PodKillAction,
|
|
|
|
Mode: v1alpha1.AllPodMode,
|
|
|
|
Scheduler: &v1alpha1.SchedulerSpec{
|
|
|
|
Cron: "@every 10m",
|
|
|
|
},
|
|
|
|
},
|
2021-01-29 19:22:43 +08:00
|
|
|
}
|
|
|
|
|
2021-02-08 10:33:06 +08:00
|
|
|
err := cli.Create(context.Background(), chaos)
|
2021-01-29 19:22:43 +08:00
|
|
|
g.Expect(err).To(BeNil())
|
2021-02-03 09:53:23 +08:00
|
|
|
}
|
|
|
|
|
2021-01-29 19:22:43 +08:00
|
|
|
func TestGetSuccessWhenEtcdKilled(t *testing.T) {
|
|
|
|
g := NewWithT(t)
|
|
|
|
e := httpexpect.New(t, host)
|
2021-02-08 10:33:06 +08:00
|
|
|
cliSet := initClientSet(g)
|
2021-01-29 19:22:43 +08:00
|
|
|
|
2021-02-03 09:53:23 +08:00
|
|
|
eSilent := httpexpect.WithConfig(httpexpect.Config{
|
|
|
|
BaseURL: host,
|
|
|
|
Reporter: httpexpect.NewAssertReporter(t),
|
|
|
|
Printers: []httpexpect.Printer{
|
|
|
|
newSilentPrinter(t),
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
2021-01-29 19:22:43 +08:00
|
|
|
// check if everything works
|
|
|
|
setRoute(e, http.StatusCreated)
|
2021-02-22 11:49:43 +08:00
|
|
|
getRouteList(e, http.StatusOK)
|
|
|
|
|
|
|
|
// to avoid route haven't been set yet
|
|
|
|
time.Sleep(10 * time.Second)
|
2021-01-29 19:22:43 +08:00
|
|
|
getRoute(e, http.StatusOK)
|
|
|
|
testPrometheusEtcdMetric(e, 1)
|
|
|
|
|
|
|
|
// run in background
|
|
|
|
go func() {
|
|
|
|
for {
|
2021-02-03 09:53:23 +08:00
|
|
|
go getRoute(eSilent, http.StatusOK)
|
2021-02-20 14:03:10 +08:00
|
|
|
time.Sleep(500 * time.Millisecond)
|
2021-01-29 19:22:43 +08:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2021-02-22 11:49:43 +08:00
|
|
|
// wait 1 seconds to let first route access returns
|
|
|
|
time.Sleep(1 * time.Second)
|
2021-02-20 14:03:10 +08:00
|
|
|
bandwidthBefore, durationBefore := getIngressBandwidthPerSecond(e, g)
|
|
|
|
bpsBefore := bandwidthBefore / durationBefore
|
2021-01-29 19:22:43 +08:00
|
|
|
g.Expect(bpsBefore).NotTo(BeZero())
|
|
|
|
|
2021-02-08 10:33:06 +08:00
|
|
|
listOption := client.MatchingLabels{"app": "apisix-gw"}
|
|
|
|
apisixPod := getPod(g, cliSet.ctrlCli, metav1.NamespaceDefault, listOption)
|
|
|
|
|
2021-01-29 19:22:43 +08:00
|
|
|
t.Run("error log not contains etcd error", func(t *testing.T) {
|
2021-05-27 20:09:31 +08:00
|
|
|
errorLog, err := log(apisixPod, cliSet.kubeCli)
|
|
|
|
g.Expect(err).To(BeNil())
|
2021-01-29 19:22:43 +08:00
|
|
|
g.Expect(strings.Contains(errorLog, "failed to fetch data from etcd")).To(BeFalse())
|
|
|
|
})
|
|
|
|
|
|
|
|
// apply chaos to kill all etcd pods
|
2021-02-08 10:33:06 +08:00
|
|
|
t.Run("kill all etcd pods", func(t *testing.T) {
|
|
|
|
createEtcdKillChaos(g, cliSet.ctrlCli)
|
|
|
|
time.Sleep(3 * time.Second)
|
|
|
|
})
|
2021-01-29 19:22:43 +08:00
|
|
|
|
|
|
|
// fail to set route since etcd is all killed
|
|
|
|
// while get route could still succeed
|
|
|
|
setRoute(e, http.StatusInternalServerError)
|
|
|
|
getRoute(e, http.StatusOK)
|
|
|
|
testPrometheusEtcdMetric(e, 0)
|
|
|
|
|
|
|
|
t.Run("error log contains etcd error", func(t *testing.T) {
|
2021-05-27 20:09:31 +08:00
|
|
|
errorLog, err := log(apisixPod, cliSet.kubeCli)
|
|
|
|
g.Expect(err).To(BeNil())
|
2021-01-29 19:22:43 +08:00
|
|
|
g.Expect(strings.Contains(errorLog, "failed to fetch data from etcd")).To(BeTrue())
|
|
|
|
})
|
|
|
|
|
2021-02-20 14:03:10 +08:00
|
|
|
bandwidthAfter, durationAfter := getIngressBandwidthPerSecond(e, g)
|
|
|
|
bpsAfter := bandwidthAfter / durationAfter
|
2021-01-29 19:22:43 +08:00
|
|
|
t.Run("ingress bandwidth per second not change much", func(t *testing.T) {
|
2021-02-20 14:03:10 +08:00
|
|
|
t.Logf("bandwidth before: %f, after: %f", bandwidthBefore, bandwidthAfter)
|
|
|
|
t.Logf("duration before: %f, after: %f", durationBefore, durationAfter)
|
2021-01-29 19:22:43 +08:00
|
|
|
t.Logf("bps before: %f, after: %f", bpsBefore, bpsAfter)
|
|
|
|
g.Expect(roughCompare(bpsBefore, bpsAfter)).To(BeTrue())
|
|
|
|
})
|
|
|
|
}
|