1
0
mirror of https://github.com/goreleaser/goreleaser.git synced 2025-01-10 03:47:03 +02:00
goreleaser/pipeline/git/git.go
Carlos Alexandro Becker 1ed299a6d7 refactor: defaulter interface
Right now the code looks weird because the defaults
of a pipe are far away of the implementation of the pipe.

the intend of this PR is to bring them closer by having a
Defaulter interface.

I also renamed the Pipe interface to Piper, and removed
the Description method in favor for fmt.Stringer.
2017-12-03 13:00:01 -02:00

111 lines
2.5 KiB
Go

// Package git implements the Pipe interface getting and validating the
// current git repository state
package git
import (
"bytes"
"regexp"
"strings"
"text/template"
"time"
"github.com/apex/log"
"github.com/goreleaser/goreleaser/context"
"github.com/goreleaser/goreleaser/internal/git"
"github.com/goreleaser/goreleaser/pipeline"
"github.com/pkg/errors"
)
// Pipe for brew deployment
type Pipe struct{}
// Description of the pipe
func (Pipe) String() string {
return "getting and validating git state"
}
// Run the pipe
func (Pipe) Run(ctx *context.Context) (err error) {
tag, commit, err := getInfo()
if err != nil {
return
}
if tag == "" && !ctx.Snapshot {
return ErrNoTag
}
ctx.Git = context.GitInfo{
CurrentTag: tag,
Commit: commit,
}
log.Infof("releasing %s, commit %s", tag, commit)
if err = setVersion(ctx, tag, commit); err != nil {
return
}
if !ctx.Validate {
return pipeline.Skip("--skip-validate is set")
}
return validate(ctx, commit, tag)
}
func setVersion(ctx *context.Context, tag, commit string) (err error) {
if ctx.Snapshot {
snapshotName, err := getSnapshotName(ctx, tag, commit)
if err != nil {
return errors.Wrap(err, "failed to generate snapshot name")
}
ctx.Version = snapshotName
return nil
}
// removes usual `v` prefix
ctx.Version = strings.TrimPrefix(tag, "v")
return
}
type snapshotNameData struct {
Commit string
Tag string
Timestamp int64
}
func getSnapshotName(ctx *context.Context, tag, commit string) (string, error) {
tmpl, err := template.New("snapshot").Parse(ctx.Config.Snapshot.NameTemplate)
var out bytes.Buffer
if err != nil {
return "", err
}
var data = snapshotNameData{
Commit: commit,
Tag: tag,
Timestamp: time.Now().Unix(),
}
err = tmpl.Execute(&out, data)
return out.String(), err
}
func validate(ctx *context.Context, commit, tag string) error {
out, err := git.Run("status", "--porcelain")
if strings.TrimSpace(out) != "" || err != nil {
return ErrDirty{out}
}
if ctx.Snapshot {
return nil
}
if !regexp.MustCompile("^[0-9.]+").MatchString(ctx.Version) {
return ErrInvalidVersionFormat{ctx.Version}
}
_, err = git.Clean(git.Run("describe", "--exact-match", "--tags", "--match", tag))
if err != nil {
return ErrWrongRef{commit, tag}
}
return nil
}
func getInfo() (tag, commit string, err error) {
tag, err = git.Clean(git.Run("describe", "--tags", "--abbrev=0"))
if err != nil {
log.WithError(err).Info("failed to retrieve current tag")
}
commit, err = git.Clean(git.Run("show", "--format='%H'", "HEAD"))
return
}