Merge "Make it easier to run a selection of tests relevant to ongoing work"

This commit is contained in:
Zuul 2019-11-22 20:58:18 +00:00 committed by Gerrit Code Review
commit e2bba0b15e
4 changed files with 163 additions and 0 deletions

View File

@ -116,6 +116,30 @@ command directly. Running ``stestr run`` will run the entire test suite.
tests in parallel). More information about stestr can be found at:
http://stestr.readthedocs.io/
Since when testing locally, running the entire test suite on a regular
basis is prohibitively expensive, the ``tools/run-tests-for-diff.sh``
script is provided as a convenient way to run selected tests using
output from ``git diff``. For example, this allows running only the
test files changed/added in the working tree::
tools/run-tests-for-diff.sh
However since it passes its arguments directly to ``git diff``, tests
can be selected in lots of other interesting ways, e.g. it can run all
tests affected by a single commit at the tip of a given branch::
tools/run-tests-for-diff.sh mybranch^!
or all those affected by a range of commits, e.g. a branch containing
a whole patch series for a blueprint::
tools/run-tests-for-diff.sh gerrit/master..bp/my-blueprint
It supports the same ``-HEAD`` invocation syntax as ``flake8wrap.sh``
(as used by the ``fast8`` tox environment)::
tools/run-tests-for-diff.sh -HEAD
By default tests log at ``INFO`` level. It is possible to make them
log at ``DEBUG`` level by exporting the ``OS_DEBUG`` environment
variable to ``True``.

View File

@ -32,6 +32,13 @@ For details on plans to report the current test coverage, refer to
Running tests and reporting results
===================================
Running tests locally
---------------------
Please see
https://opendev.org/openstack/nova/src/branch/master/HACKING.rst#running-tests
Voting in Gerrit
----------------

25
tools/run-tests-for-diff.sh Executable file
View File

@ -0,0 +1,25 @@
#!/bin/bash
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
here=`dirname $0`
if test -z "$1"; then
set -- HEAD
elif test "x$1" = "x-HEAD"; then
# Simulate behaviour from flake8wrap.sh
shift
set -- HEAD~1
fi
git diff --name-only "$@" | $here/run-tests.py

107
tools/run-tests.py Executable file
View File

@ -0,0 +1,107 @@
#!/usr/bin/env python
#
# Copyright (c) 2019 SUSE
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
# implied. See the License for the specific language governing
# permissions and limitations under the License.
# Filters list of files on STDIN for test files (can be more than one
# per line), and runs any which can be run in the current tox
# virtualenv. Automatically detects which stestr options to use,
# e.g. --test-path=./nova/tests/functional if we're in a virtualenv
# for functional tests.
import os
import re
import subprocess
import sys
def get_tox_env():
if os.getenv('VIRTUAL_ENV', '').find('/.tox/') == -1:
sys.stderr.write(
"""%s should be run within a tox virtualenv.
Please first activate the tox virtualenv you want to use, e.g.
source .tox/py36/bin/activate
""" %
os.path.realpath(__file__))
sys.exit(1)
return os.getenv('VIRTUAL_ENV')
def tox_env_is_functional():
return get_tox_env().find('.tox/functional') != -1
def get_stestr_opts():
opts = sys.argv[1:]
if tox_env_is_functional():
opts = ['--test-path=./nova/tests/functional'] + opts
return opts
def get_test_files():
test_files = []
functional = tox_env_is_functional()
for line in sys.stdin:
files = line.strip().split()
for f in files:
if not re.match(r'^nova/tests/.*\.py$', f):
# In the future we could get really clever and
# map source files to their corresponding tests,
# as is typically done by Guardfile in projects
# which use Guard: https://guardgem.org
continue
functional_re = r'^nova/tests/functional/'
if functional:
if not re.match(functional_re, f):
continue
else:
if re.match(functional_re, f):
continue
test_files.append(f[:-3].replace('/', '.'))
return test_files
def main():
stestr_opts = get_stestr_opts()
test_files = get_test_files()
if not test_files:
print("No test files found to run")
sys.exit(0)
if len(test_files) == 1:
# If there's only one module to run (or test therein), we can
# skip discovery which will chop quite a few seconds off the
# runtime.
stestr_opts = ['-n'] + stestr_opts
cmd = ['stestr', 'run', *stestr_opts] + test_files
print(' '.join(cmd))
try:
subprocess.check_call(cmd)
except subprocess.CalledProcessError as e:
print("\nstestr returned non-zero exit code %d\n" % e.returncode)
if __name__ == '__main__':
main()