2018-01-22 01:10:17 -02:00
|
|
|
// Package build provides a pipe that can build binaries for several
|
|
|
|
// languages.
|
2016-12-21 11:37:31 -02:00
|
|
|
package build
|
|
|
|
|
|
|
|
import (
|
2018-01-26 19:26:28 -02:00
|
|
|
"os"
|
|
|
|
"os/exec"
|
2017-03-25 21:29:38 -03:00
|
|
|
"path/filepath"
|
2017-01-21 20:02:51 -02:00
|
|
|
"strings"
|
2016-12-21 11:37:31 -02:00
|
|
|
|
2017-06-22 00:09:14 -03:00
|
|
|
"github.com/apex/log"
|
2017-12-17 15:24:49 -02:00
|
|
|
"github.com/pkg/errors"
|
|
|
|
"golang.org/x/sync/errgroup"
|
|
|
|
|
2018-01-21 14:31:08 -02:00
|
|
|
builders "github.com/goreleaser/goreleaser/build"
|
2017-06-27 19:20:08 -03:00
|
|
|
"github.com/goreleaser/goreleaser/config"
|
2017-01-14 20:01:32 -02:00
|
|
|
"github.com/goreleaser/goreleaser/context"
|
2018-01-21 14:31:08 -02:00
|
|
|
|
|
|
|
// langs to init
|
|
|
|
_ "github.com/goreleaser/goreleaser/internal/builders/golang"
|
2018-02-15 11:30:15 +00:00
|
|
|
"time"
|
|
|
|
"bytes"
|
|
|
|
"text/template"
|
2016-12-21 11:37:31 -02:00
|
|
|
)
|
|
|
|
|
2016-12-30 12:41:59 -02:00
|
|
|
// Pipe for build
|
2016-12-30 09:27:35 -02:00
|
|
|
type Pipe struct{}
|
|
|
|
|
2017-12-02 19:53:19 -02:00
|
|
|
func (Pipe) String() string {
|
|
|
|
return "building binaries"
|
2016-12-30 09:27:35 -02:00
|
|
|
}
|
|
|
|
|
2016-12-30 12:41:59 -02:00
|
|
|
// Run the pipe
|
2017-01-14 14:06:57 -02:00
|
|
|
func (Pipe) Run(ctx *context.Context) error {
|
2017-06-27 19:20:08 -03:00
|
|
|
for _, build := range ctx.Config.Builds {
|
2017-06-27 19:36:36 -03:00
|
|
|
log.WithField("build", build).Debug("building")
|
2017-06-27 19:20:08 -03:00
|
|
|
if err := runPipeOnBuild(ctx, build); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-12-02 19:53:19 -02:00
|
|
|
// Default sets the pipe defaults
|
|
|
|
func (Pipe) Default(ctx *context.Context) error {
|
|
|
|
for i, build := range ctx.Config.Builds {
|
|
|
|
ctx.Config.Builds[i] = buildWithDefaults(ctx, build)
|
|
|
|
}
|
|
|
|
if len(ctx.Config.Builds) == 0 {
|
|
|
|
ctx.Config.Builds = []config.Build{
|
|
|
|
buildWithDefaults(ctx, ctx.Config.SingleBuild),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func buildWithDefaults(ctx *context.Context, build config.Build) config.Build {
|
2018-01-21 14:31:08 -02:00
|
|
|
if build.Lang == "" {
|
|
|
|
build.Lang = "go"
|
|
|
|
}
|
2017-12-02 19:53:19 -02:00
|
|
|
if build.Binary == "" {
|
|
|
|
build.Binary = ctx.Config.Release.GitHub.Name
|
|
|
|
}
|
2018-02-12 20:53:57 -02:00
|
|
|
for k, v := range build.Env {
|
|
|
|
build.Env[k] = os.ExpandEnv(v)
|
|
|
|
}
|
2018-01-26 19:35:12 -02:00
|
|
|
return builders.For(build.Lang).WithDefaults(build)
|
2017-12-02 19:53:19 -02:00
|
|
|
}
|
|
|
|
|
2017-06-27 19:20:08 -03:00
|
|
|
func runPipeOnBuild(ctx *context.Context, build config.Build) error {
|
2017-12-29 17:07:06 -02:00
|
|
|
if err := runHook(ctx, build.Env, build.Hooks.Pre); err != nil {
|
2017-11-18 14:14:39 -02:00
|
|
|
return errors.Wrap(err, "pre hook failed")
|
2017-03-23 14:20:24 -03:00
|
|
|
}
|
2017-07-15 16:49:52 -03:00
|
|
|
sem := make(chan bool, ctx.Parallelism)
|
2016-12-29 14:12:54 -02:00
|
|
|
var g errgroup.Group
|
2018-01-21 22:44:06 -02:00
|
|
|
for _, target := range build.Targets {
|
2017-04-24 14:27:21 -03:00
|
|
|
sem <- true
|
|
|
|
target := target
|
2017-07-01 12:27:13 -03:00
|
|
|
build := build
|
2017-04-24 14:27:21 -03:00
|
|
|
g.Go(func() error {
|
|
|
|
defer func() {
|
|
|
|
<-sem
|
|
|
|
}()
|
2017-07-01 12:27:13 -03:00
|
|
|
return doBuild(ctx, build, target)
|
2017-04-24 14:27:21 -03:00
|
|
|
})
|
2016-12-21 11:37:31 -02:00
|
|
|
}
|
2017-03-23 18:32:27 -03:00
|
|
|
if err := g.Wait(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-12-29 17:07:06 -02:00
|
|
|
return errors.Wrap(runHook(ctx, build.Env, build.Hooks.Post), "post hook failed")
|
2017-04-09 10:43:23 -03:00
|
|
|
}
|
|
|
|
|
2017-12-29 17:07:06 -02:00
|
|
|
func runHook(ctx *context.Context, env []string, hook string) error {
|
2017-04-09 10:43:23 -03:00
|
|
|
if hook == "" {
|
|
|
|
return nil
|
2017-03-23 14:20:24 -03:00
|
|
|
}
|
2017-06-22 10:47:34 -03:00
|
|
|
log.WithField("hook", hook).Info("running hook")
|
2017-04-09 10:43:23 -03:00
|
|
|
cmd := strings.Fields(hook)
|
2018-01-26 19:26:28 -02:00
|
|
|
return run(ctx, cmd, env)
|
2016-12-29 14:12:54 -02:00
|
|
|
}
|
|
|
|
|
2018-01-21 22:44:06 -02:00
|
|
|
func doBuild(ctx *context.Context, build config.Build, target string) error {
|
|
|
|
var ext = extFor(target)
|
2018-02-15 11:30:15 +00:00
|
|
|
|
|
|
|
binary, err := binary(ctx, build)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
build.Binary = binary
|
2018-01-21 22:44:06 -02:00
|
|
|
var name = build.Binary + ext
|
|
|
|
var path = filepath.Join(ctx.Config.Dist, target, name)
|
|
|
|
log.WithField("binary", path).Info("building")
|
2018-01-21 14:31:08 -02:00
|
|
|
return builders.For(build.Lang).Build(ctx, build, builders.Options{
|
|
|
|
Target: target,
|
2018-01-21 22:44:06 -02:00
|
|
|
Name: name,
|
|
|
|
Path: path,
|
2018-01-21 14:31:08 -02:00
|
|
|
Ext: ext,
|
2017-12-17 15:24:49 -02:00
|
|
|
})
|
2016-12-21 11:37:31 -02:00
|
|
|
}
|
2018-01-21 22:44:06 -02:00
|
|
|
|
2018-02-15 11:30:15 +00:00
|
|
|
func binary(ctx *context.Context, build config.Build) (string, error) {
|
|
|
|
var data = struct {
|
|
|
|
Commit string
|
|
|
|
Tag string
|
|
|
|
Version string
|
|
|
|
Date string
|
|
|
|
Env map[string]string
|
|
|
|
}{
|
|
|
|
Commit: ctx.Git.Commit,
|
|
|
|
Tag: ctx.Git.CurrentTag,
|
|
|
|
Version: ctx.Version,
|
|
|
|
Date: time.Now().UTC().Format(time.RFC3339),
|
|
|
|
Env: ctx.Env,
|
|
|
|
}
|
|
|
|
var out bytes.Buffer
|
|
|
|
t, err := template.New("binary").
|
|
|
|
Option("missingkey=error").
|
|
|
|
Parse(build.Binary)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
err = t.Execute(&out, data)
|
|
|
|
return out.String(), err
|
|
|
|
}
|
|
|
|
|
2018-01-21 22:44:06 -02:00
|
|
|
func extFor(target string) string {
|
|
|
|
if strings.Contains(target, "windows") {
|
|
|
|
return ".exe"
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
2018-01-26 19:26:28 -02:00
|
|
|
|
|
|
|
func run(ctx *context.Context, command, env []string) error {
|
|
|
|
/* #nosec */
|
|
|
|
var cmd = exec.CommandContext(ctx, command[0], command[1:]...)
|
|
|
|
var log = log.WithField("env", env).WithField("cmd", command)
|
|
|
|
cmd.Env = append(cmd.Env, os.Environ()...)
|
|
|
|
cmd.Env = append(cmd.Env, env...)
|
|
|
|
log.WithField("cmd", command).WithField("env", env).Debug("running")
|
|
|
|
if out, err := cmd.CombinedOutput(); err != nil {
|
|
|
|
log.WithError(err).Debug("failed")
|
|
|
|
return errors.New(string(out))
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|