chore(go): route release notes through vociferate
This commit is contained in:
@@ -14,6 +14,7 @@ func main() {
|
|||||||
date := flag.String("date", "", "release date in YYYY-MM-DD format")
|
date := flag.String("date", "", "release date in YYYY-MM-DD format")
|
||||||
recommend := flag.Bool("recommend", false, "print the recommended next release tag based on the changelog")
|
recommend := flag.Bool("recommend", false, "print the recommended next release tag based on the changelog")
|
||||||
printUnreleased := flag.Bool("print-unreleased", false, "print the current Unreleased changelog body")
|
printUnreleased := flag.Bool("print-unreleased", false, "print the current Unreleased changelog body")
|
||||||
|
printReleaseNotes := flag.Bool("print-release-notes", false, "print the release notes section for --version")
|
||||||
root := flag.String("root", ".", "repository root to update")
|
root := flag.String("root", ".", "repository root to update")
|
||||||
versionFile := flag.String("version-file", "", "path to the version file, relative to --root")
|
versionFile := flag.String("version-file", "", "path to the version file, relative to --root")
|
||||||
versionPattern := flag.String("version-pattern", "", "regexp with one capture group for the version value")
|
versionPattern := flag.String("version-pattern", "", "regexp with one capture group for the version value")
|
||||||
@@ -52,8 +53,18 @@ func main() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if *printReleaseNotes {
|
||||||
|
body, err := vociferate.ReleaseNotes(absRoot, *version, opts)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "print release notes: %v\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
fmt.Print(body)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
if *version == "" || *date == "" {
|
if *version == "" || *date == "" {
|
||||||
fmt.Fprintln(os.Stderr, "usage: vociferate --version <version> --date <YYYY-MM-DD> [--root <dir>] [--version-file <path>] [--version-pattern <regexp>] [--changelog <path>] | --recommend [--root <dir>] [--version-file <path>] [--version-pattern <regexp>] [--changelog <path>] | --print-unreleased [--root <dir>] [--changelog <path>]")
|
fmt.Fprintln(os.Stderr, "usage: vociferate --version <version> --date <YYYY-MM-DD> [--root <dir>] [--version-file <path>] [--version-pattern <regexp>] [--changelog <path>] | --recommend [--root <dir>] [--version-file <path>] [--version-pattern <regexp>] [--changelog <path>] | --print-unreleased [--root <dir>] [--changelog <path>] | --print-release-notes --version <version> [--root <dir>] [--changelog <path>]")
|
||||||
os.Exit(2)
|
os.Exit(2)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -215,6 +215,11 @@ func UnreleasedBody(rootDir string, options Options) (string, error) {
|
|||||||
return defaultService().UnreleasedBody(rootDir, options)
|
return defaultService().UnreleasedBody(rootDir, options)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ReleaseNotes returns the release section for a specific semantic version.
|
||||||
|
func ReleaseNotes(rootDir, version string, options Options) (string, error) {
|
||||||
|
return defaultService().ReleaseNotes(rootDir, version, options)
|
||||||
|
}
|
||||||
|
|
||||||
// RecommendedTag returns the next semantic release tag based on current changelog state.
|
// RecommendedTag returns the next semantic release tag based on current changelog state.
|
||||||
func (s *Service) RecommendedTag(rootDir string, options Options) (string, error) {
|
func (s *Service) RecommendedTag(rootDir string, options Options) (string, error) {
|
||||||
resolved, err := resolveOptions(options)
|
resolved, err := resolveOptions(options)
|
||||||
@@ -281,6 +286,16 @@ func (s *Service) UnreleasedBody(rootDir string, options Options) (string, error
|
|||||||
return s.readUnreleasedBody(rootDir, resolved.Changelog)
|
return s.readUnreleasedBody(rootDir, resolved.Changelog)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ReleaseNotes returns the release section for a specific semantic version.
|
||||||
|
func (s *Service) ReleaseNotes(rootDir, version string, options Options) (string, error) {
|
||||||
|
resolved, err := resolveOptions(options)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
return s.readReleaseNotes(rootDir, strings.TrimPrefix(strings.TrimSpace(version), "v"), resolved.Changelog)
|
||||||
|
}
|
||||||
|
|
||||||
func sectionHasEntries(unreleasedBody, sectionName string) bool {
|
func sectionHasEntries(unreleasedBody, sectionName string) bool {
|
||||||
heading := "### " + sectionName
|
heading := "### " + sectionName
|
||||||
sectionStart := strings.Index(unreleasedBody, heading)
|
sectionStart := strings.Index(unreleasedBody, heading)
|
||||||
@@ -471,6 +486,10 @@ func readLatestChangelogVersion(rootDir, changelogPath string) (string, bool, er
|
|||||||
return defaultService().readLatestChangelogVersion(rootDir, changelogPath)
|
return defaultService().readLatestChangelogVersion(rootDir, changelogPath)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func readReleaseNotes(rootDir, version, changelogPath string) (string, error) {
|
||||||
|
return defaultService().readReleaseNotes(rootDir, version, changelogPath)
|
||||||
|
}
|
||||||
|
|
||||||
func (s *Service) readLatestChangelogVersion(rootDir, changelogPath string) (string, bool, error) {
|
func (s *Service) readLatestChangelogVersion(rootDir, changelogPath string) (string, bool, error) {
|
||||||
path := filepath.Join(rootDir, changelogPath)
|
path := filepath.Join(rootDir, changelogPath)
|
||||||
contents, err := s.fileSystem.ReadFile(path)
|
contents, err := s.fileSystem.ReadFile(path)
|
||||||
@@ -485,6 +504,38 @@ func (s *Service) readLatestChangelogVersion(rootDir, changelogPath string) (str
|
|||||||
return match[1], true, nil
|
return match[1], true, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *Service) readReleaseNotes(rootDir, version, changelogPath string) (string, error) {
|
||||||
|
if version == "" {
|
||||||
|
return "", fmt.Errorf("release version must not be empty")
|
||||||
|
}
|
||||||
|
|
||||||
|
path := filepath.Join(rootDir, changelogPath)
|
||||||
|
contents, err := s.fileSystem.ReadFile(path)
|
||||||
|
if err != nil {
|
||||||
|
return "", fmt.Errorf("read changelog: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
text := string(contents)
|
||||||
|
headingExpr := regexp.MustCompile(`(?m)^## \[` + regexp.QuoteMeta(version) + `\](?:\([^\n)]*\))? - `)
|
||||||
|
headingLoc := headingExpr.FindStringIndex(text)
|
||||||
|
if headingLoc == nil {
|
||||||
|
return "", fmt.Errorf("release notes section for %s not found in changelog", version)
|
||||||
|
}
|
||||||
|
|
||||||
|
nextSectionRelative := strings.Index(text[headingLoc[0]+1:], "\n## [")
|
||||||
|
sectionEnd := len(text)
|
||||||
|
if nextSectionRelative != -1 {
|
||||||
|
sectionEnd = headingLoc[0] + 1 + nextSectionRelative
|
||||||
|
}
|
||||||
|
|
||||||
|
section := text[headingLoc[0]:sectionEnd]
|
||||||
|
if !strings.HasSuffix(section, "\n") {
|
||||||
|
section += "\n"
|
||||||
|
}
|
||||||
|
|
||||||
|
return section, nil
|
||||||
|
}
|
||||||
|
|
||||||
func deriveRepositoryURL(rootDir string) (string, bool) {
|
func deriveRepositoryURL(rootDir string) (string, bool) {
|
||||||
return defaultService().deriveRepositoryURL(rootDir)
|
return defaultService().deriveRepositoryURL(rootDir)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -36,6 +36,19 @@ outputs:
|
|||||||
runs:
|
runs:
|
||||||
using: composite
|
using: composite
|
||||||
steps:
|
steps:
|
||||||
|
- name: Resolve vociferate binary metadata
|
||||||
|
id: resolve-binary
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
ACTION_REF: ${{ github.action_ref }}
|
||||||
|
ACTION_REPOSITORY: ${{ github.action_repository }}
|
||||||
|
CACHE_TOKEN: ${{ env.VOCIFERATE_CACHE_TOKEN }}
|
||||||
|
SERVER_URL: ${{ github.server_url }}
|
||||||
|
RUNNER_ARCH: ${{ runner.arch }}
|
||||||
|
RUNNER_TEMP: ${{ runner.temp }}
|
||||||
|
run: |
|
||||||
|
bash "$GITHUB_ACTION_PATH/../script/resolve-vociferate-runtime.sh"
|
||||||
|
|
||||||
- name: Resolve release version
|
- name: Resolve release version
|
||||||
id: resolve-version
|
id: resolve-version
|
||||||
shell: bash
|
shell: bash
|
||||||
@@ -63,9 +76,53 @@ runs:
|
|||||||
echo "tag=$tag" >> "$GITHUB_OUTPUT"
|
echo "tag=$tag" >> "$GITHUB_OUTPUT"
|
||||||
echo "version=$normalized" >> "$GITHUB_OUTPUT"
|
echo "version=$normalized" >> "$GITHUB_OUTPUT"
|
||||||
|
|
||||||
- name: Extract release notes from changelog
|
- name: Setup Go
|
||||||
id: extract-notes
|
if: steps.resolve-binary.outputs.use_binary != 'true'
|
||||||
|
uses: actions/setup-go@v5
|
||||||
|
with:
|
||||||
|
go-version: '1.26.1'
|
||||||
|
cache: true
|
||||||
|
cache-dependency-path: ${{ steps.resolve-binary.outputs.source_root }}/go.sum
|
||||||
|
|
||||||
|
- name: Restore cached vociferate binary
|
||||||
|
id: cache-vociferate
|
||||||
|
if: steps.resolve-binary.outputs.use_binary == 'true'
|
||||||
|
uses: actions/cache@v4
|
||||||
|
with:
|
||||||
|
path: ${{ steps.resolve-binary.outputs.cache_dir }}
|
||||||
|
key: vociferate-${{ steps.resolve-binary.outputs.cache_token }}-linux-${{ runner.arch }}
|
||||||
|
|
||||||
|
- name: Download vociferate binary
|
||||||
|
if: steps.resolve-binary.outputs.use_binary == 'true' && steps.cache-vociferate.outputs.cache-hit != 'true'
|
||||||
shell: bash
|
shell: bash
|
||||||
|
env:
|
||||||
|
TOKEN: ${{ github.token }}
|
||||||
|
ASSET_URL: ${{ steps.resolve-binary.outputs.asset_url }}
|
||||||
|
BINARY_PATH: ${{ steps.resolve-binary.outputs.binary_path }}
|
||||||
|
run: |
|
||||||
|
bash "${{ steps.resolve-binary.outputs.source_root }}/script/download-vociferate-binary.sh"
|
||||||
|
|
||||||
|
- name: Extract release notes from binary
|
||||||
|
id: extract-notes-binary
|
||||||
|
if: steps.resolve-binary.outputs.use_binary == 'true'
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
VOCIFERATE_BIN: ${{ steps.resolve-binary.outputs.binary_path }}
|
||||||
|
CHANGELOG: ${{ inputs.changelog != '' && inputs.changelog || 'CHANGELOG.md' }}
|
||||||
|
RELEASE_VERSION: ${{ steps.resolve-version.outputs.version }}
|
||||||
|
RUNNER_TEMP: ${{ runner.temp }}
|
||||||
|
run: |
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
notes_file="${RUNNER_TEMP}/release-notes.md"
|
||||||
|
"$VOCIFERATE_BIN" --print-release-notes --version "$RELEASE_VERSION" --root "$GITHUB_WORKSPACE" --changelog "$CHANGELOG" > "$notes_file"
|
||||||
|
printf 'notes_file=%s\n' "$notes_file" >> "$GITHUB_OUTPUT"
|
||||||
|
|
||||||
|
- name: Extract release notes from source
|
||||||
|
id: extract-notes-source
|
||||||
|
if: steps.resolve-binary.outputs.use_binary != 'true'
|
||||||
|
shell: bash
|
||||||
|
working-directory: ${{ steps.resolve-binary.outputs.source_root }}
|
||||||
env:
|
env:
|
||||||
CHANGELOG: ${{ inputs.changelog != '' && inputs.changelog || 'CHANGELOG.md' }}
|
CHANGELOG: ${{ inputs.changelog != '' && inputs.changelog || 'CHANGELOG.md' }}
|
||||||
RELEASE_VERSION: ${{ steps.resolve-version.outputs.version }}
|
RELEASE_VERSION: ${{ steps.resolve-version.outputs.version }}
|
||||||
@@ -73,22 +130,25 @@ runs:
|
|||||||
run: |
|
run: |
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
|
|
||||||
release_notes="$(awk -v version="$RELEASE_VERSION" '
|
notes_file="${RUNNER_TEMP}/release-notes.md"
|
||||||
$0 ~ "^## \\[" version "\\]" {capture=1}
|
go run ./cmd/vociferate --print-release-notes --version "$RELEASE_VERSION" --root "$GITHUB_WORKSPACE" --changelog "$CHANGELOG" > "$notes_file"
|
||||||
capture {
|
printf 'notes_file=%s\n' "$notes_file" >> "$GITHUB_OUTPUT"
|
||||||
if ($0 ~ "^## \\[" && $0 !~ "^## \\[" version "\\]") exit
|
|
||||||
print
|
|
||||||
}
|
|
||||||
' "$CHANGELOG")"
|
|
||||||
|
|
||||||
if [[ -z "${release_notes//[[:space:]]/}" ]]; then
|
- name: Finalize release notes file
|
||||||
echo "Release notes section for ${RELEASE_VERSION} was not found in ${CHANGELOG}" >&2
|
id: extract-notes
|
||||||
exit 1
|
shell: bash
|
||||||
|
env:
|
||||||
|
NOTES_FILE_BINARY: ${{ steps.extract-notes-binary.outputs.notes_file }}
|
||||||
|
NOTES_FILE_SOURCE: ${{ steps.extract-notes-source.outputs.notes_file }}
|
||||||
|
run: |
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
notes_file="$NOTES_FILE_BINARY"
|
||||||
|
if [[ -z "$notes_file" ]]; then
|
||||||
|
notes_file="$NOTES_FILE_SOURCE"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
notes_file="${RUNNER_TEMP}/release-notes.md"
|
printf 'notes_file=%s\n' "$notes_file" >> "$GITHUB_OUTPUT"
|
||||||
printf '%s\n' "$release_notes" > "$notes_file"
|
|
||||||
echo "notes_file=$notes_file" >> "$GITHUB_OUTPUT"
|
|
||||||
|
|
||||||
- name: Create or update release
|
- name: Create or update release
|
||||||
id: create-release
|
id: create-release
|
||||||
|
|||||||
9
script/download-vociferate-binary.sh
Normal file
9
script/download-vociferate-binary.sh
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
curl --fail --location \
|
||||||
|
-H "Authorization: token ${TOKEN}" \
|
||||||
|
-o "$BINARY_PATH" \
|
||||||
|
"$ASSET_URL"
|
||||||
|
chmod +x "$BINARY_PATH"
|
||||||
51
script/resolve-vociferate-runtime.sh
Normal file
51
script/resolve-vociferate-runtime.sh
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
case "${RUNNER_ARCH}" in
|
||||||
|
X64)
|
||||||
|
arch="amd64"
|
||||||
|
;;
|
||||||
|
ARM64)
|
||||||
|
arch="arm64"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "Unsupported runner architecture: ${RUNNER_ARCH}" >&2
|
||||||
|
exit 1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
source_root="${GITHUB_ACTION_PATH}"
|
||||||
|
if [[ ! -f "${source_root}/go.mod" ]]; then
|
||||||
|
source_root="$(realpath "${GITHUB_ACTION_PATH}/..")"
|
||||||
|
fi
|
||||||
|
|
||||||
|
printf 'source_root=%s\n' "$source_root" >> "$GITHUB_OUTPUT"
|
||||||
|
|
||||||
|
if [[ "${ACTION_REF:-}" == v* ]]; then
|
||||||
|
release_tag="${ACTION_REF}"
|
||||||
|
normalized_version="${release_tag#v}"
|
||||||
|
asset_name="vociferate_${normalized_version}_linux_${arch}"
|
||||||
|
cache_dir="${RUNNER_TEMP}/vociferate/${release_tag}/linux-${arch}"
|
||||||
|
binary_path="${cache_dir}/vociferate"
|
||||||
|
asset_url="${SERVER_URL}/aether/vociferate/releases/download/${release_tag}/${asset_name}"
|
||||||
|
|
||||||
|
provided_cache_token="$(printf '%s' "${CACHE_TOKEN:-}" | sed 's/^[[:space:]]\+//; s/[[:space:]]\+$//')"
|
||||||
|
if [[ -n "$provided_cache_token" ]]; then
|
||||||
|
cache_token="$provided_cache_token"
|
||||||
|
else
|
||||||
|
cache_token="${ACTION_REPOSITORY:-aether/vociferate}-${release_tag}"
|
||||||
|
fi
|
||||||
|
|
||||||
|
mkdir -p "$cache_dir"
|
||||||
|
|
||||||
|
printf 'use_binary=true\n' >> "$GITHUB_OUTPUT"
|
||||||
|
printf 'release_tag=%s\n' "$release_tag" >> "$GITHUB_OUTPUT"
|
||||||
|
printf 'cache_token=%s\n' "$cache_token" >> "$GITHUB_OUTPUT"
|
||||||
|
printf 'asset_name=%s\n' "$asset_name" >> "$GITHUB_OUTPUT"
|
||||||
|
printf 'asset_url=%s\n' "$asset_url" >> "$GITHUB_OUTPUT"
|
||||||
|
printf 'cache_dir=%s\n' "$cache_dir" >> "$GITHUB_OUTPUT"
|
||||||
|
printf 'binary_path=%s\n' "$binary_path" >> "$GITHUB_OUTPUT"
|
||||||
|
else
|
||||||
|
printf 'use_binary=false\n' >> "$GITHUB_OUTPUT"
|
||||||
|
fi
|
||||||
Reference in New Issue
Block a user