func tests: fix etag-quoter insertion in pipeline
Previously the in-process functional test setup attempted to insert etag-quoter in the proxy pipeline by replacing the substring 'cache listing_formats' with 'cache etag-quoter listing_formats'. However, the pipeline had already been modified to have 'cache domain-remap listing_formats', so the etag-quoter replacement failed silently. This patch modifies the etag-quoter replacement to match just 'cache'. Some helper functions are also introduced to make the pipeline modification more uniform and to consistently verify the intended modifications have been achieved, or otherwise raise an exception. Change-Id: I640837a4da985bcbe48f9ae5eeb56385b11034eb
This commit is contained in:
		| @@ -14,6 +14,10 @@ | |||||||
| # limitations under the License. | # limitations under the License. | ||||||
|  |  | ||||||
| from __future__ import print_function | from __future__ import print_function | ||||||
|  |  | ||||||
|  | import configparser | ||||||
|  | import contextlib | ||||||
|  |  | ||||||
| import mock | import mock | ||||||
| import os | import os | ||||||
| import six | import six | ||||||
| @@ -35,7 +39,7 @@ from shutil import rmtree | |||||||
| from tempfile import mkdtemp | from tempfile import mkdtemp | ||||||
| from unittest import SkipTest | from unittest import SkipTest | ||||||
|  |  | ||||||
| from six.moves.configparser import ConfigParser, NoSectionError | from six.moves.configparser import ConfigParser | ||||||
| from six.moves import http_client | from six.moves import http_client | ||||||
| from six.moves.http_client import HTTPException | from six.moves.http_client import HTTPException | ||||||
|  |  | ||||||
| @@ -131,24 +135,60 @@ def _debug(msg): | |||||||
|         _info('DEBUG: ' + msg) |         _info('DEBUG: ' + msg) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @contextlib.contextmanager | ||||||
|  | def _modify_conf_file(src_conf_file, dest_conf_file): | ||||||
|  |     conf = ConfigParser() | ||||||
|  |     conf.read(src_conf_file) | ||||||
|  |     try: | ||||||
|  |         yield conf | ||||||
|  |     except configparser.Error as err: | ||||||
|  |         msg = 'Error modifying conf file %s: %s' % (src_conf_file, err) | ||||||
|  |         raise InProcessException(msg) | ||||||
|  |     else: | ||||||
|  |         with open(dest_conf_file, 'w') as fp: | ||||||
|  |             conf.write(fp) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _cleanup_pipeline(pipeline_str): | ||||||
|  |     # return single-whitespace delimited and padded string | ||||||
|  |     # e.g. "x   y  " -> " x y " | ||||||
|  |     return ' %s ' % ' '.join([x for x in pipeline_str.split(' ') if x]) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _modify_pipeline_substring(conf, old_substr, new_substr): | ||||||
|  |     section = 'pipeline:main' | ||||||
|  |     old_pipeline = _cleanup_pipeline(conf.get(section, 'pipeline')) | ||||||
|  |     old_substr = _cleanup_pipeline(old_substr) | ||||||
|  |     new_substr = _cleanup_pipeline(new_substr) | ||||||
|  |     if old_substr not in old_pipeline: | ||||||
|  |         raise InProcessException( | ||||||
|  |             'Failed to replace pipeline substring: old="%s", new="%s", ' | ||||||
|  |             'pipeline="%s"' % (old_substr, new_substr, old_pipeline)) | ||||||
|  |  | ||||||
|  |     if new_substr in old_pipeline: | ||||||
|  |         _info('WARNING: "%s" is already in pipeline, not replacing' | ||||||
|  |               % new_substr) | ||||||
|  |         new_pipeline = old_pipeline | ||||||
|  |     else: | ||||||
|  |         new_pipeline = old_pipeline.replace(old_substr, new_substr) | ||||||
|  |         _debug('Replaced pipeline substring "%s" with "%s"' | ||||||
|  |                % (old_substr, new_substr)) | ||||||
|  |     # remove padding | ||||||
|  |     new_pipeline = new_pipeline.strip(' ') | ||||||
|  |     conf.set(section, 'pipeline', new_pipeline) | ||||||
|  |     _debug('Proxy pipeline is now "%s"' % new_pipeline) | ||||||
|  |  | ||||||
|  |  | ||||||
| def _in_process_setup_swift_conf(swift_conf_src, testdir): | def _in_process_setup_swift_conf(swift_conf_src, testdir): | ||||||
|     # override swift.conf contents for in-process functional test runs |     # override swift.conf contents for in-process functional test runs | ||||||
|     conf = ConfigParser() |     test_conf_file = os.path.join(testdir, 'swift.conf') | ||||||
|     conf.read(swift_conf_src) |     with _modify_conf_file(swift_conf_src, test_conf_file) as conf: | ||||||
|     try: |  | ||||||
|         section = 'swift-hash' |         section = 'swift-hash' | ||||||
|         conf.set(section, 'swift_hash_path_suffix', 'inprocfunctests') |         conf.set(section, 'swift_hash_path_suffix', 'inprocfunctests') | ||||||
|         conf.set(section, 'swift_hash_path_prefix', 'inprocfunctests') |         conf.set(section, 'swift_hash_path_prefix', 'inprocfunctests') | ||||||
|         section = 'swift-constraints' |         section = 'swift-constraints' | ||||||
|         max_file_size = (8 * 1024 * 1024) + 2  # 8 MB + 2 |         max_file_size = (8 * 1024 * 1024) + 2  # 8 MB + 2 | ||||||
|         conf.set(section, 'max_file_size', str(max_file_size)) |         conf.set(section, 'max_file_size', str(max_file_size)) | ||||||
|     except NoSectionError: |  | ||||||
|         msg = 'Conf file %s is missing section %s' % (swift_conf_src, section) |  | ||||||
|         raise InProcessException(msg) |  | ||||||
|  |  | ||||||
|     test_conf_file = os.path.join(testdir, 'swift.conf') |  | ||||||
|     with open(test_conf_file, 'w') as fp: |  | ||||||
|         conf.write(fp) |  | ||||||
|  |  | ||||||
|     return test_conf_file |     return test_conf_file | ||||||
|  |  | ||||||
| @@ -195,8 +235,7 @@ def _in_process_setup_ring(swift_conf, conf_src_dir, testdir): | |||||||
|       in process testing, and renaming it to suit policy-0 |       in process testing, and renaming it to suit policy-0 | ||||||
|     Otherwise, create a default ring file. |     Otherwise, create a default ring file. | ||||||
|     """ |     """ | ||||||
|     conf = ConfigParser() |     with _modify_conf_file(swift_conf, swift_conf) as conf: | ||||||
|     conf.read(swift_conf) |  | ||||||
|         sp_prefix = 'storage-policy:' |         sp_prefix = 'storage-policy:' | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
| @@ -205,7 +244,8 @@ def _in_process_setup_ring(swift_conf, conf_src_dir, testdir): | |||||||
|         except PolicyError as e: |         except PolicyError as e: | ||||||
|             raise InProcessException(e) |             raise InProcessException(e) | ||||||
|  |  | ||||||
|     # clear all policies from test swift.conf before adding test policy back |         # clear all policies from test swift.conf before adding test policy | ||||||
|  |         # back | ||||||
|         for policy in policies: |         for policy in policies: | ||||||
|             conf.remove_section(sp_prefix + str(policy.idx)) |             conf.remove_section(sp_prefix + str(policy.idx)) | ||||||
|  |  | ||||||
| @@ -226,9 +266,6 @@ def _in_process_setup_ring(swift_conf, conf_src_dir, testdir): | |||||||
|             conf.set(sp_zero_section, k, str(v)) |             conf.set(sp_zero_section, k, str(v)) | ||||||
|         conf.set(sp_zero_section, 'default', 'True') |         conf.set(sp_zero_section, 'default', 'True') | ||||||
|  |  | ||||||
|     with open(swift_conf, 'w') as fp: |  | ||||||
|         conf.write(fp) |  | ||||||
|  |  | ||||||
|     # look for a source ring file |     # look for a source ring file | ||||||
|     ring_file_src = ring_file_test = 'object.ring.gz' |     ring_file_src = ring_file_test = 'object.ring.gz' | ||||||
|     if policy_to_test.idx: |     if policy_to_test.idx: | ||||||
| @@ -284,6 +321,18 @@ def _in_process_setup_ring(swift_conf, conf_src_dir, testdir): | |||||||
|     return obj_sockets |     return obj_sockets | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _load_etag_quoter(proxy_conf_file, swift_conf_file, **kwargs): | ||||||
|  |     _debug('Ensuring etag-quoter is in proxy pipeline') | ||||||
|  |     test_conf_file = os.path.join(_testdir, 'proxy-server.conf') | ||||||
|  |     with _modify_conf_file(proxy_conf_file, test_conf_file) as conf: | ||||||
|  |         _modify_pipeline_substring( | ||||||
|  |             conf, | ||||||
|  |             "cache", | ||||||
|  |             "cache etag-quoter") | ||||||
|  |  | ||||||
|  |     return test_conf_file, swift_conf_file | ||||||
|  |  | ||||||
|  |  | ||||||
| def _load_encryption(proxy_conf_file, swift_conf_file, **kwargs): | def _load_encryption(proxy_conf_file, swift_conf_file, **kwargs): | ||||||
|     """ |     """ | ||||||
|     Load encryption configuration and override proxy-server.conf contents. |     Load encryption configuration and override proxy-server.conf contents. | ||||||
| @@ -302,34 +351,20 @@ def _load_encryption(proxy_conf_file, swift_conf_file, **kwargs): | |||||||
|     # conf during wsgi load_app. |     # conf during wsgi load_app. | ||||||
|     # Therefore we must modify the [pipeline:main] section. |     # Therefore we must modify the [pipeline:main] section. | ||||||
|  |  | ||||||
|     conf = ConfigParser() |     test_conf_file = os.path.join(_testdir, 'proxy-server.conf') | ||||||
|     conf.read(proxy_conf_file) |     with _modify_conf_file(proxy_conf_file, test_conf_file) as conf: | ||||||
|     try: |         _modify_pipeline_substring( | ||||||
|         section = 'pipeline:main' |             conf, | ||||||
|         pipeline = conf.get(section, 'pipeline') |  | ||||||
|         pipeline = pipeline.replace( |  | ||||||
|             "proxy-logging proxy-server", |             "proxy-logging proxy-server", | ||||||
|             "keymaster encryption proxy-logging proxy-server") |             "keymaster encryption proxy-logging proxy-server") | ||||||
|         pipeline = pipeline.replace( |  | ||||||
|             "cache listing_formats", |  | ||||||
|             "cache etag-quoter listing_formats") |  | ||||||
|         conf.set(section, 'pipeline', pipeline) |  | ||||||
|         root_secret = base64.b64encode(os.urandom(32)) |         root_secret = base64.b64encode(os.urandom(32)) | ||||||
|         if not six.PY2: |         if not six.PY2: | ||||||
|             root_secret = root_secret.decode('ascii') |             root_secret = root_secret.decode('ascii') | ||||||
|         conf.set('filter:keymaster', 'encryption_root_secret', root_secret) |         conf.set('filter:keymaster', 'encryption_root_secret', root_secret) | ||||||
|         conf.set('filter:versioned_writes', 'allow_object_versioning', 'true') |         conf.set('filter:versioned_writes', 'allow_object_versioning', 'true') | ||||||
|         conf.set('filter:etag-quoter', 'enable_by_default', 'true') |         conf.set('filter:etag-quoter', 'enable_by_default', 'true') | ||||||
|     except NoSectionError as err: |  | ||||||
|         msg = 'Error problem with proxy conf file %s: %s' % \ |  | ||||||
|               (proxy_conf_file, err) |  | ||||||
|         raise InProcessException(msg) |  | ||||||
|  |  | ||||||
|     test_conf_file = os.path.join(_testdir, 'proxy-server.conf') |     return _load_etag_quoter(test_conf_file, swift_conf_file) | ||||||
|     with open(test_conf_file, 'w') as fp: |  | ||||||
|         conf.write(fp) |  | ||||||
|  |  | ||||||
|     return test_conf_file, swift_conf_file |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def _load_ec_as_default_policy(proxy_conf_file, swift_conf_file, **kwargs): | def _load_ec_as_default_policy(proxy_conf_file, swift_conf_file, **kwargs): | ||||||
| @@ -342,8 +377,7 @@ def _load_ec_as_default_policy(proxy_conf_file, swift_conf_file, **kwargs): | |||||||
|     """ |     """ | ||||||
|     _debug('Setting configuration for default EC policy') |     _debug('Setting configuration for default EC policy') | ||||||
|  |  | ||||||
|     conf = ConfigParser() |     with _modify_conf_file(swift_conf_file, swift_conf_file) as conf: | ||||||
|     conf.read(swift_conf_file) |  | ||||||
|         # remove existing policy sections that came with swift.conf-sample |         # remove existing policy sections that came with swift.conf-sample | ||||||
|         for section in list(conf.sections()): |         for section in list(conf.sections()): | ||||||
|             if section.startswith('storage-policy'): |             if section.startswith('storage-policy'): | ||||||
| @@ -363,8 +397,6 @@ def _load_ec_as_default_policy(proxy_conf_file, swift_conf_file, **kwargs): | |||||||
|         for k, v in ec_policy_spec.items(): |         for k, v in ec_policy_spec.items(): | ||||||
|             conf.set('storage-policy:0', k, str(v)) |             conf.set('storage-policy:0', k, str(v)) | ||||||
|  |  | ||||||
|     with open(swift_conf_file, 'w') as fp: |  | ||||||
|         conf.write(fp) |  | ||||||
|     return proxy_conf_file, swift_conf_file |     return proxy_conf_file, swift_conf_file | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -390,33 +422,19 @@ def _load_domain_remap_staticweb(proxy_conf_file, swift_conf_file, **kwargs): | |||||||
|     # DEFAULTS options) then it prevents pipeline being loaded into the local |     # DEFAULTS options) then it prevents pipeline being loaded into the local | ||||||
|     # conf during wsgi load_app. |     # conf during wsgi load_app. | ||||||
|     # Therefore we must modify the [pipeline:main] section. |     # Therefore we must modify the [pipeline:main] section. | ||||||
|     conf = ConfigParser() |     test_conf_file = os.path.join(_testdir, 'proxy-server.conf') | ||||||
|     conf.read(proxy_conf_file) |     with _modify_conf_file(proxy_conf_file, test_conf_file) as conf: | ||||||
|     try: |         _modify_pipeline_substring( | ||||||
|         section = 'pipeline:main' |             conf, | ||||||
|         old_pipeline = conf.get(section, 'pipeline') |             "tempauth", | ||||||
|         pipeline = old_pipeline.replace( |             "tempauth staticweb") | ||||||
|             " tempauth ", |         _modify_pipeline_substring( | ||||||
|             " tempauth staticweb ") |             conf, | ||||||
|         pipeline = pipeline.replace( |             "listing_formats", | ||||||
|             " listing_formats ", |             "domain_remap listing_formats") | ||||||
|             " domain_remap listing_formats ") |  | ||||||
|         if pipeline == old_pipeline: |  | ||||||
|             raise InProcessException( |  | ||||||
|                 "Failed to insert domain_remap and staticweb into pipeline: %s" |  | ||||||
|                 % old_pipeline) |  | ||||||
|         conf.set(section, 'pipeline', pipeline) |  | ||||||
|         # set storage_domain in domain_remap middleware to match test config |         # set storage_domain in domain_remap middleware to match test config | ||||||
|         section = 'filter:domain_remap' |         section = 'filter:domain_remap' | ||||||
|         conf.set(section, 'storage_domain', storage_domain) |         conf.set(section, 'storage_domain', storage_domain) | ||||||
|     except NoSectionError as err: |  | ||||||
|         msg = 'Error problem with proxy conf file %s: %s' % \ |  | ||||||
|               (proxy_conf_file, err) |  | ||||||
|         raise InProcessException(msg) |  | ||||||
|  |  | ||||||
|     test_conf_file = os.path.join(_testdir, 'proxy-server.conf') |  | ||||||
|     with open(test_conf_file, 'w') as fp: |  | ||||||
|         conf.write(fp) |  | ||||||
|  |  | ||||||
|     return test_conf_file, swift_conf_file |     return test_conf_file, swift_conf_file | ||||||
|  |  | ||||||
| @@ -439,26 +457,15 @@ def _load_s3api(proxy_conf_file, swift_conf_file, **kwargs): | |||||||
|     # conf during wsgi load_app. |     # conf during wsgi load_app. | ||||||
|     # Therefore we must modify the [pipeline:main] section. |     # Therefore we must modify the [pipeline:main] section. | ||||||
|  |  | ||||||
|     conf = ConfigParser() |     test_conf_file = os.path.join(_testdir, 'proxy-server.conf') | ||||||
|     conf.read(proxy_conf_file) |     with _modify_conf_file(proxy_conf_file, test_conf_file) as conf: | ||||||
|     try: |         _modify_pipeline_substring( | ||||||
|         section = 'pipeline:main' |             conf, | ||||||
|         pipeline = conf.get(section, 'pipeline') |  | ||||||
|         pipeline = pipeline.replace( |  | ||||||
|             "tempauth", |             "tempauth", | ||||||
|             "s3api tempauth") |             "s3api tempauth") | ||||||
|         conf.set(section, 'pipeline', pipeline) |  | ||||||
|         conf.set('filter:s3api', 's3_acl', 'true') |         conf.set('filter:s3api', 's3_acl', 'true') | ||||||
|  |  | ||||||
|         conf.set('filter:versioned_writes', 'allow_object_versioning', 'true') |         conf.set('filter:versioned_writes', 'allow_object_versioning', 'true') | ||||||
|     except NoSectionError as err: |  | ||||||
|         msg = 'Error problem with proxy conf file %s: %s' % \ |  | ||||||
|               (proxy_conf_file, err) |  | ||||||
|         raise InProcessException(msg) |  | ||||||
|  |  | ||||||
|     test_conf_file = os.path.join(_testdir, 'proxy-server.conf') |  | ||||||
|     with open(test_conf_file, 'w') as fp: |  | ||||||
|         conf.write(fp) |  | ||||||
|  |  | ||||||
|     return test_conf_file, swift_conf_file |     return test_conf_file, swift_conf_file | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Alistair Coles
					Alistair Coles