-
Notifications
You must be signed in to change notification settings - Fork 370
/
conftest.py
97 lines (85 loc) Β· 3.65 KB
/
conftest.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
from typing import List, Tuple, Optional, Iterator
import pytest
import shutil
from pathlib import Path
from wasabi import msg
import srsly
import shutil
from spacy.tokens.underscore import Underscore
TESTS_DIR = "__tests__"
EXERCISES_DIR = "exercises"
META_FILE = "meta.json"
PYTEST_TEMPLATE = "pytestTemplate"
LANG_CLI_ARG = "lang"
GENERAL_TEST = "test_general.py"
@pytest.fixture(scope="module", autouse=True)
def clean_underscore():
# reset the Underscore object after the test, to avoid having state copied across tests
yield
Underscore.doc_extensions = {}
Underscore.span_extensions = {}
Underscore.token_extensions = {}
def format_test(name: str, template: str, test: str, solution: str) -> str:
full_code = template.replace("${solution}", solution).replace("${test}", test)
# Need to indent the lines to fit it into test function β can probably be less hacky
indented = "\n".join([" " + line for line in full_code.split("\n")])
return f"def test_{name}():\n{indented}"
def get_source_files(langs: List[str]) -> Iterator[Tuple[str, Path, Optional[Path]]]:
exercises_path = Path(EXERCISES_DIR)
if not exercises_path.exists():
msg.fail(f"Can't find exercises directory: {EXERCISES_DIR}", exits=1)
for lang_path in exercises_path.iterdir():
if lang_path.is_dir():
lang_name = lang_path.stem
if langs and lang_name not in langs:
continue
for py_file in lang_path.iterdir():
if py_file.name.startswith("test_"):
solution_name = f"solution_{py_file.name.split('test_')[1]}"
solution_file = lang_path / solution_name
if not solution_file.exists():
if py_file.name == GENERAL_TEST:
yield (lang_name, py_file, None)
else:
msg.warn(
f"Didn't find solution for test: {py_file.stem} ({lang_path})"
)
else:
yield (lang_name, py_file, solution_file)
def pytest_addoption(parser):
parser.addoption(f"--{LANG_CLI_ARG}", action="store", dest=LANG_CLI_ARG)
def pytest_sessionstart(session):
lang = session.config.getoption(LANG_CLI_ARG)
if lang:
lang = [lang_code.strip() for lang_code in lang.split(",")]
msg.info(f"Running only tests for {lang}")
test_dir = Path(TESTS_DIR)
if test_dir.exists():
shutil.rmtree(str(test_dir))
msg.info(f"Deleted existing test directory {TESTS_DIR}")
test_dir.mkdir()
msg.good(f"Created test directory {TESTS_DIR}")
meta = srsly.read_json(META_FILE)
n_files = 0
for test_lang, test_file, solution_file in get_source_files(lang):
test_root = test_dir / test_lang
if not test_root.exists():
test_root.mkdir()
init_path = test_root / "__init__.py"
init_path.touch()
if not solution_file: # general test file, just copy it over
shutil.copy(str(test_file), str(test_root / test_file.name))
n_files += 1
continue
with test_file.open("r", encoding="utf8") as f:
test_code = f.read()
with solution_file.open("r", encoding="utf8") as f:
solution_code = f.read()
full_code = format_test(
test_file.stem, meta[PYTEST_TEMPLATE], test_code, solution_code
)
test_path = test_root / test_file.name
with test_path.open("w", encoding="utf8") as f:
f.write(full_code)
n_files += 1
msg.good(f"Created {n_files} files for pytest in {TESTS_DIR}")