|  | // Copyright 2017 The Go Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style | 
|  | // license that can be found in the LICENSE file. | 
|  |  | 
|  | package errorstest | 
|  |  | 
|  | import ( | 
|  | "bytes" | 
|  | "fmt" | 
|  | "io/ioutil" | 
|  | "os" | 
|  | "os/exec" | 
|  | "path/filepath" | 
|  | "regexp" | 
|  | "strconv" | 
|  | "strings" | 
|  | "testing" | 
|  | ) | 
|  |  | 
|  | func path(file string) string { | 
|  | return filepath.Join("src", file) | 
|  | } | 
|  |  | 
|  | func check(t *testing.T, file string) { | 
|  | t.Run(file, func(t *testing.T) { | 
|  | t.Parallel() | 
|  |  | 
|  | contents, err := ioutil.ReadFile(path(file)) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  | var errors []*regexp.Regexp | 
|  | for i, line := range bytes.Split(contents, []byte("\n")) { | 
|  | if bytes.HasSuffix(line, []byte("ERROR HERE")) { | 
|  | re := regexp.MustCompile(regexp.QuoteMeta(fmt.Sprintf("%s:%d:", file, i+1))) | 
|  | errors = append(errors, re) | 
|  | continue | 
|  | } | 
|  |  | 
|  | frags := bytes.SplitAfterN(line, []byte("ERROR HERE: "), 2) | 
|  | if len(frags) == 1 { | 
|  | continue | 
|  | } | 
|  | re, err := regexp.Compile(string(frags[1])) | 
|  | if err != nil { | 
|  | t.Errorf("Invalid regexp after `ERROR HERE: `: %#q", frags[1]) | 
|  | continue | 
|  | } | 
|  | errors = append(errors, re) | 
|  | } | 
|  | if len(errors) == 0 { | 
|  | t.Fatalf("cannot find ERROR HERE") | 
|  | } | 
|  | expect(t, file, errors) | 
|  | }) | 
|  | } | 
|  |  | 
|  | func expect(t *testing.T, file string, errors []*regexp.Regexp) { | 
|  | dir, err := ioutil.TempDir("", filepath.Base(t.Name())) | 
|  | if err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  | defer os.RemoveAll(dir) | 
|  |  | 
|  | dst := filepath.Join(dir, strings.TrimSuffix(file, ".go")) | 
|  | cmd := exec.Command("go", "build", "-gcflags=-L", "-o="+dst, path(file)) // TODO(gri) no need for -gcflags=-L if go tool is adjusted | 
|  | out, err := cmd.CombinedOutput() | 
|  | if err == nil { | 
|  | t.Errorf("expected cgo to fail but it succeeded") | 
|  | } | 
|  |  | 
|  | lines := bytes.Split(out, []byte("\n")) | 
|  | for _, re := range errors { | 
|  | found := false | 
|  | for _, line := range lines { | 
|  | if re.Match(line) { | 
|  | t.Logf("found match for %#q: %q", re, line) | 
|  | found = true | 
|  | break | 
|  | } | 
|  | } | 
|  | if !found { | 
|  | t.Errorf("expected error output to contain %#q", re) | 
|  | } | 
|  | } | 
|  |  | 
|  | if t.Failed() { | 
|  | t.Logf("actual output:\n%s", out) | 
|  | } | 
|  | } | 
|  |  | 
|  | func sizeofLongDouble(t *testing.T) int { | 
|  | cmd := exec.Command("go", "run", path("long_double_size.go")) | 
|  | out, err := cmd.CombinedOutput() | 
|  | if err != nil { | 
|  | t.Fatalf("%#q: %v:\n%s", strings.Join(cmd.Args, " "), err, out) | 
|  | } | 
|  |  | 
|  | i, err := strconv.Atoi(strings.TrimSpace(string(out))) | 
|  | if err != nil { | 
|  | t.Fatalf("long_double_size.go printed invalid size: %s", out) | 
|  | } | 
|  | return i | 
|  | } | 
|  |  | 
|  | func TestReportsTypeErrors(t *testing.T) { | 
|  | for _, file := range []string{ | 
|  | "err1.go", | 
|  | "err2.go", | 
|  | "err3.go", | 
|  | "issue7757.go", | 
|  | "issue8442.go", | 
|  | "issue11097a.go", | 
|  | "issue11097b.go", | 
|  | "issue13129.go", | 
|  | "issue13423.go", | 
|  | "issue13467.go", | 
|  | "issue13635.go", | 
|  | "issue13830.go", | 
|  | "issue16116.go", | 
|  | "issue16591.go", | 
|  | "issue18452.go", | 
|  | "issue18889.go", | 
|  | "issue26745.go", | 
|  | "issue28721.go", | 
|  | } { | 
|  | check(t, file) | 
|  | } | 
|  |  | 
|  | if sizeofLongDouble(t) > 8 { | 
|  | for _, file := range []string{ | 
|  | "err4.go", | 
|  | "issue28069.go", | 
|  | } { | 
|  | check(t, file) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestToleratesOptimizationFlag(t *testing.T) { | 
|  | for _, cflags := range []string{ | 
|  | "", | 
|  | "-O", | 
|  | } { | 
|  | cflags := cflags | 
|  | t.Run(cflags, func(t *testing.T) { | 
|  | t.Parallel() | 
|  |  | 
|  | cmd := exec.Command("go", "build", path("issue14669.go")) | 
|  | cmd.Env = append(os.Environ(), "CGO_CFLAGS="+cflags) | 
|  | out, err := cmd.CombinedOutput() | 
|  | if err != nil { | 
|  | t.Errorf("%#q: %v:\n%s", strings.Join(cmd.Args, " "), err, out) | 
|  | } | 
|  | }) | 
|  | } | 
|  | } | 
|  |  | 
|  | func TestMallocCrashesOnNil(t *testing.T) { | 
|  | t.Parallel() | 
|  |  | 
|  | cmd := exec.Command("go", "run", path("malloc.go")) | 
|  | out, err := cmd.CombinedOutput() | 
|  | if err == nil { | 
|  | t.Logf("%#q:\n%s", strings.Join(cmd.Args, " "), out) | 
|  | t.Fatalf("succeeded unexpectedly") | 
|  | } | 
|  | } |