package goreleaserlib import ( "fmt" "io/ioutil" "os" "strings" "github.com/apex/log" "github.com/goreleaser/goreleaser/config" "github.com/goreleaser/goreleaser/context" "github.com/goreleaser/goreleaser/pipeline" "github.com/goreleaser/goreleaser/pipeline/archive" "github.com/goreleaser/goreleaser/pipeline/brew" "github.com/goreleaser/goreleaser/pipeline/build" "github.com/goreleaser/goreleaser/pipeline/checksums" "github.com/goreleaser/goreleaser/pipeline/cleandist" "github.com/goreleaser/goreleaser/pipeline/defaults" "github.com/goreleaser/goreleaser/pipeline/env" "github.com/goreleaser/goreleaser/pipeline/fpm" "github.com/goreleaser/goreleaser/pipeline/git" "github.com/goreleaser/goreleaser/pipeline/release" yaml "gopkg.in/yaml.v1" ) var pipes = []pipeline.Pipe{ defaults.Pipe{}, // load default configs git.Pipe{}, // get and validate git repo state env.Pipe{}, // load and validate environment variables cleandist.Pipe{}, // ensure ./dist is clean build.Pipe{}, // build archive.Pipe{}, // archive (tar.gz, zip, etc) fpm.Pipe{}, // archive via fpm (deb, rpm, etc) checksums.Pipe{}, // checksums of the files release.Pipe{}, // release to github brew.Pipe{}, // push to brew tap } // Flags interface represents an extractor of cli flags type Flags interface { IsSet(s string) bool String(s string) string Bool(s string) bool } // Release runs the release process with the given flags func Release(flags Flags) error { var file = getConfigFile(flags) var notes = flags.String("release-notes") if flags.Bool("debug") { log.SetLevel(log.DebugLevel) } cfg, err := config.Load(file) if err != nil { // Allow file not found errors if config file was not // explicitly specified _, statErr := os.Stat(file) if !os.IsNotExist(statErr) || flags.IsSet("config") { return err } log.WithField("file", file).Warn("could not load config, using defaults") } var ctx = context.New(cfg) ctx.Validate = !flags.Bool("skip-validate") ctx.Publish = !flags.Bool("skip-publish") if notes != "" { bts, err := ioutil.ReadFile(notes) if err != nil { return err } log.WithField("notes", notes).Info("loaded custom release notes") ctx.ReleaseNotes = string(bts) } ctx.Snapshot = flags.Bool("snapshot") if ctx.Snapshot { log.Info("publishing disabled in snapshot mode") ctx.Publish = false } ctx.RmDist = flags.Bool("rm-dist") for _, pipe := range pipes { log.Infof("\033[1m%s\033[0m", strings.ToUpper(pipe.Description())) if err := pipe.Run(ctx); err != nil { return err } } log.Infof("\033[1mSUCCESS!\033[0m") return nil } // InitProject creates an example goreleaser.yml in the current directory func InitProject(filename string) error { if _, err := os.Stat(filename); !os.IsNotExist(err) { if err != nil { return err } return fmt.Errorf("%s already exists", filename) } var ctx = context.New(config.Project{}) var pipe = defaults.Pipe{} if err := pipe.Run(ctx); err != nil { return err } out, err := yaml.Marshal(ctx.Config) if err != nil { return err } return ioutil.WriteFile(filename, out, 0644) } func getConfigFile(flags Flags) string { var config = flags.String("config") if flags.IsSet("config") { return config } for _, f := range []string{".goreleaser.yml", "goreleaser.yml"} { _, ferr := os.Stat(f) if ferr == nil || os.IsExist(ferr) { return f } } return config }