Skip to content

Commit

Permalink
packaging: generate release notes
Browse files Browse the repository at this point in the history
  • Loading branch information
nilason committed Feb 3, 2024
1 parent 33d6548 commit 3159872
Show file tree
Hide file tree
Showing 3 changed files with 325 additions and 0 deletions.
285 changes: 285 additions & 0 deletions utils/generate_release_notes.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,285 @@
#!/usr/bin/env python3

"""Generate release notes using git log or GitHub API
Needs PyYAML, Git, and GitHub CLI.
"""

import argparse
import csv
import json
import re
import subprocess
import sys
from collections import defaultdict
from pathlib import Path

import yaml

PRETTY_TEMPLATE = (
" - hash: %H%n"
" author_name: %aN%n"
" author_email: %aE%n"
" date: %ad%n"
" message: |-%n %s"
)


def remove_excluded_changes(changes, exclude):
"""Return a list of changes with excluded changes removed"""
result = []
for change in changes:
include = True
for expression in exclude["regexp"]:
if re.match(expression, change):
include = False
break
if include:
result.append(change)
return result


def round_down_to_five(value):
"""Round down to the nearest multiple of five"""
base = 5
return value - (value % base)


def split_to_categories(changes, categories):
"""Return dictionary of changes divided into categories
*categories* is a list of dictionaries (mappings) with
keys title and regexp.
"""
by_category = defaultdict(list)
for change in changes:
added = False
for category in categories:
if re.match(category["regexp"], change):
by_category[category["title"]].append(change)
added = True
break
if not added:
by_category["Other Changes"].append(change)
return by_category


def print_section_heading_2(text, file=None):
print(f"### {text}\n", file=file)


def print_section_heading_3(text, file=None):
print(f"### {text}\n", file=file)


def print_category(category, changes, file=None):
"""Print changes for one category from dictionary of changes
If *changes* don't contain a given category, nothing is printed.
"""
items = changes.get(category, None)
if not items:
return
print_section_heading_3(category, file=file)
for item in sorted(items):
print(f"* {item}", file=file)
print("")


def print_by_category(changes, categories, file=None):
"""Print changes by categories from dictionary of changes"""
for category in categories:
print_category(category["title"], changes, file=file)
print_category("Other Changes", changes, file=file)


def print_notes(
start_tag, end_tag, changes, categories, before=None, after=None, file=None
):
"""Print notes from given inputs
*changes* is a list of strings. It will be sorted and ordered by category
internally by this function.
"""
num_changes = round_down_to_five(len(changes))
print(
f"The GRASS GIS {end_tag} release provides more than "
f"{num_changes} improvements and fixes "
f"with respect to the release {start_tag}.\n"
)

if before:
print(before)
print_section_heading_2("What's Changed", file=file)
changes_by_category = split_to_categories(changes, categories=categories)
print_by_category(changes_by_category, categories=categories, file=file)
if after:
print(after)
print("")


def notes_from_gh_api(start_tag, end_tag, branch, categories, exclude):
"""Generate notes from GitHub API"""
text = subprocess.run(
[
"gh",
"api",
"repos/OSGeo/grass/releases/generate-notes",
"-f",
f"previous_tag_name={start_tag}",
"-f",
f"tag_name={end_tag}",
"-f",
f"target_commitish={branch}",
],
capture_output=True,
text=True,
check=True,
).stdout
body = json.loads(text)["body"]

lines = body.splitlines()
start_whats_changed = lines.index("## What's Changed")
end_whats_changed = lines.index("", start_whats_changed)
raw_changes = lines[start_whats_changed + 1 : end_whats_changed]
changes = []
for change in raw_changes:
if change.startswith("* ") or change.startswith("- "):
changes.append(change[2:])
else:
changes.append(change)
changes = remove_excluded_changes(changes=changes, exclude=exclude)
print_notes(
start_tag=start_tag,
end_tag=end_tag,
changes=changes,
before="\n".join(lines[:start_whats_changed]),
after="\n".join(lines[end_whats_changed + 1 :]),
categories=categories,
)


def csv_to_dict(filename, key, value):
"""Read a CSV file as a dictionary"""
result = {}
with open(filename, encoding="utf-8", newline="") as csvfile:
reader = csv.DictReader(csvfile)
for row in reader:
result[row[key]] = row[value]
return result


def notes_from_git_log(start_tag, end_tag, categories, exclude):
"""Generate notes from git log"""
text = subprocess.run(
["git", "log", f"{start_tag}..{end_tag}", f"--pretty=format:{PRETTY_TEMPLATE}"],
capture_output=True,
text=True,
check=True,
).stdout
commits = yaml.safe_load(text)
if not commits:
raise RuntimeError("No commits retrieved from git log (try different tags)")

