-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathfilter.py
118 lines (106 loc) · 4.73 KB
/
filter.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
"""
Edit a Markdown file based on an external metadata file.
Replace each author in the YAML header file by a LaTeX string
with full author information (name, affiliation(s), ORCID iD).
Add funding acknowledgements at the bottom of the manuscript.
"""
import yaml
import argparse
parser = argparse.ArgumentParser(description=__doc__)
parser.add_argument("--input", type=str, required=True,
help="input Markdown file")
parser.add_argument("--output", type=str, required=False,
help="output Markdown file")
parser.add_argument("--contributors", type=str, required=False,
default="contributors.yml",
help="contributors YAML file")
def get_contributors_metadata(content):
entries = yaml.safe_load(content)["authors"]
dictionary = {entry["author"]: entry for entry in entries}
assert len(entries) == len(dictionary), "duplicate keys found in YAML file"
return dictionary
def get_manuscript_metadata(markdown):
markdown = markdown.lstrip()
assert markdown.startswith("---")
markdown = markdown.split("\n", 1)[1]
markdown = markdown.lstrip()
metadata, body = markdown.split("\n---", 1)
body = body.split("\n", 1)[1]
return yaml.safe_load(metadata), body
def get_author_thanks(metadata):
retval = {}
for line in metadata["author"]:
author, *thanks = [x.strip() for x in line.split("|")]
retval[author] = thanks
return retval
def process_markdown(metadata, body, contributors):
authors_tex = []
authors_xmp = []
affiliations = {}
acknowledgements = ["# Acknowledgements {- #sec:acknowledgements}"]
bookkeeping_initials = {}
author_thanks = get_author_thanks(metadata)
if "acknowledgements-before" in metadata:
acknowledgements.append(metadata["acknowledgements-before"])
del metadata["acknowledgements-before"]
# parse metadata
for fullname, annotations in author_thanks.items():
author_metadata = contributors[fullname]
authors_xmp.append(fullname)
author_tex = fullname
if "orcid" in author_metadata:
author_tex += f"\\texorpdfstring{{\\thinspace\\orcidlink{{{author_metadata['orcid']}}}}}{{}}"
author_affil = []
if annotations:
author_affil.append("")
for thanks in annotations:
author_tex += f"\\texorpdfstring{{\\thanks{{{thanks}}}}}{{}}"
for affiliation in author_metadata.get("affiliations", []):
if affiliation not in affiliations:
affiliations[affiliation] = f"{len(affiliations) + 1}"
author_affil.append(affiliations[affiliation])
authors_tex.append(f"\\author[{','.join(author_affil)}]{{{author_tex}}}")
initials = author_metadata["initials"]
assert initials not in bookkeeping_initials, f"initials '{initials}' collide with '{fullname}' and '{bookkeeping_initials[initials]}'"
bookkeeping_initials[initials] = fullname
if "acknowledgements" in author_metadata:
message = author_metadata["acknowledgements"]
assert initials in message, f"'{initials}' doesn't appear in the acknowledgements of '{fullname}'"
acknowledgements.append(message)
affiliations_tex = [f"\\affil[{key}]{{{affil}}}" for affil, key in affiliations.items()]
if "acknowledgements-after" in metadata:
acknowledgements.append(metadata["acknowledgements-after"])
del metadata["acknowledgements-after"]
# update Markdown metadata
del metadata["author"]
metadata["authorxmp"] = authors_xmp
metadata["header-includes"].extend(authors_tex)
metadata["header-includes"].extend(affiliations_tex)
# rewrite Markdown file
markdown = []
markdown.append("---")
markdown.append(yaml.dump(metadata, allow_unicode=True))
markdown.append("---")
token_appendix = "\n\\appendix\n"
if token_appendix in body:
body = body.replace(token_appendix, "\n\n" + "\n".join(acknowledgements) + "\n\n" + token_appendix, 1)
markdown.append(body)
else:
markdown.append(body)
markdown.append("")
markdown.extend(acknowledgements)
return "\n".join(markdown)
if __name__ == "__main__":
args = parser.parse_args()
with open(args.contributors) as f:
contributors = get_contributors_metadata(f.read())
with open(args.input) as f:
metadata, body = get_manuscript_metadata(f.read())
for key in get_author_thanks(metadata).keys():
assert key in contributors, f"author '{key}' in {args.input} not found in {args.contributors}"
markdown = process_markdown(metadata, body, contributors)
if args.output:
with open(args.output, "w") as f:
f.write(markdown)
else:
print(markdown)