2017-01-14 12:34:22 -02:00
|
|
|
package git
|
|
|
|
|
2017-01-29 21:55:32 -02:00
|
|
|
import (
|
2020-09-21 14:47:51 -03:00
|
|
|
"fmt"
|
2021-09-04 02:06:53 +03:00
|
|
|
"net/url"
|
2020-01-31 19:38:56 +01:00
|
|
|
"os"
|
2018-08-20 23:18:43 -03:00
|
|
|
"os/exec"
|
2020-07-06 16:09:22 -04:00
|
|
|
"strconv"
|
2017-01-29 21:55:32 -02:00
|
|
|
"strings"
|
2020-07-06 16:09:22 -04:00
|
|
|
"time"
|
2017-01-29 21:55:32 -02:00
|
|
|
|
2022-06-21 21:11:15 -03:00
|
|
|
"github.com/caarlos0/log"
|
2017-08-19 12:47:04 -03:00
|
|
|
"github.com/goreleaser/goreleaser/internal/git"
|
2018-09-12 14:18:01 -03:00
|
|
|
"github.com/goreleaser/goreleaser/internal/pipe"
|
2018-08-14 23:50:20 -03:00
|
|
|
"github.com/goreleaser/goreleaser/pkg/context"
|
2017-01-29 21:55:32 -02:00
|
|
|
)
|
2017-01-14 12:34:22 -02:00
|
|
|
|
2020-05-26 00:48:10 -03:00
|
|
|
// Pipe that sets up git state.
|
2017-01-14 12:34:22 -02:00
|
|
|
type Pipe struct{}
|
|
|
|
|
2017-12-02 19:53:19 -02:00
|
|
|
func (Pipe) String() string {
|
|
|
|
return "getting and validating git state"
|
2017-01-14 12:34:22 -02:00
|
|
|
}
|
|
|
|
|
2020-05-26 00:48:10 -03:00
|
|
|
// Run the pipe.
|
2018-02-25 20:17:45 -03:00
|
|
|
func (Pipe) Run(ctx *context.Context) error {
|
2018-08-20 23:18:43 -03:00
|
|
|
if _, err := exec.LookPath("git"); err != nil {
|
|
|
|
return ErrNoGit
|
|
|
|
}
|
2018-02-25 20:17:45 -03:00
|
|
|
info, err := getInfo(ctx)
|
2017-01-14 12:34:22 -02:00
|
|
|
if err != nil {
|
2018-02-25 20:17:45 -03:00
|
|
|
return err
|
2017-01-14 12:34:22 -02:00
|
|
|
}
|
2018-02-25 20:17:45 -03:00
|
|
|
ctx.Git = info
|
2021-07-24 10:13:05 -03:00
|
|
|
log.WithField("commit", info.Commit).WithField("latest tag", info.CurrentTag).Info("building...")
|
2018-12-13 10:09:36 -02:00
|
|
|
ctx.Version = strings.TrimPrefix(ctx.Git.CurrentTag, "v")
|
2018-02-25 20:17:45 -03:00
|
|
|
return validate(ctx)
|
|
|
|
}
|
|
|
|
|
2018-11-07 22:04:49 -02:00
|
|
|
// nolint: gochecknoglobals
|
2018-03-01 20:42:47 -03:00
|
|
|
var fakeInfo = context.GitInfo{
|
2020-11-18 19:50:31 +01:00
|
|
|
Branch: "none",
|
2018-10-03 17:44:31 -03:00
|
|
|
CurrentTag: "v0.0.0",
|
|
|
|
Commit: "none",
|
|
|
|
ShortCommit: "none",
|
|
|
|
FullCommit: "none",
|
2021-11-24 20:01:56 -07:00
|
|
|
Summary: "none",
|
2018-03-01 20:42:47 -03:00
|
|
|
}
|
|
|
|
|
2018-02-25 20:17:45 -03:00
|
|
|
func getInfo(ctx *context.Context) (context.GitInfo, error) {
|
2022-04-12 08:35:19 -03:00
|
|
|
if !git.IsRepo(ctx) && ctx.Snapshot {
|
2018-03-01 20:42:47 -03:00
|
|
|
log.Warn("accepting to run without a git repo because this is a snapshot")
|
|
|
|
return fakeInfo, nil
|
2018-02-25 20:17:45 -03:00
|
|
|
}
|
2022-04-12 08:35:19 -03:00
|
|
|
if !git.IsRepo(ctx) {
|
2018-02-25 20:17:45 -03:00
|
|
|
return context.GitInfo{}, ErrNotRepository
|
|
|
|
}
|
2022-04-12 08:35:19 -03:00
|
|
|
info, err := getGitInfo(ctx)
|
2018-02-25 20:17:45 -03:00
|
|
|
if err != nil && ctx.Snapshot {
|
2018-03-01 20:42:47 -03:00
|
|
|
log.WithError(err).Warn("ignoring errors because this is a snapshot")
|
|
|
|
if info.Commit == "" {
|
|
|
|
info = fakeInfo
|
|
|
|
}
|
2018-02-25 20:17:45 -03:00
|
|
|
return info, nil
|
|
|
|
}
|
|
|
|
return info, err
|
|
|
|
}
|
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
func getGitInfo(ctx *context.Context) (context.GitInfo, error) {
|
|
|
|
branch, err := getBranch(ctx)
|
2020-11-18 19:50:31 +01:00
|
|
|
if err != nil {
|
|
|
|
return context.GitInfo{}, fmt.Errorf("couldn't get current branch: %w", err)
|
|
|
|
}
|
2022-04-12 08:35:19 -03:00
|
|
|
short, err := getShortCommit(ctx)
|
2018-10-03 17:44:31 -03:00
|
|
|
if err != nil {
|
2020-09-21 14:47:51 -03:00
|
|
|
return context.GitInfo{}, fmt.Errorf("couldn't get current commit: %w", err)
|
2018-10-03 17:44:31 -03:00
|
|
|
}
|
2022-04-12 08:35:19 -03:00
|
|
|
full, err := getFullCommit(ctx)
|
2018-02-25 20:17:45 -03:00
|
|
|
if err != nil {
|
2020-09-21 14:47:51 -03:00
|
|
|
return context.GitInfo{}, fmt.Errorf("couldn't get current commit: %w", err)
|
2017-01-14 14:06:57 -02:00
|
|
|
}
|
2022-04-12 08:35:19 -03:00
|
|
|
date, err := getCommitDate(ctx)
|
2020-07-06 16:09:22 -04:00
|
|
|
if err != nil {
|
2020-09-21 14:47:51 -03:00
|
|
|
return context.GitInfo{}, fmt.Errorf("couldn't get commit date: %w", err)
|
2020-07-06 16:09:22 -04:00
|
|
|
}
|
2022-04-12 08:35:19 -03:00
|
|
|
summary, err := getSummary(ctx)
|
2021-11-24 20:01:56 -07:00
|
|
|
if err != nil {
|
|
|
|
return context.GitInfo{}, fmt.Errorf("couldn't get summary: %w", err)
|
|
|
|
}
|
2022-04-12 08:35:19 -03:00
|
|
|
gitURL, err := getURL(ctx)
|
2018-10-04 09:38:19 -03:00
|
|
|
if err != nil {
|
2020-09-21 14:47:51 -03:00
|
|
|
return context.GitInfo{}, fmt.Errorf("couldn't get remote URL: %w", err)
|
2018-10-04 09:38:19 -03:00
|
|
|
}
|
2021-09-04 02:06:53 +03:00
|
|
|
|
|
|
|
if strings.HasPrefix(gitURL, "https://") {
|
|
|
|
u, err := url.Parse(gitURL)
|
|
|
|
if err != nil {
|
|
|
|
return context.GitInfo{}, fmt.Errorf("couldn't parse remote URL: %w", err)
|
|
|
|
}
|
|
|
|
u.User = nil
|
|
|
|
gitURL = u.String()
|
|
|
|
}
|
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
tag, err := getTag(ctx)
|
2021-11-06 16:54:01 -03:00
|
|
|
if err != nil {
|
2018-02-25 20:17:45 -03:00
|
|
|
return context.GitInfo{
|
2020-11-18 19:50:31 +01:00
|
|
|
Branch: branch,
|
2019-01-19 17:37:42 -02:00
|
|
|
Commit: full,
|
2018-10-03 17:44:31 -03:00
|
|
|
FullCommit: full,
|
|
|
|
ShortCommit: short,
|
2020-07-06 16:09:22 -04:00
|
|
|
CommitDate: date,
|
2021-09-04 02:06:53 +03:00
|
|
|
URL: gitURL,
|
2021-11-06 16:54:01 -03:00
|
|
|
CurrentTag: "v0.0.0",
|
2021-11-24 20:01:56 -07:00
|
|
|
Summary: summary,
|
2018-02-25 20:17:45 -03:00
|
|
|
}, ErrNoTag
|
2017-05-01 10:39:57 -03:00
|
|
|
}
|
2021-11-24 09:12:24 -03:00
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
subject, err := getTagWithFormat(ctx, tag, "contents:subject")
|
2021-12-05 23:23:15 -03:00
|
|
|
if err != nil {
|
|
|
|
return context.GitInfo{}, fmt.Errorf("couldn't get tag subject: %w", err)
|
|
|
|
}
|
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
contents, err := getTagWithFormat(ctx, tag, "contents")
|
2021-12-06 16:52:26 -03:00
|
|
|
if err != nil {
|
|
|
|
return context.GitInfo{}, fmt.Errorf("couldn't get tag contents: %w", err)
|
|
|
|
}
|
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
body, err := getTagWithFormat(ctx, tag, "contents:body")
|
2022-02-24 22:57:51 -03:00
|
|
|
if err != nil {
|
|
|
|
return context.GitInfo{}, fmt.Errorf("couldn't get tag content body: %w", err)
|
|
|
|
}
|
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
previous, err := getPreviousTag(ctx, tag)
|
2021-11-24 09:12:24 -03:00
|
|
|
if err != nil {
|
|
|
|
// shouldn't error, will only affect templates
|
|
|
|
log.Warnf("couldn't find any tags before %q", tag)
|
|
|
|
}
|
|
|
|
|
2018-02-25 20:17:45 -03:00
|
|
|
return context.GitInfo{
|
2020-11-18 19:50:31 +01:00
|
|
|
Branch: branch,
|
2018-10-03 17:44:31 -03:00
|
|
|
CurrentTag: tag,
|
2021-11-24 09:12:24 -03:00
|
|
|
PreviousTag: previous,
|
2019-01-19 17:37:42 -02:00
|
|
|
Commit: full,
|
2018-10-03 17:44:31 -03:00
|
|
|
FullCommit: full,
|
|
|
|
ShortCommit: short,
|
2020-07-06 16:09:22 -04:00
|
|
|
CommitDate: date,
|
2021-09-04 02:06:53 +03:00
|
|
|
URL: gitURL,
|
2021-11-24 20:01:56 -07:00
|
|
|
Summary: summary,
|
2021-12-06 16:52:26 -03:00
|
|
|
TagSubject: subject,
|
|
|
|
TagContents: contents,
|
2022-02-24 22:57:51 -03:00
|
|
|
TagBody: body,
|
2018-02-25 20:17:45 -03:00
|
|
|
}, nil
|
2017-05-01 10:39:57 -03:00
|
|
|
}
|
|
|
|
|
2018-02-25 20:17:45 -03:00
|
|
|
func validate(ctx *context.Context) error {
|
2018-02-24 17:59:08 -03:00
|
|
|
if ctx.Snapshot {
|
2018-09-12 14:18:01 -03:00
|
|
|
return pipe.ErrSnapshotEnabled
|
2018-03-08 08:42:33 -03:00
|
|
|
}
|
|
|
|
if ctx.SkipValidate {
|
2018-09-12 14:18:01 -03:00
|
|
|
return pipe.ErrSkipValidateEnabled
|
2018-02-24 17:59:08 -03:00
|
|
|
}
|
2021-03-24 08:55:13 -03:00
|
|
|
if _, err := os.Stat(".git/shallow"); err == nil {
|
2021-03-24 23:15:23 -03:00
|
|
|
log.Warn("running against a shallow clone - check your CI documentation at https://goreleaser.com/ci")
|
2021-03-24 08:55:13 -03:00
|
|
|
}
|
2022-04-12 08:35:19 -03:00
|
|
|
if err := CheckDirty(ctx); err != nil {
|
2021-06-04 23:09:12 -03:00
|
|
|
return err
|
2017-04-15 16:11:47 -03:00
|
|
|
}
|
2022-04-12 08:35:19 -03:00
|
|
|
_, err := git.Clean(git.Run(ctx, "describe", "--exact-match", "--tags", "--match", ctx.Git.CurrentTag))
|
2017-05-01 10:57:37 -03:00
|
|
|
if err != nil {
|
2018-06-19 15:53:14 -03:00
|
|
|
return ErrWrongRef{
|
|
|
|
commit: ctx.Git.Commit,
|
|
|
|
tag: ctx.Git.CurrentTag,
|
|
|
|
}
|
2017-04-15 16:11:47 -03:00
|
|
|
}
|
|
|
|
return nil
|
2017-01-14 12:34:22 -02:00
|
|
|
}
|
2017-04-15 13:05:54 -03:00
|
|
|
|
2021-06-04 23:09:12 -03:00
|
|
|
// CheckDirty returns an error if the current git repository is dirty.
|
2022-04-12 08:35:19 -03:00
|
|
|
func CheckDirty(ctx *context.Context) error {
|
|
|
|
out, err := git.Run(ctx, "status", "--porcelain")
|
2021-06-04 23:09:12 -03:00
|
|
|
if strings.TrimSpace(out) != "" || err != nil {
|
|
|
|
return ErrDirty{status: out}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
func getBranch(ctx *context.Context) (string, error) {
|
|
|
|
return git.Clean(git.Run(ctx, "rev-parse", "--abbrev-ref", "HEAD", "--quiet"))
|
2020-11-18 19:50:31 +01:00
|
|
|
}
|
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
func getCommitDate(ctx *context.Context) (time.Time, error) {
|
|
|
|
ct, err := git.Clean(git.Run(ctx, "show", "--format='%ct'", "HEAD", "--quiet"))
|
2020-07-06 16:09:22 -04:00
|
|
|
if err != nil {
|
|
|
|
return time.Time{}, err
|
|
|
|
}
|
|
|
|
if ct == "" {
|
|
|
|
return time.Time{}, nil
|
|
|
|
}
|
|
|
|
i, err := strconv.ParseInt(ct, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return time.Time{}, err
|
|
|
|
}
|
|
|
|
t := time.Unix(i, 0).UTC()
|
|
|
|
return t, nil
|
|
|
|
}
|
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
func getShortCommit(ctx *context.Context) (string, error) {
|
2022-10-17 22:39:32 -03:00
|
|
|
return git.Clean(git.Run(ctx, "show", "--format=%h", "HEAD", "--quiet"))
|
2018-10-03 17:44:31 -03:00
|
|
|
}
|
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
func getFullCommit(ctx *context.Context) (string, error) {
|
2022-10-17 22:39:32 -03:00
|
|
|
return git.Clean(git.Run(ctx, "show", "--format=%H", "HEAD", "--quiet"))
|
2018-02-25 20:17:45 -03:00
|
|
|
}
|
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
func getSummary(ctx *context.Context) (string, error) {
|
|
|
|
return git.Clean(git.Run(ctx, "describe", "--always", "--dirty", "--tags"))
|
2021-11-24 20:01:56 -07:00
|
|
|
}
|
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
func getTagWithFormat(ctx *context.Context, tag, format string) (string, error) {
|
|
|
|
out, err := git.Run(ctx, "tag", "-l", "--format='%("+format+")'", tag)
|
2022-02-24 22:57:51 -03:00
|
|
|
return strings.TrimSpace(strings.TrimSuffix(strings.ReplaceAll(out, "'", ""), "\n\n")), err
|
2021-12-06 16:52:26 -03:00
|
|
|
}
|
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
func getTag(ctx *context.Context) (string, error) {
|
2022-10-14 00:31:05 -03:00
|
|
|
for _, fn := range []func() ([]string, error){
|
|
|
|
getFromEnv("GORELEASER_CURRENT_TAG"),
|
|
|
|
func() ([]string, error) {
|
|
|
|
return gitTagsPointingAt(ctx, "HEAD")
|
2020-09-08 10:11:59 -03:00
|
|
|
},
|
2022-10-14 00:31:05 -03:00
|
|
|
func() ([]string, error) {
|
|
|
|
// this will get the last tag, even if it wasn't made against the
|
|
|
|
// last commit...
|
|
|
|
return git.CleanAllLines(gitDescribe(ctx, "HEAD"))
|
2020-09-08 10:11:59 -03:00
|
|
|
},
|
|
|
|
} {
|
2022-10-14 00:31:05 -03:00
|
|
|
tags, err := fn()
|
|
|
|
if len(tags) > 0 {
|
|
|
|
return tags[0], err
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
2020-09-08 10:11:59 -03:00
|
|
|
}
|
2020-01-31 19:38:56 +01:00
|
|
|
}
|
|
|
|
|
2022-10-14 00:31:05 -03:00
|
|
|
return "", nil
|
2017-04-23 16:33:44 -03:00
|
|
|
}
|
2018-10-04 09:38:19 -03:00
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
func getPreviousTag(ctx *context.Context, current string) (string, error) {
|
2022-10-14 00:31:05 -03:00
|
|
|
for _, fn := range []func() ([]string, error){
|
|
|
|
getFromEnv("GORELEASER_PREVIOUS_TAG"),
|
|
|
|
func() ([]string, error) {
|
|
|
|
sha, err := previousTagSha(ctx, current)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return gitTagsPointingAt(ctx, sha)
|
|
|
|
},
|
|
|
|
} {
|
|
|
|
tags, err := fn()
|
|
|
|
if len(tags) > 0 {
|
|
|
|
return tags[0], err
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2021-11-24 09:12:24 -03:00
|
|
|
}
|
|
|
|
|
2022-10-14 00:31:05 -03:00
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func gitTagsPointingAt(ctx *context.Context, ref string) ([]string, error) {
|
|
|
|
return git.CleanAllLines(git.Run(
|
|
|
|
ctx,
|
|
|
|
"tag",
|
|
|
|
"--points-at",
|
|
|
|
ref,
|
|
|
|
"--sort",
|
|
|
|
"-version:refname",
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
|
|
|
func gitDescribe(ctx *context.Context, ref string) (string, error) {
|
|
|
|
return git.Clean(git.Run(
|
|
|
|
ctx,
|
|
|
|
"describe",
|
|
|
|
"--tags",
|
|
|
|
"--abbrev=0",
|
|
|
|
ref,
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
|
|
|
func previousTagSha(ctx *context.Context, current string) (string, error) {
|
|
|
|
tag, err := gitDescribe(ctx, fmt.Sprintf("tags/%s^", current))
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2022-10-17 22:39:32 -03:00
|
|
|
return git.Clean(git.Run(ctx, "rev-list", "-n1", tag))
|
2021-11-24 09:12:24 -03:00
|
|
|
}
|
|
|
|
|
2022-04-12 08:35:19 -03:00
|
|
|
func getURL(ctx *context.Context) (string, error) {
|
|
|
|
return git.Clean(git.Run(ctx, "ls-remote", "--get-url"))
|
2018-10-04 09:38:19 -03:00
|
|
|
}
|
2022-10-14 00:31:05 -03:00
|
|
|
|
|
|
|
func getFromEnv(s string) func() ([]string, error) {
|
|
|
|
return func() ([]string, error) {
|
|
|
|
if tag := os.Getenv(s); tag != "" {
|
|
|
|
return []string{tag}, nil
|
|
|
|
}
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
}
|