125 lines
4.0 KiB
Python
125 lines
4.0 KiB
Python
|
#!/usr/bin/env python3
|
||
|
import argparse
|
||
|
import copy
|
||
|
import os
|
||
|
import subprocess
|
||
|
import sys
|
||
|
import textwrap
|
||
|
|
||
|
import xml.etree.ElementTree as ET
|
||
|
|
||
|
import yaml
|
||
|
|
||
|
|
||
|
SCRIPT_DIR = os.path.dirname(os.path.realpath(__file__))
|
||
|
CONFIG_FILE = os.path.join(SCRIPT_DIR, "flags.yaml")
|
||
|
WIDTH = 80
|
||
|
|
||
|
|
||
|
def wrap_text(s, initial_indent="# "):
|
||
|
return "\n#\n".join(
|
||
|
map(lambda x: textwrap.fill(x, WIDTH, initial_indent=initial_indent,
|
||
|
subsequent_indent="# "), s.split("\n")))
|
||
|
|
||
|
|
||
|
def extract_flags(binary_path):
|
||
|
ret = {}
|
||
|
data = subprocess.run([binary_path, "--help-xml"],
|
||
|
stdout=subprocess.PIPE).stdout.decode("utf-8")
|
||
|
root = ET.fromstring(data)
|
||
|
for child in root:
|
||
|
if child.tag == "usage" and child.text.lower().count("warning"):
|
||
|
raise Exception("You should set the usage message!")
|
||
|
if child.tag == "flag":
|
||
|
flag = {}
|
||
|
for elem in child:
|
||
|
flag[elem.tag] = elem.text if elem.text is not None else ""
|
||
|
flag["override"] = False
|
||
|
ret[flag["name"]] = flag
|
||
|
return ret
|
||
|
|
||
|
|
||
|
def apply_config_to_flags(config, flags):
|
||
|
flags = copy.deepcopy(flags)
|
||
|
for name in config["undocumented"]:
|
||
|
flags.pop(name)
|
||
|
for modification in config["modifications"]:
|
||
|
name = modification["name"]
|
||
|
if name not in flags:
|
||
|
print("WARNING: Flag '" + name + "' missing from binary!",
|
||
|
file=sys.stderr)
|
||
|
continue
|
||
|
flags[name]["default"] = modification["value"]
|
||
|
flags[name]["override"] = modification["override"]
|
||
|
return flags
|
||
|
|
||
|
|
||
|
def extract_sections(flags):
|
||
|
sections = []
|
||
|
other = []
|
||
|
current_section = ""
|
||
|
current_flags = []
|
||
|
for name in sorted(flags.keys()):
|
||
|
section = name.split("_")[0]
|
||
|
if section == current_section:
|
||
|
current_flags.append(name)
|
||
|
else:
|
||
|
if len(current_flags) < 2:
|
||
|
other.extend(current_flags)
|
||
|
else:
|
||
|
sections.append((current_section, current_flags))
|
||
|
current_section = section
|
||
|
current_flags = [name]
|
||
|
if len(current_flags) < 2:
|
||
|
other.extend(current_flags)
|
||
|
else:
|
||
|
sections.append((current_section, current_flags))
|
||
|
sections.append(("other", other))
|
||
|
assert set(sum(map(lambda x: x[1], sections), [])) == set(flags.keys()), \
|
||
|
"The section extraction algorithm lost some flags!"
|
||
|
return sections
|
||
|
|
||
|
|
||
|
def generate_config_file(sections, flags):
|
||
|
ret = wrap_text(config["header"]) + "\n\n\n"
|
||
|
for section, section_flags in sections:
|
||
|
ret += wrap_text(section.capitalize(), initial_indent="## ") + "\n\n"
|
||
|
for name in section_flags:
|
||
|
flag = flags[name]
|
||
|
helpstr = flag["meaning"] + " [" + flag["type"] + "]"
|
||
|
ret += wrap_text(helpstr) + "\n"
|
||
|
prefix = "# " if not flag["override"] else ""
|
||
|
ret += prefix + "--" + flag["name"].replace("_", "-") + \
|
||
|
"=" + flag["default"] + "\n\n"
|
||
|
ret += "\n"
|
||
|
ret += wrap_text(config["footer"])
|
||
|
return ret.strip() + "\n"
|
||
|
|
||
|
|
||
|
if __name__ == "__main__":
|
||
|
parser = argparse.ArgumentParser()
|
||
|
parser.add_argument("memgraph_binary",
|
||
|
help="path to Memgraph binary")
|
||
|
parser.add_argument("output_file",
|
||
|
help="path where to store the generated Memgraph "
|
||
|
"configuration file")
|
||
|
parser.add_argument("--config-file", default=CONFIG_FILE,
|
||
|
help="path to generator configuration file")
|
||
|
|
||
|
args = parser.parse_args()
|
||
|
flags = extract_flags(args.memgraph_binary)
|
||
|
|
||
|
with open(args.config_file) as f:
|
||
|
config = yaml.safe_load(f)
|
||
|
|
||
|
flags = apply_config_to_flags(config, flags)
|
||
|
sections = extract_sections(flags)
|
||
|
data = generate_config_file(sections, flags)
|
||
|
|
||
|
dirname = os.path.dirname(args.output_file)
|
||
|
if dirname and not os.path.exists(dirname):
|
||
|
os.makedirs(dirname)
|
||
|
|
||
|
with open(args.output_file, "w") as f:
|
||
|
f.write(data)
|