blob: 95664131682767a0e0748077c16a7bdee169bb51 [file] [log] [blame] [edit]
"""Helpers for writing tests"""
import contextlib
import os
import os.path
import re
import shutil
from typing import List, Callable, Iterator, Optional, Tuple
from mypy import build
from mypy.errors import CompileError
from mypy.options import Options
from mypy.test.data import DataSuite, DataDrivenTestCase
from mypy.test.config import test_temp_dir
from mypy.test.helpers import assert_string_arrays_equal
from mypyc import genops
from mypyc.options import CompilerOptions
from mypyc.ops import FuncIR
from mypyc.errors import Errors
from mypyc.test.config import test_data_prefix
# The builtins stub used during icode generation test cases.
ICODE_GEN_BUILTINS = os.path.join(test_data_prefix, 'fixtures/ir.py')
# The testutil support library
TESTUTIL_PATH = os.path.join(test_data_prefix, 'fixtures/testutil.py')
class MypycDataSuite(DataSuite):
# Need to list no files, since this will be picked up as a suite of tests
files = [] # type: List[str]
data_prefix = test_data_prefix
def builtins_wrapper(func: Callable[[DataDrivenTestCase], None],
path: str) -> Callable[[DataDrivenTestCase], None]:
"""Decorate a function that implements a data-driven test case to copy an
alternative builtins module implementation in place before performing the
test case. Clean up after executing the test case.
"""
return lambda testcase: perform_test(func, path, testcase)
@contextlib.contextmanager
def use_custom_builtins(builtins_path: str, testcase: DataDrivenTestCase) -> Iterator[None]:
for path, _ in testcase.files:
if os.path.basename(path) == 'builtins.pyi':
default_builtins = False
break
else:
# Use default builtins.
builtins = os.path.abspath(os.path.join(test_temp_dir, 'builtins.pyi'))
shutil.copyfile(builtins_path, builtins)
default_builtins = True
# Actually peform the test case.
yield None
if default_builtins:
# Clean up.
os.remove(builtins)
def perform_test(func: Callable[[DataDrivenTestCase], None],
builtins_path: str, testcase: DataDrivenTestCase) -> None:
for path, _ in testcase.files:
if os.path.basename(path) == 'builtins.py':
default_builtins = False
break
else:
# Use default builtins.
builtins = os.path.join(test_temp_dir, 'builtins.py')
shutil.copyfile(builtins_path, builtins)
default_builtins = True
# Actually peform the test case.
func(testcase)
if default_builtins:
# Clean up.
os.remove(builtins)
def build_ir_for_single_file(input_lines: List[str],
compiler_options: Optional[CompilerOptions] = None) -> List[FuncIR]:
program_text = '\n'.join(input_lines)
compiler_options = compiler_options or CompilerOptions()
options = Options()
options.show_traceback = True
options.use_builtins_fixtures = True
options.strict_optional = True
options.python_version = (3, 6)
options.export_types = True
options.preserve_asts = True
options.per_module_options['__main__'] = {'mypyc': True}
source = build.BuildSource('main', '__main__', program_text)
# Construct input as a single single.
# Parse and type check the input program.
result = build.build(sources=[source],
options=options,
alt_lib_path=test_temp_dir)
if result.errors:
raise CompileError(result.errors)
errors = Errors()
modules = genops.build_ir(
[result.files['__main__']], result.graph, result.types,
genops.Mapper({'__main__': None}),
compiler_options, errors)
assert errors.num_errors == 0
module = list(modules.values())[0]
return module.functions
def update_testcase_output(testcase: DataDrivenTestCase, output: List[str]) -> None:
# TODO: backport this to mypy
assert testcase.old_cwd is not None, "test was not properly set up"
testcase_path = os.path.join(testcase.old_cwd, testcase.file)
with open(testcase_path) as f:
data_lines = f.read().splitlines()
# We can't rely on the test line numbers to *find* the test, since
# we might fix multiple tests in a run. So find it by the case
# header. Give up if there are multiple tests with the same name.
test_slug = '[case {}]'.format(testcase.name)
if data_lines.count(test_slug) != 1:
return
start_idx = data_lines.index(test_slug)
stop_idx = start_idx + 11
while stop_idx < len(data_lines) and not data_lines[stop_idx].startswith('[case '):
stop_idx += 1
test = data_lines[start_idx:stop_idx]
out_start = test.index('[out]')
test[out_start + 1:] = output
data_lines[start_idx:stop_idx] = test + ['']
data = '\n'.join(data_lines)
with open(testcase_path, 'w') as f:
print(data, file=f)
def assert_test_output(testcase: DataDrivenTestCase, actual: List[str],
message: str,
expected: Optional[List[str]] = None) -> None:
expected_output = expected if expected is not None else testcase.output
if expected_output != actual and testcase.config.getoption('--update-data', False):
update_testcase_output(testcase, actual)
assert_string_arrays_equal(
expected_output, actual,
'{} ({}, line {})'.format(message, testcase.file, testcase.line))
def get_func_names(expected: List[str]) -> List[str]:
res = []
for s in expected:
m = re.match(r'def ([_a-zA-Z0-9.*$]+)\(', s)
if m:
res.append(m.group(1))
return res
def remove_comment_lines(a: List[str]) -> List[str]:
"""Return a copy of array with comments removed.
Lines starting with '--' (but not with '---') are removed.
"""
r = []
for s in a:
if s.strip().startswith('--') and not s.strip().startswith('---'):
pass
else:
r.append(s)
return r
def print_with_line_numbers(s: str) -> None:
lines = s.splitlines()
for i, line in enumerate(lines):
print('%-4d %s' % (i + 1, line))
def heading(text: str) -> None:
print('=' * 20 + ' ' + text + ' ' + '=' * 20)
def show_c(cfiles: List[List[Tuple[str, str]]]) -> None:
heading('Generated C')
for group in cfiles:
for cfile, ctext in group:
print('== {} =='.format(cfile))
print_with_line_numbers(ctext)
heading('End C')