diff --git a/swift/proxy/controllers/base.py b/swift/proxy/controllers/base.py index c2e03c8212..23ad79daad 100644 --- a/swift/proxy/controllers/base.py +++ b/swift/proxy/controllers/base.py @@ -495,45 +495,40 @@ class Controller(object): if getattr(source, 'swift_conn', None): self.close_swift_conn(source) - def _make_app_iter(self, node, source, response): + def _make_app_iter(self, node, source): """ Returns an iterator over the contents of the source (via its read func). There is also quite a bit of cleanup to ensure garbage collection works and the underlying socket of the source is closed. - :param response: The webob.Response object this iterator should be - assigned to via response.app_iter. :param source: The httplib.Response object this iterator should read from. :param node: The node the source is reading from, for logging purposes. """ try: - try: - # Spawn reader to read from the source and place in the queue. - # We then drop any reference to the source or node, for garbage - # collection purposes. - queue = Queue(1) - spawn_n(self._make_app_iter_reader, node, source, queue, - self.app.logger.thread_locals) - source = node = None - while True: - chunk = queue.get(timeout=self.app.node_timeout) - if isinstance(chunk, bool): # terminator - success = chunk - if not success: - raise Exception(_('Failed to read all data' - ' from the source')) - break - yield chunk - except Empty: - raise ChunkReadTimeout() - except (GeneratorExit, Timeout): - self.app.logger.warn(_('Client disconnected on read')) - except Exception: - self.app.logger.exception(_('Trying to send to client')) - raise - finally: - response.app_iter = None + # Spawn reader to read from the source and place in the queue. + # We then drop any reference to the source or node, for garbage + # collection purposes. + queue = Queue(1) + spawn_n(self._make_app_iter_reader, node, source, queue, + self.app.logger.thread_locals) + source = node = None + while True: + chunk = queue.get(timeout=self.app.node_timeout) + if isinstance(chunk, bool): # terminator + success = chunk + if not success: + raise Exception(_('Failed to read all data' + ' from the source')) + break + yield chunk + except Empty: + raise ChunkReadTimeout() + except (GeneratorExit, Timeout): + self.app.logger.warn(_('Client disconnected on read')) + except Exception: + self.app.logger.exception(_('Trying to send to client')) + raise def close_swift_conn(self, src): try: @@ -646,7 +641,7 @@ class Controller(object): self.close_swift_conn(src) res = Response(request=req, conditional_response=True) - res.app_iter = self._make_app_iter(node, source, res) + res.app_iter = self._make_app_iter(node, source) # See NOTE: swift_conn at top of file about this. res.swift_conn = source.swift_conn update_headers(res, source.getheaders()) diff --git a/test/unit/proxy/test_server.py b/test/unit/proxy/test_server.py index c2e2d9cf33..e7e82e9e8e 100644 --- a/test/unit/proxy/test_server.py +++ b/test/unit/proxy/test_server.py @@ -59,9 +59,29 @@ import swift.proxy.controllers logging.getLogger().addHandler(logging.StreamHandler(sys.stdout)) +_request_instances = 0 + + +def request_init(self, *args, **kwargs): + global _request_instances + self._orig_init(*args, **kwargs) + _request_instances += 1 + + +def request_del(self): + global _request_instances + if self._orig_del: + self._orig_del() + _request_instances -= 1 + + def setup(): global _testdir, _test_servers, _test_sockets, \ _orig_container_listing_limit, _test_coros + Request._orig_init = Request.__init__ + Request.__init__ = request_init + Request._orig_del = getattr(Request, '__del__', None) + Request.__del__ = request_del monkey_patch_mimetools() # Since we're starting up a lot here, we're going to test more than # just chunked puts; we're also going to test parts of @@ -153,6 +173,9 @@ def teardown(): swift.proxy.controllers.obj.CONTAINER_LISTING_LIMIT = \ _orig_container_listing_limit rmtree(os.path.dirname(_testdir)) + Request.__init__ = Request._orig_init + if Request._orig_del: + Request.__del__ = Request._orig_del def fake_http_connect(*code_iter, **kwargs): @@ -3259,6 +3282,43 @@ class TestObjectController(unittest.TestCase): self.assertEquals(resp.status_int, 400) self.assertTrue('X-Delete-At in past' in resp.body) + def test_leak_1(self): + global _request_instances + prolis = _test_sockets[0] + prosrv = _test_servers[0] + obj_len = prosrv.client_chunk_size * 2 + # PUT test file + sock = connect_tcp(('localhost', prolis.getsockname()[1])) + fd = sock.makefile() + fd.write('PUT /v1/a/c/test_leak_1 HTTP/1.1\r\n' + 'Host: localhost\r\n' + 'Connection: close\r\n' + 'X-Auth-Token: t\r\n' + 'Content-Length: %s\r\n' + 'Content-Type: application/octet-stream\r\n' + '\r\n%s' % (obj_len, 'a' * obj_len)) + fd.flush() + headers = readuntil2crlfs(fd) + exp = 'HTTP/1.1 201' + self.assertEqual(headers[:len(exp)], exp) + # Remember Request instance count + before_request_instances = _request_instances + # GET test file, but disconnect early + sock = connect_tcp(('localhost', prolis.getsockname()[1])) + fd = sock.makefile() + fd.write('GET /v1/a/c/test_leak_1 HTTP/1.1\r\n' + 'Host: localhost\r\n' + 'Connection: close\r\n' + 'X-Auth-Token: t\r\n' + '\r\n') + fd.flush() + headers = readuntil2crlfs(fd) + exp = 'HTTP/1.1 200' + self.assertEqual(headers[:len(exp)], exp) + fd.read(1) + fd.close() + sock.close() + self.assertEquals(before_request_instances, _request_instances) class TestContainerController(unittest.TestCase): "Test swift.proxy_server.ContainerController"