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:
Alistair Coles 2024-09-25 12:57:37 +01:00
parent 81547cb67d
commit 86081b7822

View File

@ -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')
pipeline = old_pipeline.replace(
"tempauth", "tempauth",
"tempauth staticweb") "tempauth staticweb")
pipeline = pipeline.replace( _modify_pipeline_substring(
conf,
"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