mirror of
https://github.com/goreleaser/goreleaser.git
synced 2025-01-10 03:47:03 +02:00
874d698564
here's an idea: `goreleaser healthcheck` It'll check if the needed dependencies (docker, git, etc) are available in the path... this way users can preemptively run it before releasing or to debug issues. What do you think? Here's how it looks like: <img width="1007" alt="CleanShot 2023-03-02 at 23 24 26@2x" src="https://user-images.githubusercontent.com/245435/222615682-d9cd0733-d900-43d1-9166-23b2be589b3a.png"> --------- Signed-off-by: Carlos A Becker <caarlos0@users.noreply.github.com>
148 lines
3.8 KiB
Go
148 lines
3.8 KiB
Go
package cmd
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/caarlos0/log"
|
|
"github.com/charmbracelet/lipgloss"
|
|
"github.com/goreleaser/goreleaser/pkg/context"
|
|
"github.com/spf13/cobra"
|
|
cobracompletefig "github.com/withfig/autocomplete-tools/integrations/cobra"
|
|
)
|
|
|
|
var (
|
|
boldStyle = lipgloss.NewStyle().Bold(true)
|
|
codeStyle = lipgloss.NewStyle().Italic(true)
|
|
)
|
|
|
|
func Execute(version string, exit func(int), args []string) {
|
|
newRootCmd(version, exit).Execute(args)
|
|
}
|
|
|
|
func (cmd *rootCmd) Execute(args []string) {
|
|
cmd.cmd.SetArgs(args)
|
|
|
|
if shouldPrependRelease(cmd.cmd, args) {
|
|
cmd.cmd.SetArgs(append([]string{"release"}, args...))
|
|
}
|
|
|
|
if err := cmd.cmd.Execute(); err != nil {
|
|
code := 1
|
|
msg := "command failed"
|
|
eerr := &exitError{}
|
|
if errors.As(err, &eerr) {
|
|
code = eerr.code
|
|
if eerr.details != "" {
|
|
msg = eerr.details
|
|
}
|
|
}
|
|
log.WithError(err).Error(msg)
|
|
cmd.exit(code)
|
|
}
|
|
}
|
|
|
|
type rootCmd struct {
|
|
cmd *cobra.Command
|
|
debug bool
|
|
exit func(int)
|
|
}
|
|
|
|
func newRootCmd(version string, exit func(int)) *rootCmd {
|
|
root := &rootCmd{
|
|
exit: exit,
|
|
}
|
|
cmd := &cobra.Command{
|
|
Use: "goreleaser",
|
|
Short: "Deliver Go binaries as fast and easily as possible",
|
|
Long: `GoReleaser is a release automation tool for Go projects.
|
|
Its goal is to simplify the build, release and publish steps while providing variant customization options for all steps.
|
|
|
|
GoReleaser is built for CI tools, you only need to download and execute it in your build script. Of course, you can also install it locally if you wish.
|
|
|
|
You can customize your entire release process through a single .goreleaser.yaml file.
|
|
|
|
Check out our website for more information, examples and documentation: https://goreleaser.com
|
|
`,
|
|
Version: version,
|
|
SilenceUsage: true,
|
|
SilenceErrors: true,
|
|
Args: cobra.NoArgs,
|
|
PersistentPreRun: func(cmd *cobra.Command, args []string) {
|
|
if root.debug {
|
|
log.SetLevel(log.DebugLevel)
|
|
log.Debug("debug logs enabled")
|
|
}
|
|
},
|
|
}
|
|
|
|
cmd.PersistentFlags().BoolVar(&root.debug, "debug", false, "Enable debug mode")
|
|
cmd.AddCommand(
|
|
newBuildCmd().cmd,
|
|
newReleaseCmd().cmd,
|
|
newCheckCmd().cmd,
|
|
newHealthcheckCmd().cmd,
|
|
newInitCmd().cmd,
|
|
newDocsCmd().cmd,
|
|
newManCmd().cmd,
|
|
newSchemaCmd().cmd,
|
|
cobracompletefig.CreateCompletionSpecCommand(),
|
|
)
|
|
root.cmd = cmd
|
|
return root
|
|
}
|
|
|
|
func shouldPrependRelease(cmd *cobra.Command, args []string) bool {
|
|
// find current cmd, if its not root, it means the user actively
|
|
// set a command, so let it go
|
|
xmd, _, _ := cmd.Find(args)
|
|
if xmd != cmd {
|
|
return false
|
|
}
|
|
|
|
// allow help and the two __complete commands.
|
|
if len(args) > 0 && (args[0] == "help" || args[0] == "completion" ||
|
|
args[0] == cobra.ShellCompRequestCmd || args[0] == cobra.ShellCompNoDescRequestCmd) {
|
|
return false
|
|
}
|
|
|
|
// if we have != 1 args, assume its a release
|
|
if len(args) != 1 {
|
|
return true
|
|
}
|
|
|
|
// given that its 1, check if its one of the valid standalone flags
|
|
// for the root cmd
|
|
for _, s := range []string{"-h", "--help", "-v", "--version"} {
|
|
if s == args[0] {
|
|
// if it is, we should run the root cmd
|
|
return false
|
|
}
|
|
}
|
|
|
|
// otherwise, we should probably prepend release
|
|
return true
|
|
}
|
|
|
|
func deprecateWarn(ctx *context.Context) {
|
|
if ctx.Deprecated {
|
|
log.Warn(boldStyle.Render("you are using deprecated options, check the output above for details"))
|
|
}
|
|
}
|
|
|
|
func timedRunE(verb string, rune func(cmd *cobra.Command, args []string) error) func(cmd *cobra.Command, args []string) error {
|
|
return func(cmd *cobra.Command, args []string) error {
|
|
start := time.Now()
|
|
|
|
log.Infof(boldStyle.Render(fmt.Sprintf("starting %s...", verb)))
|
|
|
|
if err := rune(cmd, args); err != nil {
|
|
return wrapError(err, boldStyle.Render(fmt.Sprintf("%s failed after %s", verb, time.Since(start).Truncate(time.Second))))
|
|
}
|
|
|
|
log.Infof(boldStyle.Render(fmt.Sprintf("%s succeeded after %s", verb, time.Since(start).Truncate(time.Second))))
|
|
return nil
|
|
}
|
|
}
|