blob: c0acc61987ef986e54e5b99cf167fe5e4f325fb5 [file] [log] [blame]
// Copyright 2023 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.
// Action graph execution methods related to coverage.
package work
import (
"cmd/go/internal/base"
"cmd/go/internal/cfg"
"cmd/go/internal/str"
"cmd/internal/cov/covcmd"
"context"
"encoding/json"
"fmt"
"internal/coverage"
"io"
"os"
"path/filepath"
)
// CovData invokes "go tool covdata" with the specified arguments
// as part of the execution of action 'a'.
func (b *Builder) CovData(a *Action, cmdargs ...any) ([]byte, error) {
cmdline := str.StringList(cmdargs...)
args := append([]string{}, cfg.BuildToolexec...)
args = append(args, base.Tool("covdata"))
args = append(args, cmdline...)
return b.Shell(a).runOut(a.Objdir, nil, args)
}
// BuildActionCoverMetaFile locates and returns the path of the
// meta-data file written by the "go tool cover" step as part of the
// build action for the "go test -cover" run action 'runAct'. Note
// that if the package has no functions the meta-data file will exist
// but will be empty; in this case the return is an empty string.
func BuildActionCoverMetaFile(runAct *Action) (string, error) {
p := runAct.Package
for i := range runAct.Deps {
pred := runAct.Deps[i]
if pred.Mode != "build" || pred.Package == nil {
continue
}
if pred.Package.ImportPath == p.ImportPath {
metaFile := pred.Objdir + covcmd.MetaFileForPackage(p.ImportPath)
f, err := os.Open(metaFile)
if err != nil {
return "", err
}
defer f.Close()
fi, err2 := f.Stat()
if err2 != nil {
return "", err2
}
if fi.Size() == 0 {
return "", nil
}
return metaFile, nil
}
}
return "", fmt.Errorf("internal error: unable to locate build action for package %q run action", p.ImportPath)
}
// WriteCoveragePercent writes out to the writer 'w' a "percent
// statements covered" for the package whose test-run action is
// 'runAct', based on the meta-data file 'mf'. This helper is used in
// cases where a user runs "go test -cover" on a package that has
// functions but no tests; in the normal case (package has tests)
// the percentage is written by the test binary when it runs.
func WriteCoveragePercent(b *Builder, runAct *Action, mf string, w io.Writer) error {
dir := filepath.Dir(mf)
output, cerr := b.CovData(runAct, "percent", "-i", dir)
if cerr != nil {
return b.Shell(runAct).reportCmd("", "", output, cerr)
}
_, werr := w.Write(output)
return werr
}
// WriteCoverageProfile writes out a coverage profile fragment for the
// package whose test-run action is 'runAct'; content is written to
// the file 'outf' based on the coverage meta-data info found in
// 'mf'. This helper is used in cases where a user runs "go test
// -cover" on a package that has functions but no tests.
func WriteCoverageProfile(b *Builder, runAct *Action, mf, outf string, w io.Writer) error {
dir := filepath.Dir(mf)
output, err := b.CovData(runAct, "textfmt", "-i", dir, "-o", outf)
if err != nil {
return b.Shell(runAct).reportCmd("", "", output, err)
}
_, werr := w.Write(output)
return werr
}
// WriteCoverMetaFilesFile writes out a summary file ("meta-files
// file") as part of the action function for the "writeCoverMeta"
// pseudo action employed during "go test -coverpkg" runs where there
// are multiple tests and multiple packages covered. It builds up a
// table mapping package import path to meta-data file fragment and
// writes it out to a file where it can be read by the various test
// run actions. Note that this function has to be called A) after the
// build actions are complete for all packages being tested, and B)
// before any of the "run test" actions for those packages happen.
// This requirement is enforced by adding making this action ("a")
// dependent on all test package build actions, and making all test
// run actions dependent on this action.
func WriteCoverMetaFilesFile(b *Builder, ctx context.Context, a *Action) error {
sh := b.Shell(a)
// Build the metafilecollection object.
var collection coverage.MetaFileCollection
for i := range a.Deps {
dep := a.Deps[i]
if dep.Mode != "build" {
panic("unexpected mode " + dep.Mode)
}
metaFilesFile := dep.Objdir + covcmd.MetaFileForPackage(dep.Package.ImportPath)
// Check to make sure the meta-data file fragment exists
// and has content (may be empty if package has no functions).
if fi, err := os.Stat(metaFilesFile); err != nil {
continue
} else if fi.Size() == 0 {
continue
}
collection.ImportPaths = append(collection.ImportPaths, dep.Package.ImportPath)
collection.MetaFileFragments = append(collection.MetaFileFragments, metaFilesFile)
}
// Serialize it.
data, err := json.Marshal(collection)
if err != nil {
return fmt.Errorf("marshal MetaFileCollection: %v", err)
}
data = append(data, '\n') // makes -x output more readable
// Create the directory for this action's objdir and
// then write out the serialized collection
// to a file in the directory.
if err := sh.Mkdir(a.Objdir); err != nil {
return err
}
mfpath := a.Objdir + coverage.MetaFilesFileName
if err := sh.writeFile(mfpath, data); err != nil {
return fmt.Errorf("writing metafiles file: %v", err)
}
// We're done.
return nil
}