2014-04-02 15:31:39 +01:00
|
|
|
#! /usr/bin/env python3
|
2013-12-29 22:34:42 +00:00
|
|
|
|
2014-01-27 22:53:28 +01:00
|
|
|
import os
|
|
|
|
|
import subprocess
|
|
|
|
|
import sys
|
2014-09-17 22:56:34 +01:00
|
|
|
import platform
|
2014-05-03 16:43:27 +01:00
|
|
|
import argparse
|
2015-03-20 17:25:25 +00:00
|
|
|
import re
|
2014-01-27 22:53:28 +01:00
|
|
|
from glob import glob
|
2013-12-29 22:34:42 +00:00
|
|
|
|
2014-04-04 17:28:34 +03:00
|
|
|
# Tests require at least CPython 3.3. If your default python3 executable
|
|
|
|
|
# is of lower version, you can point MICROPY_CPYTHON3 environment var
|
|
|
|
|
# to the correct executable.
|
2014-01-27 22:53:28 +01:00
|
|
|
if os.name == 'nt':
|
2014-04-04 17:28:34 +03:00
|
|
|
CPYTHON3 = os.getenv('MICROPY_CPYTHON3', 'python3.exe')
|
2014-04-17 01:22:45 +01:00
|
|
|
MICROPYTHON = os.getenv('MICROPY_MICROPYTHON', '../windows/micropython.exe')
|
2014-01-27 22:53:28 +01:00
|
|
|
else:
|
2014-04-03 22:06:35 +03:00
|
|
|
CPYTHON3 = os.getenv('MICROPY_CPYTHON3', 'python3')
|
2014-04-17 01:22:45 +01:00
|
|
|
MICROPYTHON = os.getenv('MICROPY_MICROPYTHON', '../unix/micropython')
|
2013-12-29 22:34:42 +00:00
|
|
|
|
2016-09-20 12:19:35 +10:00
|
|
|
# mpy-cross is only needed if --via-mpy command-line arg is passed
|
|
|
|
|
MPYCROSS = os.getenv('MICROPY_MPYCROSS', '../mpy-cross/mpy-cross')
|
|
|
|
|
|
2014-11-19 10:44:31 -05:00
|
|
|
# Set PYTHONIOENCODING so that CPython will use utf-8 on systems which set another encoding in the locale
|
2014-12-12 00:51:39 +02:00
|
|
|
os.environ['PYTHONIOENCODING'] = 'utf-8'
|
2014-11-19 10:44:31 -05:00
|
|
|
|
2014-03-23 01:07:30 +02:00
|
|
|
def rm_f(fname):
|
|
|
|
|
if os.path.exists(fname):
|
|
|
|
|
os.remove(fname)
|
|
|
|
|
|
2016-07-26 21:29:48 +03:00
|
|
|
|
|
|
|
|
# unescape wanted regex chars and escape unwanted ones
|
|
|
|
|
def convert_regex_escapes(line):
|
|
|
|
|
cs = []
|
|
|
|
|
escape = False
|
|
|
|
|
for c in str(line, 'utf8'):
|
|
|
|
|
if escape:
|
|
|
|
|
escape = False
|
|
|
|
|
cs.append(c)
|
|
|
|
|
elif c == '\\':
|
|
|
|
|
escape = True
|
|
|
|
|
elif c in ('(', ')', '[', ']', '{', '}', '.', '*', '+', '^', '$'):
|
|
|
|
|
cs.append('\\' + c)
|
|
|
|
|
else:
|
|
|
|
|
cs.append(c)
|
|
|
|
|
# accept carriage-return(s) before final newline
|
|
|
|
|
if cs[-1] == '\n':
|
|
|
|
|
cs[-1] = '\r*\n'
|
|
|
|
|
return bytes(''.join(cs), 'utf8')
|
|
|
|
|
|
|
|
|
|
|
2017-04-14 17:07:13 +03:00
|
|
|
def run_micropython(pyb, args, test_file, is_special=False):
|
2017-01-22 11:49:08 +11:00
|
|
|
special_tests = ('micropython/meminfo.py', 'basics/bytes_compare3.py', 'basics/builtin_help.py', 'thread/thread_exc2.py')
|
2014-10-05 22:27:12 +01:00
|
|
|
if pyb is None:
|
|
|
|
|
# run on PC
|
2016-07-26 21:29:48 +03:00
|
|
|
if test_file.startswith(('cmdline/', 'feature_check/')) or test_file in special_tests:
|
2015-03-13 10:58:34 +00:00
|
|
|
# special handling for tests of the unix cmdline program
|
2016-07-26 21:29:48 +03:00
|
|
|
is_special = True
|
2015-03-13 10:58:34 +00:00
|
|
|
|
2017-04-14 17:07:13 +03:00
|
|
|
if is_special:
|
2015-03-13 10:58:34 +00:00
|
|
|
# check for any cmdline options needed for this test
|
|
|
|
|
args = [MICROPYTHON]
|
|
|
|
|
with open(test_file, 'rb') as f:
|
|
|
|
|
line = f.readline()
|
|
|
|
|
if line.startswith(b'# cmdline:'):
|
2015-08-06 12:34:48 +02:00
|
|
|
# subprocess.check_output on Windows only accepts strings, not bytes
|
|
|
|
|
args += [str(c, 'utf-8') for c in line[10:].strip().split()]
|
2015-03-13 10:58:34 +00:00
|
|
|
|
|
|
|
|
# run the test, possibly with redirected input
|
|
|
|
|
try:
|
2015-08-30 11:36:42 +03:00
|
|
|
if 'repl_' in test_file:
|
2015-07-23 12:05:30 -07:00
|
|
|
# Need to use a PTY to test command line editing
|
2015-08-06 12:34:48 +02:00
|
|
|
try:
|
|
|
|
|
import pty
|
|
|
|
|
except ImportError:
|
|
|
|
|
# in case pty module is not available, like on Windows
|
|
|
|
|
return b'SKIP\n'
|
2015-07-23 12:05:30 -07:00
|
|
|
import select
|
|
|
|
|
|
2015-11-24 19:52:03 +02:00
|
|
|
def get(required=False):
|
2015-07-23 12:05:30 -07:00
|
|
|
rv = b''
|
|
|
|
|
while True:
|
|
|
|
|
ready = select.select([master], [], [], 0.02)
|
|
|
|
|
if ready[0] == [master]:
|
|
|
|
|
rv += os.read(master, 1024)
|
|
|
|
|
else:
|
2015-11-24 19:52:03 +02:00
|
|
|
if not required or rv:
|
|
|
|
|
return rv
|
2015-07-23 12:05:30 -07:00
|
|
|
|
|
|
|
|
def send_get(what):
|
|
|
|
|
os.write(master, what)
|
|
|
|
|
return get()
|
|
|
|
|
|
|
|
|
|
with open(test_file, 'rb') as f:
|
|
|
|
|
# instead of: output_mupy = subprocess.check_output(args, stdin=f)
|
|
|
|
|
master, slave = pty.openpty()
|
|
|
|
|
p = subprocess.Popen(args, stdin=slave, stdout=slave,
|
|
|
|
|
stderr=subprocess.STDOUT, bufsize=0)
|
2015-11-24 19:52:03 +02:00
|
|
|
banner = get(True)
|
2015-07-23 12:05:30 -07:00
|
|
|
output_mupy = banner + b''.join(send_get(line) for line in f)
|
2016-12-20 15:13:49 +11:00
|
|
|
send_get(b'\x04') # exit the REPL, so coverage info is saved
|
2015-07-23 12:05:30 -07:00
|
|
|
p.kill()
|
|
|
|
|
os.close(master)
|
|
|
|
|
os.close(slave)
|
2015-03-13 10:58:34 +00:00
|
|
|
else:
|
|
|
|
|
output_mupy = subprocess.check_output(args + [test_file])
|
|
|
|
|
except subprocess.CalledProcessError:
|
2015-09-21 12:24:17 +02:00
|
|
|
return b'CRASH'
|
2015-03-13 10:58:34 +00:00
|
|
|
|
|
|
|
|
else:
|
2016-09-20 12:19:35 +10:00
|
|
|
# a standard test run on PC
|
|
|
|
|
|
|
|
|
|
# create system command
|
|
|
|
|
cmdlist = [MICROPYTHON, '-X', 'emit=' + args.emit]
|
|
|
|
|
if args.heapsize is not None:
|
|
|
|
|
cmdlist.extend(['-X', 'heapsize=' + args.heapsize])
|
|
|
|
|
|
|
|
|
|
# if running via .mpy, first compile the .py file
|
|
|
|
|
if args.via_mpy:
|
|
|
|
|
subprocess.check_output([MPYCROSS, '-mcache-lookup-bc', '-o', 'mpytest.mpy', test_file])
|
|
|
|
|
cmdlist.extend(['-m', 'mpytest'])
|
|
|
|
|
else:
|
2016-03-15 13:04:43 +00:00
|
|
|
cmdlist.append(test_file)
|
2016-09-20 12:19:35 +10:00
|
|
|
|
|
|
|
|
# run the actual test
|
|
|
|
|
try:
|
2016-03-15 13:04:43 +00:00
|
|
|
output_mupy = subprocess.check_output(cmdlist)
|
2015-03-13 10:58:34 +00:00
|
|
|
except subprocess.CalledProcessError:
|
|
|
|
|
output_mupy = b'CRASH'
|
2016-09-20 12:19:35 +10:00
|
|
|
|
|
|
|
|
# clean up if we had an intermediate .mpy file
|
|
|
|
|
if args.via_mpy:
|
|
|
|
|
rm_f('mpytest.mpy')
|
|
|
|
|
|
2014-10-05 22:27:12 +01:00
|
|
|
else:
|
|
|
|
|
# run on pyboard
|
|
|
|
|
import pyboard
|
|
|
|
|
pyb.enter_raw_repl()
|
|
|
|
|
try:
|
2016-03-07 12:01:21 +00:00
|
|
|
output_mupy = pyb.execfile(test_file)
|
2014-10-05 22:27:12 +01:00
|
|
|
except pyboard.PyboardError:
|
|
|
|
|
output_mupy = b'CRASH'
|
|
|
|
|
|
2016-03-07 12:01:21 +00:00
|
|
|
# canonical form for all ports/platforms is to use \n for end-of-line
|
|
|
|
|
output_mupy = output_mupy.replace(b'\r\n', b'\n')
|
|
|
|
|
|
2017-01-23 10:44:29 +01:00
|
|
|
# don't try to convert the output if we should skip this test
|
2017-04-16 17:24:15 +03:00
|
|
|
if output_mupy in (b'SKIP\n', b'CRASH'):
|
2017-01-23 10:44:29 +01:00
|
|
|
return output_mupy
|
|
|
|
|
|
2016-07-26 21:29:48 +03:00
|
|
|
if is_special or test_file in special_tests:
|
|
|
|
|
# convert parts of the output that are not stable across runs
|
|
|
|
|
with open(test_file + '.exp', 'rb') as f:
|
|
|
|
|
lines_exp = []
|
|
|
|
|
for line in f.readlines():
|
|
|
|
|
if line == b'########\n':
|
|
|
|
|
line = (line,)
|
|
|
|
|
else:
|
|
|
|
|
line = (line, re.compile(convert_regex_escapes(line)))
|
|
|
|
|
lines_exp.append(line)
|
|
|
|
|
lines_mupy = [line + b'\n' for line in output_mupy.split(b'\n')]
|
|
|
|
|
if output_mupy.endswith(b'\n'):
|
|
|
|
|
lines_mupy = lines_mupy[:-1] # remove erroneous last empty line
|
|
|
|
|
i_mupy = 0
|
|
|
|
|
for i in range(len(lines_exp)):
|
|
|
|
|
if lines_exp[i][0] == b'########\n':
|
|
|
|
|
# 8x #'s means match 0 or more whole lines
|
|
|
|
|
line_exp = lines_exp[i + 1]
|
|
|
|
|
skip = 0
|
|
|
|
|
while i_mupy + skip < len(lines_mupy) and not line_exp[1].match(lines_mupy[i_mupy + skip]):
|
|
|
|
|
skip += 1
|
|
|
|
|
if i_mupy + skip >= len(lines_mupy):
|
|
|
|
|
lines_mupy[i_mupy] = b'######## FAIL\n'
|
|
|
|
|
break
|
|
|
|
|
del lines_mupy[i_mupy:i_mupy + skip]
|
|
|
|
|
lines_mupy.insert(i_mupy, b'########\n')
|
|
|
|
|
i_mupy += 1
|
|
|
|
|
else:
|
|
|
|
|
# a regex
|
|
|
|
|
if lines_exp[i][1].match(lines_mupy[i_mupy]):
|
|
|
|
|
lines_mupy[i_mupy] = lines_exp[i][0]
|
|
|
|
|
else:
|
|
|
|
|
#print("don't match: %r %s" % (lines_exp[i][1], lines_mupy[i_mupy])) # DEBUG
|
|
|
|
|
pass
|
|
|
|
|
i_mupy += 1
|
|
|
|
|
if i_mupy >= len(lines_mupy):
|
|
|
|
|
break
|
|
|
|
|
output_mupy = b''.join(lines_mupy)
|
|
|
|
|
|
2014-10-05 22:27:12 +01:00
|
|
|
return output_mupy
|
|
|
|
|
|
2017-04-14 17:07:13 +03:00
|
|
|
|
|
|
|
|
def run_feature_check(pyb, args, base_path, test_file):
|
|
|
|
|
return run_micropython(pyb, args, base_path + "/feature_check/" + test_file, is_special=True)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def run_tests(pyb, tests, args, base_path="."):
|
2014-05-03 16:43:27 +01:00
|
|
|
test_count = 0
|
|
|
|
|
testcase_count = 0
|
|
|
|
|
passed_count = 0
|
|
|
|
|
failed_tests = []
|
2014-05-10 16:52:58 +03:00
|
|
|
skipped_tests = []
|
2013-12-29 22:34:42 +00:00
|
|
|
|
2014-06-06 07:45:55 +10:00
|
|
|
skip_tests = set()
|
2014-12-13 00:50:54 +02:00
|
|
|
skip_native = False
|
2017-03-03 20:14:02 +03:00
|
|
|
skip_int_big = False
|
2017-01-05 00:16:29 +03:00
|
|
|
skip_set_type = False
|
2017-02-14 22:03:25 +03:00
|
|
|
skip_async = False
|
2017-04-02 22:51:31 +03:00
|
|
|
skip_const = False
|
2013-12-29 22:34:42 +00:00
|
|
|
|
2014-10-05 22:27:12 +01:00
|
|
|
# Check if micropython.native is supported, and skip such tests if it's not
|
2017-04-14 17:07:13 +03:00
|
|
|
native = run_feature_check(pyb, args, base_path, 'native_check.py')
|
2014-10-05 22:27:12 +01:00
|
|
|
if native == b'CRASH':
|
2014-12-13 00:50:54 +02:00
|
|
|
skip_native = True
|
2014-10-05 22:27:12 +01:00
|
|
|
|
2017-03-03 20:14:02 +03:00
|
|
|
# Check if arbitrary-precision integers are supported, and skip such tests if it's not
|
2017-04-14 17:07:13 +03:00
|
|
|
native = run_feature_check(pyb, args, base_path, 'int_big.py')
|
2017-03-10 02:11:43 +01:00
|
|
|
if native != b'1000000000000000000000000000000000000000000000\n':
|
2017-03-03 20:14:02 +03:00
|
|
|
skip_int_big = True
|
|
|
|
|
|
2017-01-05 00:16:29 +03:00
|
|
|
# Check if set type (and set literals) is supported, and skip such tests if it's not
|
2017-04-14 17:07:13 +03:00
|
|
|
native = run_feature_check(pyb, args, base_path, 'set_check.py')
|
2017-01-05 00:16:29 +03:00
|
|
|
if native == b'CRASH':
|
|
|
|
|
skip_set_type = True
|
|
|
|
|
|
2017-02-14 22:03:25 +03:00
|
|
|
# Check if async/await keywords are supported, and skip such tests if it's not
|
2017-04-14 17:07:13 +03:00
|
|
|
native = run_feature_check(pyb, args, base_path, 'async_check.py')
|
2017-02-14 22:03:25 +03:00
|
|
|
if native == b'CRASH':
|
|
|
|
|
skip_async = True
|
|
|
|
|
|
2017-04-02 22:51:31 +03:00
|
|
|
# Check if const keyword (MicroPython extension) is supported, and skip such tests if it's not
|
2017-04-14 17:07:13 +03:00
|
|
|
native = run_feature_check(pyb, args, base_path, 'const.py')
|
2017-04-02 22:51:31 +03:00
|
|
|
if native == b'CRASH':
|
|
|
|
|
skip_const = True
|
|
|
|
|
|
2015-07-29 23:16:51 -07:00
|
|
|
# Check if emacs repl is supported, and skip such tests if it's not
|
2017-04-14 17:07:13 +03:00
|
|
|
t = run_feature_check(pyb, args, base_path, 'repl_emacs_check.py')
|
2015-07-29 23:16:51 -07:00
|
|
|
if not 'True' in str(t, 'ascii'):
|
|
|
|
|
skip_tests.add('cmdline/repl_emacs_keys.py')
|
|
|
|
|
|
2017-04-14 17:07:13 +03:00
|
|
|
upy_byteorder = run_feature_check(pyb, args, base_path, 'byteorder.py')
|
2017-06-26 13:47:00 +10:00
|
|
|
upy_float_precision = int(run_feature_check(pyb, args, base_path, 'float.py'))
|
2017-04-14 17:07:13 +03:00
|
|
|
has_complex = run_feature_check(pyb, args, base_path, 'complex.py') == b'complex\n'
|
|
|
|
|
has_coverage = run_feature_check(pyb, args, base_path, 'coverage.py') == b'coverage\n'
|
|
|
|
|
cpy_byteorder = subprocess.check_output([CPYTHON3, base_path + '/feature_check/byteorder.py'])
|
2015-08-30 00:28:59 +03:00
|
|
|
skip_endian = (upy_byteorder != cpy_byteorder)
|
|
|
|
|
|
2014-06-06 07:45:55 +10:00
|
|
|
# Some tests shouldn't be run under Travis CI
|
|
|
|
|
if os.getenv('TRAVIS') == 'true':
|
|
|
|
|
skip_tests.add('basics/memoryerror.py')
|
2016-08-27 23:41:16 +10:00
|
|
|
skip_tests.add('thread/thread_gc1.py') # has reliability issues
|
2016-08-17 13:48:09 +10:00
|
|
|
skip_tests.add('thread/thread_lock4.py') # has reliability issues
|
2016-08-25 14:06:40 +10:00
|
|
|
skip_tests.add('thread/stress_heap.py') # has reliability issues
|
2016-09-08 13:06:29 +10:00
|
|
|
skip_tests.add('thread/stress_recurse.py') # has reliability issues
|
2014-04-03 22:44:51 +01:00
|
|
|
|
2017-06-26 13:47:00 +10:00
|
|
|
if upy_float_precision == 0:
|
|
|
|
|
skip_tests.add('extmod/ujson_dumps_float.py')
|
|
|
|
|
skip_tests.add('extmod/ujson_loads_float.py')
|
|
|
|
|
skip_tests.add('misc/rge_sm.py')
|
|
|
|
|
if upy_float_precision < 32:
|
|
|
|
|
skip_tests.add('float/float2int_intbig.py') # requires fp32, there's float2int_fp30_intbig.py instead
|
|
|
|
|
skip_tests.add('float/string_format.py') # requires fp32, there's string_format_fp30.py instead
|
|
|
|
|
skip_tests.add('float/bytes_construct.py') # requires fp32
|
|
|
|
|
skip_tests.add('float/bytearray_construct.py') # requires fp32
|
|
|
|
|
if upy_float_precision < 64:
|
|
|
|
|
skip_tests.add('float/float_divmod.py') # tested by float/float_divmod_relaxed.py instead
|
|
|
|
|
skip_tests.add('float/float2int_doubleprec_intbig.py')
|
|
|
|
|
|
2015-12-06 14:56:37 +02:00
|
|
|
if not has_complex:
|
|
|
|
|
skip_tests.add('float/complex1.py')
|
2017-04-04 12:26:43 +10:00
|
|
|
skip_tests.add('float/complex1_intbig.py')
|
2015-12-06 14:56:37 +02:00
|
|
|
skip_tests.add('float/int_big_float.py')
|
|
|
|
|
skip_tests.add('float/true_value.py')
|
|
|
|
|
skip_tests.add('float/types.py')
|
|
|
|
|
|
2016-12-22 11:26:06 +11:00
|
|
|
if not has_coverage:
|
|
|
|
|
skip_tests.add('cmdline/cmd_parsetree.py')
|
|
|
|
|
|
2016-07-09 16:52:57 +01:00
|
|
|
# Some tests shouldn't be run on a PC
|
|
|
|
|
if pyb is None:
|
|
|
|
|
# unix build does not have the GIL so can't run thread mutation tests
|
|
|
|
|
for t in tests:
|
|
|
|
|
if t.startswith('thread/mutate_'):
|
|
|
|
|
skip_tests.add(t)
|
|
|
|
|
|
2014-09-13 18:43:09 +01:00
|
|
|
# Some tests shouldn't be run on pyboard
|
|
|
|
|
if pyb is not None:
|
2015-10-20 23:55:50 +01:00
|
|
|
skip_tests.add('basics/exception_chain.py') # warning is not printed
|
2015-04-04 22:05:30 +01:00
|
|
|
skip_tests.add('micropython/meminfo.py') # output is very different to PC output
|
2015-10-20 23:55:50 +01:00
|
|
|
skip_tests.add('extmod/machine_mem.py') # raw memory access not supported
|
2014-09-13 18:43:09 +01:00
|
|
|
|
2015-06-23 21:24:51 +02:00
|
|
|
if args.target == 'wipy':
|
|
|
|
|
skip_tests.add('misc/print_exception.py') # requires error reporting full
|
|
|
|
|
skip_tests.add('misc/recursion.py') # requires stack checking enabled
|
|
|
|
|
skip_tests.add('misc/recursive_data.py') # requires stack checking enabled
|
|
|
|
|
skip_tests.add('misc/recursive_iternext.py') # requires stack checking enabled
|
2016-02-21 22:30:35 +01:00
|
|
|
skip_tests.update({'extmod/uctypes_%s.py' % t for t in 'bytearray le native_le ptr_le ptr_native_le sizeof sizeof_native array_assign_le array_assign_native_le'.split()}) # requires uctypes
|
2015-06-23 21:24:51 +02:00
|
|
|
skip_tests.add('extmod/zlibd_decompress.py') # requires zlib
|
2015-09-27 18:47:35 +02:00
|
|
|
skip_tests.add('extmod/uheapq1.py') # uheapq not supported by WiPy
|
2016-02-21 22:30:35 +01:00
|
|
|
skip_tests.add('extmod/urandom_basic.py') # requires urandom
|
|
|
|
|
skip_tests.add('extmod/urandom_extra.py') # requires urandom
|
2016-02-13 18:28:02 +02:00
|
|
|
elif args.target == 'esp8266':
|
|
|
|
|
skip_tests.add('misc/rge_sm.py') # too large
|
2017-04-03 10:20:41 +03:00
|
|
|
elif args.target == 'minimal':
|
|
|
|
|
skip_tests.add('misc/rge_sm.py') # too large
|
|
|
|
|
skip_tests.add('micropython/opt_level.py') # don't assume line numbers are stored
|
2015-06-23 21:24:51 +02:00
|
|
|
|
2014-09-17 22:56:34 +01:00
|
|
|
# Some tests are known to fail on 64-bit machines
|
|
|
|
|
if pyb is None and platform.architecture()[0] == '64bit':
|
2014-09-23 12:09:26 +00:00
|
|
|
pass
|
2014-09-17 22:56:34 +01:00
|
|
|
|
2014-10-04 08:39:15 +02:00
|
|
|
# Some tests use unsupported features on Windows
|
|
|
|
|
if os.name == 'nt':
|
2015-08-06 12:31:37 +02:00
|
|
|
skip_tests.add('import/import_file.py') # works but CPython prints forward slashes
|
2014-10-04 08:39:15 +02:00
|
|
|
|
2014-08-29 19:47:10 +01:00
|
|
|
# Some tests are known to fail with native emitter
|
|
|
|
|
# Remove them from the below when they work
|
|
|
|
|
if args.emit == 'native':
|
2017-01-17 00:17:44 +11:00
|
|
|
skip_tests.update({'basics/%s.py' % t for t in 'gen_yield_from gen_yield_from_close gen_yield_from_ducktype gen_yield_from_exc gen_yield_from_iter gen_yield_from_send gen_yield_from_stopped gen_yield_from_throw gen_yield_from_throw2 gen_yield_from_throw3 generator1 generator2 generator_args generator_close generator_closure generator_exc generator_return generator_send'.split()}) # require yield
|
2015-04-06 22:38:53 +01:00
|
|
|
skip_tests.update({'basics/%s.py' % t for t in 'bytes_gen class_store_class globals_del string_join'.split()}) # require yield
|
2016-10-11 12:30:32 +11:00
|
|
|
skip_tests.update({'basics/async_%s.py' % t for t in 'def await await2 for for2 with with2'.split()}) # require yield
|
2015-04-06 22:38:53 +01:00
|
|
|
skip_tests.update({'basics/%s.py' % t for t in 'try_reraise try_reraise2'.split()}) # require raise_varargs
|
2016-04-07 08:53:24 +01:00
|
|
|
skip_tests.update({'basics/%s.py' % t for t in 'with_break with_continue with_return'.split()}) # require complete with support
|
2015-03-25 23:33:48 +00:00
|
|
|
skip_tests.add('basics/array_construct2.py') # requires generators
|
|
|
|
|
skip_tests.add('basics/bool1.py') # seems to randomly fail
|
2017-07-07 11:47:38 +10:00
|
|
|
skip_tests.add('basics/builtin_hash_gen.py') # requires yield
|
2015-12-26 13:06:50 +00:00
|
|
|
skip_tests.add('basics/class_bind_self.py') # requires yield
|
2015-04-06 22:38:53 +01:00
|
|
|
skip_tests.add('basics/del_deref.py') # requires checking for unbound local
|
|
|
|
|
skip_tests.add('basics/del_local.py') # requires checking for unbound local
|
2015-09-01 11:53:27 +03:00
|
|
|
skip_tests.add('basics/exception_chain.py') # raise from is not supported
|
2015-12-08 21:39:21 +00:00
|
|
|
skip_tests.add('basics/for_range.py') # requires yield_value
|
2015-04-06 22:38:53 +01:00
|
|
|
skip_tests.add('basics/try_finally_loops.py') # requires proper try finally code
|
|
|
|
|
skip_tests.add('basics/try_finally_return.py') # requires proper try finally code
|
2015-12-24 12:49:31 +00:00
|
|
|
skip_tests.add('basics/try_finally_return2.py') # requires proper try finally code
|
2015-04-06 22:38:53 +01:00
|
|
|
skip_tests.add('basics/unboundlocal.py') # requires checking for unbound local
|
|
|
|
|
skip_tests.add('import/gen_context.py') # requires yield_value
|
|
|
|
|
skip_tests.add('misc/features.py') # requires raise_varargs
|
|
|
|
|
skip_tests.add('misc/rge_sm.py') # requires yield
|
2014-12-11 17:34:55 +00:00
|
|
|
skip_tests.add('misc/print_exception.py') # because native doesn't have proper traceback info
|
2015-04-25 03:17:41 +03:00
|
|
|
skip_tests.add('misc/sys_exc_info.py') # sys.exc_info() is not supported for native
|
2016-11-21 17:39:23 +11:00
|
|
|
skip_tests.add('micropython/heapalloc_traceback.py') # because native doesn't have proper traceback info
|
2017-02-10 16:09:20 +11:00
|
|
|
skip_tests.add('micropython/heapalloc_iter.py') # requires generators
|
2017-03-16 18:05:00 +11:00
|
|
|
skip_tests.add('micropython/schedule.py') # native code doesn't check pending events
|
2014-08-29 19:47:10 +01:00
|
|
|
|
2014-05-03 16:43:27 +01:00
|
|
|
for test_file in tests:
|
2015-08-06 12:31:37 +02:00
|
|
|
test_file = test_file.replace('\\', '/')
|
2014-06-06 07:41:30 +10:00
|
|
|
test_basename = os.path.basename(test_file)
|
|
|
|
|
test_name = os.path.splitext(test_basename)[0]
|
2014-12-13 00:50:54 +02:00
|
|
|
is_native = test_name.startswith("native_") or test_name.startswith("viper_")
|
2015-08-30 00:28:59 +03:00
|
|
|
is_endian = test_name.endswith("_endian")
|
2017-03-04 00:13:27 +03:00
|
|
|
is_int_big = test_name.startswith("int_big") or test_name.endswith("_intbig")
|
2017-01-31 00:25:09 +03:00
|
|
|
is_set_type = test_name.startswith("set_") or test_name.startswith("frozenset")
|
2017-02-14 22:03:25 +03:00
|
|
|
is_async = test_name.startswith("async_")
|
2017-04-02 22:51:31 +03:00
|
|
|
is_const = test_name.startswith("const")
|
2014-06-06 07:41:30 +10:00
|
|
|
|
2017-01-05 00:16:29 +03:00
|
|
|
skip_it = test_file in skip_tests
|
|
|
|
|
skip_it |= skip_native and is_native
|
|
|
|
|
skip_it |= skip_endian and is_endian
|
2017-03-03 20:14:02 +03:00
|
|
|
skip_it |= skip_int_big and is_int_big
|
2017-01-05 00:16:29 +03:00
|
|
|
skip_it |= skip_set_type and is_set_type
|
2017-02-14 22:03:25 +03:00
|
|
|
skip_it |= skip_async and is_async
|
2017-04-02 22:51:31 +03:00
|
|
|
skip_it |= skip_const and is_const
|
2017-01-05 00:16:29 +03:00
|
|
|
|
|
|
|
|
if skip_it:
|
2014-05-03 16:43:27 +01:00
|
|
|
print("skip ", test_file)
|
2014-05-10 16:52:58 +03:00
|
|
|
skipped_tests.append(test_name)
|
2014-05-03 16:43:27 +01:00
|
|
|
continue
|
2014-04-16 03:28:40 +01:00
|
|
|
|
2014-05-03 16:43:27 +01:00
|
|
|
# get expected output
|
|
|
|
|
test_file_expected = test_file + '.exp'
|
|
|
|
|
if os.path.isfile(test_file_expected):
|
|
|
|
|
# expected output given by a file, so read that in
|
|
|
|
|
with open(test_file_expected, 'rb') as f:
|
|
|
|
|
output_expected = f.read()
|
|
|
|
|
else:
|
2014-05-11 12:45:02 +02:00
|
|
|
# run CPython to work out expected output
|
2014-05-03 16:43:27 +01:00
|
|
|
try:
|
|
|
|
|
output_expected = subprocess.check_output([CPYTHON3, '-B', test_file])
|
2014-07-12 16:34:51 +03:00
|
|
|
if args.write_exp:
|
|
|
|
|
with open(test_file_expected, 'wb') as f:
|
|
|
|
|
f.write(output_expected)
|
2014-05-03 16:43:27 +01:00
|
|
|
except subprocess.CalledProcessError:
|
|
|
|
|
output_expected = b'CPYTHON3 CRASH'
|
2013-12-29 22:34:42 +00:00
|
|
|
|
2016-03-07 12:01:21 +00:00
|
|
|
# canonical form for all host platforms is to use \n for end-of-line
|
|
|
|
|
output_expected = output_expected.replace(b'\r\n', b'\n')
|
|
|
|
|
|
2014-07-12 16:34:51 +03:00
|
|
|
if args.write_exp:
|
|
|
|
|
continue
|
|
|
|
|
|
2014-05-03 16:43:27 +01:00
|
|
|
# run Micro Python
|
2014-10-05 22:27:12 +01:00
|
|
|
output_mupy = run_micropython(pyb, args, test_file)
|
2014-04-03 23:51:16 +01:00
|
|
|
|
2016-03-07 12:01:21 +00:00
|
|
|
if output_mupy == b'SKIP\n':
|
2014-05-10 16:52:58 +03:00
|
|
|
print("skip ", test_file)
|
|
|
|
|
skipped_tests.append(test_name)
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
testcase_count += len(output_expected.splitlines())
|
|
|
|
|
|
2014-05-03 16:43:27 +01:00
|
|
|
filename_expected = test_basename + ".exp"
|
|
|
|
|
filename_mupy = test_basename + ".out"
|
|
|
|
|
|
|
|
|
|
if output_expected == output_mupy:
|
|
|
|
|
print("pass ", test_file)
|
|
|
|
|
passed_count += 1
|
|
|
|
|
rm_f(filename_expected)
|
|
|
|
|
rm_f(filename_mupy)
|
|
|
|
|
else:
|
2014-06-28 10:29:12 +01:00
|
|
|
with open(filename_expected, "wb") as f:
|
|
|
|
|
f.write(output_expected)
|
|
|
|
|
with open(filename_mupy, "wb") as f:
|
|
|
|
|
f.write(output_mupy)
|
2014-05-03 16:43:27 +01:00
|
|
|
print("FAIL ", test_file)
|
|
|
|
|
failed_tests.append(test_name)
|
|
|
|
|
|
|
|
|
|
test_count += 1
|
|
|
|
|
|
|
|
|
|
print("{} tests performed ({} individual testcases)".format(test_count, testcase_count))
|
|
|
|
|
print("{} tests passed".format(passed_count))
|
|
|
|
|
|
2014-05-10 16:52:58 +03:00
|
|
|
if len(skipped_tests) > 0:
|
|
|
|
|
print("{} tests skipped: {}".format(len(skipped_tests), ' '.join(skipped_tests)))
|
2014-05-03 16:43:27 +01:00
|
|
|
if len(failed_tests) > 0:
|
|
|
|
|
print("{} tests failed: {}".format(len(failed_tests), ' '.join(failed_tests)))
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
# all tests succeeded
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
def main():
|
2015-10-12 00:06:25 +01:00
|
|
|
cmd_parser = argparse.ArgumentParser(description='Run tests for MicroPython.')
|
2015-06-23 21:24:51 +02:00
|
|
|
cmd_parser.add_argument('--target', default='unix', help='the target platform')
|
2015-06-28 14:01:27 +02:00
|
|
|
cmd_parser.add_argument('--device', default='/dev/ttyACM0', help='the serial device or the IP address of the pyboard')
|
|
|
|
|
cmd_parser.add_argument('-b', '--baudrate', default=115200, help='the baud rate of the serial device')
|
|
|
|
|
cmd_parser.add_argument('-u', '--user', default='micro', help='the telnet login username')
|
|
|
|
|
cmd_parser.add_argument('-p', '--password', default='python', help='the telnet login password')
|
2014-05-31 18:11:01 +01:00
|
|
|
cmd_parser.add_argument('-d', '--test-dirs', nargs='*', help='input test directories (if no files given)')
|
2014-07-12 16:34:51 +03:00
|
|
|
cmd_parser.add_argument('--write-exp', action='store_true', help='save .exp files to run tests w/o CPython')
|
2015-10-12 00:06:25 +01:00
|
|
|
cmd_parser.add_argument('--emit', default='bytecode', help='MicroPython emitter to use (bytecode or native)')
|
2016-03-15 13:04:43 +00:00
|
|
|
cmd_parser.add_argument('--heapsize', help='heapsize to use (use default if not specified)')
|
2016-09-20 12:19:35 +10:00
|
|
|
cmd_parser.add_argument('--via-mpy', action='store_true', help='compile .py files to .mpy first')
|
2015-03-24 11:55:26 +01:00
|
|
|
cmd_parser.add_argument('--keep-path', action='store_true', help='do not clear MICROPYPATH when running tests')
|
2014-05-03 16:43:27 +01:00
|
|
|
cmd_parser.add_argument('files', nargs='*', help='input test files')
|
|
|
|
|
args = cmd_parser.parse_args()
|
|
|
|
|
|
2017-04-03 10:20:41 +03:00
|
|
|
EXTERNAL_TARGETS = ('pyboard', 'wipy', 'esp8266', 'minimal')
|
2016-05-07 18:45:16 +03:00
|
|
|
if args.target in EXTERNAL_TARGETS:
|
2014-05-03 16:43:27 +01:00
|
|
|
import pyboard
|
2015-06-28 14:01:27 +02:00
|
|
|
pyb = pyboard.Pyboard(args.device, args.baudrate, args.user, args.password)
|
2014-04-13 13:48:33 +01:00
|
|
|
pyb.enter_raw_repl()
|
2015-06-23 21:24:51 +02:00
|
|
|
elif args.target == 'unix':
|
2014-05-03 16:43:27 +01:00
|
|
|
pyb = None
|
2015-06-23 21:24:51 +02:00
|
|
|
else:
|
2016-05-07 18:45:16 +03:00
|
|
|
raise ValueError('target must be either %s or unix' % ", ".join(EXTERNAL_TARGETS))
|
2013-12-29 22:34:42 +00:00
|
|
|
|
2014-05-03 16:43:27 +01:00
|
|
|
if len(args.files) == 0:
|
2014-05-28 14:27:54 +02:00
|
|
|
if args.test_dirs is None:
|
2015-06-23 21:24:51 +02:00
|
|
|
if args.target == 'pyboard':
|
2014-05-28 14:27:54 +02:00
|
|
|
# run pyboard tests
|
2016-03-15 13:20:18 +00:00
|
|
|
test_dirs = ('basics', 'micropython', 'float', 'misc', 'stress', 'extmod', 'pyb', 'pybnative', 'inlineasm')
|
2017-04-03 10:20:41 +03:00
|
|
|
elif args.target in ('esp8266', 'minimal'):
|
2016-02-13 17:01:27 +02:00
|
|
|
test_dirs = ('basics', 'micropython', 'float', 'misc', 'extmod')
|
2015-06-23 21:24:51 +02:00
|
|
|
elif args.target == 'wipy':
|
|
|
|
|
# run WiPy tests
|
2015-09-03 11:25:44 +02:00
|
|
|
test_dirs = ('basics', 'micropython', 'misc', 'extmod', 'wipy')
|
2015-06-23 21:24:51 +02:00
|
|
|
else:
|
|
|
|
|
# run PC tests
|
2016-03-15 13:20:18 +00:00
|
|
|
test_dirs = ('basics', 'micropython', 'float', 'import', 'io', 'misc', 'stress', 'unicode', 'extmod', 'unix', 'cmdline')
|
2014-05-03 16:43:27 +01:00
|
|
|
else:
|
2014-05-28 14:27:54 +02:00
|
|
|
# run tests from these directories
|
|
|
|
|
test_dirs = args.test_dirs
|
2014-05-03 16:43:27 +01:00
|
|
|
tests = sorted(test_file for test_files in (glob('{}/*.py'.format(dir)) for dir in test_dirs) for test_file in test_files)
|
2014-01-27 22:53:28 +01:00
|
|
|
else:
|
2014-05-03 16:43:27 +01:00
|
|
|
# tests explicitly given
|
|
|
|
|
tests = args.files
|
2014-01-27 22:53:28 +01:00
|
|
|
|
2015-03-24 11:55:26 +01:00
|
|
|
if not args.keep_path:
|
|
|
|
|
# clear search path to make sure tests use only builtin modules
|
|
|
|
|
os.environ['MICROPYPATH'] = ''
|
|
|
|
|
|
2017-04-14 17:07:13 +03:00
|
|
|
# Even if we run completely different tests in a different directory,
|
|
|
|
|
# we need to access feature_check's from the same directory as the
|
|
|
|
|
# run-tests script itself.
|
|
|
|
|
base_path = os.path.dirname(sys.argv[0]) or "."
|
|
|
|
|
res = run_tests(pyb, tests, args, base_path)
|
2017-04-02 20:47:44 +03:00
|
|
|
if pyb:
|
|
|
|
|
pyb.close()
|
|
|
|
|
if not res:
|
2014-05-03 16:43:27 +01:00
|
|
|
sys.exit(1)
|
2014-01-27 22:53:28 +01:00
|
|
|
|
2014-05-03 16:43:27 +01:00
|
|
|
if __name__ == "__main__":
|
|
|
|
|
main()
|