Skip to content

Commit a8d200b

Browse files
authoredSep 1, 2021
Add 'disable_response_handler' test data (#301)
Defaults to False. If true, no response handler is run to transform the response body from its raw content-type to structured data that can be used by tests managed by that content-type's handler. This is useful when a response (over which we probably have no control) claims a content-type of application/json but the response is not valid JSON. In this circumstance there will be an error, the test will fail. Sometimes we may need the test to not fail so we can make other checks (such as checking headers) without worrying about the body. In that case disable_response_handler may be set to True. In the failure case, when we do want the error, the error is now more informative. Docs are updated to reflect the new field. Also included in this change: * Fix (in .stestr.conf) to gabbi's own test runs not being properly parallelized due to a bad regex * Run pep8 tests by default Fixes #299
1 parent c2d759b commit a8d200b

11 files changed

+63
-8
lines changed
 

‎.stestr.conf

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,4 +3,4 @@ test_path=gabbi/tests
33
test_command=${PYTHON:-python} -m subunit.run discover gabbi $LISTOPT $IDOPTION
44
test_id_option=--load-list $IDFILE
55
test_list_option=--list
6-
group_regex=(?:gabbi\.suitemaker\.(test_[^_]+_[^_]+)|tests\.test_(?:intercept|inner_fixture)\.([^_]+))
6+
group_regex=(?:gabbi\.tests\.test_(?:\w+)\.([^_]+))

‎docs/source/format.rst

+8
Original file line numberDiff line numberDiff line change
@@ -72,6 +72,14 @@ Metadata
7272
so changing this may be requried. It can also be changed when
7373
:doc:`loader` or using :doc:`gabbi-run <runner>`.
7474
- defaults to ``True``
75+
* - ``disable_response_handler``
76+
- If ``True``, means that the response body will not be processed to
77+
Python data. This can be necessary if a response claims a
78+
``content-type`` but the body is not actually that type but it is still
79+
necessary to run tests against the response. In that situation, if
80+
``disable_response_handler`` is ``False`` the test will be treated as
81+
a failure.
82+
- defaults to ``False``
7583

7684

7785
.. note:: When tests are generated dynamically, the ``TestCase`` name will

‎gabbi/case.py

+9-2
Original file line numberDiff line numberDiff line change
@@ -70,6 +70,7 @@
7070
'skip': '',
7171
'poll': {},
7272
'use_prior_test': True,
73+
'disable_response_handler': False
7374
}
7475

7576

@@ -482,9 +483,15 @@ def _run_request(self, url, method, headers, body, redirect=False):
482483
decoded_output = utils.decode_response_content(response, content)
483484
self.content_type = response.get('content-type', '').lower()
484485
loader_class = self.get_content_handler(self.content_type)
485-
if decoded_output and loader_class:
486+
if (decoded_output and loader_class
487+
and not self.test_data['disable_response_handler']):
486488
# save structured response data
487-
self.response_data = loader_class.loads(decoded_output)
489+
try:
490+
self.response_data = loader_class.loads(decoded_output)
491+
except exception.GabbiDataLoadError as exc:
492+
raise AssertionError(
493+
'unable to load data as %s' % self.content_type) from exc
494+
488495
else:
489496
self.response_data = None
490497
self.output = decoded_output

‎gabbi/exception.py

+5
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,11 @@
1313
"""Gabbi specific exceptions."""
1414

1515

16+
class GabbiDataLoadError(ValueError):
17+
"""An exception to alert when data streams cannot be loaded."""
18+
pass
19+
20+
1621
class GabbiFormatError(ValueError):
1722
"""An exception to encapsulate poorly formed test data."""
1823
pass

‎gabbi/handlers/base.py

+5-1
Original file line numberDiff line numberDiff line change
@@ -109,7 +109,11 @@ def dumps(data, pretty=False, test=None):
109109

110110
@staticmethod
111111
def loads(data):
112-
"""Create structured (Python) data from a stream."""
112+
"""Create structured (Python) data from a stream.
113+
114+
If there is a failure decoding then the handler should
115+
repackage the error as a gabbi.exception.GabbiDataLoadError.
116+
"""
113117
return data
114118

115119
@staticmethod

‎gabbi/handlers/jsonhandler.py

+5-1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616

1717
import six
1818

19+
from gabbi.exception import GabbiDataLoadError
1920
from gabbi.handlers import base
2021
from gabbi import json_parser
2122

@@ -54,7 +55,10 @@ def dumps(data, pretty=False, test=None):
5455

5556
@staticmethod
5657
def loads(data):
57-
return json.loads(data)
58+
try:
59+
return json.loads(data)
60+
except ValueError as exc:
61+
raise GabbiDataLoadError('unable to parse data') from exc
5862

5963
@staticmethod
6064
def load_data_file(test, file_path):
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
# Test that disabling the response handler despite an accept match
2+
# works as required.
3+
4+
tests:
5+
6+
- name: get some not json fail
7+
desc: This will cause an error, presented as a test failure
8+
xfail: True
9+
GET: /notjson
10+
response_headers:
11+
content-type: application/json
12+
response_strings:
13+
- not valid json
14+
15+
- name: get some not json gloss
16+
desc: this will not error because we do not parse
17+
GET: /notjson
18+
response_headers:
19+
content-type: application/json
20+
disable_response_handler: True
21+
response_strings:
22+
- not valid json

‎gabbi/tests/simple_wsgi.py

+4
Original file line numberDiff line numberDiff line change
@@ -92,6 +92,10 @@ def __call__(self, environ, start_response):
9292
</body>
9393
</html>
9494
"""]
95+
# Provide response that claims to be json but is not.
96+
elif path_info.startswith('/notjson'):
97+
start_response('200 OK', [('Content-Type', 'application/json')])
98+
return [b'not valid json']
9599
elif path_info.startswith('/poller'):
96100
if CURRENT_POLL == 0:
97101
CURRENT_POLL = int(query_data.get('count', [5])[0])

‎gabbi/tests/test_live.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -47,7 +47,8 @@ def start_fixture(self):
4747
def load_tests(loader, tests, pattern):
4848
"""Provide a TestSuite to the discovery process."""
4949
test_dir = os.path.join(os.path.dirname(__file__), TESTS_DIR)
50-
return driver.build_tests(test_dir, loader, **BUILD_TEST_ARGS)
50+
return driver.build_tests(
51+
test_dir, loader, test_loader_name=__name__, **BUILD_TEST_ARGS)
5152

5253

5354
def pytest_generate_tests(metafunc):

‎test-failskip.sh

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ shopt -s nocasematch
77
[[ "${GABBI_SKIP_NETWORK:-false}" == "true" ]] && SKIP=7 || SKIP=2
88
shopt -u nocasematch
99

10-
FAILS=12
10+
FAILS=13
1111

1212
GREP_FAIL_MATCH="expected failures=$FAILS"
1313
GREP_SKIP_MATCH="skipped=$SKIP,"

‎tox.ini

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
[tox]
22
minversion = 3.1.1
33
skipsdist = True
4-
envlist = py36,py37,py38,py39,pypy3,pep8,limit,failskip,docs,py39-prefix,py39-limit,py39-verbosity,py39-failskip,py36-pytest,py37-pytest,py38-pytest,py39-pytest
4+
envlist = pep8,py36,py37,py38,py39,pypy3,pep8,limit,failskip,docs,py39-prefix,py39-limit,py39-verbosity,py39-failskip,py36-pytest,py37-pytest,py38-pytest,py39-pytest
55

66
[testenv]
77
deps = -r{toxinidir}/requirements.txt

0 commit comments

Comments
 (0)
Please sign in to comment.