Make RunBlueprint() return an error, if need be.
This is so that it doesn't need to abruptly call os.Exit(), denying callers the opportunity to do cleanups. Bug: 244730498 Test: Presubmits. Change-Id: Ifd191d3bbbf2fdea2ca49e4fb552e5d1c557b80f
This commit is contained in:
parent
6126fe8067
commit
2cd5fe6206
2 changed files with 28 additions and 29 deletions
|
@ -16,6 +16,7 @@ package bootstrap
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"os"
|
"os"
|
||||||
|
@ -44,7 +45,7 @@ type Args struct {
|
||||||
// its dependencies. These can be written to a `${args.OutFile}.d` file
|
// its dependencies. These can be written to a `${args.OutFile}.d` file
|
||||||
// so that it is correctly rebuilt when needed in case Blueprint is itself
|
// so that it is correctly rebuilt when needed in case Blueprint is itself
|
||||||
// invoked from Ninja
|
// invoked from Ninja
|
||||||
func RunBlueprint(args Args, stopBefore StopBefore, ctx *blueprint.Context, config interface{}) []string {
|
func RunBlueprint(args Args, stopBefore StopBefore, ctx *blueprint.Context, config interface{}) ([]string, error) {
|
||||||
runtime.GOMAXPROCS(runtime.NumCPU())
|
runtime.GOMAXPROCS(runtime.NumCPU())
|
||||||
|
|
||||||
if args.NoGC {
|
if args.NoGC {
|
||||||
|
@ -54,7 +55,7 @@ func RunBlueprint(args Args, stopBefore StopBefore, ctx *blueprint.Context, conf
|
||||||
if args.Cpuprofile != "" {
|
if args.Cpuprofile != "" {
|
||||||
f, err := os.Create(joinPath(ctx.SrcDir(), args.Cpuprofile))
|
f, err := os.Create(joinPath(ctx.SrcDir(), args.Cpuprofile))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fatalf("error opening cpuprofile: %s", err)
|
return nil, fmt.Errorf("error opening cpuprofile: %s", err)
|
||||||
}
|
}
|
||||||
pprof.StartCPUProfile(f)
|
pprof.StartCPUProfile(f)
|
||||||
defer f.Close()
|
defer f.Close()
|
||||||
|
@ -64,7 +65,7 @@ func RunBlueprint(args Args, stopBefore StopBefore, ctx *blueprint.Context, conf
|
||||||
if args.TraceFile != "" {
|
if args.TraceFile != "" {
|
||||||
f, err := os.Create(joinPath(ctx.SrcDir(), args.TraceFile))
|
f, err := os.Create(joinPath(ctx.SrcDir(), args.TraceFile))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fatalf("error opening trace: %s", err)
|
return nil, fmt.Errorf("error opening trace: %s", err)
|
||||||
}
|
}
|
||||||
trace.Start(f)
|
trace.Start(f)
|
||||||
defer f.Close()
|
defer f.Close()
|
||||||
|
@ -72,7 +73,7 @@ func RunBlueprint(args Args, stopBefore StopBefore, ctx *blueprint.Context, conf
|
||||||
}
|
}
|
||||||
|
|
||||||
if args.ModuleListFile == "" {
|
if args.ModuleListFile == "" {
|
||||||
fatalf("-l <moduleListFile> is required and must be nonempty")
|
return nil, fmt.Errorf("-l <moduleListFile> is required and must be nonempty")
|
||||||
}
|
}
|
||||||
ctx.SetModuleListFile(args.ModuleListFile)
|
ctx.SetModuleListFile(args.ModuleListFile)
|
||||||
|
|
||||||
|
@ -82,7 +83,7 @@ func RunBlueprint(args Args, stopBefore StopBefore, ctx *blueprint.Context, conf
|
||||||
ctx.BeginEvent("list_modules")
|
ctx.BeginEvent("list_modules")
|
||||||
var filesToParse []string
|
var filesToParse []string
|
||||||
if f, err := ctx.ListModulePaths("."); err != nil {
|
if f, err := ctx.ListModulePaths("."); err != nil {
|
||||||
fatalf("could not enumerate files: %v\n", err.Error())
|
return nil, fmt.Errorf("could not enumerate files: %v\n", err.Error())
|
||||||
} else {
|
} else {
|
||||||
filesToParse = f
|
filesToParse = f
|
||||||
}
|
}
|
||||||
|
@ -96,36 +97,36 @@ func RunBlueprint(args Args, stopBefore StopBefore, ctx *blueprint.Context, conf
|
||||||
|
|
||||||
ctx.BeginEvent("parse_bp")
|
ctx.BeginEvent("parse_bp")
|
||||||
if blueprintFiles, errs := ctx.ParseFileList(".", filesToParse, config); len(errs) > 0 {
|
if blueprintFiles, errs := ctx.ParseFileList(".", filesToParse, config); len(errs) > 0 {
|
||||||
fatalErrors(errs)
|
return nil, fatalErrors(errs)
|
||||||
} else {
|
} else {
|
||||||
ctx.EndEvent("parse_bp")
|
ctx.EndEvent("parse_bp")
|
||||||
ninjaDeps = append(ninjaDeps, blueprintFiles...)
|
ninjaDeps = append(ninjaDeps, blueprintFiles...)
|
||||||
}
|
}
|
||||||
|
|
||||||
if resolvedDeps, errs := ctx.ResolveDependencies(config); len(errs) > 0 {
|
if resolvedDeps, errs := ctx.ResolveDependencies(config); len(errs) > 0 {
|
||||||
fatalErrors(errs)
|
return nil, fatalErrors(errs)
|
||||||
} else {
|
} else {
|
||||||
ninjaDeps = append(ninjaDeps, resolvedDeps...)
|
ninjaDeps = append(ninjaDeps, resolvedDeps...)
|
||||||
}
|
}
|
||||||
|
|
||||||
if stopBefore == StopBeforePrepareBuildActions {
|
if stopBefore == StopBeforePrepareBuildActions {
|
||||||
return ninjaDeps
|
return ninjaDeps, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
if ctx.BeforePrepareBuildActionsHook != nil {
|
if ctx.BeforePrepareBuildActionsHook != nil {
|
||||||
if err := ctx.BeforePrepareBuildActionsHook(); err != nil {
|
if err := ctx.BeforePrepareBuildActionsHook(); err != nil {
|
||||||
fatalErrors([]error{err})
|
return nil, fatalErrors([]error{err})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if buildActionsDeps, errs := ctx.PrepareBuildActions(config); len(errs) > 0 {
|
if buildActionsDeps, errs := ctx.PrepareBuildActions(config); len(errs) > 0 {
|
||||||
fatalErrors(errs)
|
return nil, fatalErrors(errs)
|
||||||
} else {
|
} else {
|
||||||
ninjaDeps = append(ninjaDeps, buildActionsDeps...)
|
ninjaDeps = append(ninjaDeps, buildActionsDeps...)
|
||||||
}
|
}
|
||||||
|
|
||||||
if stopBefore == StopBeforeWriteNinja {
|
if stopBefore == StopBeforeWriteNinja {
|
||||||
return ninjaDeps
|
return ninjaDeps, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
const outFilePermissions = 0666
|
const outFilePermissions = 0666
|
||||||
|
@ -137,13 +138,13 @@ func RunBlueprint(args Args, stopBefore StopBefore, ctx *blueprint.Context, conf
|
||||||
defer ctx.EndEvent("write_files")
|
defer ctx.EndEvent("write_files")
|
||||||
if args.EmptyNinjaFile {
|
if args.EmptyNinjaFile {
|
||||||
if err := os.WriteFile(joinPath(ctx.SrcDir(), args.OutFile), []byte(nil), outFilePermissions); err != nil {
|
if err := os.WriteFile(joinPath(ctx.SrcDir(), args.OutFile), []byte(nil), outFilePermissions); err != nil {
|
||||||
fatalf("error writing empty Ninja file: %s", err)
|
return nil, fmt.Errorf("error writing empty Ninja file: %s", err)
|
||||||
}
|
}
|
||||||
out = io.Discard.(io.StringWriter)
|
out = io.Discard.(io.StringWriter)
|
||||||
} else {
|
} else {
|
||||||
f, err := os.OpenFile(joinPath(ctx.SrcDir(), args.OutFile), os.O_WRONLY|os.O_CREATE|os.O_TRUNC, outFilePermissions)
|
f, err := os.OpenFile(joinPath(ctx.SrcDir(), args.OutFile), os.O_WRONLY|os.O_CREATE|os.O_TRUNC, outFilePermissions)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fatalf("error opening Ninja file: %s", err)
|
return nil, fmt.Errorf("error opening Ninja file: %s", err)
|
||||||
}
|
}
|
||||||
defer f.Close()
|
defer f.Close()
|
||||||
buf = bufio.NewWriterSize(f, 16*1024*1024)
|
buf = bufio.NewWriterSize(f, 16*1024*1024)
|
||||||
|
@ -151,40 +152,34 @@ func RunBlueprint(args Args, stopBefore StopBefore, ctx *blueprint.Context, conf
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := ctx.WriteBuildFile(out); err != nil {
|
if err := ctx.WriteBuildFile(out); err != nil {
|
||||||
fatalf("error writing Ninja file contents: %s", err)
|
return nil, fmt.Errorf("error writing Ninja file contents: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if buf != nil {
|
if buf != nil {
|
||||||
if err := buf.Flush(); err != nil {
|
if err := buf.Flush(); err != nil {
|
||||||
fatalf("error flushing Ninja file contents: %s", err)
|
return nil, fmt.Errorf("error flushing Ninja file contents: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if f != nil {
|
if f != nil {
|
||||||
if err := f.Close(); err != nil {
|
if err := f.Close(); err != nil {
|
||||||
fatalf("error closing Ninja file: %s", err)
|
return nil, fmt.Errorf("error closing Ninja file: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if args.Memprofile != "" {
|
if args.Memprofile != "" {
|
||||||
f, err := os.Create(joinPath(ctx.SrcDir(), args.Memprofile))
|
f, err := os.Create(joinPath(ctx.SrcDir(), args.Memprofile))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fatalf("error opening memprofile: %s", err)
|
return nil, fmt.Errorf("error opening memprofile: %s", err)
|
||||||
}
|
}
|
||||||
defer f.Close()
|
defer f.Close()
|
||||||
pprof.WriteHeapProfile(f)
|
pprof.WriteHeapProfile(f)
|
||||||
}
|
}
|
||||||
|
|
||||||
return ninjaDeps
|
return ninjaDeps, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func fatalf(format string, args ...interface{}) {
|
func fatalErrors(errs []error) error {
|
||||||
fmt.Printf(format, args...)
|
|
||||||
fmt.Print("\n")
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
func fatalErrors(errs []error) {
|
|
||||||
red := "\x1b[31m"
|
red := "\x1b[31m"
|
||||||
unred := "\x1b[0m"
|
unred := "\x1b[0m"
|
||||||
|
|
||||||
|
@ -198,7 +193,8 @@ func fatalErrors(errs []error) {
|
||||||
fmt.Printf("%sinternal error:%s %s\n", red, unred, err)
|
fmt.Printf("%sinternal error:%s %s\n", red, unred, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
os.Exit(1)
|
|
||||||
|
return errors.New("fatal errors encountered")
|
||||||
}
|
}
|
||||||
|
|
||||||
func joinPath(base, path string) string {
|
func joinPath(base, path string) string {
|
||||||
|
|
|
@ -210,18 +210,21 @@ func (s *GlobSingleton) GenerateBuildActions(ctx blueprint.SingletonContext) {
|
||||||
// Writes a .ninja file that contains instructions for regenerating the glob
|
// Writes a .ninja file that contains instructions for regenerating the glob
|
||||||
// files that contain the results of every glob that was run. The list of files
|
// files that contain the results of every glob that was run. The list of files
|
||||||
// is available as the result of GlobFileListFiles().
|
// is available as the result of GlobFileListFiles().
|
||||||
func WriteBuildGlobsNinjaFile(glob *GlobSingleton, config interface{}) {
|
func WriteBuildGlobsNinjaFile(glob *GlobSingleton, config interface{}) error {
|
||||||
buffer, errs := generateGlobNinjaFile(glob, config)
|
buffer, errs := generateGlobNinjaFile(glob, config)
|
||||||
if len(errs) > 0 {
|
if len(errs) > 0 {
|
||||||
fatalErrors(errs)
|
return fatalErrors(errs)
|
||||||
}
|
}
|
||||||
|
|
||||||
const outFilePermissions = 0666
|
const outFilePermissions = 0666
|
||||||
err := ioutil.WriteFile(joinPath(glob.SrcDir, glob.GlobFile), buffer, outFilePermissions)
|
err := ioutil.WriteFile(joinPath(glob.SrcDir, glob.GlobFile), buffer, outFilePermissions)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fatalf("error writing %s: %s", glob.GlobFile, err)
|
return fmt.Errorf("error writing %s: %s", glob.GlobFile, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func generateGlobNinjaFile(glob *GlobSingleton, config interface{}) ([]byte, []error) {
|
func generateGlobNinjaFile(glob *GlobSingleton, config interface{}) ([]byte, []error) {
|
||||||
|
|
||||||
ctx := blueprint.NewContext()
|
ctx := blueprint.NewContext()
|
||||||
|
|
Loading…
Reference in a new issue