1
2
3
4
5 package errorstest
6
7 import (
8 "bytes"
9 "fmt"
10 "internal/testenv"
11 "os"
12 "os/exec"
13 "path/filepath"
14 "regexp"
15 "strconv"
16 "strings"
17 "testing"
18 )
19
20 func path(file string) string {
21 return filepath.Join("testdata", file)
22 }
23
24 func check(t *testing.T, file string) {
25 t.Run(file, func(t *testing.T) {
26 testenv.MustHaveGoBuild(t)
27 testenv.MustHaveCGO(t)
28 t.Parallel()
29
30 contents, err := os.ReadFile(path(file))
31 if err != nil {
32 t.Fatal(err)
33 }
34 var errors []*regexp.Regexp
35 for i, line := range bytes.Split(contents, []byte("\n")) {
36 if bytes.HasSuffix(line, []byte("ERROR HERE")) {
37 re := regexp.MustCompile(regexp.QuoteMeta(fmt.Sprintf("%s:%d:", file, i+1)))
38 errors = append(errors, re)
39 continue
40 }
41
42 if _, frag, ok := bytes.Cut(line, []byte("ERROR HERE: ")); ok {
43 re, err := regexp.Compile(fmt.Sprintf(":%d:.*%s", i+1, frag))
44 if err != nil {
45 t.Errorf("Invalid regexp after `ERROR HERE: `: %#q", frag)
46 continue
47 }
48 errors = append(errors, re)
49 }
50
51 if _, frag, ok := bytes.Cut(line, []byte("ERROR MESSAGE: ")); ok {
52 re, err := regexp.Compile(string(frag))
53 if err != nil {
54 t.Errorf("Invalid regexp after `ERROR MESSAGE: `: %#q", frag)
55 continue
56 }
57 errors = append(errors, re)
58 }
59 }
60 if len(errors) == 0 {
61 t.Fatalf("cannot find ERROR HERE")
62 }
63 expect(t, file, errors)
64 })
65 }
66
67 func expect(t *testing.T, file string, errors []*regexp.Regexp) {
68 dir, err := os.MkdirTemp("", filepath.Base(t.Name()))
69 if err != nil {
70 t.Fatal(err)
71 }
72 defer os.RemoveAll(dir)
73
74 dst := filepath.Join(dir, strings.TrimSuffix(file, ".go"))
75 cmd := exec.Command("go", "build", "-gcflags=-L -e", "-o="+dst, path(file))
76 out, err := cmd.CombinedOutput()
77 if err == nil {
78 t.Errorf("expected cgo to fail but it succeeded")
79 }
80
81 lines := bytes.Split(out, []byte("\n"))
82 for _, re := range errors {
83 found := false
84 for _, line := range lines {
85 if re.Match(line) {
86 t.Logf("found match for %#q: %q", re, line)
87 found = true
88 break
89 }
90 }
91 if !found {
92 t.Errorf("expected error output to contain %#q", re)
93 }
94 }
95
96 if t.Failed() {
97 t.Logf("actual output:\n%s", out)
98 }
99 }
100
101 func sizeofLongDouble(t *testing.T) int {
102 testenv.MustHaveGoRun(t)
103 testenv.MustHaveCGO(t)
104 cmd := exec.Command("go", "run", path("long_double_size.go"))
105 out, err := cmd.CombinedOutput()
106 if err != nil {
107 t.Fatalf("%#q: %v:\n%s", strings.Join(cmd.Args, " "), err, out)
108 }
109
110 i, err := strconv.Atoi(strings.TrimSpace(string(out)))
111 if err != nil {
112 t.Fatalf("long_double_size.go printed invalid size: %s", out)
113 }
114 return i
115 }
116
117 func TestReportsTypeErrors(t *testing.T) {
118 for _, file := range []string{
119 "err1.go",
120 "err2.go",
121 "err5.go",
122 "issue11097a.go",
123 "issue11097b.go",
124 "issue18452.go",
125 "issue18889.go",
126 "issue28721.go",
127 "issue33061.go",
128 "issue50710.go",
129 } {
130 check(t, file)
131 }
132
133 if sizeofLongDouble(t) > 8 {
134 for _, file := range []string{
135 "err4.go",
136 "issue28069.go",
137 } {
138 check(t, file)
139 }
140 }
141 }
142
143 func TestToleratesOptimizationFlag(t *testing.T) {
144 for _, cflags := range []string{
145 "",
146 "-O",
147 } {
148 cflags := cflags
149 t.Run(cflags, func(t *testing.T) {
150 testenv.MustHaveGoBuild(t)
151 testenv.MustHaveCGO(t)
152 t.Parallel()
153
154 cmd := exec.Command("go", "build", path("issue14669.go"))
155 cmd.Env = append(os.Environ(), "CGO_CFLAGS="+cflags)
156 out, err := cmd.CombinedOutput()
157 if err != nil {
158 t.Errorf("%#q: %v:\n%s", strings.Join(cmd.Args, " "), err, out)
159 }
160 })
161 }
162 }
163
164 func TestMallocCrashesOnNil(t *testing.T) {
165 testenv.MustHaveCGO(t)
166 testenv.MustHaveGoRun(t)
167 t.Parallel()
168
169 cmd := exec.Command("go", "run", path("malloc.go"))
170 out, err := cmd.CombinedOutput()
171 if err == nil {
172 t.Logf("%#q:\n%s", strings.Join(cmd.Args, " "), out)
173 t.Fatalf("succeeded unexpectedly")
174 }
175 }
176
177 func TestNotMatchedCFunction(t *testing.T) {
178 file := "notmatchedcfunction.go"
179 check(t, file)
180 }
181
View as plain text