2022-03-07 00:09:34 +00:00

196 lines
4.3 KiB
Go

// Package gotest is a standard Go test output parser.
package gotest
import (
"bufio"
"fmt"
"io"
"regexp"
"strconv"
"strings"
"time"
)
type Event struct {
Type string
Id int
Name string
Result string
Duration time.Duration
Data string
Indent int
CovPct float64
CovPackages []string
}
var (
regexEndTest = regexp.MustCompile(`((?: )*)--- (PASS|FAIL|SKIP): ([^ ]+) \((\d+\.\d+)(?: seconds|s)\)`)
regexStatus = regexp.MustCompile(`^(PASS|FAIL|SKIP)$`)
regexSummary = regexp.MustCompile(`^(ok|FAIL)\s+([^ ]+)\s+` +
`(?:(\d+\.\d+)s|(\(cached\)|\[\w+ failed]))` +
`(?:\s+coverage:\s+(\d+\.\d+)%\sof\sstatements(?:\sin\s(.+))?)?$`)
regexCoverage = regexp.MustCompile(`^coverage:\s+(\d+|\d+\.\d+)%\s+of\s+statements(?:\sin\s(.+))?$`)
)
// Parse parses Go test output from the given io.Reader r.
func Parse(r io.Reader) ([]Event, error) {
p := &parser{}
s := bufio.NewScanner(r)
for s.Scan() {
p.parseLine(s.Text())
}
if s.Err() != nil {
return nil, s.Err()
}
return p.events, nil
}
type parser struct {
id int
events []Event
}
func (p *parser) parseLine(line string) {
if strings.HasPrefix(line, "=== RUN ") {
p.runTest(strings.TrimSpace(line[8:]))
} else if strings.HasPrefix(line, "=== PAUSE ") {
p.pauseTest(strings.TrimSpace(line[10:]))
} else if strings.HasPrefix(line, "=== CONT ") {
p.contTest(strings.TrimSpace(line[9:]))
} else if matches := regexEndTest.FindStringSubmatch(line); len(matches) == 5 {
p.endTest(line, matches[1], matches[2], matches[3], matches[4])
} else if matches := regexStatus.FindStringSubmatch(line); len(matches) == 2 {
p.status(matches[1])
} else if matches := regexSummary.FindStringSubmatch(line); len(matches) == 7 {
p.summary(matches[1], matches[2], matches[3], matches[4], matches[5], matches[6])
} else if matches := regexCoverage.FindStringSubmatch(line); len(matches) == 3 {
p.coverage(matches[1], matches[2])
} else {
p.output(line)
}
}
func (p *parser) add(event Event) {
p.events = append(p.events, event)
}
func (p *parser) findTest(name string) int {
for i := len(p.events) - 1; i >= 0; i-- {
// TODO: should we only consider tests that haven't ended yet?
if p.events[i].Type == "run_test" && p.events[i].Name == name {
return p.events[i].Id
}
}
return -1
}
func (p *parser) runTest(name string) {
p.id += 1
p.add(Event{
Type: "run_test",
Id: p.id,
Name: name,
})
}
func (p *parser) pauseTest(name string) {
p.add(Event{
Type: "pause_test",
Id: p.findTest(name),
Name: name,
})
}
func (p *parser) contTest(name string) {
p.add(Event{
Type: "cont_test",
Id: p.findTest(name),
Name: name,
})
}
func (p *parser) endTest(line, indent, result, name, duration string) {
if idx := strings.Index(line, fmt.Sprintf("%s--- %s:", indent, result)); idx > 0 {
p.output(line[:idx])
}
_, n := stripIndent(indent)
p.add(Event{
Type: "end_test",
Id: p.findTest(name),
Name: name,
Result: result,
Indent: n,
Duration: parseSeconds(duration),
})
}
func (p *parser) status(result string) {
p.add(Event{
Type: "status",
Result: result,
})
}
func (p *parser) summary(result, name, duration, data, covpct, packages string) {
p.add(Event{
Type: "summary",
Result: result,
Name: name,
Duration: parseSeconds(duration),
Data: data,
CovPct: parseCoverage(covpct),
CovPackages: parsePackages(packages),
})
}
func (p *parser) coverage(percent, packages string) {
p.add(Event{
Type: "coverage",
CovPct: parseCoverage(percent),
CovPackages: parsePackages(packages),
})
}
func (p *parser) output(line string) {
p.add(Event{
Type: "output",
Data: line,
})
}
func parseSeconds(s string) time.Duration {
if s == "" {
return time.Duration(0)
}
// ignore error
d, _ := time.ParseDuration(s + "s")
return d
}
func parseCoverage(percent string) float64 {
if percent == "" {
return 0
}
// ignore error
pct, _ := strconv.ParseFloat(percent, 64)
return pct
}
func parsePackages(pkgList string) []string {
if len(pkgList) == 0 {
return nil
}
return strings.Split(pkgList, ", ")
}
func stripIndent(line string) (string, int) {
var indent int
for indent = 0; strings.HasPrefix(line, " "); indent++ {
line = line[4:]
}
return line, indent
}