Clean up functional test directory and entrypoint script
This PS simply reorganizes Deckhand's functional test directory to make it more maintainable and readable as right now it is hard to figure out what is covered by a functional test and what isn't. Additionally, the entrypoint for these tests in tools/functional-tests.sh has also been refactored slightly. Change-Id: I262c7e1f7cbce248c12ee013a9bab4e32b89adee
This commit is contained in:
parent
c29ad4406b
commit
1566b9541a
30
deckhand/tests/functional/README.rst
Normal file
30
deckhand/tests/functional/README.rst
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
Functional Tests
|
||||||
|
================
|
||||||
|
|
||||||
|
Deckhand uses `gabbi`_ to drive its functional tests. The entry point for
|
||||||
|
these tests is ``functional-tests.sh`` under ``tools`` directory.
|
||||||
|
|
||||||
|
Directory Test Layout
|
||||||
|
---------------------
|
||||||
|
|
||||||
|
Tests are contained in intuitively named subdirectories nested under
|
||||||
|
``deckhand/tests/functional/gabbits``. For example, layering tests are
|
||||||
|
contained under the ``layering`` subdirectory. This pattern should be strictly
|
||||||
|
followed.
|
||||||
|
|
||||||
|
Because `gabbi`_ does not support loading tests from subdirectories, logic
|
||||||
|
is included in ``test_gabbi.py`` to:
|
||||||
|
|
||||||
|
#. Create a temporary directory.
|
||||||
|
#. Create a symlink between all the test files in the nested subdirectories
|
||||||
|
and the temporary directory.
|
||||||
|
|
||||||
|
However, the test directory can still be modified:
|
||||||
|
|
||||||
|
* New subdirectories under ``gabbits`` can be added.
|
||||||
|
* New tests under any of those subdirectories can be added.
|
||||||
|
* New resource files under ``gabits/resources`` can be added. This directory
|
||||||
|
name should never be renamed.
|
||||||
|
* All other subdirectories, test files, and resources may be renamed.
|
||||||
|
|
||||||
|
.. _gabbi: https://gabbi.readthedocs.io/en/latest/gabbi.html
|
@ -12,14 +12,55 @@
|
|||||||
# See the License for the specific language governing permissions and
|
# See the License for the specific language governing permissions and
|
||||||
# limitations under the License.
|
# limitations under the License.
|
||||||
|
|
||||||
|
import atexit
|
||||||
import os
|
import os
|
||||||
|
import shutil
|
||||||
|
import tempfile
|
||||||
import yaml
|
import yaml
|
||||||
|
|
||||||
from gabbi import driver
|
from gabbi import driver
|
||||||
from gabbi.driver import test_pytest # noqa
|
from gabbi.driver import test_pytest # noqa
|
||||||
from gabbi.handlers import jsonhandler
|
from gabbi.handlers import jsonhandler
|
||||||
|
|
||||||
TESTS_DIR = 'gabbits'
|
TEST_DIR = tempfile.mkdtemp(prefix='deckhand')
|
||||||
|
|
||||||
|
|
||||||
|
def __create_temp_test_dir():
|
||||||
|
"""Hack around the fact that gabbi doesn't support loading tests contained
|
||||||
|
in subdirectories. This inconvenience leads to poor test directory layout
|
||||||
|
in which all the test files are contained in one directory.
|
||||||
|
|
||||||
|
"""
|
||||||
|
root_test_dir = os.path.join(os.path.dirname(__file__), 'gabbits')
|
||||||
|
test_files = []
|
||||||
|
|
||||||
|
for root, dirs, files in os.walk(root_test_dir):
|
||||||
|
is_test_file = (
|
||||||
|
'gabbits' in root and not root.endswith('gabbits')
|
||||||
|
)
|
||||||
|
if is_test_file:
|
||||||
|
test_files.extend([os.path.abspath(os.path.join(root, f))
|
||||||
|
for f in files])
|
||||||
|
|
||||||
|
resources_dir = os.path.join(TEST_DIR, 'resources')
|
||||||
|
if not os.path.exists(resources_dir):
|
||||||
|
os.makedirs(resources_dir)
|
||||||
|
|
||||||
|
for test_file in test_files:
|
||||||
|
basename = os.path.basename(test_file)
|
||||||
|
if 'resources' in test_file:
|
||||||
|
os.symlink(test_file, os.path.join(resources_dir, basename))
|
||||||
|
else:
|
||||||
|
os.symlink(test_file, os.path.join(TEST_DIR, basename))
|
||||||
|
|
||||||
|
|
||||||
|
__create_temp_test_dir()
|
||||||
|
|
||||||
|
|
||||||
|
@atexit.register
|
||||||
|
def __remove_temp_test_dir():
|
||||||
|
if os.path.exists(TEST_DIR):
|
||||||
|
shutil.rmtree(TEST_DIR)
|
||||||
|
|
||||||
|
|
||||||
# This is quite similar to the existing JSONHandler, so use it as the base
|
# This is quite similar to the existing JSONHandler, so use it as the base
|
||||||
@ -47,12 +88,11 @@ class MultidocJsonpaths(jsonhandler.JSONHandler):
|
|||||||
|
|
||||||
|
|
||||||
def pytest_generate_tests(metafunc):
|
def pytest_generate_tests(metafunc):
|
||||||
test_dir = os.path.join(os.path.dirname(__file__), TESTS_DIR)
|
|
||||||
# NOTE(fmontei): While only `url` or `host` is needed, strangely both
|
# NOTE(fmontei): While only `url` or `host` is needed, strangely both
|
||||||
# are needed because we use `pytest-html` which throws an error without
|
# are needed because we use `pytest-html` which throws an error without
|
||||||
# `host`.
|
# `host`.
|
||||||
driver.py_test_generator(
|
driver.py_test_generator(
|
||||||
test_dir, url=os.environ['DECKHAND_TEST_URL'], host='localhost',
|
TEST_DIR, url=os.environ['DECKHAND_TEST_URL'], host='localhost',
|
||||||
# NOTE(fmontei): When there are multiple handlers listed that accept
|
# NOTE(fmontei): When there are multiple handlers listed that accept
|
||||||
# the same content-type, the one that is earliest in the list will be
|
# the same content-type, the one that is earliest in the list will be
|
||||||
# used. Thus, we cannot specify multiple content handlers for handling
|
# used. Thus, we cannot specify multiple content handlers for handling
|
||||||
|
@ -154,6 +154,7 @@ function gen_paste {
|
|||||||
local disable_keystone=$1
|
local disable_keystone=$1
|
||||||
|
|
||||||
if $disable_keystone; then
|
if $disable_keystone; then
|
||||||
|
log_section Disabling Keystone authentication.
|
||||||
sed 's/authtoken api/api/' etc/deckhand/deckhand-paste.ini &> $CONF_DIR/deckhand-paste.ini
|
sed 's/authtoken api/api/' etc/deckhand/deckhand-paste.ini &> $CONF_DIR/deckhand-paste.ini
|
||||||
else
|
else
|
||||||
cp etc/deckhand/deckhand-paste.ini $CONF_DIR/deckhand-paste.ini
|
cp etc/deckhand/deckhand-paste.ini $CONF_DIR/deckhand-paste.ini
|
||||||
|
@ -51,7 +51,7 @@ function deploy_deckhand {
|
|||||||
gen_policy
|
gen_policy
|
||||||
|
|
||||||
if [ -z "$DECKHAND_IMAGE" ]; then
|
if [ -z "$DECKHAND_IMAGE" ]; then
|
||||||
log_section "Running Deckhand via uwsgi"
|
log_section "Running Deckhand via uwsgi."
|
||||||
|
|
||||||
alembic upgrade head
|
alembic upgrade head
|
||||||
# NOTE(fmontei): Deckhand's database is not configured to work with
|
# NOTE(fmontei): Deckhand's database is not configured to work with
|
||||||
@ -63,7 +63,7 @@ function deploy_deckhand {
|
|||||||
export DECKHAND_API_THREADS=4
|
export DECKHAND_API_THREADS=4
|
||||||
source $ROOTDIR/../entrypoint.sh server &
|
source $ROOTDIR/../entrypoint.sh server &
|
||||||
else
|
else
|
||||||
log_section "Running Deckhand via Docker"
|
log_section "Running Deckhand via Docker."
|
||||||
sudo docker run \
|
sudo docker run \
|
||||||
--rm \
|
--rm \
|
||||||
--net=host \
|
--net=host \
|
||||||
|
Loading…
Reference in New Issue
Block a user