1
0
mirror of https://github.com/goreleaser/goreleaser.git synced 2025-01-24 04:16:27 +02:00

127 lines
3.1 KiB
Go
Raw Normal View History

2017-04-14 15:39:32 -03:00
// Package build implements Pipe and can build Go projects for
// several platforms, with pre and post hook support.
2016-12-21 11:37:31 -02:00
package build
import (
2017-05-10 10:20:07 -03:00
"fmt"
2016-12-21 14:42:23 -02:00
"os"
2016-12-21 11:37:31 -02:00
"os/exec"
2017-03-25 21:29:38 -03:00
"path/filepath"
"strings"
2016-12-21 11:37:31 -02:00
2017-06-22 00:09:14 -03:00
"github.com/apex/log"
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"
2017-07-06 19:49:21 -03:00
"github.com/goreleaser/goreleaser/internal/buildtarget"
2017-05-11 00:05:51 -03:00
"github.com/goreleaser/goreleaser/internal/ext"
2017-07-01 12:27:13 -03:00
"github.com/goreleaser/goreleaser/internal/name"
2016-12-29 14:12:54 -02:00
"golang.org/x/sync/errgroup"
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-01-14 19:41:32 +01:00
// Description of the pipe
2017-01-14 15:14:35 -02:00
func (Pipe) Description() string {
2017-01-19 10:04:41 +01:00
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
}
func runPipeOnBuild(ctx *context.Context, build config.Build) error {
if err := runHook(build.Env, build.Hooks.Pre); err != nil {
2017-04-09 10:43:23 -03:00
return err
2017-03-23 14:20:24 -03:00
}
sem := make(chan bool, 4)
2016-12-29 14:12:54 -02:00
var g errgroup.Group
2017-06-27 19:20:08 -03:00
for _, target := range buildTargets(build) {
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-06-27 19:20:08 -03:00
return runHook(build.Env, build.Hooks.Post)
2017-04-09 10:43:23 -03:00
}
2017-05-11 10:43:25 -03:00
func runHook(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)
2017-07-06 19:49:21 -03:00
return run(buildtarget.Runtime, cmd, env)
2016-12-29 14:12:54 -02:00
}
2017-07-06 19:49:21 -03:00
func doBuild(ctx *context.Context, build config.Build, target buildtarget.Target) error {
folder, err := name.For(ctx, target)
2017-07-01 12:27:13 -03:00
if err != nil {
return err
}
2017-07-01 12:46:08 -03:00
ctx.AddFolder(target.String(), folder)
2017-07-06 19:49:21 -03:00
binary, err := binaryName(ctx, build, target, folder)
if err != nil {
return err
}
2017-07-01 12:27:13 -03:00
log.WithField("binary", binary).Info("building")
2017-02-22 09:25:43 -03:00
cmd := []string{"go", "build"}
2017-06-27 19:20:08 -03:00
if build.Flags != "" {
cmd = append(cmd, strings.Fields(build.Flags)...)
2017-02-22 09:25:43 -03:00
}
2017-06-27 19:36:36 -03:00
flags, err := ldflags(ctx, build)
2017-03-25 20:24:38 -03:00
if err != nil {
return err
}
2017-07-01 12:27:13 -03:00
cmd = append(cmd, "-ldflags="+flags, "-o", binary, build.Main)
2017-06-27 19:20:08 -03:00
return run(target, cmd, build.Env)
}
2017-07-06 19:49:21 -03:00
func binaryName(
ctx *context.Context,
build config.Build,
target buildtarget.Target,
folder string,
) (binary string, err error) {
2017-07-06 20:13:02 -03:00
binary = build.Binary + ext.For(target)
2017-07-06 19:49:21 -03:00
if ctx.Config.Archive.Format == "binary" {
binary, err = name.ForBuild(ctx, build, target)
if err != nil {
return
}
binary = binary + ext.For(target)
}
return filepath.Join(ctx.Config.Dist, folder, binary), nil
}
func run(target buildtarget.Target, command, env []string) error {
2017-06-22 00:25:52 -03:00
var cmd = exec.Command(command[0], command[1:]...)
2017-07-06 19:49:21 -03:00
env = append(env, target.Env()...)
var log = log.WithField("target", target.String()).
2017-06-25 11:37:40 -03:00
WithField("env", env).
WithField("cmd", command)
2017-01-23 09:27:28 -02:00
cmd.Env = append(cmd.Env, os.Environ()...)
2017-05-11 10:43:25 -03:00
cmd.Env = append(cmd.Env, env...)
2017-06-25 11:37:40 -03:00
log.Debug("running")
2017-01-30 08:17:15 -02:00
if out, err := cmd.CombinedOutput(); err != nil {
2017-06-25 11:37:40 -03:00
log.WithError(err).Debug("failed")
2017-07-06 19:49:21 -03:00
return fmt.Errorf("build failed for %s:\n%v", target.String(), string(out))
2016-12-29 14:12:54 -02:00
}
2016-12-21 11:37:31 -02:00
return nil
}