Files
deb-python-gabbi/gabbi/driver.py
Chris Dent 75394fbf61 Add verbose reporting of request and response
In normal testing setups it is not possible to see the requests and
responses being made. If a test is set to 'verbose: True' then the
test request info and the httplib2 request and response data will be
produced to STDOUT.
2015-04-27 14:07:17 +01:00

179 lines
6.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 a 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 copy
import glob
import inspect
import os
from unittest import suite
import uuid
import six
import yaml
from gabbi import case
from gabbi import handlers
from gabbi import suite as gabbi_suite
from gabbi import utils
RESPONSE_HANDLERS = [
handlers.StringResponseHandler,
handlers.JSONResponseHandler,
handlers.HeadersResponseHandler,
]
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,
response_handlers=None):
"""Read YAML files from a directory to create tests.
Each YAML file represents an ordered sequence of HTTP requests.
"""
assert bool(host) ^ bool(intercept), \
'must specify exactly one of host or intercept'
response_handlers = response_handlers or []
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
# Initialize the extensions for response handling.
for handler in RESPONSE_HANDLERS + response_handlers:
handler(case.HTTPTestCase)
# Return an empty suite if we have no host to access, either via
# a real host or an 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_update(orig_dict, new_dict):
"""Modify test in place to update with new data."""
for key, val in six.iteritems(new_dict):
if key == 'data':
orig_dict[key] = val
elif isinstance(val, dict):
orig_dict[key].update(val)
elif isinstance(val, list):
orig_dict[key] = orig_dict.get(key, []) + val
else:
orig_dict[key] = val
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 = gabbi_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 = copy.deepcopy(case.HTTPTestCase.base_test)
defaults = test_yaml.get('defaults', {})
test_update(base_test_data, 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
base_test_key_set = set(case.HTTPTestCase.base_test.keys())
for test_datum in test_data:
test = copy.deepcopy(base_test_data)
test_update(test, test_datum)
if not test['name']:
raise AssertionError('Test name missing in a test in %s.'
% test_base_name)
test_name = '%s_%s' % (test_base_name,
test['name'].lower().replace(' ', '_'))
test_key_set = set(test.keys())
if test_key_set != base_test_key_set:
raise AssertionError(
'Invalid test keys used in test %s: %s'
% (test_name, test_key_set - base_test_key_set))
# Use metaclasses to build a class of the necessary type
# and name with relevant arguments.
http_class = utils.get_http(verbose=test['verbose'])
klass = TestBuilder(test_name, (case.HTTPTestCase,),
{'test_data': test,
'test_directory': test_directory,
'fixtures': fixture_classes,
'http': http_class,
'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