package golang import ( "errors" "fmt" "go/ast" "go/parser" "go/token" "os" "os/exec" "path/filepath" "strconv" "strings" "time" "github.com/apex/log" "github.com/goreleaser/goreleaser/internal/artifact" "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" ) // Default builder instance. // nolint: gochecknoglobals var Default = &Builder{} // nolint: gochecknoinits func init() { api.Register("go", Default) } // Builder is golang builder. type Builder struct{} // WithDefaults sets the defaults for a golang build and returns it. func (*Builder) WithDefaults(build config.Build) config.Build { if build.Dir == "" { build.Dir = "." } if build.Main == "" { build.Main = "." } if len(build.Ldflags) == 0 { build.Ldflags = []string{"-s -w -X main.version={{.Version}} -X main.commit={{.Commit}} -X main.date={{.Date}} -X main.builtBy=goreleaser"} } if len(build.Targets) == 0 { if len(build.Goos) == 0 { build.Goos = []string{"linux", "darwin"} } if len(build.Goarch) == 0 { build.Goarch = []string{"amd64", "386"} } if len(build.Goarm) == 0 { build.Goarm = []string{"6"} } build.Targets = matrix(build) } if build.GoBinary == "" { build.GoBinary = "go" } return build } // Build builds a golang build. func (*Builder) Build(ctx *context.Context, build config.Build, options api.Options) error { if err := checkMain(build); err != nil { return err } target, err := newBuildTarget(options.Target) if err != nil { return err } var cmd = []string{build.GoBinary, "build"} var env = append(ctx.Env.Strings(), build.Env...) env = append(env, target.Env()...) artifact := &artifact.Artifact{ Type: artifact.Binary, Path: options.Path, Name: options.Name, Goos: target.os, Goarch: target.arch, Goarm: target.arm, Gomips: target.mips, Extra: map[string]interface{}{ "Binary": strings.TrimSuffix(filepath.Base(options.Path), options.Ext), "Ext": options.Ext, "ID": build.ID, }, } flags, err := processFlags(ctx, artifact, env, build.Flags, "") if err != nil { return err } cmd = append(cmd, flags...) asmflags, err := processFlags(ctx, artifact, env, build.Asmflags, "-asmflags=") if err != nil { return err } cmd = append(cmd, asmflags...) gcflags, err := processFlags(ctx, artifact, env, build.Gcflags, "-gcflags=") if err != nil { return err } cmd = append(cmd, gcflags...) // flag prefix is skipped because ldflags need to output a single string ldflags, err := processFlags(ctx, artifact, env, build.Ldflags, "") if err != nil { return err } // ldflags need to be single string in order to apply correctly processedLdFlags := joinLdFlags(ldflags) cmd = append(cmd, processedLdFlags) cmd = append(cmd, "-o", options.Path, build.Main) if err := run(ctx, cmd, env, build.Dir); err != nil { return fmt.Errorf("failed to build for %s: %w", options.Target, err) } if build.ModTimestamp != "" { modTimestamp, err := tmpl.New(ctx).WithEnvS(env).WithArtifact(artifact, map[string]string{}).Apply(build.ModTimestamp) if err != nil { return err } modUnix, err := strconv.ParseInt(modTimestamp, 10, 64) if err != nil { return err } modTime := time.Unix(modUnix, 0) err = os.Chtimes(options.Path, modTime, modTime) if err != nil { return fmt.Errorf("failed to change times for %s: %w", options.Target, err) } } ctx.Artifacts.Add(artifact) return nil } func processFlags(ctx *context.Context, a *artifact.Artifact, env, flags []string, flagPrefix string) ([]string, error) { processed := make([]string, 0, len(flags)) for _, rawFlag := range flags { flag, err := tmpl.New(ctx).WithEnvS(env).WithArtifact(a, map[string]string{}).Apply(rawFlag) if err != nil { return nil, err } processed = append(processed, flagPrefix+flag) } return processed, nil } func joinLdFlags(flags []string) string { ldflagString := strings.Builder{} ldflagString.WriteString("-ldflags=") ldflagString.WriteString(strings.Join(flags, " ")) return ldflagString.String() } func run(ctx *context.Context, command, env []string, dir string) error { /* #nosec */ var cmd = exec.CommandContext(ctx, command[0], command[1:]...) var log = log.WithField("env", env).WithField("cmd", command) cmd.Env = env cmd.Dir = dir log.Debug("running") if out, err := cmd.CombinedOutput(); err != nil { log.WithError(err).Debug("failed") return errors.New(string(out)) } return nil } type buildTarget struct { os, arch, arm, mips string } func newBuildTarget(s string) (buildTarget, error) { var t = buildTarget{} parts := strings.Split(s, "_") if len(parts) < 2 { return t, fmt.Errorf("%s is not a valid build target", s) } t.os = parts[0] t.arch = parts[1] if strings.HasPrefix(t.arch, "arm") && len(parts) == 3 { t.arm = parts[2] } if strings.HasPrefix(t.arch, "mips") && len(parts) == 3 { t.mips = parts[2] } return t, nil } func (b buildTarget) Env() []string { return []string{ "GOOS=" + b.os, "GOARCH=" + b.arch, "GOARM=" + b.arm, "GOMIPS=" + b.mips, "GOMIPS64=" + b.mips, } } func checkMain(build config.Build) error { var main = build.Main if main == "" { main = "." } if build.Dir != "" { main = filepath.Join(build.Dir, main) } stat, ferr := os.Stat(main) if ferr != nil { return ferr } if stat.IsDir() { packs, err := parser.ParseDir(token.NewFileSet(), main, nil, 0) if err != nil { return fmt.Errorf("failed to parse dir: %s: %w", main, err) } for _, pack := range packs { for _, file := range pack.Files { if hasMain(file) { return nil } } } return fmt.Errorf("build for %s does not contain a main function", build.Binary) } file, err := parser.ParseFile(token.NewFileSet(), main, nil, 0) if err != nil { return fmt.Errorf("failed to parse file: %s: %w", main, err) } if hasMain(file) { return nil } return fmt.Errorf("build for %s does not contain a main function", build.Binary) } func hasMain(file *ast.File) bool { for _, decl := range file.Decls { fn, isFn := decl.(*ast.FuncDecl) if !isFn { continue } if fn.Name.Name == "main" && fn.Recv == nil { return true } } return false }