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:
Felipe Monteiro 2018-04-07 14:53:48 -04:00
parent c29ad4406b
commit 1566b9541a
29 changed files with 76 additions and 5 deletions

View 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

View File

@ -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

View File

@ -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

View File

@ -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 \