2020-05-26 05:48:10 +02:00
|
|
|
// Package tmpl provides templating utilities for goreleaser.
|
2018-07-09 05:47:30 +02:00
|
|
|
package tmpl
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2020-05-24 17:05:49 +02:00
|
|
|
"fmt"
|
2020-04-12 17:13:20 +02:00
|
|
|
"path/filepath"
|
2020-07-06 22:12:41 +02:00
|
|
|
"regexp"
|
2019-04-09 14:15:05 +02:00
|
|
|
"strings"
|
2018-07-09 05:47:30 +02:00
|
|
|
"text/template"
|
|
|
|
"time"
|
|
|
|
|
2021-07-24 15:13:05 +02:00
|
|
|
"github.com/Masterminds/semver/v3"
|
2018-07-09 05:47:30 +02:00
|
|
|
"github.com/goreleaser/goreleaser/internal/artifact"
|
2020-04-12 17:13:20 +02:00
|
|
|
"github.com/goreleaser/goreleaser/pkg/build"
|
2018-08-15 04:50:20 +02:00
|
|
|
"github.com/goreleaser/goreleaser/pkg/context"
|
2022-11-24 20:21:42 +02:00
|
|
|
"golang.org/x/text/cases"
|
|
|
|
"golang.org/x/text/language"
|
2018-07-09 05:47:30 +02:00
|
|
|
)
|
|
|
|
|
2020-05-26 05:48:10 +02:00
|
|
|
// Template holds data that can be applied to a template string.
|
2018-07-09 05:47:30 +02:00
|
|
|
type Template struct {
|
2020-03-22 18:54:47 +02:00
|
|
|
fields Fields
|
2018-07-09 05:47:30 +02:00
|
|
|
}
|
|
|
|
|
2020-03-22 18:54:47 +02:00
|
|
|
// Fields that will be available to the template engine.
|
|
|
|
type Fields map[string]interface{}
|
2018-07-09 05:47:30 +02:00
|
|
|
|
2018-07-09 07:35:44 +02:00
|
|
|
const (
|
2020-08-04 05:21:26 +02:00
|
|
|
// general keys.
|
2020-07-06 22:09:22 +02:00
|
|
|
projectName = "ProjectName"
|
|
|
|
version = "Version"
|
|
|
|
rawVersion = "RawVersion"
|
|
|
|
tag = "Tag"
|
2021-11-24 14:12:24 +02:00
|
|
|
previousTag = "PreviousTag"
|
2020-11-18 20:50:31 +02:00
|
|
|
branch = "Branch"
|
2020-07-06 22:09:22 +02:00
|
|
|
commit = "Commit"
|
|
|
|
shortCommit = "ShortCommit"
|
|
|
|
fullCommit = "FullCommit"
|
|
|
|
commitDate = "CommitDate"
|
|
|
|
commitTimestamp = "CommitTimestamp"
|
|
|
|
gitURL = "GitURL"
|
2021-11-25 05:01:56 +02:00
|
|
|
summary = "Summary"
|
2021-12-06 21:52:26 +02:00
|
|
|
tagSubject = "TagSubject"
|
|
|
|
tagContents = "TagContents"
|
2022-02-25 03:57:51 +02:00
|
|
|
tagBody = "TagBody"
|
2021-11-06 20:39:43 +02:00
|
|
|
releaseURL = "ReleaseURL"
|
2023-05-02 06:04:18 +02:00
|
|
|
isGitDirty = "IsGitDirty"
|
2020-07-06 22:09:22 +02:00
|
|
|
major = "Major"
|
|
|
|
minor = "Minor"
|
|
|
|
patch = "Patch"
|
|
|
|
prerelease = "Prerelease"
|
|
|
|
isSnapshot = "IsSnapshot"
|
2023-06-06 06:07:26 +02:00
|
|
|
isNightly = "IsNightly"
|
2023-03-24 05:01:48 +02:00
|
|
|
isDraft = "IsDraft"
|
2020-07-06 22:09:22 +02:00
|
|
|
env = "Env"
|
|
|
|
date = "Date"
|
2023-03-20 03:21:43 +02:00
|
|
|
now = "Now"
|
2020-07-06 22:09:22 +02:00
|
|
|
timestamp = "Timestamp"
|
2021-03-22 13:55:01 +02:00
|
|
|
modulePath = "ModulePath"
|
2021-10-07 19:19:19 +02:00
|
|
|
releaseNotes = "ReleaseNotes"
|
2022-02-01 03:36:22 +02:00
|
|
|
runtimeK = "Runtime"
|
2018-07-09 07:35:44 +02:00
|
|
|
|
2020-08-04 05:21:26 +02:00
|
|
|
// artifact-only keys.
|
2020-05-10 18:03:49 +02:00
|
|
|
osKey = "Os"
|
2022-04-12 03:43:22 +02:00
|
|
|
amd64 = "Amd64"
|
2018-07-26 15:03:28 +02:00
|
|
|
arch = "Arch"
|
|
|
|
arm = "Arm"
|
2020-02-06 03:08:18 +02:00
|
|
|
mips = "Mips"
|
2018-07-26 15:03:28 +02:00
|
|
|
binary = "Binary"
|
|
|
|
artifactName = "ArtifactName"
|
2022-08-01 21:04:20 +02:00
|
|
|
artifactExt = "ArtifactExt"
|
2020-05-10 18:03:49 +02:00
|
|
|
artifactPath = "ArtifactPath"
|
2020-04-12 17:13:20 +02:00
|
|
|
|
2020-08-04 05:21:26 +02:00
|
|
|
// build keys.
|
2020-04-12 17:13:20 +02:00
|
|
|
name = "Name"
|
|
|
|
ext = "Ext"
|
|
|
|
path = "Path"
|
|
|
|
target = "Target"
|
2018-07-09 07:35:44 +02:00
|
|
|
)
|
2018-07-09 05:47:30 +02:00
|
|
|
|
2020-05-26 05:48:10 +02:00
|
|
|
// New Template.
|
2018-07-09 05:47:30 +02:00
|
|
|
func New(ctx *context.Context) *Template {
|
2020-05-24 17:05:49 +02:00
|
|
|
sv := ctx.Semver
|
|
|
|
rawVersionV := fmt.Sprintf("%d.%d.%d", sv.Major, sv.Minor, sv.Patch)
|
|
|
|
|
2023-03-20 04:32:33 +02:00
|
|
|
fields := map[string]interface{}{}
|
|
|
|
for k, v := range map[string]interface{}{
|
|
|
|
projectName: ctx.Config.ProjectName,
|
|
|
|
modulePath: ctx.ModulePath,
|
|
|
|
version: ctx.Version,
|
|
|
|
rawVersion: rawVersionV,
|
|
|
|
summary: ctx.Git.Summary,
|
|
|
|
tag: ctx.Git.CurrentTag,
|
|
|
|
previousTag: ctx.Git.PreviousTag,
|
|
|
|
branch: ctx.Git.Branch,
|
|
|
|
commit: ctx.Git.Commit,
|
|
|
|
shortCommit: ctx.Git.ShortCommit,
|
|
|
|
fullCommit: ctx.Git.FullCommit,
|
|
|
|
commitDate: ctx.Git.CommitDate.UTC().Format(time.RFC3339),
|
|
|
|
commitTimestamp: ctx.Git.CommitDate.UTC().Unix(),
|
|
|
|
gitURL: ctx.Git.URL,
|
2023-05-02 06:04:18 +02:00
|
|
|
isGitDirty: ctx.Git.Dirty,
|
2023-03-20 04:32:33 +02:00
|
|
|
env: ctx.Env,
|
|
|
|
date: ctx.Date.UTC().Format(time.RFC3339),
|
|
|
|
timestamp: ctx.Date.UTC().Unix(),
|
|
|
|
now: ctx.Date.UTC(),
|
|
|
|
major: ctx.Semver.Major,
|
|
|
|
minor: ctx.Semver.Minor,
|
|
|
|
patch: ctx.Semver.Patch,
|
|
|
|
prerelease: ctx.Semver.Prerelease,
|
|
|
|
isSnapshot: ctx.Snapshot,
|
2023-06-06 06:07:26 +02:00
|
|
|
isNightly: false,
|
2023-03-24 05:01:48 +02:00
|
|
|
isDraft: ctx.Config.Release.Draft,
|
2023-03-20 04:32:33 +02:00
|
|
|
releaseNotes: ctx.ReleaseNotes,
|
|
|
|
releaseURL: ctx.ReleaseURL,
|
|
|
|
tagSubject: ctx.Git.TagSubject,
|
|
|
|
tagContents: ctx.Git.TagContents,
|
|
|
|
tagBody: ctx.Git.TagBody,
|
|
|
|
runtimeK: ctx.Runtime,
|
|
|
|
} {
|
|
|
|
fields[k] = v
|
|
|
|
}
|
|
|
|
|
2018-07-09 05:47:30 +02:00
|
|
|
return &Template{
|
2023-03-20 04:32:33 +02:00
|
|
|
fields: fields,
|
2018-07-09 05:47:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-09 14:15:05 +02:00
|
|
|
// WithEnvS overrides template's env field with the given KEY=VALUE list of
|
2020-05-26 05:48:10 +02:00
|
|
|
// environment variables.
|
2019-04-09 14:15:05 +02:00
|
|
|
func (t *Template) WithEnvS(envs []string) *Template {
|
2021-03-22 13:55:01 +02:00
|
|
|
result := map[string]string{}
|
2019-04-09 14:15:05 +02:00
|
|
|
for _, env := range envs {
|
2022-11-24 22:12:54 +02:00
|
|
|
k, v, ok := strings.Cut(env, "=")
|
|
|
|
if !ok || k == "" {
|
|
|
|
continue
|
|
|
|
}
|
2022-03-17 04:28:13 +02:00
|
|
|
result[k] = v
|
2019-04-09 14:15:05 +02:00
|
|
|
}
|
|
|
|
return t.WithEnv(result)
|
|
|
|
}
|
|
|
|
|
2020-05-26 05:48:10 +02:00
|
|
|
// WithEnv overrides template's env field with the given environment map.
|
2019-04-09 14:15:05 +02:00
|
|
|
func (t *Template) WithEnv(e map[string]string) *Template {
|
|
|
|
t.fields[env] = e
|
|
|
|
return t
|
|
|
|
}
|
|
|
|
|
2020-03-22 18:54:47 +02:00
|
|
|
// WithExtraFields allows to add new more custom fields to the template.
|
|
|
|
// It will override fields with the same name.
|
|
|
|
func (t *Template) WithExtraFields(f Fields) *Template {
|
|
|
|
for k, v := range f {
|
|
|
|
t.fields[k] = v
|
|
|
|
}
|
|
|
|
return t
|
|
|
|
}
|
|
|
|
|
2022-11-25 20:26:14 +02:00
|
|
|
// WithArtifact populates Fields from the artifact.
|
|
|
|
func (t *Template) WithArtifact(a *artifact.Artifact) *Template {
|
|
|
|
t.fields[osKey] = a.Goos
|
|
|
|
t.fields[arch] = a.Goarch
|
|
|
|
t.fields[arm] = a.Goarm
|
|
|
|
t.fields[mips] = a.Gomips
|
|
|
|
t.fields[amd64] = a.Goamd64
|
|
|
|
t.fields[binary] = artifact.ExtraOr(*a, binary, t.fields[projectName].(string))
|
|
|
|
t.fields[artifactName] = a.Name
|
|
|
|
t.fields[artifactExt] = artifact.ExtraOr(*a, artifact.ExtraExt, "")
|
|
|
|
t.fields[artifactPath] = a.Path
|
|
|
|
return t
|
|
|
|
}
|
|
|
|
|
2020-04-12 17:13:20 +02:00
|
|
|
func (t *Template) WithBuildOptions(opts build.Options) *Template {
|
|
|
|
return t.WithExtraFields(buildOptsToFields(opts))
|
|
|
|
}
|
|
|
|
|
|
|
|
func buildOptsToFields(opts build.Options) Fields {
|
|
|
|
return Fields{
|
|
|
|
target: opts.Target,
|
|
|
|
ext: opts.Ext,
|
|
|
|
name: opts.Name,
|
|
|
|
path: opts.Path,
|
2021-09-11 18:01:57 +02:00
|
|
|
osKey: opts.Goos,
|
|
|
|
arch: opts.Goarch,
|
|
|
|
arm: opts.Goarm,
|
|
|
|
mips: opts.Gomips,
|
2020-04-12 17:13:20 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-29 04:21:43 +02:00
|
|
|
// Bool Apply the given string, and converts it to a bool.
|
|
|
|
func (t *Template) Bool(s string) (bool, error) {
|
|
|
|
r, err := t.Apply(s)
|
|
|
|
return strings.TrimSpace(strings.ToLower(r)) == "true", err
|
|
|
|
}
|
|
|
|
|
2020-03-22 18:54:47 +02:00
|
|
|
// Apply applies the given string against the Fields stored in the template.
|
2018-07-09 05:47:30 +02:00
|
|
|
func (t *Template) Apply(s string) (string, error) {
|
|
|
|
var out bytes.Buffer
|
|
|
|
tmpl, err := template.New("tmpl").
|
|
|
|
Option("missingkey=error").
|
|
|
|
Funcs(template.FuncMap{
|
2019-11-07 19:49:36 +02:00
|
|
|
"replace": strings.ReplaceAll,
|
2022-08-07 16:39:04 +02:00
|
|
|
"split": strings.Split,
|
2018-07-09 05:47:30 +02:00
|
|
|
"time": func(s string) string {
|
|
|
|
return time.Now().UTC().Format(s)
|
|
|
|
},
|
2022-02-25 03:57:43 +02:00
|
|
|
"tolower": strings.ToLower,
|
|
|
|
"toupper": strings.ToUpper,
|
|
|
|
"trim": strings.TrimSpace,
|
|
|
|
"trimprefix": strings.TrimPrefix,
|
|
|
|
"trimsuffix": strings.TrimSuffix,
|
2022-11-24 20:21:42 +02:00
|
|
|
"title": cases.Title(language.English).String,
|
2022-02-25 03:57:43 +02:00
|
|
|
"dir": filepath.Dir,
|
2023-03-04 17:16:37 +02:00
|
|
|
"base": filepath.Base,
|
2022-02-25 03:57:43 +02:00
|
|
|
"abs": filepath.Abs,
|
|
|
|
"incmajor": incMajor,
|
|
|
|
"incminor": incMinor,
|
|
|
|
"incpatch": incPatch,
|
|
|
|
"filter": filter(false),
|
|
|
|
"reverseFilter": filter(true),
|
2023-05-27 05:17:02 +02:00
|
|
|
"mdv2escape": mdv2Escape,
|
2023-06-14 05:16:01 +02:00
|
|
|
"envOrDefault": t.envOrDefault,
|
2018-07-09 05:47:30 +02:00
|
|
|
}).
|
|
|
|
Parse(s)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = tmpl.Execute(&out, t.fields)
|
|
|
|
return out.String(), err
|
|
|
|
}
|
|
|
|
|
2023-06-14 05:16:01 +02:00
|
|
|
func (t *Template) envOrDefault(name, value string) string {
|
|
|
|
s, ok := t.fields[env].(context.Env)[name]
|
|
|
|
if !ok {
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
2020-07-06 22:12:41 +02:00
|
|
|
type ExpectedSingleEnvErr struct{}
|
|
|
|
|
|
|
|
func (e ExpectedSingleEnvErr) Error() string {
|
|
|
|
return "expected {{ .Env.VAR_NAME }} only (no plain-text or other interpolation)"
|
|
|
|
}
|
|
|
|
|
|
|
|
// ApplySingleEnvOnly enforces template to only contain a single environment variable
|
|
|
|
// and nothing else.
|
|
|
|
func (t *Template) ApplySingleEnvOnly(s string) (string, error) {
|
|
|
|
s = strings.TrimSpace(s)
|
|
|
|
if len(s) == 0 {
|
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// text/template/parse (lexer) could be used here too,
|
|
|
|
// but regexp reduces the complexity and should be sufficient,
|
|
|
|
// given the context is mostly discouraging users from bad practice
|
|
|
|
// of hard-coded credentials, rather than catch all possible cases
|
|
|
|
envOnlyRe := regexp.MustCompile(`^{{\s*\.Env\.[^.\s}]+\s*}}$`)
|
|
|
|
if !envOnlyRe.Match([]byte(s)) {
|
|
|
|
return "", ExpectedSingleEnvErr{}
|
|
|
|
}
|
|
|
|
|
|
|
|
var out bytes.Buffer
|
|
|
|
tmpl, err := template.New("tmpl").
|
|
|
|
Option("missingkey=error").
|
|
|
|
Parse(s)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = tmpl.Execute(&out, t.fields)
|
|
|
|
return out.String(), err
|
|
|
|
}
|
|
|
|
|
2021-07-24 15:13:05 +02:00
|
|
|
func incMajor(v string) string {
|
|
|
|
return prefix(v) + semver.MustParse(v).IncMajor().String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func incMinor(v string) string {
|
|
|
|
return prefix(v) + semver.MustParse(v).IncMinor().String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func incPatch(v string) string {
|
|
|
|
return prefix(v) + semver.MustParse(v).IncPatch().String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func prefix(v string) string {
|
|
|
|
if v != "" && v[0] == 'v' {
|
|
|
|
return "v"
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
2022-02-25 03:57:43 +02:00
|
|
|
|
|
|
|
func filter(reverse bool) func(content, exp string) string {
|
|
|
|
return func(content, exp string) string {
|
|
|
|
re := regexp.MustCompilePOSIX(exp)
|
|
|
|
var lines []string
|
|
|
|
for _, line := range strings.Split(content, "\n") {
|
|
|
|
if reverse && re.MatchString(line) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if !reverse && !re.MatchString(line) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
lines = append(lines, line)
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.Join(lines, "\n")
|
|
|
|
}
|
|
|
|
}
|
2023-05-27 05:17:02 +02:00
|
|
|
|
|
|
|
func mdv2Escape(s string) string {
|
|
|
|
return strings.NewReplacer(
|
|
|
|
"_", "\\_",
|
|
|
|
"*", "\\*",
|
|
|
|
"[", "\\[",
|
|
|
|
"]", "\\]",
|
|
|
|
"(", "\\(",
|
|
|
|
")", "\\)",
|
|
|
|
"~", "\\~",
|
|
|
|
"`", "\\`",
|
|
|
|
">", "\\>",
|
|
|
|
"#", "\\#",
|
|
|
|
"+", "\\+",
|
|
|
|
"-", "\\-",
|
|
|
|
"=", "\\=",
|
|
|
|
"|", "\\|",
|
|
|
|
"{", "\\{",
|
|
|
|
"}", "\\}",
|
|
|
|
".", "\\.",
|
|
|
|
"!", "\\!",
|
|
|
|
).Replace(s)
|
|
|
|
}
|