Files
deb-python-gabbi/gabbi/driver.py
Chris Dent c65dbba2b9 Delay wsgi interception until after fixtures
Some WSGI applications require a lot of prior setup before they can
be use. Since we want most of that setup to happen in fixtures,
wsgi-interception is delayed until after all extant fixtures have been
started. This is done with a built in default fixture that is run as
the most deeply nested of the context managers.

The intecept callable is passed on along to the tests so that suite
can be made aware of and use it when using the intercept context
manager. If the callable is a function, when it is passed into the
class builder, it can become bound to the TestCase. This binding is
unbound in the caller.
2015-01-22 21:49:11 +00:00

158 lines
5.0 KiB
Python

# Copyright 2014, 2015 Red Hat
#
# Authors: Chris Dent <chdent@redhat.com>
#
# 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.
"""Generate HTTP tests from YAML files
Each HTTP request is its own TestCase and can be requested to be run in
isolation from other tests. If it is member of a sequence of requests,
prior requests will be run.
A sequence is represented by an ordered list in a single YAML file.
Each sequence becomes a TestSuite.
An entire directory of YAML files is a TestSuite of TestSuites.
"""
import glob
import inspect
import os
from unittest import suite
import uuid
import httplib2
import yaml
from .suite import GabbiSuite
from .case import HTTPTestCase
# Empty test from which all others inherit
BASE_TEST = {
'name': '',
'desc': '',
'ssl': False,
'redirects': False,
'method': 'GET',
'url': '',
'status': '200',
'request_headers': {},
'response_headers': {},
'response_strings': None,
'response_json_paths': None,
'data': '',
}
class TestBuilder(type):
"""Metaclass to munge a dynamically created test.
"""
required_attributes = {'has_run': False}
def __new__(mcs, name, bases, attributes):
attributes.update(mcs.required_attributes)
return type.__new__(mcs, name, bases, attributes)
def build_tests(path, loader, host=None, port=8001, intercept=None,
test_loader_name=None, fixture_module=None):
"""Read YAML files from a directory to create tests.
Each YAML file represents an ordered sequence of HTTP requests.
"""
top_suite = suite.TestSuite()
if test_loader_name is None:
test_loader_name = inspect.stack()[1]
test_loader_name = os.path.splitext(os.path.basename(
test_loader_name[1]))[0]
yaml_file_glob = '%s/*.yaml' % path
# Return an empty suite if we have no host to access, either via
# a real host or an intercept
if host or intercept:
for test_file in glob.iglob(yaml_file_glob):
if intercept:
host = str(uuid.uuid4())
test_yaml = load_yaml(test_file)
test_name = '%s_%s' % (test_loader_name,
os.path.splitext(
os.path.basename(test_file))[0])
file_suite = test_suite_from_yaml(loader, test_name, test_yaml,
path, host, port, fixture_module,
intercept)
top_suite.addTest(file_suite)
return top_suite
def load_yaml(yaml_file):
"""Read and parse any YAML file. Let exceptions flow where they may."""
with open(yaml_file) as source:
return yaml.safe_load(source.read())
def test_suite_from_yaml(loader, test_base_name, test_yaml, test_directory,
host, port, fixture_module, intercept):
"""Generate a TestSuite from YAML data."""
file_suite = GabbiSuite()
test_data = test_yaml['tests']
fixtures = test_yaml.get('fixtures', None)
# Set defaults from BASE_TESTS then update those defaults
# with any defaults set in the YAML file.
base_test_data = dict(BASE_TEST)
base_test_data.update(test_yaml.get('defaults', {}))
# Establish any fixture classes.
fixture_classes = []
if fixtures and fixture_module:
for fixture_class in fixtures:
fixture_classes.append(getattr(fixture_module, fixture_class))
prior_test = None
for test_datum in test_data:
test = dict(base_test_data)
test.update(test_datum)
test_name = '%s_%s' % (test_base_name,
test['name'].lower().replace(' ', '_'))
if set(test.keys()) != set(BASE_TEST.keys()):
raise AssertionError('Invalid test keys used in test: %s'
% test_name)
# Use metaclasses to build a class of the necessary type
# with relevant arguments.
klass = TestBuilder(test_name, (HTTPTestCase,),
{'test_data': test,
'test_directory': test_directory,
'fixtures': fixture_classes,
'http': httplib2.Http(),
'host': host,
'intercept': intercept,
'port': port,
'prior': prior_test})
tests = loader.loadTestsFromTestCase(klass)
this_test = tests._tests[0]
file_suite.addTest(this_test)
prior_test = this_test
return file_suite