// 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.
|
|
|
|
// This file implements the visitor that computes the (line, column)-(line-column) range for each function.
|
|
|
|
package main
|
|
|
|
import (
|
|
"bufio"
|
|
"fmt"
|
|
"go/ast"
|
|
"go/build"
|
|
"go/parser"
|
|
"go/token"
|
|
"os"
|
|
"path/filepath"
|
|
"text/tabwriter"
|
|
|
|
"golang.org/x/tools/cover"
|
|
)
|
|
|
|
// funcOutput takes two file names as arguments, a coverage profile to read as input and an output
|
|
// file to write ("" means to write to standard output). The function reads the profile and produces
|
|
// as output the coverage data broken down by function, like this:
|
|
//
|
|
// fmt/format.go:30: init 100.0%
|
|
// fmt/format.go:57: clearflags 100.0%
|
|
// ...
|
|
// fmt/scan.go:1046: doScan 100.0%
|
|
// fmt/scan.go:1075: advance 96.2%
|
|
// fmt/scan.go:1119: doScanf 96.8%
|
|
// total: (statements) 91.9%
|
|
|
|
func funcOutput(profile, outputFile string) error {
|
|
profiles, err := cover.ParseProfiles(profile)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var out *bufio.Writer
|
|
if outputFile == "" {
|
|
out = bufio.NewWriter(os.Stdout)
|
|
} else {
|
|
fd, err := os.Create(outputFile)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer fd.Close()
|
|
out = bufio.NewWriter(fd)
|
|
}
|
|
defer out.Flush()
|
|
|
|
tabber := tabwriter.NewWriter(out, 1, 8, 1, '\t', 0)
|
|
defer tabber.Flush()
|
|
|
|
var total, covered int64
|
|
for _, profile := range profiles {
|
|
fn := profile.FileName
|
|
file, err := findFile(fn)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
funcs, err := findFuncs(file)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// Now match up functions and profile blocks.
|
|
for _, f := range funcs {
|
|
c, t := f.coverage(profile)
|
|
fmt.Fprintf(tabber, "%s:%d:\t%s\t%.1f%%\n", fn, f.startLine, f.name, 100.0*float64(c)/float64(t))
|
|
total += t
|
|
covered += c
|
|
}
|
|
}
|
|
fmt.Fprintf(tabber, "total:\t(statements)\t%.1f%%\n", 100.0*float64(covered)/float64(total))
|
|
|
|
return nil
|
|
}
|
|
|
|
// findFuncs parses the file and returns a slice of FuncExtent descriptors.
|
|
func findFuncs(name string) ([]*FuncExtent, error) {
|
|
fset := token.NewFileSet()
|
|
parsedFile, err := parser.ParseFile(fset, name, nil, 0)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
visitor := &FuncVisitor{
|
|
fset: fset,
|
|
name: name,
|
|
astFile: parsedFile,
|
|
}
|
|
ast.Walk(visitor, visitor.astFile)
|
|
return visitor.funcs, nil
|
|
}
|
|
|
|
// FuncExtent describes a function's extent in the source by file and position.
|
|
type FuncExtent struct {
|
|
name string
|
|
startLine int
|
|
startCol int
|
|
endLine int
|
|
endCol int
|
|
}
|
|
|
|
// FuncVisitor implements the visitor that builds the function position list for a file.
|
|
type FuncVisitor struct {
|
|
fset *token.FileSet
|
|
name string // Name of file.
|
|
astFile *ast.File
|
|
funcs []*FuncExtent
|
|
}
|
|
|
|
// Visit implements the ast.Visitor interface.
|
|
func (v *FuncVisitor) Visit(node ast.Node) ast.Visitor {
|
|
switch n := node.(type) {
|
|
case *ast.FuncDecl:
|
|
start := v.fset.Position(n.Pos())
|
|
end := v.fset.Position(n.End())
|
|
fe := &FuncExtent{
|
|
name: n.Name.Name,
|
|
startLine: start.Line,
|
|
startCol: start.Column,
|
|
endLine: end.Line,
|
|
endCol: end.Column,
|
|
}
|
|
v.funcs = append(v.funcs, fe)
|
|
}
|
|
return v
|
|
}
|
|
|
|
// coverage returns the fraction of the statements in the function that were covered, as a numerator and denominator.
|
|
func (f *FuncExtent) coverage(profile *cover.Profile) (num, den int64) {
|
|
// We could avoid making this n^2 overall by doing a single scan and annotating the functions,
|
|
// but the sizes of the data structures is never very large and the scan is almost instantaneous.
|
|
var covered, total int64
|
|
// The blocks are sorted, so we can stop counting as soon as we reach the end of the relevant block.
|
|
for _, b := range profile.Blocks {
|
|
if b.StartLine > f.endLine || (b.StartLine == f.endLine && b.StartCol >= f.endCol) {
|
|
// Past the end of the function.
|
|
break
|
|
}
|
|
if b.EndLine < f.startLine || (b.EndLine == f.startLine && b.EndCol <= f.startCol) {
|
|
// Before the beginning of the function
|
|
continue
|
|
}
|
|
total += int64(b.NumStmt)
|
|
if b.Count > 0 {
|
|
covered += int64(b.NumStmt)
|
|
}
|
|
}
|
|
if total == 0 {
|
|
total = 1 // Avoid zero denominator.
|
|
}
|
|
return covered, total
|
|
}
|
|
|
|
// findFile finds the location of the named file in GOROOT, GOPATH etc.
|
|
func findFile(file string) (string, error) {
|
|
dir, file := filepath.Split(file)
|
|
pkg, err := build.Import(dir, ".", build.FindOnly)
|
|
if err != nil {
|
|
return "", fmt.Errorf("can't find %q: %v", file, err)
|
|
}
|
|
return filepath.Join(pkg.Dir, file), nil
|
|
}
|