Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
cython
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Xavier Thompson
cython
Commits
e51caa48
Commit
e51caa48
authored
Aug 28, 2018
by
scoder
Committed by
GitHub
Aug 28, 2018
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #2581 from cython/test_stats
Print run time stats in the test runner
parents
db64e492
f59a3fa7
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
193 additions
and
131 deletions
+193
-131
.coveragerc
.coveragerc
+7
-0
.travis.yml
.travis.yml
+45
-50
runtests.py
runtests.py
+141
-81
No files found.
.coveragerc
0 → 100644
View file @
e51caa48
[run]
branch = True
parallel = True
concurrency = multiprocessing,thread
include = Cython/*
source = Cython
omit = Test*
.travis.yml
View file @
e51caa48
os
:
linux
os
:
linux
dist
:
trusty
dist
:
trusty
language
:
python
# 'sudo' is enabled automatically by the 'apt' addon below.
# 'sudo' is enabled automatically by the 'apt' addon below.
#sudo: false
#sudo: false
...
@@ -18,77 +19,75 @@ cache:
...
@@ -18,77 +19,75 @@ cache:
directories
:
directories
:
-
$HOME/.ccache
-
$HOME/.ccache
language
:
python
python
:
-
2.7
-
3.6
-
2.6
-
3.4
-
3.5
-
pypy
-
pypy3
env
:
env
:
global
:
global
:
-
USE_CCACHE=1
-
USE_CCACHE=1
-
CCACHE_SLOPPINESS=pch_defines,time_macros
-
CCACHE_SLOPPINESS=pch_defines,time_macros
-
CCACHE_COMPRESS=1
-
CCACHE_COMPRESS=1
-
CCACHE_MAXSIZE=
1
50M
-
CCACHE_MAXSIZE=
2
50M
-
PATH="/usr/lib/ccache:$HOME/miniconda/bin:$PATH"
-
PATH="/usr/lib/ccache:$HOME/miniconda/bin:$PATH"
matrix
:
-
BACKEND=c,cpp
-
BACKEND=c
-
BACKEND=cpp
matrix
:
matrix
:
include
:
include
:
-
python
:
2.7
env
:
BACKEND=c
-
python
:
2.7
env
:
BACKEND=cpp
-
python
:
3.7
-
python
:
3.7
dist
:
xenial
# Required for Python 3.7
dist
:
xenial
# Required for Python 3.7
sudo
:
required
# travis-ci/travis-ci#9069
sudo
:
required
# travis-ci/travis-ci#9069
env
:
TEST_CODE_STYLE=1
env
:
BACKEND=c
-
python
:
3.7
-
python
:
3.7
dist
:
xenial
# Required for Python 3.7
dist
:
xenial
# Required for Python 3.7
sudo
:
required
# travis-ci/travis-ci#9069
sudo
:
required
# travis-ci/travis-ci#9069
env
:
BACKEND=cpp
-
python
:
2.6
env
:
BACKEND=c
env
:
BACKEND=c
-
python
:
2.6
env
:
BACKEND=cpp
# Disabled: coverage analysis takes excessively long, several times longer than without.
# - python: 3.7
# dist: xenial # Required for Python 3.7
# sudo: required # travis-ci/travis-ci#9069
# env: COVERAGE=1
-
python
:
3.7
-
python
:
3.7
dist
:
xenial
# Required for Python 3.7
dist
:
xenial
# Required for Python 3.7
sudo
:
required
# travis-ci/travis-ci#9069
sudo
:
required
# travis-ci/travis-ci#9069
env
:
TEST_CODE_STYLE=1
-
python
:
3.4
env
:
BACKEND=c
-
python
:
3.4
env
:
BACKEND=cpp
env
:
BACKEND=cpp
-
python
:
3.8-dev
-
python
:
3.5
dist
:
xenial
# Required for Python 3.7
sudo
:
required
# travis-ci/travis-ci#9069
env
:
BACKEND=c
env
:
BACKEND=c
-
python
:
3.5
env
:
BACKEND=cpp
-
python
:
3.6
env
:
BACKEND=c
-
python
:
3.6
env
:
BACKEND=cpp
-
python
:
3.8-dev
-
python
:
3.8-dev
dist
:
xenial
# Required for Python 3.7
dist
:
xenial
# Required for Python 3.7
sudo
:
required
# travis-ci/travis-ci#9069
sudo
:
required
# travis-ci/travis-ci#9069
env
:
BACKEND=cpp
-
os
:
osx
-
os
:
osx
osx_image
:
xcode6.4
osx_image
:
xcode6.4
env
:
BACKEND=c
PY=2
env
:
PY=2
python
:
2
python
:
2
language
:
c
language
:
c
compiler
:
clang
compiler
:
clang
cache
:
false
cache
:
false
-
os
:
osx
-
os
:
osx
osx_image
:
xcode6.4
osx_image
:
xcode6.4
env
:
BACKEND=cpp PY=2
env
:
PY=3
python
:
2
language
:
cpp
compiler
:
clang
cache
:
false
-
os
:
osx
osx_image
:
xcode6.4
env
:
BACKEND=c PY=3
python
:
3
python
:
3
language
:
c
language
:
c
compiler
:
clang
compiler
:
clang
cache
:
false
cache
:
false
-
os
:
osx
-
python
:
pypy
osx_image
:
xcode6.4
env
:
BACKEND=c
env
:
BACKEND=cpp PY=3
-
python
:
pypy3
python
:
3
env
:
BACKEND=c
language
:
cpp
compiler
:
clang
cache
:
false
-
env
:
STACKLESS=true BACKEND=c PY=2
-
env
:
STACKLESS=true BACKEND=c PY=2
python
:
2.7
python
:
2.7
-
env
:
STACKLESS=true BACKEND=c PY=3
-
env
:
STACKLESS=true BACKEND=c PY=3
...
@@ -99,11 +98,6 @@ matrix:
...
@@ -99,11 +98,6 @@ matrix:
-
python
:
3.8-dev
-
python
:
3.8-dev
-
env
:
STACKLESS=true BACKEND=c PY=2
-
env
:
STACKLESS=true BACKEND=c PY=2
-
env
:
STACKLESS=true BACKEND=c PY=3
-
env
:
STACKLESS=true BACKEND=c PY=3
exclude
:
-
python
:
pypy
env
:
BACKEND=cpp
-
python
:
pypy3
env
:
BACKEND=cpp
branches
:
branches
:
only
:
only
:
...
@@ -115,11 +109,11 @@ before_install:
...
@@ -115,11 +109,11 @@ before_install:
if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then
if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then
# adding apt repos in travis is really fragile => retry a couple of times.
# adding apt repos in travis is really fragile => retry a couple of times.
for i in {1..10}; do travis_retry sudo apt-add-repository --yes 'ppa:ubuntu-toolchain-r/test' && break; sleep 2; done
for i in {1..10}; do travis_retry sudo apt-add-repository --yes 'ppa:ubuntu-toolchain-r/test' && break; sleep 2; done
for i in {1..10}; do travis_retry sudo apt-get update && travis_retry sudo apt-get install --yes gcc-8 $(if [
"$BACKEND" = cpp
]; then echo -n "g++-8"; fi ) && break; sleep 2; done
for i in {1..10}; do travis_retry sudo apt-get update && travis_retry sudo apt-get install --yes gcc-8 $(if [
-z "${BACKEND##*cpp*}"
]; then echo -n "g++-8"; fi ) && break; sleep 2; done
sudo update-alternatives --install /usr/bin/gcc gcc /usr/bin/gcc-8 60 $(if [
"$BACKEND" = cpp
]; then echo " --slave /usr/bin/g++ g++ /usr/bin/g++-8"; fi)
sudo update-alternatives --install /usr/bin/gcc gcc /usr/bin/gcc-8 60 $(if [
-z "${BACKEND##*cpp*}"
]; then echo " --slave /usr/bin/g++ g++ /usr/bin/g++-8"; fi)
sudo update-alternatives --set gcc /usr/bin/gcc-8
sudo update-alternatives --set gcc /usr/bin/gcc-8
export CC=gcc
export CC=gcc
if [
"$BACKEND" = cpp
]; then sudo update-alternatives --set g++ /usr/bin/g++-8; export CXX=g++; fi
if [
-z "${BACKEND##*cpp*}"
]; then sudo update-alternatives --set g++ /usr/bin/g++-8; export CXX=g++; fi
fi
fi
-
|
-
|
...
@@ -145,19 +139,20 @@ before_install:
...
@@ -145,19 +139,20 @@ before_install:
install
:
install
:
-
python -c 'import sys; print("Python %s" % (sys.version,))'
-
python -c 'import sys; print("Python %s" % (sys.version,))'
-
if [ -n "${TRAVIS_PYTHON_VERSION##*-dev}" -a -n "${TRAVIS_PYTHON_VERSION##2.6*}" ]; then pip install -r test-requirements.txt $( [ -z "${TRAVIS_PYTHON_VERSION##pypy*}" -o -z "${TRAVIS_PYTHON_VERSION##3.7*}" ] || echo " -r test-requirements-cpython.txt" ) ; fi
-
if [ -n "${TRAVIS_PYTHON_VERSION##*-dev}" -a -n "${TRAVIS_PYTHON_VERSION##2.6*}" ]; then pip install -r test-requirements.txt $( [ -z "${TRAVIS_PYTHON_VERSION##pypy*}" -o -z "${TRAVIS_PYTHON_VERSION##3.7*}" ] || echo " -r test-requirements-cpython.txt" ) ; fi
-
CFLAGS="-O2 -ggdb -Wall -Wextra $(python -c 'import sys; print("-fno-strict-aliasing" if sys.version_info[0] == 2 else "")')" python setup.py build
#
- CFLAGS="-O2 -ggdb -Wall -Wextra $(python -c 'import sys; print("-fno-strict-aliasing" if sys.version_info[0] == 2 else "")')" python setup.py build
before_script
:
ccache -s ||
true
before_script
:
ccache -s ||
true
script
:
script
:
-
PYTHON_DBG="python$( python -c 'import sys; print("%d.%d" % sys.version_info[:2])' )-dbg"
-
if [ "$TEST_CODE_STYLE" = "1" ]; then
-
if [ "$TEST_CODE_STYLE" = "1" ]; then
STYLE_ARGS="--no-unit --no-doctest --no-file --no-pyregr --no-examples";
STYLE_ARGS="--no-unit --no-doctest --no-file --no-pyregr --no-examples";
else
else
STYLE_ARGS=--no-code-style;
STYLE_ARGS=--no-code-style;
if $PYTHON_DBG -V >&2; then CFLAGS="-O0 -ggdb" $PYTHON_DBG runtests.py -vv --no-code-style Debugger --backends=$BACKEND; fi;
if [ -z "${BACKEND##*cpp*}" -a -n "${TRAVIS_PYTHON_VERSION##2.6*}" ]; then pip install pythran; fi;
if [ "$BACKEND" != "cpp" -a -n "${TRAVIS_PYTHON_VERSION##2*}" ]; then pip install mypy; fi;
fi
fi
-
PYTHON_DBG="python$( python -c 'import sys; print("%d.%d" % sys.version_info[:2])' )-dbg"
-
if [ "$COVERAGE" != "1" ]; then CFLAGS="-O2 -ggdb -Wall -Wextra $(python -c 'import sys; print("-fno-strict-aliasing" if sys.version_info[0] == 2 else "")')" python setup.py build_ext -i; fi
-
if $PYTHON_DBG -V >&2; then CFLAGS="-O0 -ggdb" $PYTHON_DBG runtests.py -vv $STYLE_ARGS Debugger --backends=$BACKEND; fi
-
CFLAGS="-O0 -ggdb -Wall -Wextra" python runtests.py -vv $STYLE_ARGS -x Debugger --backends=$BACKEND $(if [ "$COVERAGE" == "1" ]; then echo " --coverage"; fi) $(if [ -z "$TEST_CODE_STYLE" ]; then echo " -j7 "; fi)
-
if [ "$BACKEND" = "cpp" -a -n "${TRAVIS_PYTHON_VERSION##2.6*}" ]; then pip install pythran; fi
-
ccache -s ||
true
-
if [ "$BACKEND" = "c" -a -n "${TRAVIS_PYTHON_VERSION##2*}" ]; then pip install mypy; fi
-
CFLAGS="-O2 -ggdb -Wall -Wextra $(python -c 'import sys; print("-fno-strict-aliasing" if sys.version_info[0] == 2 else "")')" python setup.py build_ext -i
-
CFLAGS="-O0 -ggdb -Wall -Wextra" python runtests.py -vv $STYLE_ARGS -x Debugger --backends=$BACKEND $(if [ -z "$TEST_CODE_STYLE" ]; then echo " -j7 "; fi)
runtests.py
View file @
e51caa48
...
@@ -7,6 +7,7 @@ import os
...
@@ -7,6 +7,7 @@ import os
import
sys
import
sys
import
re
import
re
import
gc
import
gc
import
heapq
import
locale
import
locale
import
shutil
import
shutil
import
time
import
time
...
@@ -564,11 +565,56 @@ class ErrorWriter(object):
...
@@ -564,11 +565,56 @@ class ErrorWriter(object):
pass # ignore, only to match file-like interface
pass # ignore, only to match file-like interface
class Stats(object):
def __init__(self, top_n=8):
self.top_n = top_n
self.test_counts = defaultdict(int)
self.test_times = defaultdict(float)
self.top_tests = defaultdict(list)
def add_time(self, name, language, metric, t):
self.test_counts[metric] += 1
self.test_times[metric] += t
top = self.top_tests[metric]
push = heapq.heappushpop if len(top) >= self.top_n else heapq.heappush
# min-heap => pop smallest/shortest until longest times remain
push(top, (t, name, language))
@contextmanager
def time(self, name, language, metric):
t = time.time()
yield
t = time.time() - t
self.add_time(name, language, metric, t)
def update(self, stats):
# type: (Stats) -> None
for metric, t in stats.test_times.items():
self.test_times[metric] += t
self.test_counts[metric] += stats.test_counts[metric]
top = self.top_tests[metric]
for entry in stats.top_tests[metric]:
push = heapq.heappushpop if len(top) >= self.top_n else heapq.heappush
push(top, entry)
def print_stats(self, out=sys.stderr):
if not self.test_times:
return
lines = ['Times:
\
n
']
for metric, t in sorted(self.test_times.items()):
count = self.test_counts[metric]
top = self.top_tests[metric]
lines.append("%-12s: %8.2f sec (%4d, %6.3f / run) - slowest: %s
\
n
" % (
metric, t, count, t / count,
', '.join("'{2}:{1}' ({0:.2f}s)".format(*item) for item in heapq.nlargest(self.top_n, top))))
out.write(''.join(lines))
class TestBuilder(object):
class TestBuilder(object):
def __init__(self, rootdir, workdir, selectors, exclude_selectors, options,
def __init__(self, rootdir, workdir, selectors, exclude_selectors, options,
with_pyregr, languages, test_bugs, language_level,
with_pyregr, languages, test_bugs, language_level,
common_utility_dir, pythran_dir=None,
common_utility_dir, pythran_dir=None,
default_mode='run',
default_mode='run',
stats=None,
add_embedded_test=False):
add_embedded_test=False):
self.rootdir = rootdir
self.rootdir = rootdir
self.workdir = workdir
self.workdir = workdir
...
@@ -588,6 +634,7 @@ class TestBuilder(object):
...
@@ -588,6 +634,7 @@ class TestBuilder(object):
self.common_utility_dir = common_utility_dir
self.common_utility_dir = common_utility_dir
self.pythran_dir = pythran_dir
self.pythran_dir = pythran_dir
self.default_mode = default_mode
self.default_mode = default_mode
self.stats = stats
self.add_embedded_test = add_embedded_test
self.add_embedded_test = add_embedded_test
def build_suite(self):
def build_suite(self):
...
@@ -646,7 +693,7 @@ class TestBuilder(object):
...
@@ -646,7 +693,7 @@ class TestBuilder(object):
if ext == '.srctree':
if ext == '.srctree':
if 'cpp' not in tags['tag'] or 'cpp' in self.languages:
if 'cpp' not in tags['tag'] or 'cpp' in self.languages:
suite.addTest(EndToEndTest(filepath, workdir, self.cleanup_workdir))
suite.addTest(EndToEndTest(filepath, workdir, self.cleanup_workdir
, stats=self.stats
))
continue
continue
# Choose the test suite.
# Choose the test suite.
...
@@ -676,7 +723,7 @@ class TestBuilder(object):
...
@@ -676,7 +723,7 @@ class TestBuilder(object):
if pyver
if pyver
]
]
if not min_py_ver or any(sys.version_info >= min_ver for min_ver in min_py_ver):
if not min_py_ver or any(sys.version_info >= min_ver for min_ver in min_py_ver):
suite.addTest(PureDoctestTestCase(module, os.path.join(path, filename), tags))
suite.addTest(PureDoctestTestCase(module, os.path.join(path, filename), tags
, stats=self.stats
))
return suite
return suite
...
@@ -736,7 +783,8 @@ class TestBuilder(object):
...
@@ -736,7 +783,8 @@ class TestBuilder(object):
warning_errors=warning_errors,
warning_errors=warning_errors,
test_determinism=self.test_determinism,
test_determinism=self.test_determinism,
common_utility_dir=self.common_utility_dir,
common_utility_dir=self.common_utility_dir,
pythran_dir=pythran_dir)
pythran_dir=pythran_dir,
stats=self.stats)
def skip_c(tags):
def skip_c(tags):
...
@@ -773,7 +821,7 @@ class CythonCompileTestCase(unittest.TestCase):
...
@@ -773,7 +821,7 @@ class CythonCompileTestCase(unittest.TestCase):
cleanup_sharedlibs=True, cleanup_failures=True, cython_only=False,
cleanup_sharedlibs=True, cleanup_failures=True, cython_only=False,
fork=True, language_level=2, warning_errors=False,
fork=True, language_level=2, warning_errors=False,
test_determinism=False,
test_determinism=False,
common_utility_dir=None, pythran_dir=None):
common_utility_dir=None, pythran_dir=None
, stats=None
):
self.test_directory = test_directory
self.test_directory = test_directory
self.tags = tags
self.tags = tags
self.workdir = workdir
self.workdir = workdir
...
@@ -794,6 +842,7 @@ class CythonCompileTestCase(unittest.TestCase):
...
@@ -794,6 +842,7 @@ class CythonCompileTestCase(unittest.TestCase):
self.test_determinism = test_determinism
self.test_determinism = test_determinism
self.common_utility_dir = common_utility_dir
self.common_utility_dir = common_utility_dir
self.pythran_dir = pythran_dir
self.pythran_dir = pythran_dir
self.stats = stats
unittest.TestCase.__init__(self)
unittest.TestCase.__init__(self)
def shortDescription(self):
def shortDescription(self):
...
@@ -1083,7 +1132,8 @@ class CythonCompileTestCase(unittest.TestCase):
...
@@ -1083,7 +1132,8 @@ class CythonCompileTestCase(unittest.TestCase):
old_stderr = sys.stderr
old_stderr = sys.stderr
try:
try:
sys.stderr = ErrorWriter()
sys.stderr = ErrorWriter()
self.run_cython(test_directory, module, workdir, incdir, annotate)
with self.stats.time(self.name, self.language, 'cython'):
self.run_cython(test_directory, module, workdir, incdir, annotate)
errors, warnings = sys.stderr.getall()
errors, warnings = sys.stderr.getall()
finally:
finally:
sys.stderr = old_stderr
sys.stderr = old_stderr
...
@@ -1126,7 +1176,8 @@ class CythonCompileTestCase(unittest.TestCase):
...
@@ -1126,7 +1176,8 @@ class CythonCompileTestCase(unittest.TestCase):
try:
try:
with captured_fd(1) as get_stdout:
with captured_fd(1) as get_stdout:
with captured_fd(2) as get_stderr:
with captured_fd(2) as get_stderr:
so_path = self.run_distutils(test_directory, module, workdir, incdir)
with self.stats.time(self.name, self.language, 'compile-%s' % self.language):
so_path = self.run_distutils(test_directory, module, workdir, incdir)
except Exception as exc:
except Exception as exc:
if ('cerror' in self.tags['tag'] and
if ('cerror' in self.tags['tag'] and
((get_stderr and get_stderr()) or
((get_stderr and get_stderr()) or
...
@@ -1214,18 +1265,22 @@ class CythonRunTestCase(CythonCompileTestCase):
...
@@ -1214,18 +1265,22 @@ class CythonRunTestCase(CythonCompileTestCase):
pass
pass
def run_tests(self, result, ext_so_path):
def run_tests(self, result, ext_so_path):
self.run_doctests(self.module, result, ext_so_path)
with self.stats.time(self.name, self.language, 'run'):
self.run_doctests(self.module, result, ext_so_path)
def run_doctests(self, module_or_name, result, ext_so_path):
def run_doctests(self, module_or_name, result, ext_so_path):
def run_test(result):
def run_test(result):
if isinstance(module_or_name, basestring):
if isinstance(module_or_name, basestring):
module = import_ext(module_or_name, ext_so_path)
with self.stats.time(self.name, self.language, 'import'):
module = import_ext(module_or_name, ext_so_path)
else:
else:
module = module_or_name
module = module_or_name
tests = doctest.DocTestSuite(module)
tests = doctest.DocTestSuite(module)
tests.run(result)
with self.stats.time(self.name, self.language, 'run'):
tests.run(result)
run_forked_test(result, run_test, self.shortDescription(), self.fork)
run_forked_test(result, run_test, self.shortDescription(), self.fork)
def run_forked_test(result, run_func, test_name, fork=True):
def run_forked_test(result, run_func, test_name, fork=True):
if not fork or sys.version_info[0] >= 3 or not hasattr(os, 'fork'):
if not fork or sys.version_info[0] >= 3 or not hasattr(os, 'fork'):
run_func(result)
run_func(result)
...
@@ -1301,11 +1356,13 @@ def run_forked_test(result, run_func, test_name, fork=True):
...
@@ -1301,11 +1356,13 @@ def run_forked_test(result, run_func, test_name, fork=True):
except:
except:
pass
pass
class PureDoctestTestCase(unittest.TestCase):
class PureDoctestTestCase(unittest.TestCase):
def __init__(self, module_name, module_path, tags):
def __init__(self, module_name, module_path, tags
, stats=None
):
self.tags = tags
self.tags = tags
self.module_name = module_name
self.module_name =
self.name =
module_name
self.module_path = module_path
self.module_path = module_path
self.stats = stats
unittest.TestCase.__init__(self, 'run')
unittest.TestCase.__init__(self, 'run')
def shortDescription(self):
def shortDescription(self):
...
@@ -1320,9 +1377,11 @@ class PureDoctestTestCase(unittest.TestCase):
...
@@ -1320,9 +1377,11 @@ class PureDoctestTestCase(unittest.TestCase):
self.setUp()
self.setUp()
import imp
import imp
m = imp.load_source(loaded_module_name, self.module_path)
with self.stats.time(self.name, 'py', 'pyimport'):
m = imp.load_source(loaded_module_name, self.module_path)
try:
try:
doctest.DocTestSuite(m).run(result)
with self.stats.time(self.name, 'py', 'pyrun'):
doctest.DocTestSuite(m).run(result)
finally:
finally:
del m
del m
if loaded_module_name in sys.modules:
if loaded_module_name in sys.modules:
...
@@ -1336,22 +1395,20 @@ class PureDoctestTestCase(unittest.TestCase):
...
@@ -1336,22 +1395,20 @@ class PureDoctestTestCase(unittest.TestCase):
except Exception:
except Exception:
pass
pass
try:
if 'mypy' in self.tags['tag']:
from mypy import api as mypy_api
try:
nomypy = False
from mypy import api as mypy_api
except ImportError:
except ImportError:
nomypy = True
pass
if 'mypy' in self.tags['tag'] and not nomypy:
else:
with self.stats.time(self.name, 'py', 'mypy'):
mypy_result = mypy_api.run((
mypy_result = mypy_api.run((
self.module_path,
self.module_path,
'--ignore-missing-imports',
'--ignore-missing-imports',
'--follow-imports', 'skip',
'--follow-imports', 'skip',
))
))
if mypy_result[2]:
if mypy_result[2]:
self.fail(mypy_result[0])
import pdb; pdb.set_trace()
self.fail(mypy_result[0])
is_private_field = re.compile('^_[^_]').match
is_private_field = re.compile('^_[^_]').match
...
@@ -1420,7 +1477,8 @@ class CythonUnitTestCase(CythonRunTestCase):
...
@@ -1420,7 +1477,8 @@ class CythonUnitTestCase(CythonRunTestCase):
return "compiling (%s) tests in %s" % (self.language, self.name)
return "compiling (%s) tests in %s" % (self.language, self.name)
def run_tests(self, result, ext_so_path):
def run_tests(self, result, ext_so_path):
module = import_ext(self.module, ext_so_path)
with self.stats.time(self.name, self.language, 'import'):
module = import_ext(self.module, ext_so_path)
unittest.defaultTestLoader.loadTestsFromModule(module).run(result)
unittest.defaultTestLoader.loadTestsFromModule(module).run(result)
...
@@ -1452,10 +1510,12 @@ class CythonPyregrTestCase(CythonRunTestCase):
...
@@ -1452,10 +1510,12 @@ class CythonPyregrTestCase(CythonRunTestCase):
suite.addTest(cls)
suite.addTest(cls)
else:
else:
suite.addTest(unittest.makeSuite(cls))
suite.addTest(unittest.makeSuite(cls))
suite.run(result)
with self.stats.time(self.name, self.language, 'run'):
suite.run(result)
def _run_doctest(self, result, module):
def _run_doctest(self, result, module):
self.run_doctests(module, result, None)
with self.stats.time(self.name, self.language, 'run'):
self.run_doctests(module, result, None)
def run_tests(self, result, ext_so_path):
def run_tests(self, result, ext_so_path):
try:
try:
...
@@ -1606,11 +1666,12 @@ class EndToEndTest(unittest.TestCase):
...
@@ -1606,11 +1666,12 @@ class EndToEndTest(unittest.TestCase):
"""
"""
cython_root = os.path.dirname(os.path.abspath(__file__))
cython_root = os.path.dirname(os.path.abspath(__file__))
def __init__(self, treefile, workdir, cleanup_workdir=True):
def __init__(self, treefile, workdir, cleanup_workdir=True
, stats=None
):
self.name = os.path.splitext(os.path.basename(treefile))[0]
self.name = os.path.splitext(os.path.basename(treefile))[0]
self.treefile = treefile
self.treefile = treefile
self.workdir = os.path.join(workdir, self.name)
self.workdir = os.path.join(workdir, self.name)
self.cleanup_workdir = cleanup_workdir
self.cleanup_workdir = cleanup_workdir
self.stats = stats
cython_syspath = [self.cython_root]
cython_syspath = [self.cython_root]
for path in sys.path:
for path in sys.path:
if path.startswith(self.cython_root) and path not in cython_syspath:
if path.startswith(self.cython_root) and path not in cython_syspath:
...
@@ -1658,12 +1719,13 @@ class EndToEndTest(unittest.TestCase):
...
@@ -1658,12 +1719,13 @@ class EndToEndTest(unittest.TestCase):
env = dict(os.environ)
env = dict(os.environ)
env['PYTHONPATH'] = self.cython_syspath + os.pathsep + (old_path or '')
env['PYTHONPATH'] = self.cython_syspath + os.pathsep + (old_path or '')
for command in filter(None, commands.splitlines()):
for command in filter(None, commands.splitlines()):
p = subprocess.Popen(command,
with self.stats.time(self.name, 'c', 'endtoend'):
stderr=subprocess.PIPE,
p = subprocess.Popen(command,
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
shell=True,
stdout=subprocess.PIPE,
env=env)
shell=True,
out, err = p.communicate()
env=env)
out, err = p.communicate()
res = p.returncode
res = p.returncode
if res != 0:
if res != 0:
sys.stderr.write("%s
\
n
%s
\
n
%s
\
n
" % (
sys.stderr.write("%s
\
n
%s
\
n
%s
\
n
" % (
...
@@ -1707,8 +1769,8 @@ class EmbedTest(unittest.TestCase):
...
@@ -1707,8 +1769,8 @@ class EmbedTest(unittest.TestCase):
if sys.version_info[0] >=3 and CY3_DIR:
if sys.version_info[0] >=3 and CY3_DIR:
cython = os.path.join(CY3_DIR, cython)
cython = os.path.join(CY3_DIR, cython)
cython = os.path.abspath(os.path.join('..', '..', cython))
cython = os.path.abspath(os.path.join('..', '..', cython))
self.assert
True(
os.system(
self.assert
Equal(0,
os.system(
"make PYTHON='%s' CYTHON='%s' LIBDIR1='%s' test > make.output" % (sys.executable, cython, libdir))
== 0
)
"make PYTHON='%s' CYTHON='%s' LIBDIR1='%s' test > make.output" % (sys.executable, cython, libdir)))
try:
try:
os.remove('make.output')
os.remove('make.output')
except OSError:
except OSError:
...
@@ -2026,15 +2088,14 @@ def main():
...
@@ -2026,15 +2088,14 @@ def main():
coverage = None
coverage = None
if options.coverage or options.coverage_xml or options.coverage_html:
if options.coverage or options.coverage_xml or options.coverage_html:
if options.shard_count <= 1 and options.shard_num < 0:
if not WITH_CYTHON:
if not WITH_CYTHON:
options.coverage = options.coverage_xml = options.coverage_html = False
options.coverage = options.coverage_xml = options.coverage_html = False
elif options.shard_num == -1:
else:
print("Enabling coverage analysis")
print("Enabling coverage analysis")
from coverage import coverage as _coverage
from coverage import coverage as _coverage
coverage = _coverage()
coverage = _coverage(branch=True, omit=['Test*'])
coverage.erase()
coverage.erase()
coverage.start()
coverage.start()
if options.xml_output_dir:
if options.xml_output_dir:
shutil.rmtree(options.xml_output_dir, ignore_errors=True)
shutil.rmtree(options.xml_output_dir, ignore_errors=True)
...
@@ -2046,12 +2107,14 @@ def main():
...
@@ -2046,12 +2107,14 @@ def main():
errors = []
errors = []
# NOTE: create process pool before time stamper thread to avoid forking issues.
# NOTE: create process pool before time stamper thread to avoid forking issues.
total_time = time.time()
total_time = time.time()
stats = Stats()
with time_stamper_thread():
with time_stamper_thread():
for shard_num, return_code in pool.imap_unordered(runtests_callback, tasks):
for shard_num,
shard_stats,
return_code in pool.imap_unordered(runtests_callback, tasks):
if return_code != 0:
if return_code != 0:
errors.append(shard_num)
errors.append(shard_num)
sys.stderr.write("FAILED (%s/%s)
\
n
" % (shard_num, options.shard_count))
sys.stderr.write("FAILED (%s/%s)
\
n
" % (shard_num, options.shard_count))
sys.stderr.write("ALL DONE (%s/%s)
\
n
" % (shard_num, options.shard_count))
sys.stderr.write("ALL DONE (%s/%s)
\
n
" % (shard_num, options.shard_count))
stats.update(shard_stats)
pool.close()
pool.close()
pool.join()
pool.join()
total_time = time.time() - total_time
total_time = time.time() - total_time
...
@@ -2063,7 +2126,17 @@ def main():
...
@@ -2063,7 +2126,17 @@ def main():
return_code = 0
return_code = 0
else:
else:
with time_stamper_thread():
with time_stamper_thread():
_, return_code = runtests(options, cmd_args, coverage)
_, stats, return_code = runtests(options, cmd_args, coverage)
if coverage:
if options.shard_count > 1 and options.shard_num == -1:
coverage.combine()
coverage.stop()
stats.print_stats(sys.stderr)
if coverage:
save_coverage(coverage, options)
sys.stderr.write("ALL DONE
\
n
")
sys.stderr.write("ALL DONE
\
n
")
sys.stderr.flush()
sys.stderr.flush()
...
@@ -2132,6 +2205,15 @@ def configure_cython(options):
...
@@ -2132,6 +2205,15 @@ def configure_cython(options):
Cython.Compiler.Version.watermark = options.watermark
Cython.Compiler.Version.watermark = options.watermark
def save_coverage(coverage, options):
if options.coverage:
coverage.report(show_missing=0)
if options.coverage_xml:
coverage.xml_report(outfile="coverage-report.xml")
if options.coverage_html:
coverage.html_report(directory="coverage-report-html")
def runtests_callback(args):
def runtests_callback(args):
options, cmd_args, shard_num = args
options, cmd_args, shard_num = args
options.shard_num = shard_num
options.shard_num = shard_num
...
@@ -2299,6 +2381,7 @@ def runtests(options, cmd_args, coverage=None):
...
@@ -2299,6 +2381,7 @@ def runtests(options, cmd_args, coverage=None):
sys.stderr.write("
\
n
")
sys.stderr.write("
\
n
")
test_suite = unittest.TestSuite()
test_suite = unittest.TestSuite()
stats = Stats()
if options.unittests:
if options.unittests:
collect_unittests(UNITTEST_ROOT, UNITTEST_MODULE + ".", test_suite, selectors, exclude_selectors)
collect_unittests(UNITTEST_ROOT, UNITTEST_MODULE + ".", test_suite, selectors, exclude_selectors)
...
@@ -2310,7 +2393,7 @@ def runtests(options, cmd_args, coverage=None):
...
@@ -2310,7 +2393,7 @@ def runtests(options, cmd_args, coverage=None):
filetests = TestBuilder(ROOTDIR, WORKDIR, selectors, exclude_selectors,
filetests = TestBuilder(ROOTDIR, WORKDIR, selectors, exclude_selectors,
options, options.pyregr, languages, test_bugs,
options, options.pyregr, languages, test_bugs,
options.language_level, common_utility_dir,
options.language_level, common_utility_dir,
options.pythran_dir, add_embedded_test=True)
options.pythran_dir, add_embedded_test=True
, stats=stats
)
test_suite.addTest(filetests.build_suite())
test_suite.addTest(filetests.build_suite())
if options.examples and languages:
if options.examples and languages:
for subdirectory in glob.glob(os.path.join(options.examples_dir, "*/")):
for subdirectory in glob.glob(os.path.join(options.examples_dir, "*/")):
...
@@ -2318,7 +2401,7 @@ def runtests(options, cmd_args, coverage=None):
...
@@ -2318,7 +2401,7 @@ def runtests(options, cmd_args, coverage=None):
options, options.pyregr, languages, test_bugs,
options, options.pyregr, languages, test_bugs,
options.language_level, common_utility_dir,
options.language_level, common_utility_dir,
options.pythran_dir,
options.pythran_dir,
default_mode='compile')
default_mode='compile'
, stats=stats
)
test_suite.addTest(filetests.build_suite())
test_suite.addTest(filetests.build_suite())
if options.system_pyregr and languages:
if options.system_pyregr and languages:
...
@@ -2328,7 +2411,7 @@ def runtests(options, cmd_args, coverage=None):
...
@@ -2328,7 +2411,7 @@ def runtests(options, cmd_args, coverage=None):
if os.path.isdir(sys_pyregr_dir):
if os.path.isdir(sys_pyregr_dir):
filetests = TestBuilder(ROOTDIR, WORKDIR, selectors, exclude_selectors,
filetests = TestBuilder(ROOTDIR, WORKDIR, selectors, exclude_selectors,
options, True, languages, test_bugs,
options, True, languages, test_bugs,
sys.version_info[0], common_utility_dir)
sys.version_info[0], common_utility_dir
, stats=stats
)
sys.stderr.write("Including CPython regression tests in %s
\
n
" % sys_pyregr_dir)
sys.stderr.write("Including CPython regression tests in %s
\
n
" % sys_pyregr_dir)
test_suite.addTest(filetests.handle_directory(sys_pyregr_dir, 'pyregr'))
test_suite.addTest(filetests.handle_directory(sys_pyregr_dir, 'pyregr'))
...
@@ -2370,29 +2453,6 @@ def runtests(options, cmd_args, coverage=None):
...
@@ -2370,29 +2453,6 @@ def runtests(options, cmd_args, coverage=None):
if common_utility_dir and options.shard_num < 0 and options.cleanup_workdir:
if common_utility_dir and options.shard_num < 0 and options.cleanup_workdir:
shutil.rmtree(common_utility_dir)
shutil.rmtree(common_utility_dir)
if coverage is not None:
coverage.stop()
ignored_modules = set(
'Cython.Compiler.' + name
for name in ('Version', 'DebugFlags', 'CmdLine')) | set(
'Cython.' + name
for name in ('Debugging',))
ignored_packages = ['Cython.Runtime', 'Cython.Tempita']
modules = [
module for name, module in sys.modules.items()
if module is not None and
name.startswith('Cython.') and
'.Tests' not in name and
name not in ignored_modules and
not any(name.startswith(package) for package in ignored_packages)
]
if options.coverage:
coverage.report(modules, show_missing=0)
if options.coverage_xml:
coverage.xml_report(modules, outfile="coverage-report.xml")
if options.coverage_html:
coverage.html_report(modules, directory="coverage-report-html")
if missing_dep_excluder.tests_missing_deps:
if missing_dep_excluder.tests_missing_deps:
sys.stderr.write("Following tests excluded because of missing dependencies on your system:
\
n
")
sys.stderr.write("Following tests excluded because of missing dependencies on your system:
\
n
")
for test in missing_dep_excluder.tests_missing_deps:
for test in missing_dep_excluder.tests_missing_deps:
...
@@ -2403,9 +2463,9 @@ def runtests(options, cmd_args, coverage=None):
...
@@ -2403,9 +2463,9 @@ def runtests(options, cmd_args, coverage=None):
sys.stderr.write("
\
n
".join([repr(x) for x in refnanny.reflog]))
sys.stderr.write("
\
n
".join([repr(x) for x in refnanny.reflog]))
if options.exit_ok:
if options.exit_ok:
return options.shard_num, 0
return options.shard_num,
stats,
0
else:
else:
return options.shard_num, not result.wasSuccessful()
return options.shard_num,
stats,
not result.wasSuccessful()
if __name__ == '__main__':
if __name__ == '__main__':
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment