mirror of
https://github.com/goreleaser/goreleaser.git
synced 2025-01-08 03:31:59 +02:00
260 lines
8.2 KiB
Go
260 lines
8.2 KiB
Go
package main
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/alecthomas/kingpin"
|
|
"github.com/apex/log"
|
|
"github.com/apex/log/handlers/cli"
|
|
"github.com/caarlos0/ctrlc"
|
|
"github.com/fatih/color"
|
|
|
|
"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/before"
|
|
"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"
|
|
)
|
|
|
|
var (
|
|
version = "dev"
|
|
commit = "none"
|
|
date = "unknown"
|
|
)
|
|
|
|
var pipes = []Piper{
|
|
defaults.Pipe{}, // load default configs
|
|
before.Pipe{}, // run global hooks before build
|
|
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
|
|
}
|
|
|
|
// 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
|
|
}
|
|
|
|
type releaseOptions struct {
|
|
Config string
|
|
ReleaseNotes string
|
|
Snapshot bool
|
|
SkipPublish bool
|
|
SkipValidate bool
|
|
RmDist bool
|
|
Debug bool
|
|
Parallelism int
|
|
Timeout time.Duration
|
|
}
|
|
|
|
func init() {
|
|
log.SetHandler(cli.Default)
|
|
}
|
|
|
|
func main() {
|
|
fmt.Println()
|
|
defer fmt.Println()
|
|
|
|
var app = kingpin.New("goreleaser", "Deliver Go binaries as fast and easily as possible")
|
|
var initCmd = app.Command("init", "Generates a .goreleaser.yml file").Alias("i")
|
|
var releaseCmd = app.Command("release", "Releases the current project").Alias("r").Default()
|
|
var config = releaseCmd.Flag("config", "Load configuration from file").Short('c').Short('f').PlaceHolder(".goreleaser.yml").String()
|
|
var releaseNotes = releaseCmd.Flag("release-notes", "Load custom release notes from a markdown file").PlaceHolder("notes.md").String()
|
|
var snapshot = releaseCmd.Flag("snapshot", "Generate an unversioned snapshot release, skipping all validations and without publishing any artifacts").Bool()
|
|
var skipPublish = releaseCmd.Flag("skip-publish", "Generates all artifacts but does not publish them anywhere").Bool()
|
|
var skipValidate = releaseCmd.Flag("skip-validate", "Skips all git sanity checks").Bool()
|
|
var rmDist = releaseCmd.Flag("rm-dist", "Remove the dist folder before building").Bool()
|
|
var parallelism = releaseCmd.Flag("parallelism", "Amount of slow tasks to do in concurrently").Short('p').Default("4").Int() // TODO: use runtime.NumCPU here?
|
|
var debug = releaseCmd.Flag("debug", "Enable debug mode").Bool()
|
|
var timeout = releaseCmd.Flag("timeout", "Timeout to the entire release process").Default("30m").Duration()
|
|
|
|
app.Version(fmt.Sprintf("%v, commit %v, built at %v", version, commit, date))
|
|
app.VersionFlag.Short('v')
|
|
app.HelpFlag.Short('h')
|
|
|
|
switch kingpin.MustParse(app.Parse(os.Args[1:])) {
|
|
case initCmd.FullCommand():
|
|
var filename = ".goreleaser.yml"
|
|
if err := initProject(filename); err != nil {
|
|
log.WithError(err).Error("failed to init project")
|
|
terminate(1)
|
|
return
|
|
}
|
|
log.WithField("file", filename).Info("config created; please edit accordingly to your needs")
|
|
case releaseCmd.FullCommand():
|
|
start := time.Now()
|
|
log.Infof(color.New(color.Bold).Sprintf("releasing using goreleaser %s...", version))
|
|
var options = releaseOptions{
|
|
Config: *config,
|
|
ReleaseNotes: *releaseNotes,
|
|
Snapshot: *snapshot,
|
|
SkipPublish: *skipPublish,
|
|
SkipValidate: *skipValidate,
|
|
RmDist: *rmDist,
|
|
Parallelism: *parallelism,
|
|
Debug: *debug,
|
|
Timeout: *timeout,
|
|
}
|
|
if err := releaseProject(options); err != nil {
|
|
log.WithError(err).Errorf(color.New(color.Bold).Sprintf("release failed after %0.2fs", time.Since(start).Seconds()))
|
|
terminate(1)
|
|
return
|
|
}
|
|
log.Infof(color.New(color.Bold).Sprintf("release succeeded after %0.2fs", time.Since(start).Seconds()))
|
|
}
|
|
}
|
|
|
|
func terminate(status int) {
|
|
os.Exit(status)
|
|
}
|
|
|
|
func releaseProject(options releaseOptions) error {
|
|
if options.Debug {
|
|
log.SetLevel(log.DebugLevel)
|
|
}
|
|
cfg, err := loadConfig(options.Config)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
ctx, cancel := context.NewWithTimeout(cfg, options.Timeout)
|
|
defer cancel()
|
|
ctx.Parallelism = options.Parallelism
|
|
ctx.Debug = options.Debug
|
|
log.Debugf("parallelism: %v", ctx.Parallelism)
|
|
if options.ReleaseNotes != "" {
|
|
bts, err := ioutil.ReadFile(options.ReleaseNotes)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
log.WithField("file", options.ReleaseNotes).Info("loaded custom release notes")
|
|
log.WithField("file", options.ReleaseNotes).Debugf("custom release notes: \n%s", string(bts))
|
|
ctx.ReleaseNotes = string(bts)
|
|
}
|
|
ctx.Snapshot = options.Snapshot
|
|
ctx.SkipPublish = ctx.Snapshot || options.SkipPublish
|
|
ctx.SkipValidate = ctx.Snapshot || options.SkipValidate
|
|
ctx.RmDist = options.RmDist
|
|
return doRelease(ctx)
|
|
}
|
|
|
|
func doRelease(ctx *context.Context) error {
|
|
defer func() { cli.Default.Padding = 3 }()
|
|
var release = func() error {
|
|
for _, pipe := range pipes {
|
|
cli.Default.Padding = 3
|
|
log.Infof(color.New(color.Bold).Sprint(strings.ToUpper(pipe.String())))
|
|
cli.Default.Padding = 6
|
|
if err := handle(pipe.Run(ctx)); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
return ctrlc.Default.Run(ctx, release)
|
|
}
|
|
|
|
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)
|
|
}
|
|
log.Infof(color.New(color.Bold).Sprintf("Generating %s file", filename))
|
|
return ioutil.WriteFile(filename, []byte(exampleConfig), 0644)
|
|
}
|
|
|
|
func loadConfig(path string) (config.Project, error) {
|
|
if path != "" {
|
|
return config.Load(path)
|
|
}
|
|
for _, f := range [4]string{
|
|
".goreleaser.yml",
|
|
".goreleaser.yaml",
|
|
"goreleaser.yml",
|
|
"goreleaser.yaml",
|
|
} {
|
|
proj, err := config.Load(f)
|
|
if err != nil && os.IsNotExist(err) {
|
|
continue
|
|
}
|
|
return proj, err
|
|
}
|
|
// the user didn't specified a config file and the known files
|
|
// doest not exist, so, return an empty config and a nil err.
|
|
log.Warn("could not load config, using defaults")
|
|
return config.Project{}, nil
|
|
}
|
|
|
|
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:'
|
|
`
|