2021-08-21 04:45:52 +00:00
|
|
|
#!/usr/bin/env python
|
|
|
|
"""Download markdown files with AWS managed ConfigRule info and convert to JSON.
|
|
|
|
|
|
|
|
The first markdown file is read to obtain the names of markdown files for
|
|
|
|
all the AWS managed config rules. Then each of those markdown files are read
|
|
|
|
and info is extracted with the final results written to a JSON file.
|
|
|
|
|
|
|
|
The JSON output will look as follows:
|
|
|
|
|
|
|
|
{
|
|
|
|
"ManagedRules": [
|
|
|
|
{
|
|
|
|
"ACCESS_KEYS_ROTATED": {
|
|
|
|
"AWS Region": "All supported AWS regions",
|
|
|
|
"Parameters": [
|
|
|
|
{
|
|
|
|
"Default": "90",
|
|
|
|
"Name": "maxAccessKeyAgeType",
|
|
|
|
"Optional": false,
|
|
|
|
"Type": "intDefault"
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"Trigger type": "Periodic"
|
|
|
|
},
|
|
|
|
},
|
|
|
|
...
|
|
|
|
]
|
|
|
|
}
|
|
|
|
"""
|
|
|
|
import json
|
|
|
|
import re
|
|
|
|
import sys
|
|
|
|
|
|
|
|
import requests
|
|
|
|
|
|
|
|
MANAGED_RULES_OUTPUT_FILENAME = "../moto/config/resources/aws_managed_rules.json"
|
|
|
|
|
|
|
|
AWS_MARKDOWN_URL_START = "https://raw.githubusercontent.com/awsdocs/aws-config-developer-guide/main/doc_source/"
|
|
|
|
|
|
|
|
LIST_OF_MARKDOWNS_URL = "managed-rules-by-aws-config.md"
|
|
|
|
|
|
|
|
|
2021-08-25 14:11:32 +00:00
|
|
|
def extract_param_info(line):
|
|
|
|
"""Return dict containing parameter info extracted from line."""
|
|
|
|
# Examples of parameter definitions:
|
|
|
|
# maxAccessKeyAgeType: intDefault: 90
|
|
|
|
# IgnorePublicAcls \(Optional\)Type: StringDefault: True
|
|
|
|
# MasterAccountId \(Optional\)Type: String
|
|
|
|
# endpointConfigurationTypesType: String
|
|
|
|
|
|
|
|
values = re.split(r":\s?", line)
|
|
|
|
name = values[0]
|
|
|
|
param_type = values[1]
|
|
|
|
|
|
|
|
# If there is no Optional keyword, then sometimes there
|
|
|
|
# isn't a space between the parameter name and "Type".
|
|
|
|
name = re.sub("Type$", "", name)
|
|
|
|
|
|
|
|
# Sometimes there isn't a space between the type and the
|
|
|
|
# word "Default".
|
|
|
|
if "Default" in param_type:
|
|
|
|
param_type = re.sub("Default$", "", param_type)
|
|
|
|
|
|
|
|
optional = False
|
|
|
|
if "Optional" in line:
|
|
|
|
optional = True
|
|
|
|
# Remove "Optional" from the line.
|
|
|
|
name = name.split()[0]
|
|
|
|
|
|
|
|
param_info = {
|
|
|
|
"Name": name,
|
|
|
|
"Optional": optional,
|
|
|
|
"Type": param_type,
|
|
|
|
}
|
|
|
|
|
|
|
|
# A default value isn't always provided.
|
|
|
|
if len(values) > 2:
|
|
|
|
param_info["Default"] = values[2]
|
|
|
|
|
|
|
|
return param_info
|
|
|
|
|
|
|
|
|
|
|
|
def extract_managed_rule_info(lines):
|
2021-08-21 04:45:52 +00:00
|
|
|
"""Return dict of qualifiers/rules extracted from a markdown file."""
|
|
|
|
rule_info = {}
|
|
|
|
label_pattern = re.compile(r"(?:\*\*)(?P<label>[^\*].*)\:\*\*\s?(?P<value>.*)?")
|
|
|
|
|
|
|
|
collecting_params = False
|
|
|
|
params = []
|
|
|
|
for line in lines:
|
|
|
|
if not line:
|
|
|
|
continue
|
|
|
|
line = line.replace("\\", "").strip()
|
|
|
|
|
|
|
|
# Parameters are listed in the lines following the label, so they
|
|
|
|
# require special processing.
|
|
|
|
if collecting_params:
|
|
|
|
# A new header marks the end of the parameters.
|
|
|
|
if line.startswith("##"):
|
|
|
|
rule_info["Parameters"] = params
|
|
|
|
break
|
|
|
|
|
|
|
|
if "Type: " in line:
|
2021-08-25 14:11:32 +00:00
|
|
|
params.append(extract_param_info(line))
|
2021-08-21 04:45:52 +00:00
|
|
|
continue
|
|
|
|
|
|
|
|
# Check for a label starting with two asterisks.
|
|
|
|
matches = re.match(label_pattern, line)
|
|
|
|
if not matches:
|
|
|
|
continue
|
|
|
|
|
2021-08-25 14:11:32 +00:00
|
|
|
# Look for "Identifier", "Trigger type", "AWS Region" and
|
|
|
|
# "Parameters" labels and store the values for all but parameters.
|
|
|
|
# Parameters values aren't on the same line as labels.
|
2021-08-21 04:45:52 +00:00
|
|
|
label = matches.group("label")
|
|
|
|
value = matches.group("value")
|
|
|
|
if label in ["Identifier", "Trigger type", "AWS Region"]:
|
2021-08-25 14:11:32 +00:00
|
|
|
rule_info[label] = value
|
2021-08-21 04:45:52 +00:00
|
|
|
elif label == "Parameters":
|
|
|
|
collecting_params = True
|
|
|
|
else:
|
|
|
|
print(f"ERROR: Unknown label: '{label}', line: '{line}'", file=sys.stderr)
|
|
|
|
return rule_info
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
"""Create a JSON file containing info pulled from AWS markdown files."""
|
|
|
|
req = requests.get(AWS_MARKDOWN_URL_START + LIST_OF_MARKDOWNS_URL)
|
|
|
|
|
|
|
|
# Extract the list of all the markdown files on the page.
|
|
|
|
link_pattern = re.compile(r"\+ \[[^\]]+\]\(([^)]+)\)")
|
|
|
|
markdown_files = link_pattern.findall(req.text)
|
|
|
|
|
2021-08-25 14:11:32 +00:00
|
|
|
# For each of those markdown files, extract the id, region, trigger type
|
|
|
|
# and parameter information.
|
|
|
|
managed_rules = {"ManagedRules": {}}
|
2021-08-21 04:45:52 +00:00
|
|
|
for markdown_file in markdown_files:
|
|
|
|
req = requests.get(AWS_MARKDOWN_URL_START + markdown_file)
|
2021-08-25 14:11:32 +00:00
|
|
|
rules = extract_managed_rule_info(req.text.split("\n"))
|
|
|
|
|
2021-08-21 04:45:52 +00:00
|
|
|
rule_id = rules.pop("Identifier")
|
2021-08-25 14:11:32 +00:00
|
|
|
managed_rules["ManagedRules"][rule_id] = rules
|
2021-08-21 04:45:52 +00:00
|
|
|
|
2021-08-25 14:11:32 +00:00
|
|
|
# Create a JSON file with the extracted managed rule info.
|
2021-08-21 04:45:52 +00:00
|
|
|
with open(MANAGED_RULES_OUTPUT_FILENAME, "w") as fhandle:
|
2021-08-25 14:11:32 +00:00
|
|
|
json.dump(managed_rules, fhandle, sort_keys=True, indent=2)
|
2021-08-21 04:45:52 +00:00
|
|
|
return 0
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
sys.exit(main())
|