blob: 37f33b77f6eda21dcd204595b043492ae56217ac [file] [log] [blame]
// Copyright 2016 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 benchfmt provides readers and writers for the Go benchmark format.
//
// The format is documented at https://golang.org/design/14313-benchmark-format
package benchfmt
import (
"bufio"
"fmt"
"io"
"sort"
"strconv"
"strings"
"unicode"
)
// Reader reads benchmark results from an io.Reader.
// Use Next to advance through the results.
//
// br := benchfmt.NewReader(r)
// for br.Next() {
// res := br.Result()
// ...
// }
// err = br.Err() // get any error encountered during iteration
// ...
type Reader struct {
s *bufio.Scanner
labels Labels
// permLabels are permanent labels read from the start of the
// file or provided by AddLabels. They cannot be overridden.
permLabels Labels
lineNum int
// cached from last call to newResult, to save on allocations
lastName string
lastNameLabels Labels
// cached from the last call to Next
result *Result
err error
}
// NewReader creates a BenchmarkReader that reads from r.
func NewReader(r io.Reader) *Reader {
return &Reader{
s: bufio.NewScanner(r),
labels: make(Labels),
}
}
// AddLabels adds additional labels as if they had been read from the header of a file.
// It must be called before the first call to r.Next.
func (r *Reader) AddLabels(labels Labels) {
r.permLabels = labels.copy()
for k, v := range labels {
r.labels[k] = v
}
}
// Result represents a single line from a benchmark file.
// All information about that line is self-contained in the Result.
type Result struct {
// Labels is the set of persistent labels that apply to the result.
// Labels must not be modified.
Labels Labels
// NameLabels is the set of ephemeral labels that were parsed
// from the benchmark name/line.
// NameLabels must not be modified.
NameLabels Labels
// LineNum is the line number on which the result was found
LineNum int
// Content is the verbatim input line of the benchmark file, beginning with the string "Benchmark".
Content string
}
// Labels is a set of key-value strings.
type Labels map[string]string
// TODO(quentin): Add String and Equal methods to Labels?
// Keys returns a sorted list of the keys in l.
func (l Labels) Keys() []string {
var out []string
for k := range l {
out = append(out, k)
}
sort.Strings(out)
return out
}
// A Printer prints a sequence of benchmark results.
type Printer struct {
w io.Writer
labels Labels
}
// NewPrinter constructs a BenchmarkPrinter writing to w.
func NewPrinter(w io.Writer) *Printer {
return &Printer{w: w}
}
// Print writes the lines necessary to recreate r.
func (p *Printer) Print(r *Result) error {
var keys []string
// Print removed keys first.
for k := range p.labels {
if r.Labels[k] == "" {
keys = append(keys, k)
}
}
sort.Strings(keys)
for _, k := range keys {
if _, err := fmt.Fprintf(p.w, "%s:\n", k); err != nil {
return err
}
}
// Then print new or changed keys.
keys = keys[:0]
for k, v := range r.Labels {
if v != "" && p.labels[k] != v {
keys = append(keys, k)
}
}
sort.Strings(keys)
for _, k := range keys {
if _, err := fmt.Fprintf(p.w, "%s: %s\n", k, r.Labels[k]); err != nil {
return err
}
}
// Finally print the actual line itself.
if _, err := fmt.Fprintf(p.w, "%s\n", r.Content); err != nil {
return err
}
p.labels = r.Labels
return nil
}
// parseNameLabels extracts extra labels from a benchmark name and sets them in labels.
func parseNameLabels(name string, labels Labels) {
dash := strings.LastIndex(name, "-")
if dash >= 0 {
// Accept -N as an alias for /gomaxprocs=N
_, err := strconv.Atoi(name[dash+1:])
if err == nil {
labels["gomaxprocs"] = name[dash+1:]
name = name[:dash]
}
}
parts := strings.Split(name, "/")
labels["name"] = parts[0]
for i, sub := range parts[1:] {
equals := strings.Index(sub, "=")
var key string
if equals >= 0 {
key, sub = sub[:equals], sub[equals+1:]
} else {
key = fmt.Sprintf("sub%d", i+1)
}
labels[key] = sub
}
}
// newResult parses a line and returns a Result object for the line.
func (r *Reader) newResult(labels Labels, lineNum int, name, content string) *Result {
res := &Result{
Labels: labels,
LineNum: lineNum,
Content: content,
}
if r.lastName != name {
r.lastName = name
r.lastNameLabels = make(Labels)
parseNameLabels(name, r.lastNameLabels)
}
res.NameLabels = r.lastNameLabels
return res
}
// copy returns a new copy of the labels map, to protect against
// future modifications to labels.
func (l Labels) copy() Labels {
new := make(Labels)
for k, v := range l {
new[k] = v
}
return new
}
// TODO(quentin): How to represent and efficiently group multiple lines?
// Next returns the next benchmark result from the file. If there are
// no further results, it returns nil, io.EOF.
func (r *Reader) Next() bool {
if r.err != nil {
return false
}
copied := false
havePerm := r.permLabels != nil
for r.s.Scan() {
r.lineNum++
line := r.s.Text()
if key, value, ok := parseKeyValueLine(line); ok {
if _, ok := r.permLabels[key]; ok {
continue
}
if !copied {
copied = true
r.labels = r.labels.copy()
}
// TODO(quentin): Spec says empty value is valid, but
// we need a way to cancel previous labels, so we'll
// treat an empty value as a removal.
if value == "" {
delete(r.labels, key)
} else {
r.labels[key] = value
}
continue
}
// Blank line delimits the header. If we find anything else, the file must not have a header.
if !havePerm {
if line == "" {
r.permLabels = r.labels.copy()
} else {
r.permLabels = Labels{}
}
}
if fullName, ok := parseBenchmarkLine(line); ok {
r.result = r.newResult(r.labels, r.lineNum, fullName, line)
return true
}
}
if err := r.s.Err(); err != nil {
r.err = err
return false
}
r.err = io.EOF
return false
}
// Result returns the most recent result generated by a call to Next.
func (r *Reader) Result() *Result {
return r.result
}
// Err returns the error state of the reader.
func (r *Reader) Err() error {
if r.err == io.EOF {
return nil
}
return r.err
}
// parseKeyValueLine attempts to parse line as a key: value pair. ok
// indicates whether the line could be parsed.
func parseKeyValueLine(line string) (key, val string, ok bool) {
for i, c := range line {
if i == 0 && !unicode.IsLower(c) {
return
}
if unicode.IsSpace(c) || unicode.IsUpper(c) {
return
}
if i > 0 && c == ':' {
key = line[:i]
val = line[i+1:]
break
}
}
if key == "" {
return
}
if val == "" {
ok = true
return
}
for len(val) > 0 && (val[0] == ' ' || val[0] == '\t') {
val = val[1:]
ok = true
}
return
}
// parseBenchmarkLine attempts to parse line as a benchmark result. If
// successful, fullName is the name of the benchmark with the
// "Benchmark" prefix stripped, and ok is true.
func parseBenchmarkLine(line string) (fullName string, ok bool) {
space := strings.IndexFunc(line, unicode.IsSpace)
if space < 0 {
return
}
name := line[:space]
if !strings.HasPrefix(name, "Benchmark") {
return
}
return name[len("Benchmark"):], true
}