#!/usr/bin/env python3
# pylint: disable=too-many-locals,unused-argument,keyword-arg-before-vararg
# A script to generate enums in different languages from a CSV file.
# A CSV File contains
# enum1, 0
# enum2, ..
# ...
# enumn, n
# where 0,...,n are numbers.
#
# Written by Supakorn "Jamie" Rassameemasmuang <
[email protected]>
import argparse
import io
import os
import re
import sys
import time
from datetime import datetime, timezone
from typing import List, Tuple, Union
def cleanComment(s):
return re.sub(r" *#", " ", s)
def parse_args():
parser = argparse.ArgumentParser()
parser.add_argument("-language", "--language", type=str, required=True)
parser.add_argument("-o", "--output", type=str, required=True)
parser.add_argument("-i", "--input", type=str, required=True)
parser.add_argument("-name", "--name", type=str, required=True)
parser.add_argument("-xopt", "--xopt", type=str, nargs="*")
return parser.parse_args()
def create_enums(filename: str) -> List[Union[Tuple[str, int, str], Tuple[str, int]]]:
final_list = []
with io.open(filename, newline="", encoding="utf-8") as rawfile:
for line in rawfile.readlines():
if line.startswith("#") or line.strip() == "":
continue
raw_line = line.strip().split(",")
raw_str, raw_number = raw_line[0:2]
comment = None
if len(raw_line) >= 3:
comment = raw_line[-1]
final_list.append((raw_str.strip(), int(raw_number.strip()), comment))
else:
final_list.append((raw_str.strip(), int(raw_number.strip())))
return final_list
def datetime_now():
return datetime.fromtimestamp(
int(os.environ.get("SOURCE_DATE_EPOCH", time.time())), tz=timezone.utc
)
def generate_enum_cpp(outname, enums, name, comment=None, *args, **kwargs):
with io.open(outname, "w", encoding="utf-8") as fil:
fil.write(f"// Enum class for {name}\n")
if comment is not None:
fil.write(f"// {comment}\n")
if "namespace" in kwargs:
fil.write(f"namespace {kwargs['namespace']}\n")
fil.write("{\n")
fil.write(f"enum {name} : uint32_t\n")
fil.write("{\n")
for enumTxt, enumNum, *ar in enums:
if len(ar) > 0:
comment = cleanComment(ar[-1])
if comment is not None:
fil.write(f"// {comment.strip()}\n")
fil.write(f"{enumTxt}={enumNum},\n\n")
fil.write("};\n\n")
if "namespace" in kwargs:
fil.write(f"}} // namespace {kwargs['namespace']}\n")
fil.write("// End of File\n")
def generate_enum_java(outname, enums, name, comment=None, *args, **kwargs):
with io.open(outname, "w", encoding="utf-8") as fil:
fil.write(f"// Enum class for {name}\n")
if comment is not None:
fil.write(f"// {comment}\n")
if "package" in kwargs:
fil.write(f"package {kwargs['package']};\n")
fil.write("\n")
fil.write(f"public enum {name} {{\n")
spaces = kwargs.get("spaces", 4)
spaces_tab = " " * spaces
for i, enum in enumerate(enums):
enumTxt, enumNum, *ar = enum
endsep = "," if i < len(enums) - 1 else ";"
fil.write(f"{spaces_tab}{enumTxt}({enumNum}){endsep}\n")
if len(ar) > 0:
comment = cleanComment(ar[-1])
if comment is not None:
fil.write(f"// {comment.strip()}\n\n")
out_lines = [
"",
f"{name}(int value) {{",
f"{spaces_tab}this.value=value;",
"}",
"public String toString() {",
f"{spaces_tab}return Integer.toString(value);",
"}",
"private int value;",
]
for line in out_lines:
fil.write(spaces_tab)
fil.write(line)
fil.write("\n")
fil.write("};\n\n")
fil.write("// End of File\n")
def generate_enum_asy(outname, enums, name, comment=None, *args, **kwargs):
with io.open(outname, "w", encoding="utf-8") as fil:
fil.write(f"// Enum class for {name}\n")
if comment is not None:
fil.write(f"// {comment}\n")
fil.write(f"struct {name}\n")
fil.write("{\n")
for enumTxt, enumNum, *ar in enums:
fil.write(f" int {enumTxt}={enumNum};\n")
if len(ar) > 0:
comment = cleanComment(ar[-1])
if comment is not None:
fil.write(f"// {comment.strip()}\n\n")
fil.write("};\n\n")
fil.write(f"{name} {name};")
fil.write("// End of File\n")
def generate_enum_py(outname, enums, name, comment=None, *args, **kwargs):
with io.open(outname, "w", encoding="utf-8") as fil:
fil.write("#!/usr/bin/env python3\n")
fil.write(f"# Enum class for {name}\n")
if comment is not None:
fil.write(f'""" {comment} """\n')
fil.write(f"class {name}:\n")
for enumTxt, enumNum, *ar in enums:
fil.write(f" {name}_{enumTxt}={enumNum}\n")
if len(ar) > 0:
comment = cleanComment(ar[-1])
if comment is not None:
fil.write(f" # {comment.strip()}\n\n")
fil.write("# End of File\n")
def main():
arg = parse_args()
if arg.language in {"python", "py"}:
fn = generate_enum_py
elif arg.language in {"cxx", "c++", "cpp"}:
fn = generate_enum_cpp
elif arg.language in {"asy", "asymptote"}:
fn = generate_enum_asy
elif arg.language in {"java"}:
fn = generate_enum_java
else:
return 1
custom_args = {}
if arg.xopt is not None:
for xopt in arg.xopt:
key, val = xopt.split("=")
custom_args[key] = val
enums = create_enums(arg.input)
fn(arg.output, enums, arg.name, "AUTO-GENERATED from " + arg.input, **custom_args)
return 0
if __name__ == "__main__":
sys.exit(main() or 0)