| | |
| | |
| | |
| |
|
| | package main |
| |
|
| | import ( |
| | "bufio" |
| | "cmd/internal/browser" |
| | "fmt" |
| | "html/template" |
| | "io" |
| | "math" |
| | "os" |
| | "path/filepath" |
| | "strings" |
| |
|
| | "golang.org/x/tools/cover" |
| | ) |
| |
|
| | |
| | |
| | |
| | func htmlOutput(profile, outfile string) error { |
| | profiles, err := cover.ParseProfiles(profile) |
| | if err != nil { |
| | return err |
| | } |
| |
|
| | var d templateData |
| |
|
| | dirs, err := findPkgs(profiles) |
| | if err != nil { |
| | return err |
| | } |
| |
|
| | for _, profile := range profiles { |
| | fn := profile.FileName |
| | if profile.Mode == "set" { |
| | d.Set = true |
| | } |
| | file, err := findFile(dirs, fn) |
| | if err != nil { |
| | return err |
| | } |
| | src, err := os.ReadFile(file) |
| | if err != nil { |
| | return fmt.Errorf("can't read %q: %v", fn, err) |
| | } |
| | var buf strings.Builder |
| | err = htmlGen(&buf, src, profile.Boundaries(src)) |
| | if err != nil { |
| | return err |
| | } |
| | d.Files = append(d.Files, &templateFile{ |
| | Name: fn, |
| | Body: template.HTML(buf.String()), |
| | Coverage: percentCovered(profile), |
| | }) |
| | } |
| |
|
| | var out *os.File |
| | if outfile == "" { |
| | var dir string |
| | dir, err = os.MkdirTemp("", "cover") |
| | if err != nil { |
| | return err |
| | } |
| | out, err = os.Create(filepath.Join(dir, "coverage.html")) |
| | } else { |
| | out, err = os.Create(outfile) |
| | } |
| | if err != nil { |
| | return err |
| | } |
| | err = htmlTemplate.Execute(out, d) |
| | if err2 := out.Close(); err == nil { |
| | err = err2 |
| | } |
| | if err != nil { |
| | return err |
| | } |
| |
|
| | if outfile == "" { |
| | if !browser.Open("file://" + out.Name()) { |
| | fmt.Fprintf(os.Stderr, "HTML output written to %s\n", out.Name()) |
| | } |
| | } |
| |
|
| | return nil |
| | } |
| |
|
| | |
| | |
| | |
| | func percentCovered(p *cover.Profile) float64 { |
| | var total, covered int64 |
| | for _, b := range p.Blocks { |
| | total += int64(b.NumStmt) |
| | if b.Count > 0 { |
| | covered += int64(b.NumStmt) |
| | } |
| | } |
| | if total == 0 { |
| | return 0 |
| | } |
| | return float64(covered) / float64(total) * 100 |
| | } |
| |
|
| | |
| | |
| | func htmlGen(w io.Writer, src []byte, boundaries []cover.Boundary) error { |
| | dst := bufio.NewWriter(w) |
| | for i := range src { |
| | for len(boundaries) > 0 && boundaries[0].Offset == i { |
| | b := boundaries[0] |
| | if b.Start { |
| | n := 0 |
| | if b.Count > 0 { |
| | n = int(math.Floor(b.Norm*9)) + 1 |
| | } |
| | fmt.Fprintf(dst, `<span class="cov%v" title="%v">`, n, b.Count) |
| | } else { |
| | dst.WriteString("</span>") |
| | } |
| | boundaries = boundaries[1:] |
| | } |
| | switch b := src[i]; b { |
| | case '>': |
| | dst.WriteString(">") |
| | case '<': |
| | dst.WriteString("<") |
| | case '&': |
| | dst.WriteString("&") |
| | case '\t': |
| | dst.WriteString(" ") |
| | default: |
| | dst.WriteByte(b) |
| | } |
| | } |
| | return dst.Flush() |
| | } |
| |
|
| | |
| | |
| | func rgb(n int) string { |
| | if n == 0 { |
| | return "rgb(192, 0, 0)" |
| | } |
| | |
| | r := 128 - 12*(n-1) |
| | g := 128 + 12*(n-1) |
| | b := 128 + 3*(n-1) |
| | return fmt.Sprintf("rgb(%v, %v, %v)", r, g, b) |
| | } |
| |
|
| | |
| | func colors() template.CSS { |
| | var buf strings.Builder |
| | for i := 0; i < 11; i++ { |
| | fmt.Fprintf(&buf, ".cov%v { color: %v }\n", i, rgb(i)) |
| | } |
| | return template.CSS(buf.String()) |
| | } |
| |
|
| | var htmlTemplate = template.Must(template.New("html").Funcs(template.FuncMap{ |
| | "colors": colors, |
| | }).Parse(tmplHTML)) |
| |
|
| | type templateData struct { |
| | Files []*templateFile |
| | Set bool |
| | } |
| |
|
| | |
| | |
| | |
| | |
| | |
| | |
| | func (td templateData) PackageName() string { |
| | if len(td.Files) == 0 { |
| | return "" |
| | } |
| | fileName := td.Files[0].Name |
| | elems := strings.Split(fileName, "/") |
| | |
| | for i := len(elems) - 2; i >= 0; i-- { |
| | if elems[i] != "" { |
| | return elems[i] |
| | } |
| | } |
| | return "" |
| | } |
| |
|
| | type templateFile struct { |
| | Name string |
| | Body template.HTML |
| | Coverage float64 |
| | } |
| |
|
| | const tmplHTML = ` |
| | <!DOCTYPE html> |
| | <html> |
| | <head> |
| | <meta http-equiv="Content-Type" content="text/html; charset=utf-8"> |
| | <title>{{$pkg := .PackageName}}{{if $pkg}}{{$pkg}}: {{end}}Go Coverage Report</title> |
| | <style> |
| | body { |
| | background: black; |
| | color: rgb(80, 80, 80); |
| | } |
| | body, pre, #legend span { |
| | font-family: Menlo, monospace; |
| | font-weight: bold; |
| | } |
| | #topbar { |
| | background: black; |
| | position: fixed; |
| | top: 0; left: 0; right: 0; |
| | height: 42px; |
| | border-bottom: 1px solid rgb(80, 80, 80); |
| | } |
| | #content { |
| | margin-top: 50px; |
| | } |
| | #nav, #legend { |
| | float: left; |
| | margin-left: 10px; |
| | } |
| | #legend { |
| | margin-top: 12px; |
| | } |
| | #nav { |
| | margin-top: 10px; |
| | } |
| | #legend span { |
| | margin: 0 5px; |
| | } |
| | {{colors}} |
| | </style> |
| | </head> |
| | <body> |
| | <div id="topbar"> |
| | <div id="nav"> |
| | <select id="files"> |
| | {{range $i, $f := .Files}} |
| | <option value="file{{$i}}">{{$f.Name}} ({{printf "%.1f" $f.Coverage}}%)</option> |
| | {{end}} |
| | </select> |
| | </div> |
| | <div id="legend"> |
| | <span>not tracked</span> |
| | {{if .Set}} |
| | <span class="cov0">not covered</span> |
| | <span class="cov8">covered</span> |
| | {{else}} |
| | <span class="cov0">no coverage</span> |
| | <span class="cov1">low coverage</span> |
| | <span class="cov2">*</span> |
| | <span class="cov3">*</span> |
| | <span class="cov4">*</span> |
| | <span class="cov5">*</span> |
| | <span class="cov6">*</span> |
| | <span class="cov7">*</span> |
| | <span class="cov8">*</span> |
| | <span class="cov9">*</span> |
| | <span class="cov10">high coverage</span> |
| | {{end}} |
| | </div> |
| | </div> |
| | <div id="content"> |
| | {{range $i, $f := .Files}} |
| | <pre class="file" id="file{{$i}}" style="display: none">{{$f.Body}}</pre> |
| | {{end}} |
| | </div> |
| | </body> |
| | <script> |
| | (function() { |
| | var files = document.getElementById('files'); |
| | var visible; |
| | files.addEventListener('change', onChange, false); |
| | function select(part) { |
| | if (visible) |
| | visible.style.display = 'none'; |
| | visible = document.getElementById(part); |
| | if (!visible) |
| | return; |
| | files.value = part; |
| | visible.style.display = 'block'; |
| | location.hash = part; |
| | } |
| | function onChange() { |
| | select(files.value); |
| | window.scrollTo(0, 0); |
| | } |
| | if (location.hash != "") { |
| | select(location.hash.substr(1)); |
| | } |
| | if (!visible) { |
| | select("file0"); |
| | } |
| | })(); |
| | </script> |
| | </html> |
| | ` |
| |
|