2017-12-23 06:00:46 +01:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
2018-01-27 06:36:57 +01:00
|
|
|
import re
|
2017-12-23 06:00:46 +01:00
|
|
|
import sys
|
|
|
|
|
|
|
|
anchor = '###'
|
2018-01-27 06:36:57 +01:00
|
|
|
min_entries_per_section = 3
|
2017-12-23 06:00:46 +01:00
|
|
|
auth_keys = ['apiKey', 'OAuth', 'X-Mashape-Key', 'No']
|
|
|
|
punctuation = ['.', '?', '!']
|
|
|
|
https_keys = ['Yes', 'No']
|
2018-01-14 19:43:19 +01:00
|
|
|
cors_keys = ['Yes', 'No', 'Unknown']
|
2017-12-23 06:00:46 +01:00
|
|
|
|
|
|
|
index_title = 0
|
|
|
|
index_desc = 1
|
|
|
|
index_auth = 2
|
|
|
|
index_https = 3
|
2018-01-14 19:43:19 +01:00
|
|
|
index_cors = 4
|
|
|
|
index_link = 5
|
2018-09-10 03:38:36 +02:00
|
|
|
num_segments = 5
|
2017-12-23 06:00:46 +01:00
|
|
|
|
|
|
|
errors = []
|
2018-03-12 03:04:14 +01:00
|
|
|
title_links = []
|
2018-05-07 00:53:12 +02:00
|
|
|
previous_links = []
|
2018-03-12 03:15:21 +01:00
|
|
|
anchor_re = re.compile(anchor + '\s(.+)')
|
2018-03-12 03:04:14 +01:00
|
|
|
section_title_re = re.compile('\*\s\[(.*)\]')
|
2018-09-10 03:38:36 +02:00
|
|
|
link_re = re.compile('\[(.+)\]\((http.*)\)')
|
2017-12-23 06:00:46 +01:00
|
|
|
|
|
|
|
|
|
|
|
def add_error(line_num, message):
|
|
|
|
"""adds an error to the dynamic error list"""
|
2018-01-28 21:48:53 +01:00
|
|
|
err = '(L{:03d}) {}'.format(line_num + 1, message)
|
2017-12-23 06:00:46 +01:00
|
|
|
errors.append(err)
|
|
|
|
|
|
|
|
|
2018-01-28 21:48:53 +01:00
|
|
|
def check_alphabetical(lines):
|
2017-12-23 06:00:46 +01:00
|
|
|
"""
|
2018-01-28 21:48:53 +01:00
|
|
|
checks if all entries per section are in alphabetical order based in entry title
|
2017-12-23 06:00:46 +01:00
|
|
|
"""
|
|
|
|
sections = {}
|
|
|
|
section_line_num = {}
|
|
|
|
for line_num, line in enumerate(lines):
|
|
|
|
if line.startswith(anchor):
|
|
|
|
category = line.split(anchor)[1].strip()
|
|
|
|
sections[category] = []
|
|
|
|
section_line_num[category] = line_num
|
|
|
|
continue
|
|
|
|
if not line.startswith('|') or line.startswith('|---'):
|
|
|
|
continue
|
2018-09-10 03:38:36 +02:00
|
|
|
raw_title = [x.strip() for x in line.split('|')[1:-1]][0]
|
|
|
|
title_re_match = link_re.match(raw_title)
|
|
|
|
if title_re_match:
|
|
|
|
sections[category].append(title_re_match.group(1).upper())
|
2017-12-23 06:00:46 +01:00
|
|
|
|
|
|
|
for category, entries in sections.items():
|
|
|
|
if sorted(entries) != entries:
|
2018-01-02 19:15:40 +01:00
|
|
|
add_error(section_line_num[category], "{} section is not in alphabetical order".format(category))
|
2017-12-23 06:00:46 +01:00
|
|
|
|
2018-01-28 21:48:53 +01:00
|
|
|
|
|
|
|
def check_entry(line_num, segments):
|
|
|
|
# START Title
|
2018-09-10 03:38:36 +02:00
|
|
|
raw_title = segments[index_title]
|
|
|
|
title_re_match = link_re.match(raw_title)
|
|
|
|
# url should be wrapped in '[TITLE](LINK)' Markdown syntax
|
|
|
|
if not title_re_match:
|
|
|
|
add_error(line_num, 'Title syntax should be "[TITLE](LINK)"')
|
|
|
|
else:
|
|
|
|
# do not allow "... API" in the entry title
|
|
|
|
title = title_re_match.group(1)
|
|
|
|
if title.upper().endswith(' API'):
|
|
|
|
add_error(line_num, 'Title should not end with "... API". Every entry is an API here!')
|
|
|
|
# do not allow duplicate links
|
|
|
|
link = title_re_match.group(2)
|
|
|
|
if link in previous_links:
|
|
|
|
add_error(line_num, 'Duplicate link - entries should only be included in one section')
|
|
|
|
else:
|
|
|
|
previous_links.append(link)
|
2018-01-28 21:48:53 +01:00
|
|
|
# END Title
|
|
|
|
# START Description
|
|
|
|
# first character should be capitalized
|
|
|
|
char = segments[index_desc][0]
|
|
|
|
if char.upper() != char:
|
|
|
|
add_error(line_num, "first character of description is not capitalized")
|
|
|
|
# last character should not punctuation
|
|
|
|
char = segments[index_desc][-1]
|
|
|
|
if char in punctuation:
|
|
|
|
add_error(line_num, "description should not end with {}".format(char))
|
2018-02-13 03:43:45 +01:00
|
|
|
desc_length = len(segments[index_desc])
|
|
|
|
if desc_length > 100:
|
|
|
|
add_error(line_num, "description should not exceed 100 characters (currently {})".format(desc_length))
|
2018-01-28 21:48:53 +01:00
|
|
|
# END Description
|
|
|
|
# START Auth
|
|
|
|
# values should conform to valid options only
|
2018-02-10 06:55:36 +01:00
|
|
|
auth = segments[index_auth]
|
|
|
|
if auth != 'No' and (not auth.startswith('`') or not auth.endswith('`')):
|
|
|
|
add_error(line_num, "auth value is not enclosed with `backticks`")
|
|
|
|
if auth.replace('`', '') not in auth_keys:
|
2018-01-28 21:48:53 +01:00
|
|
|
add_error(line_num, "{} is not a valid Auth option".format(auth))
|
|
|
|
# END Auth
|
|
|
|
# START HTTPS
|
|
|
|
# values should conform to valid options only
|
|
|
|
https = segments[index_https]
|
|
|
|
if https not in https_keys:
|
|
|
|
add_error(line_num, "{} is not a valid HTTPS option".format(https))
|
|
|
|
# END HTTPS
|
|
|
|
# START CORS
|
|
|
|
# values should conform to valid options only
|
|
|
|
cors = segments[index_cors]
|
|
|
|
if cors not in cors_keys:
|
|
|
|
add_error(line_num, "{} is not a valid CORS option".format(cors))
|
|
|
|
# END CORS
|
|
|
|
|
|
|
|
|
|
|
|
def check_format(filename):
|
|
|
|
"""
|
|
|
|
validates that each line is formatted correctly,
|
|
|
|
appending to error list as needed
|
|
|
|
"""
|
|
|
|
with open(filename) as fp:
|
|
|
|
lines = list(line.rstrip() for line in fp)
|
|
|
|
check_alphabetical(lines)
|
2017-12-23 06:00:46 +01:00
|
|
|
# START Check Entries
|
2018-01-27 06:36:57 +01:00
|
|
|
num_in_category = min_entries_per_section + 1
|
2018-01-28 21:48:53 +01:00
|
|
|
category = ""
|
2018-01-27 17:37:28 +01:00
|
|
|
category_line = 0
|
2017-12-23 06:00:46 +01:00
|
|
|
for line_num, line in enumerate(lines):
|
2018-03-12 03:04:14 +01:00
|
|
|
if section_title_re.match(line):
|
|
|
|
title_links.append(section_title_re.match(line).group(1))
|
2018-01-27 06:36:57 +01:00
|
|
|
# check each section for the minimum number of entries
|
|
|
|
if line.startswith(anchor):
|
2018-03-12 03:04:14 +01:00
|
|
|
match = anchor_re.match(line)
|
|
|
|
if match:
|
|
|
|
if match.group(1) not in title_links:
|
|
|
|
add_error(line_num, "section header ({}) not added as a title link".format(match.group(1)))
|
|
|
|
else:
|
2018-01-27 06:36:57 +01:00
|
|
|
add_error(line_num, "section header is not formatted correctly")
|
|
|
|
if num_in_category < min_entries_per_section:
|
|
|
|
add_error(category_line, "{} section does not have the minimum {} entries (only has {})".format(
|
|
|
|
category, min_entries_per_section, num_in_category))
|
|
|
|
category = line.split(' ')[1]
|
|
|
|
category_line = line_num
|
|
|
|
num_in_category = 0
|
|
|
|
continue
|
2018-03-12 03:15:21 +01:00
|
|
|
# skips lines that we do not care about
|
2017-12-23 06:00:46 +01:00
|
|
|
if not line.startswith('|') or line.startswith('|---'):
|
|
|
|
continue
|
2018-01-27 06:36:57 +01:00
|
|
|
num_in_category += 1
|
2017-12-23 06:00:46 +01:00
|
|
|
segments = line.split('|')[1:-1]
|
2018-04-21 20:22:41 +02:00
|
|
|
if len(segments) < num_segments:
|
|
|
|
add_error(line_num, "entry does not have all the required sections (have {}, need {})".format(
|
|
|
|
len(segments), num_segments))
|
|
|
|
continue
|
2017-12-23 06:00:46 +01:00
|
|
|
# START Global
|
|
|
|
for segment in segments:
|
|
|
|
# every line segment should start and end with exactly 1 space
|
|
|
|
if len(segment) - len(segment.lstrip()) != 1 or len(segment) - len(segment.rstrip()) != 1:
|
|
|
|
add_error(line_num, "each segment must start and end with exactly 1 space")
|
|
|
|
# END Global
|
|
|
|
segments = [seg.strip() for seg in segments]
|
2018-01-28 21:48:53 +01:00
|
|
|
check_entry(line_num, segments)
|
2017-12-23 06:00:46 +01:00
|
|
|
# END Check Entries
|
|
|
|
|
2018-01-28 21:48:53 +01:00
|
|
|
|
2017-12-23 06:00:46 +01:00
|
|
|
def main():
|
2018-03-12 03:15:21 +01:00
|
|
|
if len(sys.argv) < 2:
|
2018-01-02 19:15:40 +01:00
|
|
|
print("No file passed (file should contain Markdown table syntax)")
|
2017-12-23 06:00:46 +01:00
|
|
|
sys.exit(1)
|
|
|
|
check_format(sys.argv[1])
|
|
|
|
if len(errors) > 0:
|
|
|
|
for err in errors:
|
|
|
|
print(err)
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|