2017-10-15 20:21:35 -02:00
|
|
|
// Package changelog provides the release changelog to goreleaser.
|
|
|
|
package changelog
|
|
|
|
|
|
|
|
import (
|
2017-10-19 20:44:12 -02:00
|
|
|
"errors"
|
2017-10-15 20:21:35 -02:00
|
|
|
"fmt"
|
2020-01-31 19:38:56 +01:00
|
|
|
"os"
|
2018-05-01 20:54:16 -07:00
|
|
|
"path/filepath"
|
2017-10-17 23:45:19 -02:00
|
|
|
"regexp"
|
2017-10-19 20:44:12 -02:00
|
|
|
"sort"
|
2017-10-15 20:40:53 -02:00
|
|
|
"strings"
|
2017-10-15 20:21:35 -02:00
|
|
|
|
2022-06-21 21:11:15 -03:00
|
|
|
"github.com/caarlos0/log"
|
2021-10-04 09:32:30 -03:00
|
|
|
"github.com/goreleaser/goreleaser/internal/client"
|
2017-10-15 20:21:35 -02:00
|
|
|
"github.com/goreleaser/goreleaser/internal/git"
|
2020-05-31 09:18:48 -03:00
|
|
|
"github.com/goreleaser/goreleaser/internal/tmpl"
|
2018-08-14 23:50:20 -03:00
|
|
|
"github.com/goreleaser/goreleaser/pkg/context"
|
2017-10-15 20:21:35 -02:00
|
|
|
)
|
|
|
|
|
2020-05-26 00:48:10 -03:00
|
|
|
// ErrInvalidSortDirection happens when the sort order is invalid.
|
2017-10-19 20:44:12 -02:00
|
|
|
var ErrInvalidSortDirection = errors.New("invalid sort direction")
|
|
|
|
|
2021-11-24 09:11:42 -03:00
|
|
|
const li = "* "
|
|
|
|
|
2022-01-08 09:24:36 -03:00
|
|
|
type useChangelog string
|
|
|
|
|
|
|
|
func (u useChangelog) formatable() bool {
|
|
|
|
return u != "github-native"
|
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
|
|
|
useGit = "git"
|
|
|
|
useGitHub = "github"
|
|
|
|
useGitLab = "gitlab"
|
|
|
|
useGitHubNative = "github-native"
|
|
|
|
)
|
|
|
|
|
2020-05-26 00:48:10 -03:00
|
|
|
// Pipe for checksums.
|
2017-10-15 20:21:35 -02:00
|
|
|
type Pipe struct{}
|
|
|
|
|
2023-03-03 09:49:41 -03:00
|
|
|
func (Pipe) String() string { return "generating changelog" }
|
|
|
|
func (Pipe) Skip(ctx *context.Context) (bool, error) {
|
|
|
|
if ctx.Snapshot {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
return tmpl.New(ctx).Bool(ctx.Config.Changelog.Skip)
|
|
|
|
}
|
2017-10-15 20:21:35 -02:00
|
|
|
|
2020-05-26 00:48:10 -03:00
|
|
|
// Run the pipe.
|
2017-10-15 20:33:39 -02:00
|
|
|
func (Pipe) Run(ctx *context.Context) error {
|
2021-05-21 21:07:47 -03:00
|
|
|
notes, err := loadContent(ctx, ctx.ReleaseNotesFile, ctx.ReleaseNotesTmpl)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2019-03-03 14:00:00 -03:00
|
|
|
}
|
2021-05-21 21:07:47 -03:00
|
|
|
ctx.ReleaseNotes = notes
|
|
|
|
|
2022-02-06 22:09:46 -03:00
|
|
|
if ctx.ReleaseNotesFile != "" || ctx.ReleaseNotesTmpl != "" {
|
2019-03-03 14:00:00 -03:00
|
|
|
return nil
|
|
|
|
}
|
2021-05-21 21:07:47 -03:00
|
|
|
|
|
|
|
footer, err := loadContent(ctx, ctx.ReleaseFooterFile, ctx.ReleaseFooterTmpl)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2019-11-15 13:22:11 +00:00
|
|
|
}
|
2021-05-21 21:07:47 -03:00
|
|
|
|
|
|
|
header, err := loadContent(ctx, ctx.ReleaseHeaderFile, ctx.ReleaseHeaderTmpl)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2019-11-15 13:22:11 +00:00
|
|
|
}
|
|
|
|
|
2017-10-19 20:44:12 -02:00
|
|
|
if err := checkSortDirection(ctx.Config.Changelog.Sort); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-11-15 13:22:11 +00:00
|
|
|
|
2017-10-19 20:44:12 -02:00
|
|
|
entries, err := buildChangelog(ctx)
|
2017-10-15 20:21:35 -02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-06-29 16:02:40 +02:00
|
|
|
|
2022-01-08 09:24:36 -03:00
|
|
|
changes, err := formatChangelog(ctx, entries)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2021-11-24 02:23:53 +01:00
|
|
|
}
|
2022-01-08 09:24:36 -03:00
|
|
|
changelogElements := []string{changes}
|
2021-11-24 02:23:53 +01:00
|
|
|
|
2021-05-21 21:07:47 -03:00
|
|
|
if header != "" {
|
|
|
|
changelogElements = append([]string{header}, changelogElements...)
|
2020-11-25 01:09:34 +01:00
|
|
|
}
|
2021-05-21 21:07:47 -03:00
|
|
|
if footer != "" {
|
|
|
|
changelogElements = append(changelogElements, footer)
|
2020-11-25 01:09:34 +01:00
|
|
|
}
|
2020-11-25 03:39:59 +01:00
|
|
|
|
2020-11-25 01:09:34 +01:00
|
|
|
ctx.ReleaseNotes = strings.Join(changelogElements, "\n\n")
|
2020-11-25 03:39:59 +01:00
|
|
|
if !strings.HasSuffix(ctx.ReleaseNotes, "\n") {
|
|
|
|
ctx.ReleaseNotes += "\n"
|
|
|
|
}
|
2019-11-15 13:22:11 +00:00
|
|
|
|
2021-04-19 09:31:57 -03:00
|
|
|
path := filepath.Join(ctx.Config.Dist, "CHANGELOG.md")
|
2018-05-01 20:54:16 -07:00
|
|
|
log.WithField("changelog", path).Info("writing")
|
2021-04-25 13:00:51 -03:00
|
|
|
return os.WriteFile(path, []byte(ctx.ReleaseNotes), 0o644) //nolint: gosec
|
2017-10-19 20:44:12 -02:00
|
|
|
}
|
|
|
|
|
2022-06-26 22:00:23 -03:00
|
|
|
type changelogGroup struct {
|
|
|
|
title string
|
|
|
|
entries []string
|
|
|
|
order int
|
|
|
|
}
|
|
|
|
|
2023-01-07 18:33:15 -03:00
|
|
|
func title(s string, level int) string {
|
|
|
|
if s == "" {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%s %s", strings.Repeat("#", level), s)
|
|
|
|
}
|
|
|
|
|
|
|
|
func newLineFor(ctx *context.Context) string {
|
2022-01-08 09:24:36 -03:00
|
|
|
if ctx.TokenType == context.TokenTypeGitLab || ctx.TokenType == context.TokenTypeGitea {
|
|
|
|
// We need two or more whitespace to let markdown interpret
|
|
|
|
// it as newline. See https://docs.gitlab.com/ee/user/markdown.html#newlines for details
|
|
|
|
log.Debug("is gitlab or gitea changelog")
|
2023-01-07 18:33:15 -03:00
|
|
|
return " \n"
|
2022-01-08 09:24:36 -03:00
|
|
|
}
|
|
|
|
|
2023-01-07 18:33:15 -03:00
|
|
|
return "\n"
|
|
|
|
}
|
|
|
|
|
|
|
|
func abbrevEntry(s string, abbr int) string {
|
|
|
|
switch abbr {
|
|
|
|
case 0:
|
|
|
|
return s
|
|
|
|
case -1:
|
|
|
|
_, rest, _ := strings.Cut(s, " ")
|
|
|
|
return rest
|
|
|
|
default:
|
|
|
|
commit, rest, _ := strings.Cut(s, " ")
|
|
|
|
if abbr > len(commit) {
|
|
|
|
return s
|
2022-08-30 12:04:01 -03:00
|
|
|
}
|
2023-01-07 18:33:15 -03:00
|
|
|
return fmt.Sprintf("%s %s", commit[:abbr], rest)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func abbrev(entries []string, abbr int) []string {
|
|
|
|
result := make([]string, 0, len(entries))
|
|
|
|
for _, entry := range entries {
|
|
|
|
result = append(result, abbrevEntry(entry, abbr))
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func formatChangelog(ctx *context.Context, entries []string) (string, error) {
|
|
|
|
if !useChangelog(ctx.Config.Changelog.Use).formatable() {
|
|
|
|
return strings.Join(entries, newLineFor(ctx)), nil
|
2022-08-30 12:04:01 -03:00
|
|
|
}
|
|
|
|
|
2023-01-07 18:33:15 -03:00
|
|
|
entries = abbrev(entries, ctx.Config.Changelog.Abbrev)
|
|
|
|
|
|
|
|
result := []string{title("Changelog", 2)}
|
2022-01-08 09:24:36 -03:00
|
|
|
if len(ctx.Config.Changelog.Groups) == 0 {
|
|
|
|
log.Debug("not grouping entries")
|
2023-01-07 18:33:15 -03:00
|
|
|
return strings.Join(append(result, filterAndPrefixItems(entries)...), newLineFor(ctx)), nil
|
2022-01-08 09:24:36 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
log.Debug("grouping entries")
|
2022-06-26 22:00:23 -03:00
|
|
|
var groups []changelogGroup
|
|
|
|
for _, group := range ctx.Config.Changelog.Groups {
|
|
|
|
item := changelogGroup{
|
2023-01-07 18:33:15 -03:00
|
|
|
title: title(group.Title, 3),
|
2022-06-26 22:00:23 -03:00
|
|
|
order: group.Order,
|
|
|
|
}
|
2022-01-08 09:24:36 -03:00
|
|
|
if group.Regexp == "" {
|
|
|
|
// If no regexp is provided, we purge all strikethrough entries and add remaining entries to the list
|
2022-06-26 22:00:23 -03:00
|
|
|
item.entries = filterAndPrefixItems(entries)
|
2022-01-08 09:24:36 -03:00
|
|
|
// clear array
|
|
|
|
entries = nil
|
|
|
|
} else {
|
2023-01-07 18:33:15 -03:00
|
|
|
re, err := regexp.Compile(group.Regexp)
|
2022-01-08 09:24:36 -03:00
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("failed to group into %q: %w", group.Title, err)
|
|
|
|
}
|
2022-11-07 12:28:52 +00:00
|
|
|
|
|
|
|
log.Debugf("group: %#v", group)
|
|
|
|
i := 0
|
|
|
|
for _, entry := range entries {
|
2023-01-07 18:33:15 -03:00
|
|
|
match := re.MatchString(entry)
|
2022-11-07 12:28:52 +00:00
|
|
|
log.Debugf("entry: %s match: %b\n", entry, match)
|
2022-01-08 09:24:36 -03:00
|
|
|
if match {
|
2022-06-26 22:00:23 -03:00
|
|
|
item.entries = append(item.entries, li+entry)
|
2022-11-07 12:28:52 +00:00
|
|
|
} else {
|
|
|
|
// Keep unmatched entry.
|
|
|
|
entries[i] = entry
|
|
|
|
i++
|
2022-01-08 09:24:36 -03:00
|
|
|
}
|
|
|
|
}
|
2022-11-07 12:28:52 +00:00
|
|
|
entries = entries[:i]
|
2022-01-08 09:24:36 -03:00
|
|
|
}
|
2022-06-26 22:00:23 -03:00
|
|
|
groups = append(groups, item)
|
2022-11-07 12:28:52 +00:00
|
|
|
|
|
|
|
if len(entries) == 0 {
|
|
|
|
break // No more entries to process.
|
|
|
|
}
|
2022-01-08 09:24:36 -03:00
|
|
|
}
|
|
|
|
|
2023-01-07 18:33:15 -03:00
|
|
|
sort.Slice(groups, groupSort(groups))
|
2022-06-26 22:00:23 -03:00
|
|
|
for _, group := range groups {
|
|
|
|
if len(group.entries) > 0 {
|
2023-01-07 18:33:15 -03:00
|
|
|
result = append(result, group.title)
|
2022-06-26 22:00:23 -03:00
|
|
|
result = append(result, group.entries...)
|
|
|
|
}
|
|
|
|
}
|
2023-01-07 18:33:15 -03:00
|
|
|
return strings.Join(result, newLineFor(ctx)), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func groupSort(groups []changelogGroup) func(i, j int) bool {
|
|
|
|
return func(i, j int) bool {
|
|
|
|
return groups[i].order < groups[j].order
|
|
|
|
}
|
2021-12-21 14:21:05 -03:00
|
|
|
}
|
|
|
|
|
2022-01-08 09:24:36 -03:00
|
|
|
func filterAndPrefixItems(ss []string) []string {
|
2021-11-24 02:23:53 +01:00
|
|
|
var r []string
|
2021-11-24 09:11:42 -03:00
|
|
|
for _, s := range ss {
|
|
|
|
if s != "" {
|
|
|
|
r = append(r, li+s)
|
2021-11-24 02:23:53 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2019-01-22 01:56:16 -02:00
|
|
|
func loadFromFile(file string) (string, error) {
|
2021-04-25 13:00:51 -03:00
|
|
|
bts, err := os.ReadFile(file)
|
2019-01-22 01:56:16 -02:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2021-10-26 23:05:25 -03:00
|
|
|
log.WithField("file", file).Debugf("read %d bytes", len(bts))
|
2019-01-22 01:56:16 -02:00
|
|
|
return string(bts), nil
|
|
|
|
}
|
|
|
|
|
2017-10-19 20:44:12 -02:00
|
|
|
func checkSortDirection(mode string) error {
|
|
|
|
switch mode {
|
2022-12-30 22:57:24 -03:00
|
|
|
case "", "asc", "desc":
|
2017-10-19 20:44:12 -02:00
|
|
|
return nil
|
2022-12-30 22:57:24 -03:00
|
|
|
default:
|
|
|
|
return ErrInvalidSortDirection
|
2017-10-19 20:44:12 -02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func buildChangelog(ctx *context.Context) ([]string, error) {
|
2022-12-29 10:41:59 -03:00
|
|
|
l, err := getChangeloger(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
log, err := l.Log(ctx)
|
2017-10-19 20:44:12 -02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-04-19 09:31:57 -03:00
|
|
|
entries := strings.Split(log, "\n")
|
2021-10-06 01:15:31 -03:00
|
|
|
if lastLine := entries[len(entries)-1]; strings.TrimSpace(lastLine) == "" {
|
|
|
|
entries = entries[0 : len(entries)-1]
|
|
|
|
}
|
2022-01-08 09:24:36 -03:00
|
|
|
if !useChangelog(ctx.Config.Changelog.Use).formatable() {
|
|
|
|
return entries, nil
|
|
|
|
}
|
2017-10-19 20:44:12 -02:00
|
|
|
entries, err = filterEntries(ctx, entries)
|
|
|
|
if err != nil {
|
|
|
|
return entries, err
|
|
|
|
}
|
|
|
|
return sortEntries(ctx, entries), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func filterEntries(ctx *context.Context, entries []string) ([]string, error) {
|
2023-06-16 13:31:23 -03:00
|
|
|
filters := ctx.Config.Changelog.Filters
|
|
|
|
if len(filters.Include) > 0 {
|
|
|
|
var newEntries []string
|
|
|
|
for _, filter := range filters.Include {
|
|
|
|
r, err := regexp.Compile(filter)
|
|
|
|
if err != nil {
|
|
|
|
return entries, err
|
|
|
|
}
|
|
|
|
newEntries = append(newEntries, keep(r, entries)...)
|
|
|
|
}
|
|
|
|
return newEntries, nil
|
|
|
|
}
|
|
|
|
for _, filter := range filters.Exclude {
|
2017-10-17 23:45:19 -02:00
|
|
|
r, err := regexp.Compile(filter)
|
|
|
|
if err != nil {
|
2017-10-19 20:44:12 -02:00
|
|
|
return entries, err
|
2017-10-17 23:45:19 -02:00
|
|
|
}
|
|
|
|
entries = remove(r, entries)
|
2017-10-15 20:40:53 -02:00
|
|
|
}
|
2017-10-19 20:44:12 -02:00
|
|
|
return entries, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func sortEntries(ctx *context.Context, entries []string) []string {
|
2021-04-19 09:31:57 -03:00
|
|
|
direction := ctx.Config.Changelog.Sort
|
2017-10-19 20:44:12 -02:00
|
|
|
if direction == "" {
|
|
|
|
return entries
|
|
|
|
}
|
2021-04-19 09:31:57 -03:00
|
|
|
result := make([]string, len(entries))
|
2017-10-19 20:44:12 -02:00
|
|
|
copy(result, entries)
|
|
|
|
sort.Slice(result, func(i, j int) bool {
|
2021-04-19 09:31:57 -03:00
|
|
|
imsg := extractCommitInfo(result[i])
|
|
|
|
jmsg := extractCommitInfo(result[j])
|
2017-10-19 20:44:12 -02:00
|
|
|
if direction == "asc" {
|
|
|
|
return strings.Compare(imsg, jmsg) < 0
|
|
|
|
}
|
|
|
|
return strings.Compare(imsg, jmsg) > 0
|
|
|
|
})
|
|
|
|
return result
|
2017-10-15 20:21:35 -02:00
|
|
|
}
|
|
|
|
|
2023-06-16 13:31:23 -03:00
|
|
|
func keep(filter *regexp.Regexp, entries []string) (result []string) {
|
|
|
|
for _, entry := range entries {
|
|
|
|
if filter.MatchString(extractCommitInfo(entry)) {
|
|
|
|
result = append(result, entry)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
2017-10-17 23:45:19 -02:00
|
|
|
func remove(filter *regexp.Regexp, entries []string) (result []string) {
|
2017-10-15 20:40:53 -02:00
|
|
|
for _, entry := range entries {
|
2019-10-09 16:07:51 -03:00
|
|
|
if !filter.MatchString(extractCommitInfo(entry)) {
|
2017-10-15 20:40:53 -02:00
|
|
|
result = append(result, entry)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
2019-10-09 16:07:51 -03:00
|
|
|
func extractCommitInfo(line string) string {
|
|
|
|
return strings.Join(strings.Split(line, " ")[1:], " ")
|
2017-10-17 23:45:19 -02:00
|
|
|
}
|
|
|
|
|
2021-10-04 09:32:30 -03:00
|
|
|
func getChangeloger(ctx *context.Context) (changeloger, error) {
|
|
|
|
switch ctx.Config.Changelog.Use {
|
2022-01-08 09:24:36 -03:00
|
|
|
case useGit:
|
2021-10-04 09:32:30 -03:00
|
|
|
fallthrough
|
|
|
|
case "":
|
|
|
|
return gitChangeloger{}, nil
|
2022-01-08 09:24:36 -03:00
|
|
|
case useGitHub:
|
2021-11-06 16:58:07 -03:00
|
|
|
fallthrough
|
2022-01-08 09:24:36 -03:00
|
|
|
case useGitLab:
|
2021-11-06 16:58:07 -03:00
|
|
|
return newSCMChangeloger(ctx)
|
2022-01-08 09:24:36 -03:00
|
|
|
case useGitHubNative:
|
2021-11-07 12:53:28 -03:00
|
|
|
return newGithubChangeloger(ctx)
|
2021-10-04 09:32:30 -03:00
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("invalid changelog.use: %q", ctx.Config.Changelog.Use)
|
|
|
|
}
|
2017-10-15 20:21:35 -02:00
|
|
|
}
|
|
|
|
|
2021-11-07 12:53:28 -03:00
|
|
|
func newGithubChangeloger(ctx *context.Context) (changeloger, error) {
|
2023-04-06 22:58:06 -03:00
|
|
|
cli, err := client.NewGitHubReleaseNotesGenerator(ctx, ctx.Token)
|
2021-11-07 12:53:28 -03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-04-12 08:35:19 -03:00
|
|
|
repo, err := git.ExtractRepoFromConfig(ctx)
|
2021-11-07 12:53:28 -03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-05-09 09:32:43 -03:00
|
|
|
if err := repo.CheckSCM(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-11-07 12:53:28 -03:00
|
|
|
return &githubNativeChangeloger{
|
|
|
|
client: cli,
|
|
|
|
repo: client.Repo{
|
|
|
|
Owner: repo.Owner,
|
|
|
|
Name: repo.Name,
|
|
|
|
},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2021-11-06 16:58:07 -03:00
|
|
|
func newSCMChangeloger(ctx *context.Context) (changeloger, error) {
|
2021-10-10 09:52:46 -04:00
|
|
|
cli, err := client.New(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-04-12 08:35:19 -03:00
|
|
|
repo, err := git.ExtractRepoFromConfig(ctx)
|
2021-10-10 09:52:46 -04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-05-09 09:32:43 -03:00
|
|
|
if err := repo.CheckSCM(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-10-10 09:52:46 -04:00
|
|
|
return &scmChangeloger{
|
|
|
|
client: cli,
|
|
|
|
repo: client.Repo{
|
|
|
|
Owner: repo.Owner,
|
|
|
|
Name: repo.Name,
|
|
|
|
},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2021-05-21 21:07:47 -03:00
|
|
|
func loadContent(ctx *context.Context, fileName, tmplName string) (string, error) {
|
|
|
|
if tmplName != "" {
|
2022-02-06 14:24:47 -03:00
|
|
|
log.Debugf("loading template %q", tmplName)
|
2021-05-21 21:07:47 -03:00
|
|
|
content, err := loadFromFile(tmplName)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2022-02-06 14:24:47 -03:00
|
|
|
content, err = tmpl.New(ctx).Apply(content)
|
|
|
|
if strings.TrimSpace(content) == "" && err == nil {
|
|
|
|
log.Warnf("loaded %q, but it evaluates to an empty string", tmplName)
|
|
|
|
}
|
|
|
|
return content, err
|
2021-05-21 21:07:47 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
if fileName != "" {
|
2022-02-06 14:24:47 -03:00
|
|
|
log.Debugf("loading file %q", fileName)
|
|
|
|
content, err := loadFromFile(fileName)
|
|
|
|
if strings.TrimSpace(content) == "" && err == nil {
|
|
|
|
log.Warnf("loaded %q, but it is empty", fileName)
|
|
|
|
}
|
|
|
|
return content, err
|
2021-05-21 21:07:47 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
return "", nil
|
|
|
|
}
|
2021-10-04 09:32:30 -03:00
|
|
|
|
|
|
|
type changeloger interface {
|
2022-12-29 10:41:59 -03:00
|
|
|
Log(ctx *context.Context) (string, error)
|
2021-10-04 09:32:30 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
type gitChangeloger struct{}
|
|
|
|
|
2022-12-29 14:07:23 -03:00
|
|
|
var validSHA1 = regexp.MustCompile(`^[a-fA-F0-9]{40}$`)
|
|
|
|
|
2022-12-29 10:41:59 -03:00
|
|
|
func (g gitChangeloger) Log(ctx *context.Context) (string, error) {
|
2021-10-04 09:32:30 -03:00
|
|
|
args := []string{"log", "--pretty=oneline", "--abbrev-commit", "--no-decorate", "--no-color"}
|
2022-12-29 14:07:23 -03:00
|
|
|
prev, current := comparePair(ctx)
|
|
|
|
if validSHA1.MatchString(prev) {
|
|
|
|
args = append(args, prev, current)
|
2021-10-04 09:32:30 -03:00
|
|
|
} else {
|
2022-12-29 10:41:59 -03:00
|
|
|
args = append(args, fmt.Sprintf("tags/%s..tags/%s", ctx.Git.PreviousTag, ctx.Git.CurrentTag))
|
2021-10-04 09:32:30 -03:00
|
|
|
}
|
2022-04-12 08:35:19 -03:00
|
|
|
return git.Run(ctx, args...)
|
2021-10-04 09:32:30 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
type scmChangeloger struct {
|
|
|
|
client client.Client
|
2021-10-04 23:35:28 -03:00
|
|
|
repo client.Repo
|
2021-10-04 09:32:30 -03:00
|
|
|
}
|
|
|
|
|
2022-12-29 10:41:59 -03:00
|
|
|
func (c *scmChangeloger) Log(ctx *context.Context) (string, error) {
|
2022-12-29 14:07:23 -03:00
|
|
|
prev, current := comparePair(ctx)
|
|
|
|
return c.client.Changelog(ctx, c.repo, prev, current)
|
2021-10-04 09:32:30 -03:00
|
|
|
}
|
2021-11-07 12:53:28 -03:00
|
|
|
|
|
|
|
type githubNativeChangeloger struct {
|
2023-04-06 22:58:06 -03:00
|
|
|
client client.ReleaseNotesGenerator
|
2021-11-07 12:53:28 -03:00
|
|
|
repo client.Repo
|
|
|
|
}
|
|
|
|
|
2022-12-29 10:41:59 -03:00
|
|
|
func (c *githubNativeChangeloger) Log(ctx *context.Context) (string, error) {
|
|
|
|
return c.client.GenerateReleaseNotes(ctx, c.repo, ctx.Git.PreviousTag, ctx.Git.CurrentTag)
|
2021-11-07 12:53:28 -03:00
|
|
|
}
|
2022-12-29 14:07:23 -03:00
|
|
|
|
|
|
|
func comparePair(ctx *context.Context) (prev string, current string) {
|
|
|
|
prev = ctx.Git.PreviousTag
|
|
|
|
current = ctx.Git.CurrentTag
|
|
|
|
if prev == "" {
|
|
|
|
prev = ctx.Git.FirstCommit
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|