config_directory = Path("utils")
github_name_by_git_author = csv_to_dict(
config_directory / "git_author_github_name.csv",
key="git_author",
value="github_name",
)

lines = []
for commit in commits:
if commit["author_email"].endswith("users.noreply.github.com"):
github_name = commit["author_email"].split("@")[0]
if "+" in github_name:
github_name = github_name.split("+")[1]
github_name = f"@{github_name}"
else:
# Emails are stored with @ replaced by a space.
email = commit["author_email"].replace("@", " ")
git_author = f"{commit['author_name']} <{email}>"
if (
git_author in github_name_by_git_author
):
github_name = github_name_by_git_author[git_author]
github_name = f"@{github_name}"
else:
github_name = git_author
lines.append(f"{commit['message']} by {github_name}")
lines = remove_excluded_changes(changes=lines, exclude=exclude)
print_notes(
start_tag=start_tag,
end_tag=end_tag,
changes=lines,
after=(
"**Full Changelog**: "
f"https://github.com/OSGeo/gdal-grass/compare/{start_tag}...{end_tag}"
),
categories=categories,
)


def create_release_notes(args):
"""Create release notes based on parsed command line parameters"""
end_tag = args.end_tag
if not end_tag:
# git log has default, but the others do not.
end_tag = subprocess.run(
["git", "rev-parse", "--verify", "HEAD"],
capture_output=True,
text=True,
check=True,
).stdout.strip()

config_directory = Path("utils")
with open(config_directory / "release.yml", encoding="utf-8") as file:
config = yaml.safe_load(file.read())["notes"]

if args.backend == "api":
notes_from_gh_api(
start_tag=args.start_tag,
end_tag=end_tag,
branch=args.branch,
categories=config["categories"],
exclude=config["exclude"],
)
else:
notes_from_git_log(
start_tag=args.start_tag,
end_tag=end_tag,
categories=config["categories"],
exclude=config["exclude"],
)


def main():
"""Parse command line arguments and create release notes"""
parser = argparse.ArgumentParser(
description="Generate release notes from git log or GitHub API.",
epilog="Run in utils directory to access the helper files.",
)
parser.add_argument(
"backend", choices=["log", "api"], help="use git log or GitHub API"
)
parser.add_argument(
"branch", help="needed for the GitHub API when tag does not exist"
)
parser.add_argument("start_tag", help="old tag to compare against")
parser.add_argument(
"end_tag",
help=(
"new tag; "
"if not created yet, "
"an empty string for git log will use the current revision"
),
)
args = parser.parse_args()
try:
create_release_notes(args)
except subprocess.CalledProcessError as error:
sys.exit(f"Subprocess '{' '.join(error.cmd)}' failed with: {error.stderr}")


if __name__ == "__main__":
main()
14 changes: 14 additions & 0 deletions utils/git_author_github_name.csv
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
git_author,github_name
Alessandro Amici <alexamici tiscali.it>,alexamici
Radim Blazek <radim.blazek gmail.com>,blazek
Huidae Cho <grass4u gmail.com>,HuidaeCho
Bas Couwenberg <sebastic xs4all.nl>,sebastic
Andrey Kiselev <dron ak4719.spb.edu>,strezen
Nicklas Larsson <n_larsson yahoo.com>,nilason
Mateusz Loskot <mateusz loskot.net>,mloskot
Markus Metz <markus.metz.giswork gmail.com>,metzm
Markus Neteler <neteler gmail.com>,neteler
Markus Neteler <neteler mundialis.de>,neteler
Even Rouault <even.rouault spatialys.com>,rouault
Even Rouault (bot) <even.rouault.bot gmail.com>,rouault-bot
Frank Warmerdam <warmerdam pobox.com>,warmerdam
26 changes: 26 additions & 0 deletions utils/release.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
---
notes:
categories:
- title: Documentation and Messages
regexp: '(docs?|man|manual|manual pages|[Ss]phinx|mkhtml|messages?|README): '

- title: Libraries and General Functionality
regexp: '(grass_|lib|TGIS|tgis|raster|vector)[^ ]*: '

- title: Windows
regexp: '(winGRASS|win|[Ww]indows): '

- title: Packaging, Configuration, Portability, and Compilation
regexp: '(packaging|pkg|rpm|deb|pkg-config|configure|config|[Mm]ake|build): '

- title: Continuous Integration, Tests, Code Quality, and Checks
regexp: '(CI|ci|CI\(deps\)|ci\(deps\)|[Tt]ests|[Cc]hecks|pytest): '

- title: Contributing and Management
regexp: '(contributing|CONTRIBUTING.md|contributors|contributors.csv): '

exclude:
regexp:
- '[Hh]appy [Nn]ew [Yy]ear'
- 'version: '
- 'RFC: '

0 comments on commit 3159872

Please sign in to comment.