2017-04-21 15:25:32 -03:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2017-04-21 21:58:59 -03:00
|
|
|
"fmt"
|
2018-02-21 20:04:22 -03:00
|
|
|
"io/ioutil"
|
2017-04-21 15:25:32 -03:00
|
|
|
"os"
|
2018-02-21 20:04:22 -03:00
|
|
|
"strings"
|
2017-12-06 01:02:52 -02:00
|
|
|
"time"
|
2017-04-21 15:25:32 -03:00
|
|
|
|
2017-06-22 00:09:14 -03:00
|
|
|
"github.com/apex/log"
|
2017-06-22 00:38:24 -03:00
|
|
|
lcli "github.com/apex/log/handlers/cli"
|
2018-02-24 17:59:08 -03:00
|
|
|
"github.com/caarlos0/ctrlc"
|
2018-01-17 18:48:01 -02:00
|
|
|
"github.com/fatih/color"
|
2017-04-21 15:25:32 -03:00
|
|
|
"github.com/urfave/cli"
|
2018-02-21 20:04:22 -03:00
|
|
|
|
|
|
|
"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/artifactory"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/brew"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/build"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/changelog"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/checksums"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/defaults"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/dist"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/docker"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/effectiveconfig"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/env"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/git"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/nfpm"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/release"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/scoop"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/sign"
|
|
|
|
"github.com/goreleaser/goreleaser/pipeline/snapcraft"
|
2017-04-21 15:25:32 -03:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
version = "dev"
|
|
|
|
commit = "none"
|
|
|
|
date = "unknown"
|
2018-01-17 18:48:01 -02:00
|
|
|
|
2018-02-21 20:04:22 -03:00
|
|
|
bold = color.New(color.Bold)
|
|
|
|
normalPadding = lcli.Default.Padding
|
|
|
|
increasedPadding = normalPadding * 2
|
|
|
|
|
2018-03-08 18:46:15 -03:00
|
|
|
pipes = []Piper{
|
2018-02-21 20:04:22 -03:00
|
|
|
defaults.Pipe{}, // load default configs
|
|
|
|
dist.Pipe{}, // ensure ./dist is clean
|
|
|
|
git.Pipe{}, // get and validate git repo state
|
|
|
|
effectiveconfig.Pipe{}, // writes the actual config (with defaults et al set) to dist
|
|
|
|
changelog.Pipe{}, // builds the release changelog
|
|
|
|
env.Pipe{}, // load and validate environment variables
|
|
|
|
build.Pipe{}, // build
|
|
|
|
archive.Pipe{}, // archive in tar.gz, zip or binary (which does no archiving at all)
|
|
|
|
nfpm.Pipe{}, // archive via fpm (deb, rpm) using "native" go impl
|
|
|
|
snapcraft.Pipe{}, // archive via snapcraft (snap)
|
|
|
|
checksums.Pipe{}, // checksums of the files
|
|
|
|
sign.Pipe{}, // sign artifacts
|
|
|
|
docker.Pipe{}, // create and push docker images
|
|
|
|
artifactory.Pipe{}, // push to artifactory
|
|
|
|
release.Pipe{}, // release to github
|
|
|
|
brew.Pipe{}, // push to brew tap
|
|
|
|
scoop.Pipe{}, // push to scoop bucket
|
|
|
|
}
|
2017-04-21 15:25:32 -03:00
|
|
|
)
|
|
|
|
|
2018-03-08 18:46:15 -03:00
|
|
|
// Piper defines a pipe, which can be part of a pipeline (a serie of pipes).
|
|
|
|
type Piper interface {
|
|
|
|
fmt.Stringer
|
|
|
|
|
|
|
|
// Run the pipe
|
|
|
|
Run(ctx *context.Context) error
|
|
|
|
}
|
|
|
|
|
2018-02-21 20:04:22 -03:00
|
|
|
// Flags interface represents an extractor of cli flags
|
|
|
|
type Flags interface {
|
|
|
|
IsSet(s string) bool
|
|
|
|
String(s string) string
|
|
|
|
Int(s string) int
|
|
|
|
Bool(s string) bool
|
|
|
|
Duration(s string) time.Duration
|
|
|
|
}
|
|
|
|
|
2017-06-22 00:09:14 -03:00
|
|
|
func init() {
|
2017-12-08 21:34:44 -02:00
|
|
|
log.SetHandler(lcli.Default)
|
2017-06-22 00:09:14 -03:00
|
|
|
}
|
|
|
|
|
2017-04-21 15:25:32 -03:00
|
|
|
func main() {
|
2017-11-26 21:48:05 -02:00
|
|
|
fmt.Println()
|
|
|
|
defer fmt.Println()
|
2017-04-21 15:25:32 -03:00
|
|
|
var app = cli.NewApp()
|
|
|
|
app.Name = "goreleaser"
|
2017-04-21 21:58:59 -03:00
|
|
|
app.Version = fmt.Sprintf("%v, commit %v, built at %v", version, commit, date)
|
2017-04-21 15:25:32 -03:00
|
|
|
app.Usage = "Deliver Go binaries as fast and easily as possible"
|
|
|
|
app.Flags = []cli.Flag{
|
|
|
|
cli.StringFlag{
|
|
|
|
Name: "config, file, c, f",
|
|
|
|
Usage: "Load configuration from `FILE`",
|
2017-07-04 23:51:45 -03:00
|
|
|
Value: ".goreleaser.yml",
|
2017-04-21 15:25:32 -03:00
|
|
|
},
|
|
|
|
cli.StringFlag{
|
|
|
|
Name: "release-notes",
|
|
|
|
Usage: "Load custom release notes from a markdown `FILE`",
|
|
|
|
},
|
2017-04-29 12:49:22 +02:00
|
|
|
cli.BoolFlag{
|
|
|
|
Name: "snapshot",
|
2018-02-24 18:03:54 -03:00
|
|
|
Usage: "Generate an unversioned snapshot release, skipping all validations and without publishing any artifacts",
|
2017-04-29 12:49:22 +02:00
|
|
|
},
|
2018-03-01 01:12:58 -03:00
|
|
|
cli.BoolFlag{
|
|
|
|
Name: "skip-publish",
|
|
|
|
Usage: "Generates all artifacts but does not publish them anywhere",
|
|
|
|
},
|
2018-03-08 08:42:33 -03:00
|
|
|
cli.BoolFlag{
|
|
|
|
Name: "skip-validate",
|
|
|
|
Usage: "Skips all git state checks",
|
|
|
|
},
|
2017-07-04 22:53:50 -03:00
|
|
|
cli.BoolFlag{
|
|
|
|
Name: "rm-dist",
|
2018-02-24 17:59:08 -03:00
|
|
|
Usage: "Remove the dist folder before building",
|
2017-07-04 22:53:50 -03:00
|
|
|
},
|
2017-07-15 16:49:52 -03:00
|
|
|
cli.IntFlag{
|
|
|
|
Name: "parallelism, p",
|
|
|
|
Usage: "Amount of builds launch in parallel",
|
|
|
|
Value: 4,
|
|
|
|
},
|
2017-06-22 00:25:52 -03:00
|
|
|
cli.BoolFlag{
|
|
|
|
Name: "debug",
|
|
|
|
Usage: "Enable debug mode",
|
|
|
|
},
|
2017-12-29 17:07:06 -02:00
|
|
|
cli.DurationFlag{
|
|
|
|
Name: "timeout",
|
|
|
|
Usage: "How much time the entire release process is allowed to take",
|
|
|
|
Value: 30 * time.Minute,
|
|
|
|
},
|
2017-04-21 15:25:32 -03:00
|
|
|
}
|
|
|
|
app.Action = func(c *cli.Context) error {
|
2017-12-06 01:02:52 -02:00
|
|
|
start := time.Now()
|
2018-01-17 18:48:01 -02:00
|
|
|
log.Infof(bold.Sprint("releasing..."))
|
2018-02-21 20:04:22 -03:00
|
|
|
if err := releaseProject(c); err != nil {
|
2018-01-17 18:48:01 -02:00
|
|
|
log.WithError(err).Errorf(bold.Sprintf("release failed after %0.2fs", time.Since(start).Seconds()))
|
2017-07-04 22:53:50 -03:00
|
|
|
return cli.NewExitError("\n", 1)
|
2017-04-21 15:25:32 -03:00
|
|
|
}
|
2018-01-17 18:48:01 -02:00
|
|
|
log.Infof(bold.Sprintf("release succeeded after %0.2fs", time.Since(start).Seconds()))
|
2017-04-21 15:25:32 -03:00
|
|
|
return nil
|
|
|
|
}
|
2017-04-27 11:03:26 +02:00
|
|
|
app.Commands = []cli.Command{
|
|
|
|
{
|
|
|
|
Name: "init",
|
|
|
|
Aliases: []string{"i"},
|
2017-07-04 23:51:45 -03:00
|
|
|
Usage: "generate .goreleaser.yml",
|
2017-04-27 11:03:26 +02:00
|
|
|
Action: func(c *cli.Context) error {
|
2017-07-04 23:51:45 -03:00
|
|
|
var filename = ".goreleaser.yml"
|
2018-02-21 20:04:22 -03:00
|
|
|
if err := initProject(filename); err != nil {
|
2017-07-04 22:55:24 -03:00
|
|
|
log.WithError(err).Error("failed to init project")
|
|
|
|
return cli.NewExitError("\n", 1)
|
2017-04-27 11:03:26 +02:00
|
|
|
}
|
|
|
|
|
2017-06-22 10:47:34 -03:00
|
|
|
log.WithField("file", filename).
|
2017-07-15 21:01:49 +12:00
|
|
|
Info("config created; please edit accordingly to your needs")
|
2017-04-27 11:03:26 +02:00
|
|
|
return nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
2017-04-21 15:25:32 -03:00
|
|
|
if err := app.Run(os.Args); err != nil {
|
2017-06-22 10:47:34 -03:00
|
|
|
log.WithError(err).Fatal("failed")
|
2017-04-21 15:25:32 -03:00
|
|
|
}
|
|
|
|
}
|
2018-02-21 20:04:22 -03:00
|
|
|
|
|
|
|
func releaseProject(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")
|
|
|
|
}
|
|
|
|
ctx, cancel := context.NewWithTimeout(cfg, flags.Duration("timeout"))
|
|
|
|
defer cancel()
|
|
|
|
ctx.Parallelism = flags.Int("parallelism")
|
|
|
|
ctx.Debug = flags.Bool("debug")
|
|
|
|
log.Debugf("parallelism: %v", ctx.Parallelism)
|
|
|
|
if notes != "" {
|
|
|
|
bts, err := ioutil.ReadFile(notes)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
log.WithField("file", notes).Info("loaded custom release notes")
|
|
|
|
log.WithField("file", notes).Debugf("custom release notes: \n%s", string(bts))
|
|
|
|
ctx.ReleaseNotes = string(bts)
|
|
|
|
}
|
|
|
|
ctx.Snapshot = flags.Bool("snapshot")
|
2018-03-01 01:12:58 -03:00
|
|
|
ctx.SkipPublish = ctx.Snapshot || flags.Bool("skip-publish")
|
2018-03-08 08:42:33 -03:00
|
|
|
ctx.SkipValidate = ctx.Snapshot || flags.Bool("skip-validate")
|
2018-02-21 20:04:22 -03:00
|
|
|
ctx.RmDist = flags.Bool("rm-dist")
|
|
|
|
return doRelease(ctx)
|
|
|
|
}
|
|
|
|
|
|
|
|
func doRelease(ctx *context.Context) error {
|
|
|
|
defer restoreOutputPadding()
|
|
|
|
return ctrlc.Default.Run(ctx, func() error {
|
|
|
|
for _, pipe := range pipes {
|
|
|
|
restoreOutputPadding()
|
|
|
|
log.Infof(color.New(color.Bold).Sprint(strings.ToUpper(pipe.String())))
|
|
|
|
lcli.Default.Padding = increasedPadding
|
|
|
|
if err := handle(pipe.Run(ctx)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func restoreOutputPadding() {
|
|
|
|
lcli.Default.Padding = normalPadding
|
|
|
|
}
|
|
|
|
|
|
|
|
func handle(err error) error {
|
|
|
|
if err == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if pipeline.IsSkip(err) {
|
|
|
|
log.WithField("reason", err.Error()).Warn("skipped")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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)
|
|
|
|
}
|
2018-02-25 18:49:34 -03:00
|
|
|
log.Infof(color.New(color.Bold).Sprint("Generating .goreleaser.yml file"))
|
2018-03-01 23:37:15 -03:00
|
|
|
return ioutil.WriteFile(filename, []byte(exampleConfig), 0644)
|
2018-02-21 20:04:22 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
func getConfigFile(flags Flags) string {
|
|
|
|
var config = flags.String("config")
|
|
|
|
if flags.IsSet("config") {
|
|
|
|
return config
|
|
|
|
}
|
|
|
|
for _, f := range []string{
|
|
|
|
".goreleaser.yml",
|
|
|
|
".goreleaser.yaml",
|
|
|
|
"goreleaser.yml",
|
|
|
|
"goreleaser.yaml",
|
|
|
|
} {
|
|
|
|
_, ferr := os.Stat(f)
|
|
|
|
if ferr == nil || os.IsExist(ferr) {
|
|
|
|
return f
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return config
|
|
|
|
}
|
2018-03-01 23:37:15 -03:00
|
|
|
|
|
|
|
var exampleConfig = `# This is an example goreleaser.yaml file with some sane defaults.
|
|
|
|
# Make sure to check the documentation at http://goreleaser.com
|
|
|
|
builds:
|
|
|
|
- env:
|
|
|
|
- CGO_ENABLED=0
|
|
|
|
archive:
|
|
|
|
replacements:
|
|
|
|
darwin: Darwin
|
|
|
|
linux: Linux
|
|
|
|
windows: Windows
|
|
|
|
386: i386
|
|
|
|
amd64: x86_64
|
|
|
|
checksum:
|
|
|
|
name_template: 'checksums.txt'
|
|
|
|
snapshot:
|
|
|
|
name_template: "{{ .Tag }}-next"
|
|
|
|
changelog:
|
|
|
|
sort: asc
|
|
|
|
filters:
|
|
|
|
exclude:
|
|
|
|
- '^docs:'
|
|
|
|
- '^test:'
|
|
|
|
`
|