Joël Stemmer 85f2715ac9 parser/gotest: Create JSONEventReader in internal reader package
The JSONEventReader implements reading lines with metadata and replaces
the gotest.jsonReader struct.
2022-08-15 22:07:52 +01:00

123 lines
3.0 KiB
Go

package reader
import (
"bufio"
"bytes"
"encoding/json"
"io"
"strings"
"time"
)
// LineReader is an interface to read lines with optional Metadata.
type LineReader interface {
ReadLine() (string, *Metadata, error)
}
// Metadata contains metadata that belongs to a line.
type Metadata struct {
Package string
}
// LimitedLineReader reads lines from an io.Reader object with a configurable
// line size limit. Lines exceeding the limit will be truncated, but read
// completely from the underlying io.Reader.
type LimitedLineReader struct {
r *bufio.Reader
limit int
}
var _ LineReader = &LimitedLineReader{}
// NewLimitedLineReader returns a LimitedLineReader to read lines from r with a
// maximum line size of limit.
func NewLimitedLineReader(r io.Reader, limit int) *LimitedLineReader {
return &LimitedLineReader{r: bufio.NewReader(r), limit: limit}
}
// ReadLine returns the next line from the underlying reader. The length of the
// line will not exceed the configured limit. ReadLine either returns a line or
// it returns an error, never both.
func (r *LimitedLineReader) ReadLine() (string, *Metadata, error) {
line, isPrefix, err := r.r.ReadLine()
if err != nil {
return "", nil, err
}
if !isPrefix {
return string(line), nil, nil
}
// Line is incomplete, keep reading until we reach the end of the line.
var buf bytes.Buffer
buf.Write(line) // ignore err, always nil
for isPrefix {
line, isPrefix, err = r.r.ReadLine()
if err != nil {
return "", nil, err
}
if buf.Len() >= r.limit {
// Stop writing to buf if we exceed the limit. We continue reading
// however to make sure we consume the entire line.
continue
}
buf.Write(line) // ignore err, always nil
}
if buf.Len() > r.limit {
buf.Truncate(r.limit)
}
return buf.String(), nil, nil
}
// Event represents a JSON event emitted by `go test -json`.
type Event struct {
Time time.Time
Action string
Package string
Test string
Elapsed float64 // seconds
Output string
}
// JSONEventReader reads JSON events from an io.Reader object.
type JSONEventReader struct {
r *LimitedLineReader
}
var _ LineReader = &JSONEventReader{}
// jsonLineLimit is the maximum size of a single JSON line emitted by `go test
// -json`.
const jsonLineLimit = 64 * 1024
// NewJSONEventReader returns a JSONEventReader to read the data in JSON
// events from r.
func NewJSONEventReader(r io.Reader) *JSONEventReader {
return &JSONEventReader{NewLimitedLineReader(r, jsonLineLimit)}
}
// ReadLine returns the next line from the underlying reader.
func (r *JSONEventReader) ReadLine() (string, *Metadata, error) {
for {
line, _, err := r.r.ReadLine()
if err != nil {
return "", nil, err
}
if len(line) == 0 || line[0] != '{' {
return line, nil, nil
}
event := &Event{}
if err := json.Unmarshal([]byte(line), event); err != nil {
return "", nil, err
}
if event.Output == "" {
// Skip events without output
continue
}
return strings.TrimSuffix(event.Output, "\n"), &Metadata{Package: event.Package}, nil
}
}