123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256 |
- // Copyright 2013 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 cover provides support for parsing coverage profiles
- // generated by "go test -coverprofile=cover.out".
- package cover // import "golang.org/x/tools/cover"
- import (
- "bufio"
- "errors"
- "fmt"
- "math"
- "os"
- "sort"
- "strconv"
- "strings"
- )
- // Profile represents the profiling data for a specific file.
- type Profile struct {
- FileName string
- Mode string
- Blocks []ProfileBlock
- }
- // ProfileBlock represents a single block of profiling data.
- type ProfileBlock struct {
- StartLine, StartCol int
- EndLine, EndCol int
- NumStmt, Count int
- }
- type byFileName []*Profile
- func (p byFileName) Len() int { return len(p) }
- func (p byFileName) Less(i, j int) bool { return p[i].FileName < p[j].FileName }
- func (p byFileName) Swap(i, j int) { p[i], p[j] = p[j], p[i] }
- // ParseProfiles parses profile data in the specified file and returns a
- // Profile for each source file described therein.
- func ParseProfiles(fileName string) ([]*Profile, error) {
- pf, err := os.Open(fileName)
- if err != nil {
- return nil, err
- }
- defer pf.Close()
- files := make(map[string]*Profile)
- buf := bufio.NewReader(pf)
- // First line is "mode: foo", where foo is "set", "count", or "atomic".
- // Rest of file is in the format
- // encoding/base64/base64.go:34.44,37.40 3 1
- // where the fields are: name.go:line.column,line.column numberOfStatements count
- s := bufio.NewScanner(buf)
- mode := ""
- for s.Scan() {
- line := s.Text()
- if mode == "" {
- const p = "mode: "
- if !strings.HasPrefix(line, p) || line == p {
- return nil, fmt.Errorf("bad mode line: %v", line)
- }
- mode = line[len(p):]
- continue
- }
- fn, b, err := parseLine(line)
- if err != nil {
- return nil, fmt.Errorf("line %q doesn't match expected format: %v", line, err)
- }
- p := files[fn]
- if p == nil {
- p = &Profile{
- FileName: fn,
- Mode: mode,
- }
- files[fn] = p
- }
- p.Blocks = append(p.Blocks, b)
- }
- if err := s.Err(); err != nil {
- return nil, err
- }
- for _, p := range files {
- sort.Sort(blocksByStart(p.Blocks))
- // Merge samples from the same location.
- j := 1
- for i := 1; i < len(p.Blocks); i++ {
- b := p.Blocks[i]
- last := p.Blocks[j-1]
- if b.StartLine == last.StartLine &&
- b.StartCol == last.StartCol &&
- b.EndLine == last.EndLine &&
- b.EndCol == last.EndCol {
- if b.NumStmt != last.NumStmt {
- return nil, fmt.Errorf("inconsistent NumStmt: changed from %d to %d", last.NumStmt, b.NumStmt)
- }
- if mode == "set" {
- p.Blocks[j-1].Count |= b.Count
- } else {
- p.Blocks[j-1].Count += b.Count
- }
- continue
- }
- p.Blocks[j] = b
- j++
- }
- p.Blocks = p.Blocks[:j]
- }
- // Generate a sorted slice.
- profiles := make([]*Profile, 0, len(files))
- for _, profile := range files {
- profiles = append(profiles, profile)
- }
- sort.Sort(byFileName(profiles))
- return profiles, nil
- }
- // parseLine parses a line from a coverage file.
- // It is equivalent to the regex
- // ^(.+):([0-9]+)\.([0-9]+),([0-9]+)\.([0-9]+) ([0-9]+) ([0-9]+)$
- //
- // However, it is much faster: https://golang.org/cl/179377
- func parseLine(l string) (fileName string, block ProfileBlock, err error) {
- end := len(l)
- b := ProfileBlock{}
- b.Count, end, err = seekBack(l, ' ', end, "Count")
- if err != nil {
- return "", b, err
- }
- b.NumStmt, end, err = seekBack(l, ' ', end, "NumStmt")
- if err != nil {
- return "", b, err
- }
- b.EndCol, end, err = seekBack(l, '.', end, "EndCol")
- if err != nil {
- return "", b, err
- }
- b.EndLine, end, err = seekBack(l, ',', end, "EndLine")
- if err != nil {
- return "", b, err
- }
- b.StartCol, end, err = seekBack(l, '.', end, "StartCol")
- if err != nil {
- return "", b, err
- }
- b.StartLine, end, err = seekBack(l, ':', end, "StartLine")
- if err != nil {
- return "", b, err
- }
- fn := l[0:end]
- if fn == "" {
- return "", b, errors.New("a FileName cannot be blank")
- }
- return fn, b, nil
- }
- // seekBack searches backwards from end to find sep in l, then returns the
- // value between sep and end as an integer.
- // If seekBack fails, the returned error will reference what.
- func seekBack(l string, sep byte, end int, what string) (value int, nextSep int, err error) {
- // Since we're seeking backwards and we know only ASCII is legal for these values,
- // we can ignore the possibility of non-ASCII characters.
- for start := end - 1; start >= 0; start-- {
- if l[start] == sep {
- i, err := strconv.Atoi(l[start+1 : end])
- if err != nil {
- return 0, 0, fmt.Errorf("couldn't parse %q: %v", what, err)
- }
- if i < 0 {
- return 0, 0, fmt.Errorf("negative values are not allowed for %s, found %d", what, i)
- }
- return i, start, nil
- }
- }
- return 0, 0, fmt.Errorf("couldn't find a %s before %s", string(sep), what)
- }
- type blocksByStart []ProfileBlock
- func (b blocksByStart) Len() int { return len(b) }
- func (b blocksByStart) Swap(i, j int) { b[i], b[j] = b[j], b[i] }
- func (b blocksByStart) Less(i, j int) bool {
- bi, bj := b[i], b[j]
- return bi.StartLine < bj.StartLine || bi.StartLine == bj.StartLine && bi.StartCol < bj.StartCol
- }
- // Boundary represents the position in a source file of the beginning or end of a
- // block as reported by the coverage profile. In HTML mode, it will correspond to
- // the opening or closing of a <span> tag and will be used to colorize the source
- type Boundary struct {
- Offset int // Location as a byte offset in the source file.
- Start bool // Is this the start of a block?
- Count int // Event count from the cover profile.
- Norm float64 // Count normalized to [0..1].
- }
- // Boundaries returns a Profile as a set of Boundary objects within the provided src.
- func (p *Profile) Boundaries(src []byte) (boundaries []Boundary) {
- // Find maximum count.
- max := 0
- for _, b := range p.Blocks {
- if b.Count > max {
- max = b.Count
- }
- }
- // Divisor for normalization.
- divisor := math.Log(float64(max))
- // boundary returns a Boundary, populating the Norm field with a normalized Count.
- boundary := func(offset int, start bool, count int) Boundary {
- b := Boundary{Offset: offset, Start: start, Count: count}
- if !start || count == 0 {
- return b
- }
- if max <= 1 {
- b.Norm = 0.8 // Profile is in"set" mode; we want a heat map. Use cov8 in the CSS.
- } else if count > 0 {
- b.Norm = math.Log(float64(count)) / divisor
- }
- return b
- }
- line, col := 1, 2 // TODO: Why is this 2?
- for si, bi := 0, 0; si < len(src) && bi < len(p.Blocks); {
- b := p.Blocks[bi]
- if b.StartLine == line && b.StartCol == col {
- boundaries = append(boundaries, boundary(si, true, b.Count))
- }
- if b.EndLine == line && b.EndCol == col || line > b.EndLine {
- boundaries = append(boundaries, boundary(si, false, 0))
- bi++
- continue // Don't advance through src; maybe the next block starts here.
- }
- if src[si] == '\n' {
- line++
- col = 0
- }
- col++
- si++
- }
- sort.Sort(boundariesByPos(boundaries))
- return
- }
- type boundariesByPos []Boundary
- func (b boundariesByPos) Len() int { return len(b) }
- func (b boundariesByPos) Swap(i, j int) { b[i], b[j] = b[j], b[i] }
- func (b boundariesByPos) Less(i, j int) bool {
- if b[i].Offset == b[j].Offset {
- return !b[i].Start && b[j].Start
- }
- return b[i].Offset < b[j].Offset
- }
|