e2e.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278
  1. /*
  2. Copyright 2015 The Kubernetes Authors.
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package e2e
  14. import (
  15. "fmt"
  16. "io/ioutil"
  17. "os"
  18. "path"
  19. "testing"
  20. "time"
  21. "github.com/onsi/ginkgo"
  22. "github.com/onsi/ginkgo/config"
  23. "github.com/onsi/ginkgo/reporters"
  24. "github.com/onsi/gomega"
  25. "k8s.io/klog"
  26. metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
  27. runtimeutils "k8s.io/apimachinery/pkg/util/runtime"
  28. clientset "k8s.io/client-go/kubernetes"
  29. "k8s.io/component-base/logs"
  30. "k8s.io/kubernetes/pkg/version"
  31. commontest "k8s.io/kubernetes/test/e2e/common"
  32. "k8s.io/kubernetes/test/e2e/framework"
  33. "k8s.io/kubernetes/test/e2e/framework/ginkgowrapper"
  34. e2elog "k8s.io/kubernetes/test/e2e/framework/log"
  35. "k8s.io/kubernetes/test/e2e/framework/metrics"
  36. "k8s.io/kubernetes/test/e2e/manifest"
  37. testutils "k8s.io/kubernetes/test/utils"
  38. // ensure auth plugins are loaded
  39. _ "k8s.io/client-go/plugin/pkg/client/auth"
  40. // ensure that cloud providers are loaded
  41. _ "k8s.io/kubernetes/test/e2e/framework/providers/aws"
  42. _ "k8s.io/kubernetes/test/e2e/framework/providers/azure"
  43. _ "k8s.io/kubernetes/test/e2e/framework/providers/gce"
  44. _ "k8s.io/kubernetes/test/e2e/framework/providers/kubemark"
  45. _ "k8s.io/kubernetes/test/e2e/framework/providers/openstack"
  46. _ "k8s.io/kubernetes/test/e2e/framework/providers/vsphere"
  47. )
  48. var (
  49. cloudConfig = &framework.TestContext.CloudConfig
  50. nodeKillerStopCh = make(chan struct{})
  51. )
  52. // There are certain operations we only want to run once per overall test invocation
  53. // (such as deleting old namespaces, or verifying that all system pods are running.
  54. // Because of the way Ginkgo runs tests in parallel, we must use SynchronizedBeforeSuite
  55. // to ensure that these operations only run on the first parallel Ginkgo node.
  56. //
  57. // This function takes two parameters: one function which runs on only the first Ginkgo node,
  58. // returning an opaque byte array, and then a second function which runs on all Ginkgo nodes,
  59. // accepting the byte array.
  60. var _ = ginkgo.SynchronizedBeforeSuite(func() []byte {
  61. // Run only on Ginkgo node 1
  62. switch framework.TestContext.Provider {
  63. case "gce", "gke":
  64. framework.LogClusterImageSources()
  65. }
  66. c, err := framework.LoadClientset()
  67. if err != nil {
  68. klog.Fatal("Error loading client: ", err)
  69. }
  70. // Delete any namespaces except those created by the system. This ensures no
  71. // lingering resources are left over from a previous test run.
  72. if framework.TestContext.CleanStart {
  73. deleted, err := framework.DeleteNamespaces(c, nil, /* deleteFilter */
  74. []string{
  75. metav1.NamespaceSystem,
  76. metav1.NamespaceDefault,
  77. metav1.NamespacePublic,
  78. })
  79. if err != nil {
  80. framework.Failf("Error deleting orphaned namespaces: %v", err)
  81. }
  82. klog.Infof("Waiting for deletion of the following namespaces: %v", deleted)
  83. if err := framework.WaitForNamespacesDeleted(c, deleted, framework.NamespaceCleanupTimeout); err != nil {
  84. framework.Failf("Failed to delete orphaned namespaces %v: %v", deleted, err)
  85. }
  86. }
  87. // In large clusters we may get to this point but still have a bunch
  88. // of nodes without Routes created. Since this would make a node
  89. // unschedulable, we need to wait until all of them are schedulable.
  90. framework.ExpectNoError(framework.WaitForAllNodesSchedulable(c, framework.TestContext.NodeSchedulableTimeout))
  91. // If NumNodes is not specified then auto-detect how many are scheduleable and not tainted
  92. if framework.TestContext.CloudConfig.NumNodes == framework.DefaultNumNodes {
  93. framework.TestContext.CloudConfig.NumNodes = len(framework.GetReadySchedulableNodesOrDie(c).Items)
  94. }
  95. // Ensure all pods are running and ready before starting tests (otherwise,
  96. // cluster infrastructure pods that are being pulled or started can block
  97. // test pods from running, and tests that ensure all pods are running and
  98. // ready will fail).
  99. podStartupTimeout := framework.TestContext.SystemPodsStartupTimeout
  100. // TODO: In large clusters, we often observe a non-starting pods due to
  101. // #41007. To avoid those pods preventing the whole test runs (and just
  102. // wasting the whole run), we allow for some not-ready pods (with the
  103. // number equal to the number of allowed not-ready nodes).
  104. if err := framework.WaitForPodsRunningReady(c, metav1.NamespaceSystem, int32(framework.TestContext.MinStartupPods), int32(framework.TestContext.AllowedNotReadyNodes), podStartupTimeout, map[string]string{}); err != nil {
  105. framework.DumpAllNamespaceInfo(c, metav1.NamespaceSystem)
  106. framework.LogFailedContainers(c, metav1.NamespaceSystem, e2elog.Logf)
  107. runKubernetesServiceTestContainer(c, metav1.NamespaceDefault)
  108. framework.Failf("Error waiting for all pods to be running and ready: %v", err)
  109. }
  110. if err := framework.WaitForDaemonSets(c, metav1.NamespaceSystem, int32(framework.TestContext.AllowedNotReadyNodes), framework.TestContext.SystemDaemonsetStartupTimeout); err != nil {
  111. e2elog.Logf("WARNING: Waiting for all daemonsets to be ready failed: %v", err)
  112. }
  113. // Log the version of the server and this client.
  114. e2elog.Logf("e2e test version: %s", version.Get().GitVersion)
  115. dc := c.DiscoveryClient
  116. serverVersion, serverErr := dc.ServerVersion()
  117. if serverErr != nil {
  118. e2elog.Logf("Unexpected server error retrieving version: %v", serverErr)
  119. }
  120. if serverVersion != nil {
  121. e2elog.Logf("kube-apiserver version: %s", serverVersion.GitVersion)
  122. }
  123. // Reference common test to make the import valid.
  124. commontest.CurrentSuite = commontest.E2E
  125. if framework.TestContext.NodeKiller.Enabled {
  126. nodeKiller := framework.NewNodeKiller(framework.TestContext.NodeKiller, c, framework.TestContext.Provider)
  127. nodeKillerStopCh = make(chan struct{})
  128. go nodeKiller.Run(nodeKillerStopCh)
  129. }
  130. return nil
  131. }, func(data []byte) {
  132. // Run on all Ginkgo nodes
  133. })
  134. // Similar to SynchronizedBeforeSuite, we want to run some operations only once (such as collecting cluster logs).
  135. // Here, the order of functions is reversed; first, the function which runs everywhere,
  136. // and then the function that only runs on the first Ginkgo node.
  137. var _ = ginkgo.SynchronizedAfterSuite(func() {
  138. // Run on all Ginkgo nodes
  139. e2elog.Logf("Running AfterSuite actions on all nodes")
  140. framework.RunCleanupActions()
  141. }, func() {
  142. // Run only Ginkgo on node 1
  143. e2elog.Logf("Running AfterSuite actions on node 1")
  144. if framework.TestContext.ReportDir != "" {
  145. framework.CoreDump(framework.TestContext.ReportDir)
  146. }
  147. if framework.TestContext.GatherSuiteMetricsAfterTest {
  148. if err := gatherTestSuiteMetrics(); err != nil {
  149. e2elog.Logf("Error gathering metrics: %v", err)
  150. }
  151. }
  152. if framework.TestContext.NodeKiller.Enabled {
  153. close(nodeKillerStopCh)
  154. }
  155. })
  156. func gatherTestSuiteMetrics() error {
  157. e2elog.Logf("Gathering metrics")
  158. c, err := framework.LoadClientset()
  159. if err != nil {
  160. return fmt.Errorf("error loading client: %v", err)
  161. }
  162. // Grab metrics for apiserver, scheduler, controller-manager, kubelet (for non-kubemark case) and cluster autoscaler (optionally).
  163. grabber, err := metrics.NewMetricsGrabber(c, nil, !framework.ProviderIs("kubemark"), true, true, true, framework.TestContext.IncludeClusterAutoscalerMetrics)
  164. if err != nil {
  165. return fmt.Errorf("failed to create MetricsGrabber: %v", err)
  166. }
  167. received, err := grabber.Grab()
  168. if err != nil {
  169. return fmt.Errorf("failed to grab metrics: %v", err)
  170. }
  171. metricsForE2E := (*framework.MetricsForE2E)(&received)
  172. metricsJSON := metricsForE2E.PrintJSON()
  173. if framework.TestContext.ReportDir != "" {
  174. filePath := path.Join(framework.TestContext.ReportDir, "MetricsForE2ESuite_"+time.Now().Format(time.RFC3339)+".json")
  175. if err := ioutil.WriteFile(filePath, []byte(metricsJSON), 0644); err != nil {
  176. return fmt.Errorf("error writing to %q: %v", filePath, err)
  177. }
  178. } else {
  179. e2elog.Logf("\n\nTest Suite Metrics:\n%s\n", metricsJSON)
  180. }
  181. return nil
  182. }
  183. // RunE2ETests checks configuration parameters (specified through flags) and then runs
  184. // E2E tests using the Ginkgo runner.
  185. // If a "report directory" is specified, one or more JUnit test reports will be
  186. // generated in this directory, and cluster logs will also be saved.
  187. // This function is called on each Ginkgo node in parallel mode.
  188. func RunE2ETests(t *testing.T) {
  189. runtimeutils.ReallyCrash = true
  190. logs.InitLogs()
  191. defer logs.FlushLogs()
  192. gomega.RegisterFailHandler(ginkgowrapper.Fail)
  193. // Disable skipped tests unless they are explicitly requested.
  194. if config.GinkgoConfig.FocusString == "" && config.GinkgoConfig.SkipString == "" {
  195. config.GinkgoConfig.SkipString = `\[Flaky\]|\[Feature:.+\]`
  196. }
  197. // Run tests through the Ginkgo runner with output to console + JUnit for Jenkins
  198. var r []ginkgo.Reporter
  199. if framework.TestContext.ReportDir != "" {
  200. // TODO: we should probably only be trying to create this directory once
  201. // rather than once-per-Ginkgo-node.
  202. if err := os.MkdirAll(framework.TestContext.ReportDir, 0755); err != nil {
  203. klog.Errorf("Failed creating report directory: %v", err)
  204. } else {
  205. r = append(r, reporters.NewJUnitReporter(path.Join(framework.TestContext.ReportDir, fmt.Sprintf("junit_%v%02d.xml", framework.TestContext.ReportPrefix, config.GinkgoConfig.ParallelNode))))
  206. }
  207. }
  208. klog.Infof("Starting e2e run %q on Ginkgo node %d", framework.RunID, config.GinkgoConfig.ParallelNode)
  209. ginkgo.RunSpecsWithDefaultAndCustomReporters(t, "Kubernetes e2e suite", r)
  210. }
  211. // Run a test container to try and contact the Kubernetes api-server from a pod, wait for it
  212. // to flip to Ready, log its output and delete it.
  213. func runKubernetesServiceTestContainer(c clientset.Interface, ns string) {
  214. path := "test/images/clusterapi-tester/pod.yaml"
  215. e2elog.Logf("Parsing pod from %v", path)
  216. p, err := manifest.PodFromManifest(path)
  217. if err != nil {
  218. e2elog.Logf("Failed to parse clusterapi-tester from manifest %v: %v", path, err)
  219. return
  220. }
  221. p.Namespace = ns
  222. if _, err := c.CoreV1().Pods(ns).Create(p); err != nil {
  223. e2elog.Logf("Failed to create %v: %v", p.Name, err)
  224. return
  225. }
  226. defer func() {
  227. if err := c.CoreV1().Pods(ns).Delete(p.Name, nil); err != nil {
  228. e2elog.Logf("Failed to delete pod %v: %v", p.Name, err)
  229. }
  230. }()
  231. timeout := 5 * time.Minute
  232. if err := framework.WaitForPodCondition(c, ns, p.Name, "clusterapi-tester", timeout, testutils.PodRunningReady); err != nil {
  233. e2elog.Logf("Pod %v took longer than %v to enter running/ready: %v", p.Name, timeout, err)
  234. return
  235. }
  236. logs, err := framework.GetPodLogs(c, ns, p.Name, p.Spec.Containers[0].Name)
  237. if err != nil {
  238. e2elog.Logf("Failed to retrieve logs from %v: %v", p.Name, err)
  239. } else {
  240. e2elog.Logf("Output of clusterapi-tester:\n%v", logs)
  241. }
  242. }