167 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			167 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Go
		
	
	
	
// 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"
 | 
						|
 | 
						|
	"code.google.com/p/go.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:	init			100.0%
 | 
						|
//	fmt/format.go:	computePadding		84.6%
 | 
						|
//	...
 | 
						|
//	fmt/scan.go:	doScan			100.0%
 | 
						|
//	fmt/scan.go:	advance			96.2%
 | 
						|
//	fmt/scan.go:	doScanf			96.8%
 | 
						|
//	total:		(statements)		91.4%
 | 
						|
 | 
						|
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:\t%s\t%.1f%%\n", fn, 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
 | 
						|
}
 |