-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathgenerate_readme.py
193 lines (158 loc) · 7.86 KB
/
generate_readme.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
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
import os
import json
class MarkdownGenerator:
def __init__(self, data_file):
self._readme_filename = "README.md"
self._readme_template_filename = "README.template"
with open(data_file, "r") as f:
data = json.load(f)
self._year = int(data["year"])
self._repository_url = data["repository_url"]
self._challenge_titles = {}
for day, challenge_title in data["challenge_titles"].items():
self._challenge_titles[int(day)] = challenge_title
def get_repository_url(self) -> str:
return self._repository_url
def get_year(self) -> int:
return self._year
def get_challenge_titles(self) -> dict[str:str]:
return self._challenge_titles
def get_challenge_slug(self, day: int) -> str:
challenge_title = self.get_challenge_titles()[day]
slug = challenge_title.lower().replace(" ", "_")
slug = "".join(c for c in slug if c.isalnum() or c == "_")
return slug
def get_solution_filename(self, day: int) -> str:
challenge_slug = self.get_challenge_slug(day)
day_zerofill = str(day).zfill(2)
return f"{day_zerofill}_{challenge_slug}.py"
def get_markdown_filename(self, day: int) -> str:
challenge_slug = self.get_challenge_slug(day)
day_zerofill = str(day).zfill(2)
return f"{day_zerofill}_{challenge_slug}.ipynb"
def get_aoc_challenge_link(self, day: int) -> str:
return f"https://adventofcode.com/{self._year}/day/{day}"
def get_aoc_solution_github_link(self, day: int) -> str:
filename = self.get_solution_filename(day)
return f"{self.get_repository_url()}/blob/main/solutions/{filename}"
def get_aoc_dialogue_github_link(self, day: int) -> str:
filename = self.get_markdown_filename(day)
return f"{self.get_repository_url()}/blob/main/dialogues/{filename}"
def get_path_to_local_solution_filename(self, day: int) -> str:
challenge_slug = self.get_challenge_slug(day)
day_zerofill = str(day).zfill(2)
return f"solutions/{day_zerofill}_{challenge_slug}.py"
def get_end_of_line_starting_with(self, filename: str, startswith: str, default: str) -> str:
with open(filename, "r") as f:
for line in f:
line = line.strip()
while line.startswith("#"): # remove leading comment indicators
line = line[1:].strip()
if line.lower().startswith(startswith.lower()):
return line[len(startswith):].strip()
return default
def get_space_complexity(self, day: int) -> str:
filename = self.get_path_to_local_solution_filename(day)
return self.get_end_of_line_starting_with(filename, "Space complexity: ", "O(...)")
def get_time_complexity(self, day: int) -> str:
filename = self.get_path_to_local_solution_filename(day)
return self.get_end_of_line_starting_with(filename, "Time complexity: ", "O(...)")
def has_been_solved(self, day: int) -> bool:
def contains_python_code(filename: str) -> bool:
"""
Check if a file contains non-empty lines that are not comments
:param filename:
:return:
"""
with open(filename, "r") as f:
in_comment = False
for line in f:
line = line.strip()
# ignore empty lines
if not line:
continue
# ignore single line comments
if line.startswith('#'):
continue
# ignore multi-line comments
if line.startswith('"""'):
in_comment = not in_comment
continue
if line.endswith('"""'):
in_comment = not in_comment
continue
if in_comment:
continue
return True
solution_filename = self.get_path_to_local_solution_filename(day)
if os.path.exists(solution_filename):
return contains_python_code(solution_filename)
return False
def generate_markdown_table(self) -> str:
def generate_table_header() -> str:
return "Day | Challenge | Solution Code | Solution Dialog | Time Complexity | Space Complexity | Challenge Link"
def generate_row(day: int) -> str:
day_zerofill = str(day).zfill(2)
challenge_title = self.get_challenge_titles()[day]
challenge_link = self.get_aoc_challenge_link(day)
if self.has_been_solved(day):
solution_link = f"[link]({self.get_aoc_solution_github_link(day)})"
dialogue_link = f"[link]({self.get_aoc_dialogue_github_link(day)})"
time_complexity = self.get_time_complexity(day)
space_complexity = self.get_space_complexity(day)
else:
solution_link = "Unsolved"
dialogue_link = "-"
time_complexity = "-"
space_complexity = "-"
return f"{day} | {challenge_title} | {solution_link} | {dialogue_link} | {time_complexity} | {space_complexity} | [adventofcode.com]({challenge_link})"
def format_markdown_table(table: list[str]) -> list[str]:
# Find the maximum width of each column
max_column_width = []
for row in table:
cols = row.split("|")
for i, col in enumerate(cols):
col = col.strip() # here we remove leading and training whitespaces from the cell
if len(max_column_width) <= i:
max_column_width.append(0)
max_column_width[i] = int(max(max_column_width[i], len(col)))
formatted_table = []
# Format the table
for row_nr, row in enumerate(table):
cols = row.split("|")
for i, col in enumerate(cols):
cols[i] = ' ' + col.strip().ljust(
max_column_width[i]) + ' ' # here we add a leading and a trailing whitespace to the cell
formatted_table.append('| ' + "|".join(cols).strip() + " |")
if row_nr == 0:
# add separator row after the header
separator_row = "|".join(['-' * (max_column_width[i] + 2) for i in range(len(cols))])
formatted_table.append('|' + separator_row + "|")
return formatted_table
markdown_table = [generate_table_header()]
for day in range(1, 25 + 1):
markdown_table.append(generate_row(day))
markdown_table = format_markdown_table(markdown_table)
markdown = "\n".join(markdown_table)
return markdown
def generate_readme(self) -> None:
if not os.path.exists(self._readme_template_filename):
raise FileNotFoundError(f"File {self._readme_template_filename} not found.")
table = self.generate_markdown_table()
generated_readme = []
inserted_table_flag = False
with open(self._readme_template_filename, "r") as f:
lines = f.readlines()
for line in lines:
if line.strip() == "<!-- INSERT CHALLENGE TABLE HERE -->":
generated_readme.append(table)
inserted_table_flag = True
else:
generated_readme.append(line.rstrip())
if not inserted_table_flag:
raise ValueError("Could not find the placeholder for the challenge table in the README template.")
with open(self._readme_filename, "w") as f:
f.write("\n".join(generated_readme))
if __name__ == '__main__':
generator = MarkdownGenerator("challenges.json")
generator.generate_readme()