123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201 |
- /*
- Copyright 2019 The Kubernetes 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_test
- import (
- "errors"
- "regexp"
- "sort"
- "strings"
- "testing"
- "github.com/onsi/ginkgo"
- "github.com/onsi/ginkgo/config"
- "github.com/onsi/ginkgo/reporters"
- "github.com/onsi/gomega"
- "k8s.io/kubernetes/test/e2e/framework"
- )
- // The line number of the following code is checked in TestFailureOutput below.
- // Be careful when moving it around or changing the import statements above.
- // Here are some intentionally blank lines that can be removed to compensate
- // for future additional import statements.
- //
- //
- //
- //
- //
- //
- func runTests(t *testing.T, reporter ginkgo.Reporter) {
- // This source code line will be part of the stack dump comparison.
- ginkgo.RunSpecsWithDefaultAndCustomReporters(t, "Logging Suite", []ginkgo.Reporter{reporter})
- }
- var _ = ginkgo.Describe("log", func() {
- ginkgo.BeforeEach(func() {
- framework.Logf("before")
- })
- ginkgo.It("fails", func() {
- func() {
- framework.Failf("I'm failing.")
- }()
- })
- ginkgo.It("asserts", func() {
- framework.ExpectEqual(false, true, "false is never true")
- })
- ginkgo.It("error", func() {
- err := errors.New("an error with a long, useless description")
- framework.ExpectNoError(err, "hard-coded error")
- })
- ginkgo.It("equal", func() {
- framework.ExpectEqual(0, 1, "of course it's not equal...")
- })
- ginkgo.AfterEach(func() {
- framework.Logf("after")
- framework.ExpectEqual(true, false, "true is never false either")
- })
- })
- func TestFailureOutput(t *testing.T) {
- // Run the Ginkgo suite with output collected by a custom
- // reporter in adddition to the default one. To see what the full
- // Ginkgo report looks like, run this test with "go test -v".
- config.DefaultReporterConfig.FullTrace = true
- gomega.RegisterFailHandler(framework.Fail)
- fakeT := &testing.T{}
- reporter := reporters.NewFakeReporter()
- runTests(fakeT, reporter)
- // Now check the output.
- actual := normalizeReport(*reporter)
- // output from AfterEach
- commonOutput := "\n\nINFO: after\nFAIL: true is never false either\nExpected\n <bool>: true\nto equal\n <bool>: false\n\nFull Stack Trace\nk8s.io/kubernetes/test/e2e/framework_test.glob..func1.6()\n\tlog_test.go:71\nk8s.io/kubernetes/test/e2e/framework_test.runTests()\n\tlog_test.go:47\n\n"
- // Sorted by name!
- expected := suiteResults{
- testResult{
- name: "[Top Level] log asserts",
- output: "INFO: before\nFAIL: false is never true\nExpected\n <bool>: false\nto equal\n <bool>: true\n\nFull Stack Trace\nk8s.io/kubernetes/test/e2e/framework_test.glob..func1.3()\n\tlog_test.go:60\nk8s.io/kubernetes/test/e2e/framework_test.runTests()\n\tlog_test.go:47" + commonOutput,
- failure: "false is never true\nExpected\n <bool>: false\nto equal\n <bool>: true",
- stack: "k8s.io/kubernetes/test/e2e/framework_test.glob..func1.3()\n\tlog_test.go:60\nk8s.io/kubernetes/test/e2e/framework_test.runTests()\n\tlog_test.go:47\n",
- },
- testResult{
- name: "[Top Level] log equal",
- output: "INFO: before\nFAIL: of course it's not equal...\nExpected\n <int>: 0\nto equal\n <int>: 1\n\nFull Stack Trace\nk8s.io/kubernetes/test/e2e/framework_test.glob..func1.5()\n\tlog_test.go:67\nk8s.io/kubernetes/test/e2e/framework_test.runTests()\n\tlog_test.go:47" + commonOutput,
- failure: "of course it's not equal...\nExpected\n <int>: 0\nto equal\n <int>: 1",
- stack: "k8s.io/kubernetes/test/e2e/framework_test.glob..func1.5()\n\tlog_test.go:67\nk8s.io/kubernetes/test/e2e/framework_test.runTests()\n\tlog_test.go:47\n",
- },
- testResult{
- name: "[Top Level] log error",
- output: "INFO: before\nFAIL: hard-coded error\nUnexpected error:\n <*errors.errorString>: {\n s: \"an error with a long, useless description\",\n }\n an error with a long, useless description\noccurred\n\nFull Stack Trace\nk8s.io/kubernetes/test/e2e/framework_test.glob..func1.4()\n\tlog_test.go:64\nk8s.io/kubernetes/test/e2e/framework_test.runTests()\n\tlog_test.go:47" + commonOutput,
- failure: "hard-coded error\nUnexpected error:\n <*errors.errorString>: {\n s: \"an error with a long, useless description\",\n }\n an error with a long, useless description\noccurred",
- stack: "k8s.io/kubernetes/test/e2e/framework_test.glob..func1.4()\n\tlog_test.go:64\nk8s.io/kubernetes/test/e2e/framework_test.runTests()\n\tlog_test.go:47\n",
- },
- testResult{
- name: "[Top Level] log fails",
- output: "INFO: before\nFAIL: I'm failing.\n\nFull Stack Trace\nk8s.io/kubernetes/test/e2e/framework_test.glob..func1.2.1(...)\n\tlog_test.go:56\nk8s.io/kubernetes/test/e2e/framework_test.glob..func1.2()\n\tlog_test.go:57\nk8s.io/kubernetes/test/e2e/framework_test.runTests()\n\tlog_test.go:47" + commonOutput,
- failure: "I'm failing.",
- stack: "k8s.io/kubernetes/test/e2e/framework_test.glob..func1.2.1(...)\n\tlog_test.go:56\nk8s.io/kubernetes/test/e2e/framework_test.glob..func1.2()\n\tlog_test.go:57\nk8s.io/kubernetes/test/e2e/framework_test.runTests()\n\tlog_test.go:47\n",
- },
- }
- // Compare individual fields. Comparing the slices leads to unreadable error output when there is any mismatch.
- framework.ExpectEqual(len(actual), len(expected), "%d entries in %v", len(expected), actual)
- for i, a := range actual {
- b := expected[i]
- framework.ExpectEqual(a.name, b.name, "name in %d", i)
- framework.ExpectEqual(a.output, b.output, "output in %d", i)
- framework.ExpectEqual(a.failure, b.failure, "failure in %d", i)
- // There may be additional stack entries from the "testing" package at the
- // end. We ignore those in the comparison because the line number in them
- // varies.
- framework.ExpectEqual(a.stack, b.stack, "stack in %d: %s", i, a.stack)
- }
- }
- type testResult struct {
- name string
- // output written to GinkgoWriter during test.
- output string
- // failure is SpecSummary.Failure.Message with varying parts stripped.
- failure string
- // stack is a normalized version (just file names, function parametes stripped) of
- // Ginkgo's FullStackTrace of a failure. Empty if no failure.
- stack string
- }
- type suiteResults []testResult
- func normalizeReport(report reporters.FakeReporter) suiteResults {
- var results suiteResults
- for _, spec := range report.SpecSummaries {
- results = append(results, testResult{
- name: strings.Join(spec.ComponentTexts, " "),
- output: normalizeLocation(stripAddresses(stripTimes(spec.CapturedOutput))),
- failure: stripAddresses(stripTimes(spec.Failure.Message)),
- stack: normalizeLocation(spec.Failure.Location.FullStackTrace),
- })
- }
- sort.Slice(results, func(i, j int) bool {
- return strings.Compare(results[i].name, results[j].name) < 0
- })
- return results
- }
- // timePrefix matches "Jul 17 08:08:25.950: " at the beginning of each line.
- var timePrefix = regexp.MustCompile(`(?m)^[[:alpha:]]{3} +[[:digit:]]{1,2} +[[:digit:]]{2}:[[:digit:]]{2}:[[:digit:]]{2}.[[:digit:]]{3}: `)
- func stripTimes(in string) string {
- return timePrefix.ReplaceAllString(in, "")
- }
- // instanceAddr matches " | 0xc0003dec60>"
- var instanceAddr = regexp.MustCompile(` \| 0x[0-9a-fA-F]+>`)
- func stripAddresses(in string) string {
- return instanceAddr.ReplaceAllString(in, ">")
- }
- // stackLocation matches "<some path>/<file>.go:75 +0x1f1" after a slash (built
- // locally) or one of a few relative paths (built in the Kubernetes CI).
- var stackLocation = regexp.MustCompile(`(?:/|vendor/|test/|GOROOT/).*/([[:^space:]]+.go:[[:digit:]]+)( \+0x[0-9a-fA-F]+)?`)
- // functionArgs matches "<function name>(...)".
- var functionArgs = regexp.MustCompile(`([[:alpha:]]+)\(.*\)`)
- // testFailureOutput matches TestFailureOutput() and its source followed by additional stack entries:
- //
- // k8s.io/kubernetes/test/e2e/framework_test.TestFailureOutput(0xc000558800)
- // /nvme/gopath/src/k8s.io/kubernetes/test/e2e/framework/log/log_test.go:73 +0x1c9
- // testing.tRunner(0xc000558800, 0x1af2848)
- // /nvme/gopath/go/src/testing/testing.go:865 +0xc0
- // created by testing.(*T).Run
- // /nvme/gopath/go/src/testing/testing.go:916 +0x35a
- var testFailureOutput = regexp.MustCompile(`(?m)^k8s.io/kubernetes/test/e2e/framework_test\.TestFailureOutput\(.*\n\t.*(\n.*\n\t.*)*`)
- // normalizeLocation removes path prefix and function parameters and certain stack entries
- // that we don't care about.
- func normalizeLocation(in string) string {
- out := in
- out = stackLocation.ReplaceAllString(out, "$1")
- out = functionArgs.ReplaceAllString(out, "$1()")
- out = testFailureOutput.ReplaceAllString(out, "")
- return out
- }
|