package build import ( "errors" "fmt" "io/ioutil" "os" "path/filepath" "testing" "github.com/goreleaser/goreleaser/internal/artifact" "github.com/goreleaser/goreleaser/internal/testlib" "github.com/goreleaser/goreleaser/internal/tmpl" api "github.com/goreleaser/goreleaser/pkg/build" "github.com/goreleaser/goreleaser/pkg/config" "github.com/goreleaser/goreleaser/pkg/context" "github.com/stretchr/testify/assert" ) var fakeArtifact = artifact.Artifact{ Name: "fake", } type fakeBuilder struct { fail bool } func (*fakeBuilder) WithDefaults(build config.Build) config.Build { return build } var errFailedBuild = errors.New("fake builder failed") func (f *fakeBuilder) Build(ctx *context.Context, build config.Build, options api.Options) error { if f.fail { return errFailedBuild } if err := os.MkdirAll(filepath.Dir(options.Path), 0755); err != nil { return err } if err := ioutil.WriteFile(options.Path, []byte("foo"), 0755); err != nil { return err } ctx.Artifacts.Add(fakeArtifact) return nil } func init() { api.Register("fake", &fakeBuilder{}) api.Register("fakeFail", &fakeBuilder{ fail: true, }) } func TestPipeDescription(t *testing.T) { assert.NotEmpty(t, Pipe{}.String()) } func TestBuild(t *testing.T) { folder, back := testlib.Mktmp(t) defer back() var config = config.Project{ Dist: folder, Builds: []config.Build{ { Lang: "fake", Binary: "testing.v{{.Version}}", Flags: []string{"-n"}, Env: []string{"BLAH=1"}, }, }, } var ctx = &context.Context{ Artifacts: artifact.New(), Git: context.GitInfo{ CurrentTag: "v1.2.3", Commit: "123", }, Version: "1.2.3", Config: config, } error := doBuild(ctx, ctx.Config.Builds[0], "darwin_amd64") assert.NoError(t, error) } func TestRunPipe(t *testing.T) { folder, back := testlib.Mktmp(t) defer back() var config = config.Project{ Dist: folder, Builds: []config.Build{ { Lang: "fake", Binary: "testing", Flags: []string{"-v"}, Ldflags: []string{"-X main.test=testing"}, Targets: []string{"whatever"}, }, }, } var ctx = context.New(config) ctx.Git.CurrentTag = "2.4.5" assert.NoError(t, Pipe{}.Run(ctx)) assert.Equal(t, ctx.Artifacts.List(), []artifact.Artifact{fakeArtifact}) } func TestRunFullPipe(t *testing.T) { folder, back := testlib.Mktmp(t) defer back() var pre = filepath.Join(folder, "pre") var post = filepath.Join(folder, "post") var config = config.Project{ Builds: []config.Build{ { ID: "build1", Lang: "fake", Binary: "testing", Flags: []string{"-v"}, Ldflags: []string{"-X main.test=testing"}, Hooks: config.Hooks{ Pre: "touch " + pre, Post: "touch " + post, }, Targets: []string{"whatever"}, }, }, Dist: folder, } var ctx = context.New(config) ctx.Git.CurrentTag = "2.4.5" assert.NoError(t, Pipe{}.Run(ctx)) assert.Equal(t, ctx.Artifacts.List(), []artifact.Artifact{fakeArtifact}) assert.FileExists(t, post) assert.FileExists(t, pre) assert.FileExists(t, filepath.Join(folder, "build1_whatever", "testing")) } func TestRunFullPipeFail(t *testing.T) { folder, back := testlib.Mktmp(t) defer back() var pre = filepath.Join(folder, "pre") var post = filepath.Join(folder, "post") var config = config.Project{ Dist: folder, Builds: []config.Build{ { Lang: "fakeFail", Binary: "testing", Flags: []string{"-v"}, Ldflags: []string{"-X main.test=testing"}, Hooks: config.Hooks{ Pre: "touch " + pre, Post: "touch " + post, }, Targets: []string{"whatever"}, }, }, } var ctx = context.New(config) ctx.Git.CurrentTag = "2.4.5" assert.EqualError(t, Pipe{}.Run(ctx), errFailedBuild.Error()) assert.Empty(t, ctx.Artifacts.List()) assert.FileExists(t, pre) } func TestRunPipeFailingHooks(t *testing.T) { folder, back := testlib.Mktmp(t) defer back() var config = config.Project{ Dist: folder, Builds: []config.Build{ { Lang: "fake", Binary: "hooks", Hooks: config.Hooks{}, Targets: []string{"whatever"}, }, }, } t.Run("pre-hook", func(t *testing.T) { var ctx = context.New(config) ctx.Git.CurrentTag = "2.3.4" ctx.Config.Builds[0].Hooks.Pre = "exit 1" ctx.Config.Builds[0].Hooks.Post = "echo post" assert.EqualError(t, Pipe{}.Run(ctx), `pre hook failed: `) }) t.Run("post-hook", func(t *testing.T) { var ctx = context.New(config) ctx.Git.CurrentTag = "2.3.4" ctx.Config.Builds[0].Hooks.Pre = "echo pre" ctx.Config.Builds[0].Hooks.Post = "exit 1" assert.EqualError(t, Pipe{}.Run(ctx), `post hook failed: `) }) } func TestDefaultNoBuilds(t *testing.T) { var ctx = &context.Context{ Config: config.Project{}, } assert.NoError(t, Pipe{}.Default(ctx)) } func TestDefaultExpandEnv(t *testing.T) { assert.NoError(t, os.Setenv("BAR", "FOOBAR")) var ctx = &context.Context{ Config: config.Project{ Builds: []config.Build{ { Env: []string{ "FOO=bar_$BAR", }, }, }, }, } assert.NoError(t, Pipe{}.Default(ctx)) var env = ctx.Config.Builds[0].Env[0] assert.Equal(t, "FOO=bar_FOOBAR", env) } func TestDefaultEmptyBuild(t *testing.T) { var ctx = &context.Context{ Config: config.Project{ ProjectName: "foo", Builds: []config.Build{ {}, }, }, } assert.NoError(t, Pipe{}.Default(ctx)) var build = ctx.Config.Builds[0] assert.Equal(t, ctx.Config.ProjectName, build.ID) assert.Equal(t, ctx.Config.ProjectName, build.Binary) assert.Equal(t, ".", build.Main) assert.Equal(t, []string{"linux", "darwin"}, build.Goos) assert.Equal(t, []string{"amd64", "386"}, build.Goarch) assert.Equal(t, []string{"6"}, build.Goarm) assert.Len(t, build.Ldflags, 1) assert.Equal(t, "-s -w -X main.version={{.Version}} -X main.commit={{.Commit}} -X main.date={{.Date}} -X main.builtBy=goreleaser", build.Ldflags[0]) } func TestDefaultBuildID(t *testing.T) { var ctx = &context.Context{ Config: config.Project{ ProjectName: "foo", Builds: []config.Build{ { Binary: "{{.Env.FOO}}", }, { Binary: "bar", }, }, }, } assert.EqualError(t, Pipe{}.Default(ctx), "found 2 builds with the ID 'foo', please fix your config") var build = ctx.Config.Builds[0] assert.Equal(t, ctx.Config.ProjectName, build.ID) } func TestSeveralBuildsWithTheSameID(t *testing.T) { var ctx = &context.Context{ Config: config.Project{ Builds: []config.Build{ { ID: "a", Binary: "bar", }, { ID: "a", Binary: "foo", }, }, }, } assert.EqualError(t, Pipe{}.Default(ctx), "found 2 builds with the ID 'a', please fix your config") } func TestDefaultPartialBuilds(t *testing.T) { var ctx = &context.Context{ Config: config.Project{ Builds: []config.Build{ { ID: "build1", Binary: "bar", Goos: []string{"linux"}, Main: "./cmd/main.go", }, { ID: "build2", Binary: "foo", Ldflags: []string{"-s -w"}, Goarch: []string{"386"}, }, }, }, } assert.NoError(t, Pipe{}.Default(ctx)) t.Run("build0", func(t *testing.T) { var build = ctx.Config.Builds[0] assert.Equal(t, "bar", build.Binary) assert.Equal(t, "./cmd/main.go", build.Main) assert.Equal(t, []string{"linux"}, build.Goos) assert.Equal(t, []string{"amd64", "386"}, build.Goarch) assert.Equal(t, []string{"6"}, build.Goarm) assert.Len(t, build.Ldflags, 1) assert.Equal(t, "-s -w -X main.version={{.Version}} -X main.commit={{.Commit}} -X main.date={{.Date}} -X main.builtBy=goreleaser", build.Ldflags[0]) }) t.Run("build1", func(t *testing.T) { var build = ctx.Config.Builds[1] assert.Equal(t, "foo", build.Binary) assert.Equal(t, ".", build.Main) assert.Equal(t, []string{"linux", "darwin"}, build.Goos) assert.Equal(t, []string{"386"}, build.Goarch) assert.Equal(t, []string{"6"}, build.Goarm) assert.Len(t, build.Ldflags, 1) assert.Equal(t, "-s -w", build.Ldflags[0]) }) } func TestDefaultFillSingleBuild(t *testing.T) { _, back := testlib.Mktmp(t) defer back() var ctx = &context.Context{ Config: config.Project{ ProjectName: "foo", SingleBuild: config.Build{ Main: "testreleaser", }, }, } assert.NoError(t, Pipe{}.Default(ctx)) assert.Len(t, ctx.Config.Builds, 1) assert.Equal(t, ctx.Config.Builds[0].Binary, "foo") } func TestExtWindows(t *testing.T) { assert.Equal(t, ".exe", extFor("windows_amd64")) assert.Equal(t, ".exe", extFor("windows_386")) } func TestExtOthers(t *testing.T) { assert.Empty(t, "", extFor("linux_amd64")) assert.Empty(t, "", extFor("linuxwin_386")) assert.Empty(t, "", extFor("winasdasd_sad")) } func TestTemplate(t *testing.T) { var ctx = context.New(config.Project{}) ctx.Git = context.GitInfo{ CurrentTag: "v1.2.3", Commit: "123", } ctx.Version = "1.2.3" ctx.Env = map[string]string{"FOO": "123"} binary, err := tmpl.New(ctx). Apply(`-s -w -X main.version={{.Version}} -X main.tag={{.Tag}} -X main.date={{.Date}} -X main.commit={{.Commit}} -X "main.foo={{.Env.FOO}}"`) assert.NoError(t, err) assert.Contains(t, binary, "-s -w") assert.Contains(t, binary, "-X main.version=1.2.3") assert.Contains(t, binary, "-X main.tag=v1.2.3") assert.Contains(t, binary, "-X main.commit=123") assert.Contains(t, binary, "-X main.date=") assert.Contains(t, binary, `-X "main.foo=123"`) } func TestHookEnvs(t *testing.T) { tmp, back := testlib.Mktmp(t) defer back() var build = config.Build{ Env: []string{ fmt.Sprintf("FOO=%s/foo", tmp), fmt.Sprintf("BAR=%s/bar", tmp), }, } t.Run("valid template", func(t *testing.T) { var err = runHook(context.New(config.Project{ Builds: []config.Build{ build, }, }), build.Env, "touch {{ .Env.FOO }}") assert.NoError(t, err) assert.FileExists(t, filepath.Join(tmp, "foo")) }) t.Run("invalid template", func(t *testing.T) { var err = runHook(context.New(config.Project{ Builds: []config.Build{ build, }, }), build.Env, "touch {{ .Env.FOOss }}") assert.EqualError(t, err, `template: tmpl:1:13: executing "tmpl" at <.Env.FOOss>: map has no entry for key "FOOss"`) }) t.Run("env inside shell", func(t *testing.T) { t.Skip("this fails on travis for some reason") var shell = `#!/bin/sh -e touch "$BAR"` ioutil.WriteFile(filepath.Join(tmp, "test.sh"), []byte(shell), 0750) var err = runHook(context.New(config.Project{ Builds: []config.Build{ build, }, }), build.Env, "sh test.sh") assert.NoError(t, err) assert.FileExists(t, filepath.Join(tmp, "bar")) }) }