2020-05-15 16:19:20 +02:00
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
2021-04-21 21:36:53 +02:00
|
|
|
"fmt"
|
|
|
|
"os"
|
2021-01-13 19:21:04 +02:00
|
|
|
"runtime"
|
2020-05-15 16:19:20 +02:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/apex/log"
|
|
|
|
"github.com/caarlos0/ctrlc"
|
|
|
|
"github.com/fatih/color"
|
2021-09-18 15:21:29 +02:00
|
|
|
"github.com/goreleaser/goreleaser/internal/middleware/errhandler"
|
|
|
|
"github.com/goreleaser/goreleaser/internal/middleware/logging"
|
|
|
|
"github.com/goreleaser/goreleaser/internal/middleware/skip"
|
2020-05-15 16:19:20 +02:00
|
|
|
"github.com/goreleaser/goreleaser/internal/pipeline"
|
2021-04-21 21:36:53 +02:00
|
|
|
"github.com/goreleaser/goreleaser/pkg/config"
|
2020-05-15 16:19:20 +02:00
|
|
|
"github.com/goreleaser/goreleaser/pkg/context"
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
)
|
|
|
|
|
|
|
|
type buildCmd struct {
|
|
|
|
cmd *cobra.Command
|
|
|
|
opts buildOpts
|
|
|
|
}
|
|
|
|
|
|
|
|
type buildOpts struct {
|
|
|
|
config string
|
2021-04-21 21:36:53 +02:00
|
|
|
id string
|
2020-05-15 16:19:20 +02:00
|
|
|
snapshot bool
|
|
|
|
skipValidate bool
|
|
|
|
skipPostHooks bool
|
|
|
|
rmDist bool
|
|
|
|
deprecated bool
|
|
|
|
parallelism int
|
|
|
|
timeout time.Duration
|
2021-04-21 21:36:53 +02:00
|
|
|
singleTarget bool
|
2020-05-15 16:19:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func newBuildCmd() *buildCmd {
|
2021-04-21 21:36:53 +02:00
|
|
|
root := &buildCmd{}
|
2020-05-15 16:19:20 +02:00
|
|
|
// nolint: dupl
|
2021-04-21 21:36:53 +02:00
|
|
|
cmd := &cobra.Command{
|
2021-04-25 19:20:49 +02:00
|
|
|
Use: "build",
|
|
|
|
Aliases: []string{"b"},
|
|
|
|
Short: "Builds the current project",
|
2021-04-22 15:58:58 +02:00
|
|
|
Long: `The build command allows you to execute only a subset of the pipeline, i.e. only the build step with its dependencies.
|
|
|
|
|
|
|
|
It allows you to quickly check if your GoReleaser build configurations are doing what you expect.
|
|
|
|
|
2021-04-25 19:20:49 +02:00
|
|
|
Finally, it allows you to generate a local build for your current machine only using the ` + "`--single-target`" + ` option, and specific build IDs using the ` + "`--id`" + ` option.
|
2021-04-22 15:58:58 +02:00
|
|
|
`,
|
2020-05-15 16:19:20 +02:00
|
|
|
SilenceUsage: true,
|
|
|
|
SilenceErrors: true,
|
2020-11-12 22:25:27 +02:00
|
|
|
Args: cobra.NoArgs,
|
2020-05-15 16:19:20 +02:00
|
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
|
|
start := time.Now()
|
|
|
|
|
|
|
|
log.Infof(color.New(color.Bold).Sprint("building..."))
|
|
|
|
|
|
|
|
ctx, err := buildProject(root.opts)
|
|
|
|
if err != nil {
|
|
|
|
return wrapError(err, color.New(color.Bold).Sprintf("build failed after %0.2fs", time.Since(start).Seconds()))
|
|
|
|
}
|
|
|
|
|
|
|
|
if ctx.Deprecated {
|
|
|
|
log.Warn(color.New(color.Bold).Sprintf("your config is using deprecated properties, check logs above for details"))
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Infof(color.New(color.Bold).Sprintf("build succeeded after %0.2fs", time.Since(start).Seconds()))
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
cmd.Flags().StringVarP(&root.opts.config, "config", "f", "", "Load configuration from file")
|
2021-04-22 15:58:58 +02:00
|
|
|
cmd.Flags().BoolVar(&root.opts.snapshot, "snapshot", false, "Generate an unversioned snapshot build, skipping all validations")
|
2020-05-15 16:19:20 +02:00
|
|
|
cmd.Flags().BoolVar(&root.opts.skipValidate, "skip-validate", false, "Skips several sanity checks")
|
|
|
|
cmd.Flags().BoolVar(&root.opts.skipPostHooks, "skip-post-hooks", false, "Skips all post-build hooks")
|
|
|
|
cmd.Flags().BoolVar(&root.opts.rmDist, "rm-dist", false, "Remove the dist folder before building")
|
2021-04-22 15:45:36 +02:00
|
|
|
cmd.Flags().IntVarP(&root.opts.parallelism, "parallelism", "p", 0, "Amount tasks to run concurrently (default: number of CPUs)")
|
2020-05-15 16:19:20 +02:00
|
|
|
cmd.Flags().DurationVar(&root.opts.timeout, "timeout", 30*time.Minute, "Timeout to the entire build process")
|
2021-04-21 21:36:53 +02:00
|
|
|
cmd.Flags().BoolVar(&root.opts.singleTarget, "single-target", false, "Builds only for current GOOS and GOARCH")
|
|
|
|
cmd.Flags().StringVar(&root.opts.id, "id", "", "Builds only the specified build id")
|
2020-05-15 16:19:20 +02:00
|
|
|
cmd.Flags().BoolVar(&root.opts.deprecated, "deprecated", false, "Force print the deprecation message - tests only")
|
|
|
|
_ = cmd.Flags().MarkHidden("deprecated")
|
|
|
|
|
|
|
|
root.cmd = cmd
|
|
|
|
return root
|
|
|
|
}
|
|
|
|
|
|
|
|
func buildProject(options buildOpts) (*context.Context, error) {
|
|
|
|
cfg, err := loadConfig(options.config)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
ctx, cancel := context.NewWithTimeout(cfg, options.timeout)
|
|
|
|
defer cancel()
|
2021-04-21 21:36:53 +02:00
|
|
|
if err := setupBuildContext(ctx, options); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-05-15 16:19:20 +02:00
|
|
|
return ctx, ctrlc.Default.Run(ctx, func() error {
|
|
|
|
for _, pipe := range pipeline.BuildPipeline {
|
2021-09-18 15:21:29 +02:00
|
|
|
if err := skip.Maybe(
|
|
|
|
pipe,
|
|
|
|
logging.Log(
|
|
|
|
pipe.String(),
|
|
|
|
errhandler.Handle(pipe.Run),
|
|
|
|
logging.DefaultInitialPadding,
|
|
|
|
),
|
2020-05-15 16:19:20 +02:00
|
|
|
)(ctx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-04-21 21:36:53 +02:00
|
|
|
func setupBuildContext(ctx *context.Context, options buildOpts) error {
|
2021-04-22 15:45:36 +02:00
|
|
|
ctx.Parallelism = runtime.NumCPU()
|
|
|
|
if options.parallelism > 0 {
|
|
|
|
ctx.Parallelism = options.parallelism
|
|
|
|
}
|
2020-05-15 16:19:20 +02:00
|
|
|
log.Debugf("parallelism: %v", ctx.Parallelism)
|
|
|
|
ctx.Snapshot = options.snapshot
|
|
|
|
ctx.SkipValidate = ctx.Snapshot || options.skipValidate
|
|
|
|
ctx.SkipPostBuildHooks = options.skipPostHooks
|
|
|
|
ctx.RmDist = options.rmDist
|
|
|
|
ctx.SkipTokenCheck = true
|
|
|
|
|
2021-04-21 21:36:53 +02:00
|
|
|
if options.singleTarget {
|
|
|
|
setupBuildSingleTarget(ctx)
|
|
|
|
}
|
|
|
|
|
|
|
|
if options.id != "" {
|
|
|
|
if err := setupBuildID(ctx, options.id); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-15 16:19:20 +02:00
|
|
|
// test only
|
|
|
|
ctx.Deprecated = options.deprecated
|
2021-04-21 21:36:53 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func setupBuildSingleTarget(ctx *context.Context) {
|
|
|
|
goos := os.Getenv("GOOS")
|
|
|
|
if goos == "" {
|
|
|
|
goos = runtime.GOOS
|
|
|
|
}
|
|
|
|
goarch := os.Getenv("GOARCH")
|
|
|
|
if goarch == "" {
|
|
|
|
goarch = runtime.GOARCH
|
|
|
|
}
|
|
|
|
log.Infof("building only for %s/%s", goos, goarch)
|
|
|
|
if len(ctx.Config.Builds) == 0 {
|
|
|
|
ctx.Config.Builds = append(ctx.Config.Builds, config.Build{})
|
|
|
|
}
|
|
|
|
for i := range ctx.Config.Builds {
|
|
|
|
build := &ctx.Config.Builds[i]
|
|
|
|
build.Goos = []string{goos}
|
|
|
|
build.Goarch = []string{goarch}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func setupBuildID(ctx *context.Context, id string) error {
|
|
|
|
if len(ctx.Config.Builds) < 2 {
|
|
|
|
log.Warn("single build in config, '--id' ignored")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var keep []config.Build
|
|
|
|
for _, build := range ctx.Config.Builds {
|
|
|
|
if build.ID == id {
|
|
|
|
keep = append(keep, build)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(keep) == 0 {
|
|
|
|
return fmt.Errorf("no builds with id '%s'", id)
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.Config.Builds = keep
|
|
|
|
return nil
|
2020-05-15 16:19:20 +02:00
|
|
|
}
|