mirror of
				https://github.com/go-gitea/gitea
				synced 2025-10-31 03:18:24 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			262 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Go
		
	
	
	
		
			Vendored
		
	
	
	
			
		
		
	
	
			262 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Go
		
	
	
	
		
			Vendored
		
	
	
	
| // 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].
 | |
| 	Index  int     // Order in input file.
 | |
| }
 | |
| 
 | |
| // 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.
 | |
| 	index := 0
 | |
| 	boundary := func(offset int, start bool, count int) Boundary {
 | |
| 		b := Boundary{Offset: offset, Start: start, Count: count, Index: index}
 | |
| 		index++
 | |
| 		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 {
 | |
| 		// Boundaries at the same offset should be ordered according to
 | |
| 		// their original position.
 | |
| 		return b[i].Index < b[j].Index
 | |
| 	}
 | |
| 	return b[i].Offset < b[j].Offset
 | |
| }
 |