1
2
3
4
5
6
7
8
9 package test2json
10
11 import (
12 "bytes"
13 "encoding/json"
14 "fmt"
15 "io"
16 "strconv"
17 "strings"
18 "time"
19 "unicode"
20 "unicode/utf8"
21 )
22
23
24 type Mode int
25
26 const (
27 Timestamp Mode = 1 << iota
28 )
29
30
31 type event struct {
32 Time *time.Time `json:",omitempty"`
33 Action string
34 Package string `json:",omitempty"`
35 Test string `json:",omitempty"`
36 Elapsed *float64 `json:",omitempty"`
37 Output *textBytes `json:",omitempty"`
38 FailedBuild string `json:",omitempty"`
39 }
40
41
42
43
44
45 type textBytes []byte
46
47 func (b textBytes) MarshalText() ([]byte, error) { return b, nil }
48
49
50
51
52 type Converter struct {
53 w io.Writer
54 pkg string
55 mode Mode
56 start time.Time
57 testName string
58 report []*event
59 result string
60 input lineBuffer
61 output lineBuffer
62 needMarker bool
63
64
65
66 failedBuild string
67 }
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88 var (
89 inBuffer = 4096
90 outBuffer = 1024
91 )
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109 func NewConverter(w io.Writer, pkg string, mode Mode) *Converter {
110 c := new(Converter)
111 *c = Converter{
112 w: w,
113 pkg: pkg,
114 mode: mode,
115 start: time.Now(),
116 input: lineBuffer{
117 b: make([]byte, 0, inBuffer),
118 line: c.handleInputLine,
119 part: c.output.write,
120 },
121 output: lineBuffer{
122 b: make([]byte, 0, outBuffer),
123 line: c.writeOutputEvent,
124 part: c.writeOutputEvent,
125 },
126 }
127 c.writeEvent(&event{Action: "start"})
128 return c
129 }
130
131
132 func (c *Converter) Write(b []byte) (int, error) {
133 c.input.write(b)
134 return len(b), nil
135 }
136
137
138 func (c *Converter) Exited(err error) {
139 if err == nil {
140 if c.result != "skip" {
141 c.result = "pass"
142 }
143 } else {
144 c.result = "fail"
145 }
146 }
147
148
149
150
151 func (c *Converter) SetFailedBuild(pkgID string) {
152 c.failedBuild = pkgID
153 }
154
155 const marker = byte(0x16)
156
157 var (
158
159 bigPass = []byte("PASS")
160
161
162 bigFail = []byte("FAIL")
163
164
165
166 bigFailErrorPrefix = []byte("FAIL\t")
167
168
169 emptyName = []byte("=== NAME")
170 emptyNameLine = []byte("=== NAME \n")
171
172 updates = [][]byte{
173 []byte("=== RUN "),
174 []byte("=== PAUSE "),
175 []byte("=== CONT "),
176 []byte("=== NAME "),
177 []byte("=== PASS "),
178 []byte("=== FAIL "),
179 []byte("=== SKIP "),
180 }
181
182 reports = [][]byte{
183 []byte("--- PASS: "),
184 []byte("--- FAIL: "),
185 []byte("--- SKIP: "),
186 []byte("--- BENCH: "),
187 }
188
189 fourSpace = []byte(" ")
190
191 skipLinePrefix = []byte("? \t")
192 skipLineSuffix = []byte("\t[no test files]")
193 )
194
195
196
197
198 func (c *Converter) handleInputLine(line []byte) {
199 if len(line) == 0 {
200 return
201 }
202 sawMarker := false
203 if c.needMarker && line[0] != marker {
204 c.output.write(line)
205 return
206 }
207 if line[0] == marker {
208 c.output.flush()
209 sawMarker = true
210 line = line[1:]
211 }
212
213
214 trim := line
215 if len(trim) > 0 && trim[len(trim)-1] == '\n' {
216 trim = trim[:len(trim)-1]
217 if len(trim) > 0 && trim[len(trim)-1] == '\r' {
218 trim = trim[:len(trim)-1]
219 }
220 }
221
222
223 if bytes.Equal(trim, emptyName) {
224 line = emptyNameLine
225 trim = line[:len(line)-1]
226 }
227
228
229 if bytes.Equal(trim, bigPass) || bytes.Equal(trim, bigFail) || bytes.HasPrefix(trim, bigFailErrorPrefix) {
230 c.flushReport(0)
231 c.testName = ""
232 c.needMarker = sawMarker
233 c.output.write(line)
234 if bytes.Equal(trim, bigPass) {
235 c.result = "pass"
236 } else {
237 c.result = "fail"
238 }
239 return
240 }
241
242
243
244 if bytes.HasPrefix(line, skipLinePrefix) && bytes.HasSuffix(trim, skipLineSuffix) && len(c.report) == 0 {
245 c.result = "skip"
246 }
247
248
249
250
251 actionColon := false
252 origLine := line
253 ok := false
254 indent := 0
255 for _, magic := range updates {
256 if bytes.HasPrefix(line, magic) {
257 ok = true
258 break
259 }
260 }
261 if !ok {
262
263
264
265
266
267 for bytes.HasPrefix(line, fourSpace) {
268 line = line[4:]
269 indent++
270 }
271 for _, magic := range reports {
272 if bytes.HasPrefix(line, magic) {
273 actionColon = true
274 ok = true
275 break
276 }
277 }
278 }
279
280
281 if !ok {
282
283
284
285
286
287
288
289 if indent > 0 && indent <= len(c.report) {
290 c.testName = c.report[indent-1].Test
291 }
292 c.output.write(origLine)
293 return
294 }
295
296
297 i := 0
298 if actionColon {
299 i = bytes.IndexByte(line, ':') + 1
300 }
301 if i == 0 {
302 i = len(updates[0])
303 }
304 action := strings.ToLower(strings.TrimSuffix(strings.TrimSpace(string(line[4:i])), ":"))
305 name := strings.TrimSpace(string(line[i:]))
306
307 e := &event{Action: action}
308 if line[0] == '-' {
309
310 if i := strings.Index(name, " ("); i >= 0 {
311 if strings.HasSuffix(name, "s)") {
312 t, err := strconv.ParseFloat(name[i+2:len(name)-2], 64)
313 if err == nil {
314 if c.mode&Timestamp != 0 {
315 e.Elapsed = &t
316 }
317 }
318 }
319 name = name[:i]
320 }
321 if len(c.report) < indent {
322
323
324 c.output.write(origLine)
325 return
326 }
327
328 c.needMarker = sawMarker
329 c.flushReport(indent)
330 e.Test = name
331 c.testName = name
332 c.report = append(c.report, e)
333 c.output.write(origLine)
334 return
335 }
336
337
338 c.needMarker = sawMarker
339 c.flushReport(0)
340 c.testName = name
341
342 if action == "name" {
343
344
345 return
346 }
347
348 if action == "pause" {
349
350
351
352 c.output.write(origLine)
353 }
354 c.writeEvent(e)
355 if action != "pause" {
356 c.output.write(origLine)
357 }
358
359 return
360 }
361
362
363 func (c *Converter) flushReport(depth int) {
364 c.testName = ""
365 for len(c.report) > depth {
366 e := c.report[len(c.report)-1]
367 c.report = c.report[:len(c.report)-1]
368 c.writeEvent(e)
369 }
370 }
371
372
373
374
375 func (c *Converter) Close() error {
376 c.input.flush()
377 c.output.flush()
378 if c.result != "" {
379 e := &event{Action: c.result}
380 if c.mode&Timestamp != 0 {
381 dt := time.Since(c.start).Round(1 * time.Millisecond).Seconds()
382 e.Elapsed = &dt
383 }
384 if c.result == "fail" {
385 e.FailedBuild = c.failedBuild
386 }
387 c.writeEvent(e)
388 }
389 return nil
390 }
391
392
393 func (c *Converter) writeOutputEvent(out []byte) {
394 c.writeEvent(&event{
395 Action: "output",
396 Output: (*textBytes)(&out),
397 })
398 }
399
400
401
402 func (c *Converter) writeEvent(e *event) {
403 e.Package = c.pkg
404 if c.mode&Timestamp != 0 {
405 t := time.Now()
406 e.Time = &t
407 }
408 if e.Test == "" {
409 e.Test = c.testName
410 }
411 js, err := json.Marshal(e)
412 if err != nil {
413
414 fmt.Fprintf(c.w, "testjson internal error: %v\n", err)
415 return
416 }
417 js = append(js, '\n')
418 c.w.Write(js)
419 }
420
421
422
423
424
425
426
427
428
429
430
431 type lineBuffer struct {
432 b []byte
433 mid bool
434 line func([]byte)
435 part func([]byte)
436 }
437
438
439 func (l *lineBuffer) write(b []byte) {
440 for len(b) > 0 {
441
442 m := copy(l.b[len(l.b):cap(l.b)], b)
443 l.b = l.b[:len(l.b)+m]
444 b = b[m:]
445
446
447 i := 0
448 for i < len(l.b) {
449 j, w := indexEOL(l.b[i:])
450 if j < 0 {
451 if !l.mid {
452 if j := bytes.IndexByte(l.b[i:], '\t'); j >= 0 {
453 if isBenchmarkName(bytes.TrimRight(l.b[i:i+j], " ")) {
454 l.part(l.b[i : i+j+1])
455 l.mid = true
456 i += j + 1
457 }
458 }
459 }
460 break
461 }
462 e := i + j + w
463 if l.mid {
464
465 l.part(l.b[i:e])
466 l.mid = false
467 } else {
468
469 l.line(l.b[i:e])
470 }
471 i = e
472 }
473
474
475 if i == 0 && len(l.b) == cap(l.b) {
476
477
478 t := trimUTF8(l.b)
479 l.part(l.b[:t])
480 l.b = l.b[:copy(l.b, l.b[t:])]
481 l.mid = true
482 }
483
484
485
486 if i > 0 {
487 l.b = l.b[:copy(l.b, l.b[i:])]
488 }
489 }
490 }
491
492
493
494
495
496
497 func indexEOL(b []byte) (pos, wid int) {
498 for i, c := range b {
499 if c == '\n' {
500 return i, 1
501 }
502 if c == marker && i > 0 {
503 return i, 0
504 }
505 }
506 return -1, 0
507 }
508
509
510 func (l *lineBuffer) flush() {
511 if len(l.b) > 0 {
512
513 l.part(l.b)
514 l.b = l.b[:0]
515 }
516 }
517
518 var benchmark = []byte("Benchmark")
519
520
521
522 func isBenchmarkName(b []byte) bool {
523 if !bytes.HasPrefix(b, benchmark) {
524 return false
525 }
526 if len(b) == len(benchmark) {
527 return true
528 }
529 r, _ := utf8.DecodeRune(b[len(benchmark):])
530 return !unicode.IsLower(r)
531 }
532
533
534
535
536
537
538 func trimUTF8(b []byte) int {
539
540 for i := 1; i < utf8.UTFMax && i <= len(b); i++ {
541 if c := b[len(b)-i]; c&0xc0 != 0x80 {
542 switch {
543 case c&0xe0 == 0xc0:
544 if i < 2 {
545 return len(b) - i
546 }
547 case c&0xf0 == 0xe0:
548 if i < 3 {
549 return len(b) - i
550 }
551 case c&0xf8 == 0xf0:
552 if i < 4 {
553 return len(b) - i
554 }
555 }
556 break
557 }
558 }
559 return len(b)
560 }
561
View as plain text