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:
parent
8061dfb1c3
commit
2500fbeea9
@ -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.
|
||||
|
@ -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 \
|
||||
|
@ -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')
|
||||
|
@ -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'
|
||||
|
Loading…
x
Reference in New Issue
Block a user