This is an automated email from the ASF dual-hosted git repository.
sebb pushed a commit to branch master
in repository https://gitbox.apache.org/repos/asf/whimsy.git
The following commit(s) were added to refs/heads/master by this push:
new 001fbef7 Board minute mark-up tool
001fbef7 is described below
commit 001fbef75d1b403585562a96b9fba1177b38ae96
Author: Sebb <[email protected]>
AuthorDate: Sun Oct 27 15:51:40 2024 +0000
Board minute mark-up tool
---
tools/boardminutes2html.py | 174 +++++++++++++++++++++++++++++++++++++++++++++
1 file changed, 174 insertions(+)
diff --git a/tools/boardminutes2html.py b/tools/boardminutes2html.py
new file mode 100755
index 00000000..53f1a876
--- /dev/null
+++ b/tools/boardminutes2html.py
@@ -0,0 +1,174 @@
+#!/usr/bin/env python3
+
+"""
+Convert board minutes to HTML with anchors and index
+
+Processes minutes to add the following:
+- anchors for internal sections
+- links to internal sections
+- links to external http(s) URLs
+- links to board_minute references
+- index to sections (excluding committee report sections which are just
references)
+
+N.B. The naming convention for internal anchors is:
+ section-xx or attachment-xx
+These anchors are intended to be referenced externally, so the format must not
be changed
+"""
+
+import sys
+import re
+from html import escape
+
+MINUTES = 'https://www.apache.org/foundation/records/minutes/'
+
+def pod_anchor(podling):
+ """convert podling name to anchor"""
+ return podling.strip().lower().replace(' ', '')
+
+# <a class="selflink" id="section-10" href="#section-10">10</a>
+def add_anchor(current_s, line, links, info):
+ """Add anchors"""
+ # main section
+ mat = re.match(r'^([ \d]\d)(\. .+)', line)
+ if mat:
+ sect = mat.group(1)
+ off = ''
+ if sect.startswith(' '):
+ off = ''
+ sid = sect.replace(' ','')
+ rest = mat.group(2)
+ sname = f"section-{sid}"
+ line = f'{off}<a class="selflink" id="{sname}"
href="#{sname}">{sid}{rest}</a>\n'
+ links[sname] = rest.lstrip('. ')
+ # flag when in committee reports
+ if 'Committee Reports' in rest:
+ info['crsection'] = sid
+ else:
+ info.pop('crsection', None)
+ return sid, line # return the updated section number
+
+ # subsections
+ mat = re.match(r'^( {3,4})([A-Z]+)(\. .+)', line)
+ if mat:
+ off = mat.group(1)
+ sect = mat.group(2)
+ sid = current_s + sect.lstrip(' ')
+ sname = f"section-{sid}"
+ rest = mat.group(3)
+ line = f'{off}<a class="selflink" id="{sname}"
href="#{sname}">{sect}{rest}</a>\n'
+ links[sname] = rest.lstrip('. ')
+ return current_s, line
+
+ # Attachments
+ mat = re.match(r'^Attachment (\w+)(: .+)', line)
+ if mat:
+ sect = mat.group(1)
+ sname = 'attachment-' + sect
+ rest = mat.group(2)
+ info['sname'] = rest
+ line = f'<a class="selflink" id="{sname}" href="#{sname}">Attachment
{sect}{rest}</a>\n'
+ links[sname] = rest.lstrip(':')
+ return current_s, line
+
+ # Links to attachments
+ mat = re.match(r'^ +(See Attachment (\w+))', line)
+ if mat:
+ ref = mat.group(1)
+ sect = mat.group(2)
+ line = line.replace(ref, f'<a href="#attachment-{sect}">{ref}</a>')
+ # drop link to CR section if there is an attachment
+ crsect = info.get('crsection')
+ if crsect:
+ links.pop(f'section-{crsect}{sect}')
+ return current_s, line
+
+ # board minutes
+ mat = re.search(r' (board_minutes_(\d\d\d\d)_\d\d_\d\d.txt)', line)
+ if mat:
+ minutes = mat.group(1)
+ year = mat.group(2)
+ line = line.replace(minutes, f'<a
href="{MINUTES}{year}/{minutes}">{minutes}</a>')
+ return current_s, line
+
+ # external URLs TODO: tighten matching ..
+ mat = re.search(r'(https?://[^\s,)]+)', line)
+ if mat:
+ url = mat.group(1).rstrip(".")
+ line = line.replace(url, f'<a href="{url}">{url}</a>')
+ return current_s, line
+
+ # Podling ToC?
+ # [Podling](#podling)
+ mat = re.match(r'\[[^]]+\]\((#[^)]+)\)', line)
+ if mat:
+ anchor = mat.group(1)
+ line = line.replace(anchor, f'<a
href="{pod_anchor(anchor)}">{anchor}</a>')
+ return current_s, line
+
+ # we are in a podling report
+ if info['podhdr'] and line.strip() != '':
+ info['podhdr'] = False
+ pod = line.lstrip('# ').strip()
+ anchor = pod_anchor(pod)
+ if not pod.startswith('---'): # --- indicates end of podlings
+ line = f'<a class="selflink" id="{anchor}"
href="#{anchor}">{line.strip()}</a>\n'
+ links[anchor] = "-- " + pod
+ return current_s, line
+
+ # Start of a podling section?
+ if line.strip() == '--------------------' and 'Incubator Project' in
info['sname']:
+ info['podhdr'] = True
+
+ # anything else
+ return current_s, line
+
+HDR="""<html>
+<head>
+<meta charset="UTF-8">
+<style>
+.selflink {text-decoration: none}
+</style>
+</head>
+<body>
+<a href="#index">Index</a>
+"""
+
+FTR="""</body>
+</html>
+"""
+
+def text2html(inp, out):
+ """html-ise text"""
+ links = {}
+ info = {}
+ # init entries
+ info['sname'] = ''
+ info['podhdr'] = False
+ out.write(HDR)
+ out.write('<pre>')
+ cur_s = None
+ for line in inp:
+ line = escape(line, quote=False) # probably don't need to escape quotes
+ cur_s, line = add_anchor(cur_s, line, links, info)
+ out.write(line)
+ out.write('</pre>\n')
+ out.write('<h2 id="index">Index</h2>\n')
+ out.write('<ul>\n')
+ for link, text in links.items():
+ out.write(f'<li><a href="#{link}">{text}</a></li>\n')
+ out.write('</ul>\n')
+ out.write(FTR)
+
+def process_files(infile, outfile):
+ with open(infile, 'r', encoding='utf8') as inp:
+ with open(outfile, 'w', encoding='utf8') as out:
+ text2html(inp,out)
+
+def main():
+ """Main"""
+ infile = sys.argv[1]
+ outfile = sys.argv[2]
+ process_files(infile, outfile)
+
+if __name__ == '__main__':
+ main()