proxy: don't use recoverable_node_timeout with x-newest

Object GET requests with a truthy X-Newest header are not resumed if a
backend request times out. The GetOrHeadHandler therefore uses the
regular node_timeout when waiting for a backend connection response,
rather than the possibly shorter recoverable_node_timeout. However,
previously while reading data from a backend response the
recoverable_node_timeout would still be used with X-Newest requests.

This patch simplifies GetOrHeadHandler to never use
recoverable_node_timeout when X-Newest is truthy.

Change-Id: I326278ecb21465f519b281c9f6c2dedbcbb5ff14
This commit is contained in:
Alistair Coles 2024-02-07 10:57:18 +00:00
parent 8061dfb1c3
commit 2500fbeea9
4 changed files with 15 additions and 12 deletions

View File

@ -175,6 +175,7 @@ use = egg:swift#proxy
# Timeouts from these requests can be recovered from so setting this to
# something lower than node_timeout would provide quicker error recovery
# while allowing for a longer timeout for non-recoverable requests (PUTs).
# Does not apply to requests with a truthy X-Newest header value.
# Defaults to node_timeout, should be overridden if node_timeout is set to a
# high number to prevent client timeouts from firing before the proxy server
# has a chance to retry.

View File

@ -1385,7 +1385,8 @@ class GetOrHeadHandler(GetterBase):
"""
def __init__(self, app, req, server_type, node_iter, partition, path,
backend_headers, concurrency=1, policy=None, logger=None):
if server_type == 'Object':
newest = config_true_value(req.headers.get('x-newest', 'f'))
if server_type == 'Object' and not newest:
node_timeout = app.recoverable_node_timeout
else:
node_timeout = app.node_timeout
@ -1394,6 +1395,7 @@ class GetOrHeadHandler(GetterBase):
policy=policy, path=path, backend_headers=backend_headers,
node_timeout=node_timeout, resource_type=server_type.lower(),
logger=logger)
self.newest = newest
self.server_type = server_type
self.used_nodes = []
self.used_source_etag = None
@ -1403,7 +1405,6 @@ class GetOrHeadHandler(GetterBase):
self.rebalance_missing_suppression_count = min(
policy_options.rebalance_missing_suppression_count,
node_iter.num_primary_nodes - 1)
self.newest = config_true_value(req.headers.get('x-newest', 'f'))
# populated when finding source
self.statuses = []
@ -1530,7 +1531,7 @@ class GetOrHeadHandler(GetterBase):
else:
return None
def _make_node_request(self, node, node_timeout, logger_thread_locals):
def _make_node_request(self, node, logger_thread_locals):
# make a backend request; return True if the response is deemed good
# (has an acceptable status code), useful (matches any previously
# discovered etag) and sufficient (a single good response is
@ -1538,6 +1539,7 @@ class GetOrHeadHandler(GetterBase):
self.logger.thread_locals = logger_thread_locals
if node in self.used_nodes:
return False
req_headers = dict(self.backend_headers)
ip, port = get_ip_port(node, req_headers)
start_node_timing = time.time()
@ -1550,7 +1552,7 @@ class GetOrHeadHandler(GetterBase):
query_string=self.req.query_string)
self.app.set_node_timing(node, time.time() - start_node_timing)
with Timeout(node_timeout):
with Timeout(self.node_timeout):
possible_source = conn.getresponse()
# See NOTE: swift_conn at top of file about this.
possible_source.swift_conn = conn
@ -1642,14 +1644,10 @@ class GetOrHeadHandler(GetterBase):
nodes = GreenthreadSafeIterator(self.node_iter)
node_timeout = self.app.node_timeout
if self.server_type == 'Object' and not self.newest:
node_timeout = self.app.recoverable_node_timeout
pile = GreenAsyncPile(self.concurrency)
for node in nodes:
pile.spawn(self._make_node_request, node, node_timeout,
pile.spawn(self._make_node_request, node,
self.logger.thread_locals)
_timeout = self.app.get_policy_options(
self.policy).concurrency_timeout \

View File

@ -1681,10 +1681,10 @@ class TestGetOrHeadHandler(BaseTest):
# x-newest set
req = Request.blank('/v1/a/c/o', headers={'X-Newest': 'true'})
node_iter = Namespace(num_primary_nodes=3)
# app.recoverable_node_timeout
# app.node_timeout
getter = GetOrHeadHandler(
app, req, 'Object', node_iter, None, None, {})
self.assertEqual(3, getter.node_timeout)
self.assertEqual(5, getter.node_timeout)
# x-newest not set
req = Request.blank('/v1/a/c/o')

View File

@ -1934,7 +1934,11 @@ class TestReplicatedObjController(CommonObjectControllerMixin,
self.assertIn('Trying to read object during GET', line)
def test_GET_newest_will_not_resume(self):
self.app.recoverable_node_timeout = 0.01
# verify that request with x-newest use node_timeout and don't resume
self.app.node_timeout = 0.01
# set recoverable_node_timeout crazy high to verify that this is not
# the timeout value that is used
self.app.recoverable_node_timeout = 1000
self.app.client_timeout = 0.1
self.app.object_chunk_size = 10
resp_body = b'length 8'