init
This commit is contained in:
commit
38355d2442
9083 changed files with 1225834 additions and 0 deletions
266
.venv/lib/python3.8/site-packages/mypyc/test/testutil.py
Normal file
266
.venv/lib/python3.8/site-packages/mypyc/test/testutil.py
Normal file
|
|
@ -0,0 +1,266 @@
|
|||
"""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.options import CompilerOptions
|
||||
from mypyc.analysis.ircheck import assert_func_ir_valid
|
||||
from mypyc.ir.func_ir import FuncIR
|
||||
from mypyc.errors import Errors
|
||||
from mypyc.irbuild.main import build_ir
|
||||
from mypyc.irbuild.mapper import Mapper
|
||||
from mypyc.test.config import test_data_prefix
|
||||
from mypyc.common import IS_32_BIT_PLATFORM, PLATFORM_SIZE
|
||||
|
||||
# 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: 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 perform the test case.
|
||||
try:
|
||||
yield None
|
||||
finally:
|
||||
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 perform 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)
|
||||
|
||||
# By default generate IR compatible with the earliest supported Python C API.
|
||||
# If a test needs more recent API features, this should be overridden.
|
||||
compiler_options = compiler_options or CompilerOptions(capi_version=(3, 5))
|
||||
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 = build_ir(
|
||||
[result.files['__main__']], result.graph, result.types,
|
||||
Mapper({'__main__': None}),
|
||||
compiler_options, errors)
|
||||
if errors.num_errors:
|
||||
raise CompileError(errors.new_messages())
|
||||
|
||||
module = list(modules.values())[0]
|
||||
for fn in module.functions:
|
||||
assert_func_ir_valid(fn)
|
||||
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,
|
||||
formatted: Optional[List[str]] = None) -> None:
|
||||
__tracebackhide__ = True
|
||||
|
||||
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')
|
||||
|
||||
|
||||
def fudge_dir_mtimes(dir: str, delta: int) -> None:
|
||||
for dirpath, _, filenames in os.walk(dir):
|
||||
for name in filenames:
|
||||
path = os.path.join(dirpath, name)
|
||||
new_mtime = os.stat(path).st_mtime + delta
|
||||
os.utime(path, times=(new_mtime, new_mtime))
|
||||
|
||||
|
||||
def replace_word_size(text: List[str]) -> List[str]:
|
||||
"""Replace WORDSIZE with platform specific word sizes"""
|
||||
result = []
|
||||
for line in text:
|
||||
index = line.find('WORD_SIZE')
|
||||
if index != -1:
|
||||
# get 'WORDSIZE*n' token
|
||||
word_size_token = line[index:].split()[0]
|
||||
n = int(word_size_token[10:])
|
||||
replace_str = str(PLATFORM_SIZE * n)
|
||||
result.append(line.replace(word_size_token, replace_str))
|
||||
else:
|
||||
result.append(line)
|
||||
return result
|
||||
|
||||
|
||||
def infer_ir_build_options_from_test_name(name: str) -> Optional[CompilerOptions]:
|
||||
"""Look for magic substrings in test case name to set compiler options.
|
||||
|
||||
Return None if the test case should be skipped (always pass).
|
||||
|
||||
Supported naming conventions:
|
||||
|
||||
*_64bit*:
|
||||
Run test case only on 64-bit platforms
|
||||
*_32bit*:
|
||||
Run test caseonly on 32-bit platforms
|
||||
*_python3_8* (or for any Python version):
|
||||
Use Python 3.8+ C API features (default: lowest supported version)
|
||||
*StripAssert*:
|
||||
Don't generate code for assert statements
|
||||
"""
|
||||
# If this is specific to some bit width, always pass if platform doesn't match.
|
||||
if '_64bit' in name and IS_32_BIT_PLATFORM:
|
||||
return None
|
||||
if '_32bit' in name and not IS_32_BIT_PLATFORM:
|
||||
return None
|
||||
options = CompilerOptions(strip_asserts='StripAssert' in name,
|
||||
capi_version=(3, 5))
|
||||
# A suffix like _python3.8 is used to set the target C API version.
|
||||
m = re.search(r'_python([3-9]+)_([0-9]+)(_|\b)', name)
|
||||
if m:
|
||||
options.capi_version = (int(m.group(1)), int(m.group(2)))
|
||||
elif '_py' in name or '_Python' in name:
|
||||
assert False, 'Invalid _py* suffix (should be _pythonX_Y): {}'.format(name)
|
||||
return options
|
||||
Loading…
Add table
Add a link
Reference in a new issue