2019-03-28 00:28:20 +08:00
|
|
|
import argparse
|
2019-04-07 17:50:39 +10:00
|
|
|
from pathlib import Path
|
2019-03-28 00:28:20 +08:00
|
|
|
import sys
|
2019-07-26 22:40:49 +02:00
|
|
|
from typing import Iterable
|
2019-03-28 00:28:20 +08:00
|
|
|
|
|
|
|
from ruamel.yaml import (
|
|
|
|
YAML,
|
|
|
|
)
|
|
|
|
|
2019-07-25 23:13:33 +02:00
|
|
|
from gen_base.gen_typing import TestProvider
|
2019-03-28 00:28:20 +08:00
|
|
|
|
|
|
|
|
|
|
|
def validate_output_dir(path_str):
|
2019-04-07 17:50:39 +10:00
|
|
|
path = Path(path_str)
|
2019-03-28 00:28:20 +08:00
|
|
|
|
|
|
|
if not path.exists():
|
|
|
|
raise argparse.ArgumentTypeError("Output directory must exist")
|
|
|
|
|
|
|
|
if not path.is_dir():
|
|
|
|
raise argparse.ArgumentTypeError("Output path must lead to a directory")
|
|
|
|
|
|
|
|
return path
|
|
|
|
|
|
|
|
|
2019-04-07 17:50:39 +10:00
|
|
|
def validate_configs_dir(path_str):
|
|
|
|
path = Path(path_str)
|
|
|
|
|
|
|
|
if not path.exists():
|
|
|
|
raise argparse.ArgumentTypeError("Configs directory must exist")
|
|
|
|
|
|
|
|
if not path.is_dir():
|
|
|
|
raise argparse.ArgumentTypeError("Config path must lead to a directory")
|
|
|
|
|
|
|
|
return path
|
|
|
|
|
|
|
|
|
2019-07-25 23:13:33 +02:00
|
|
|
def run_generator(generator_name, test_providers: Iterable[TestProvider]):
|
2019-03-28 00:28:20 +08:00
|
|
|
"""
|
|
|
|
Implementation for a general test generator.
|
|
|
|
:param generator_name: The name of the generator. (lowercase snake_case)
|
2019-07-25 23:13:33 +02:00
|
|
|
:param test_providers: A list of test provider,
|
|
|
|
each of these returns a callable that returns an iterable of test cases.
|
|
|
|
The call to get the iterable may set global configuration,
|
|
|
|
and the iterable should not be resumed after a pause with a change of that configuration.
|
2019-03-28 00:28:20 +08:00
|
|
|
:return:
|
|
|
|
"""
|
2019-07-26 19:19:36 +02:00
|
|
|
|
2019-03-28 00:28:20 +08:00
|
|
|
parser = argparse.ArgumentParser(
|
|
|
|
prog="gen-" + generator_name,
|
|
|
|
description=f"Generate YAML test suite files for {generator_name}",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"-o",
|
|
|
|
"--output-dir",
|
|
|
|
dest="output_dir",
|
|
|
|
required=True,
|
|
|
|
type=validate_output_dir,
|
|
|
|
help="directory into which the generated YAML files will be dumped"
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
"-f",
|
|
|
|
"--force",
|
|
|
|
action="store_true",
|
|
|
|
default=False,
|
2019-07-26 19:19:36 +02:00
|
|
|
help="if set re-generate and overwrite test files if they already exist",
|
2019-03-28 00:28:20 +08:00
|
|
|
)
|
2019-04-07 16:32:48 +10:00
|
|
|
parser.add_argument(
|
|
|
|
"-c",
|
|
|
|
"--configs-path",
|
|
|
|
dest="configs_path",
|
2019-04-07 17:50:39 +10:00
|
|
|
required=True,
|
|
|
|
type=validate_configs_dir,
|
2019-04-07 16:32:48 +10:00
|
|
|
help="specify the path of the configs directory (containing constants_presets and fork_timelines)",
|
|
|
|
)
|
2019-03-28 00:28:20 +08:00
|
|
|
|
|
|
|
args = parser.parse_args()
|
|
|
|
output_dir = args.output_dir
|
|
|
|
if not args.force:
|
|
|
|
file_mode = "x"
|
|
|
|
else:
|
|
|
|
file_mode = "w"
|
|
|
|
|
|
|
|
yaml = YAML(pure=True)
|
2019-04-07 23:36:05 +10:00
|
|
|
yaml.default_flow_style = None
|
2019-03-28 00:28:20 +08:00
|
|
|
|
2019-07-25 23:13:33 +02:00
|
|
|
print(f"Generating tests into {output_dir}...")
|
2019-04-07 16:32:48 +10:00
|
|
|
print(f"Reading config presets and fork timelines from {args.configs_path}")
|
2019-03-28 00:28:20 +08:00
|
|
|
|
2019-07-25 23:13:33 +02:00
|
|
|
for tprov in test_providers:
|
|
|
|
# loads configuration etc.
|
|
|
|
config_name = tprov.prepare(args.configs_path)
|
|
|
|
for test_case in tprov.make_cases():
|
2019-07-26 19:19:36 +02:00
|
|
|
case_dir = Path(output_dir) / Path(config_name) / Path(test_case.fork_name) \
|
|
|
|
/ Path(test_case.runner_name) / Path(test_case.handler_name) \
|
|
|
|
/ Path(test_case.suite_name) / Path(test_case.case_name)
|
|
|
|
|
|
|
|
if case_dir.exists():
|
|
|
|
if not args.force:
|
|
|
|
print(f'Skipping already existing test: {case_dir}')
|
|
|
|
continue
|
|
|
|
print(f'Warning, output directory {case_dir} already exist,'
|
|
|
|
f' old files are not deleted but will be overwritten when a new version is produced')
|
|
|
|
|
|
|
|
print(f'Generating test: {case_dir}')
|
|
|
|
try:
|
2019-07-25 23:13:33 +02:00
|
|
|
case_dir.mkdir(parents=True, exist_ok=True)
|
2019-07-26 19:19:36 +02:00
|
|
|
meta = dict()
|
|
|
|
for (name, out_kind, data) in test_case.case_fn():
|
|
|
|
if out_kind == "meta":
|
|
|
|
meta[name] = data
|
2019-07-27 13:34:19 +02:00
|
|
|
if out_kind == "data":
|
2019-07-26 19:19:36 +02:00
|
|
|
try:
|
|
|
|
out_path = case_dir / Path(name + '.yaml')
|
|
|
|
with out_path.open(file_mode) as f:
|
|
|
|
yaml.dump(data, f)
|
|
|
|
except IOError as e:
|
2019-07-27 13:34:19 +02:00
|
|
|
sys.exit(f'Error when dumping test "{case_dir}", part "{name}", kind "{out_kind}": {e}')
|
|
|
|
if out_kind == "ssz":
|
|
|
|
try:
|
|
|
|
out_path = case_dir / Path(name + '.ssz')
|
|
|
|
with out_path.open(file_mode + 'b') as f: # write in raw binary mode
|
|
|
|
f.write(data)
|
|
|
|
except IOError as e:
|
|
|
|
sys.exit(f'Error when dumping test "{case_dir}", part "{name}", kind "{out_kind}": {e}')
|
2019-07-26 19:19:36 +02:00
|
|
|
# Once all meta data is collected (if any), write it to a meta data file.
|
|
|
|
if len(meta) != 0:
|
|
|
|
try:
|
|
|
|
out_path = case_dir / Path('meta.yaml')
|
|
|
|
with out_path.open(file_mode) as f:
|
|
|
|
yaml.dump(meta, f)
|
|
|
|
except IOError as e:
|
|
|
|
sys.exit(f'Error when dumping test "{case_dir}" meta data": {e}')
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
print(f"ERROR: failed to generate vector(s) for test {case_dir}: {e}")
|
2019-07-25 23:13:33 +02:00
|
|
|
print(f"completed {generator_name}")
|