This repository has been archived by the owner on Nov 1, 2024. It is now read-only.
forked from new-frontiers-14/frontier-station-14
-
Notifications
You must be signed in to change notification settings - Fork 37
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add manual changelog tool (new-frontiers-14#1548)
- Loading branch information
1 parent
289590a
commit ad14949
Showing
1 changed file
with
150 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,150 @@ | ||
#!/usr/bin/python | ||
# Update the changelog manually, for when you don't want to bother setting up the bot. | ||
|
||
import argparse | ||
import datetime | ||
from re import compile as re_compile, I as re_I | ||
from sys import stdin | ||
from yaml import safe_load as yaml_safe_load, dump as yaml_dump | ||
|
||
|
||
def make_timestamp(): | ||
now = datetime.datetime.now(datetime.timezone.utc) | ||
return now.isoformat() | ||
|
||
def load_changelog(infile): | ||
print(f"Loading changelog {infile.name} ...") | ||
return yaml_safe_load(infile) | ||
|
||
def make_change(change_type, message): | ||
change_type = change_type.lower().capitalize() | ||
|
||
assert(type(message) == str) | ||
assert(change_type in ['Add', 'Remove', 'Fix', 'Tweak']) | ||
|
||
return { | ||
'message': message, | ||
'type': change_type, | ||
} | ||
|
||
def get_last_id(changelog): | ||
return changelog['Entries'][-1]['id'] | ||
|
||
def insert_entry(changelog, author, changes): | ||
new_id = 1 + get_last_id(changelog) | ||
|
||
assert(type(author) == str) | ||
|
||
entry = { | ||
'author': author, | ||
'changes': changes, | ||
'id': new_id, | ||
'time': make_timestamp(), | ||
} | ||
|
||
changelog['Entries'].append(entry) | ||
|
||
def prune_entries(changelog, max_entries=500): | ||
print(f"Pruning changelog to a maxmimum of {max_entries} entries ...") | ||
changelog['Entries'] = changelog['Entries'][-max_entries:] | ||
|
||
def save_changelog(changelog, outfile): | ||
print(f"Saving changelog to {outfile.name} ...") | ||
yaml_dump(changelog, outfile) | ||
|
||
def parse_github_pull_request(changelog, stream): | ||
re_changelog_header = re_compile(r'^\s*:?cl:?\s*(?P<author>.*)', re_I) | ||
re_change = re_compile(r'^\s*-?\s*(?P<type>add|fix|remove|tweak):(?P<message>.+)', re_I) | ||
|
||
pending_entries_with_unspecified_author = [] | ||
author = '' | ||
stored_changes = [] | ||
|
||
def flush_changes(changes): | ||
if len(changes) > 0: | ||
if author == '': | ||
pending_entries_with_unspecified_author.append(changes[:]) | ||
else: | ||
insert_entry(changelog, author, changes[:]) | ||
|
||
changes[:] = [] | ||
|
||
print("Type or paste in changelog entries from GitHub.\n") | ||
|
||
try: | ||
# Parse the stream. | ||
for line in stream: | ||
line = line.strip() | ||
|
||
# Read change lines. | ||
match = re_change.match(line) | ||
|
||
if match: | ||
group = match.groupdict() | ||
|
||
ctype = group['type'].strip() | ||
cmessage = group['message'].strip() | ||
|
||
stored_changes.append(make_change( | ||
ctype, | ||
cmessage, | ||
)) | ||
continue | ||
|
||
# Read author lines. | ||
match = re_changelog_header.match(line) | ||
|
||
if match: | ||
# Flush any changes before setting the new author. | ||
flush_changes(stored_changes) | ||
|
||
group = match.groupdict() | ||
author = group['author'].strip() | ||
except KeyboardInterrupt: | ||
pass | ||
|
||
# Finished reading from stream. | ||
|
||
# Flush any changes. | ||
flush_changes(stored_changes) | ||
|
||
print("") | ||
|
||
# Deal with unspecified authors. | ||
for entry in pending_entries_with_unspecified_author: | ||
print(f"[!] Entry with unspecified author:\n{entry}") | ||
print("\nEnter author> ", end='') | ||
author = input() | ||
insert_entry(changelog, author, entry) | ||
|
||
|
||
def main(): | ||
default_filename = 'Resources/Changelog/Changelog.yml' | ||
|
||
parser = argparse.ArgumentParser(description='Update the changelog manually.') | ||
|
||
parser.add_argument('--infile', type=str, | ||
default=default_filename, | ||
help='which file to load current entries from') | ||
|
||
parser.add_argument('--outfile', type=str, | ||
default=default_filename, | ||
help='which file to save results to') | ||
|
||
args = parser.parse_args() | ||
|
||
infile = open(args.infile, 'r') | ||
cl = load_changelog(infile) | ||
infile.close() | ||
|
||
parse_github_pull_request(cl, stdin) | ||
prune_entries(cl) | ||
|
||
outfile = open(args.outfile, 'w') | ||
save_changelog(cl, outfile) | ||
outfile.close() | ||
|
||
print("Done!") | ||
|
||
if __name__ == "__main__": | ||
main() |