Commit b6c86fd8 authored by Serhiy Storchaka's avatar Serhiy Storchaka

Issue #16723: httplib.HTTPResponse no longer marked closed when the connection

is automatically closed.
parents e201e9d5 b5b9c8cd
...@@ -332,7 +332,7 @@ class HTTPResponse(io.RawIOBase): ...@@ -332,7 +332,7 @@ class HTTPResponse(io.RawIOBase):
# empty version will cause next test to fail. # empty version will cause next test to fail.
version = "" version = ""
if not version.startswith("HTTP/"): if not version.startswith("HTTP/"):
self.close() self._close_conn()
raise BadStatusLine(line) raise BadStatusLine(line)
# The status code is a three-digit number # The status code is a three-digit number
...@@ -454,22 +454,25 @@ class HTTPResponse(io.RawIOBase): ...@@ -454,22 +454,25 @@ class HTTPResponse(io.RawIOBase):
# otherwise, assume it will close # otherwise, assume it will close
return True return True
def _close_conn(self):
fp = self.fp
self.fp = None
fp.close()
def close(self): def close(self):
super().close() # set "closed" flag
if self.fp: if self.fp:
self.fp.close() self._close_conn()
self.fp = None
# These implementations are for the benefit of io.BufferedReader. # These implementations are for the benefit of io.BufferedReader.
# XXX This class should probably be revised to act more like # XXX This class should probably be revised to act more like
# the "raw stream" that BufferedReader expects. # the "raw stream" that BufferedReader expects.
@property
def closed(self):
return self.isclosed()
def flush(self): def flush(self):
self.fp.flush() super().flush()
if self.fp:
self.fp.flush()
def readable(self): def readable(self):
return True return True
...@@ -477,6 +480,7 @@ class HTTPResponse(io.RawIOBase): ...@@ -477,6 +480,7 @@ class HTTPResponse(io.RawIOBase):
# End of "raw stream" methods # End of "raw stream" methods
def isclosed(self): def isclosed(self):
"""True if the connection is closed."""
# NOTE: it is possible that we will not ever call self.close(). This # NOTE: it is possible that we will not ever call self.close(). This
# case occurs when will_close is TRUE, length is None, and we # case occurs when will_close is TRUE, length is None, and we
# read up to the last byte, but NOT past it. # read up to the last byte, but NOT past it.
...@@ -490,7 +494,7 @@ class HTTPResponse(io.RawIOBase): ...@@ -490,7 +494,7 @@ class HTTPResponse(io.RawIOBase):
return b"" return b""
if self._method == "HEAD": if self._method == "HEAD":
self.close() self._close_conn()
return b"" return b""
if amt is not None: if amt is not None:
...@@ -510,10 +514,10 @@ class HTTPResponse(io.RawIOBase): ...@@ -510,10 +514,10 @@ class HTTPResponse(io.RawIOBase):
try: try:
s = self._safe_read(self.length) s = self._safe_read(self.length)
except IncompleteRead: except IncompleteRead:
self.close() self._close_conn()
raise raise
self.length = 0 self.length = 0
self.close() # we read everything self._close_conn() # we read everything
return s return s
def readinto(self, b): def readinto(self, b):
...@@ -521,7 +525,7 @@ class HTTPResponse(io.RawIOBase): ...@@ -521,7 +525,7 @@ class HTTPResponse(io.RawIOBase):
return 0 return 0
if self._method == "HEAD": if self._method == "HEAD":
self.close() self._close_conn()
return 0 return 0
if self.chunked: if self.chunked:
...@@ -539,11 +543,11 @@ class HTTPResponse(io.RawIOBase): ...@@ -539,11 +543,11 @@ class HTTPResponse(io.RawIOBase):
if not n: if not n:
# Ideally, we would raise IncompleteRead if the content-length # Ideally, we would raise IncompleteRead if the content-length
# wasn't satisfied, but it might break compatibility. # wasn't satisfied, but it might break compatibility.
self.close() self._close_conn()
elif self.length is not None: elif self.length is not None:
self.length -= n self.length -= n
if not self.length: if not self.length:
self.close() self._close_conn()
return n return n
def _read_next_chunk_size(self): def _read_next_chunk_size(self):
...@@ -559,7 +563,7 @@ class HTTPResponse(io.RawIOBase): ...@@ -559,7 +563,7 @@ class HTTPResponse(io.RawIOBase):
except ValueError: except ValueError:
# close the connection as protocol synchronisation is # close the connection as protocol synchronisation is
# probably lost # probably lost
self.close() self._close_conn()
raise raise
def _read_and_discard_trailer(self): def _read_and_discard_trailer(self):
...@@ -597,7 +601,7 @@ class HTTPResponse(io.RawIOBase): ...@@ -597,7 +601,7 @@ class HTTPResponse(io.RawIOBase):
self._read_and_discard_trailer() self._read_and_discard_trailer()
# we read everything; close the "file" # we read everything; close the "file"
self.close() self._close_conn()
return b''.join(value) return b''.join(value)
...@@ -638,7 +642,7 @@ class HTTPResponse(io.RawIOBase): ...@@ -638,7 +642,7 @@ class HTTPResponse(io.RawIOBase):
self._read_and_discard_trailer() self._read_and_discard_trailer()
# we read everything; close the "file" # we read everything; close the "file"
self.close() self._close_conn()
return total_bytes return total_bytes
......
...@@ -164,6 +164,9 @@ class BasicTest(TestCase): ...@@ -164,6 +164,9 @@ class BasicTest(TestCase):
resp.begin() resp.begin()
self.assertEqual(resp.read(), b"Text") self.assertEqual(resp.read(), b"Text")
self.assertTrue(resp.isclosed()) self.assertTrue(resp.isclosed())
self.assertFalse(resp.closed)
resp.close()
self.assertTrue(resp.closed)
body = "HTTP/1.1 400.100 Not Ok\r\n\r\nText" body = "HTTP/1.1 400.100 Not Ok\r\n\r\nText"
sock = FakeSocket(body) sock = FakeSocket(body)
...@@ -185,6 +188,9 @@ class BasicTest(TestCase): ...@@ -185,6 +188,9 @@ class BasicTest(TestCase):
self.assertFalse(resp.isclosed()) self.assertFalse(resp.isclosed())
self.assertEqual(resp.read(2), b'xt') self.assertEqual(resp.read(2), b'xt')
self.assertTrue(resp.isclosed()) self.assertTrue(resp.isclosed())
self.assertFalse(resp.closed)
resp.close()
self.assertTrue(resp.closed)
def test_partial_readintos(self): def test_partial_readintos(self):
# if we have a length, the system knows when to close itself # if we have a length, the system knows when to close itself
...@@ -202,6 +208,9 @@ class BasicTest(TestCase): ...@@ -202,6 +208,9 @@ class BasicTest(TestCase):
self.assertEqual(n, 2) self.assertEqual(n, 2)
self.assertEqual(bytes(b), b'xt') self.assertEqual(bytes(b), b'xt')
self.assertTrue(resp.isclosed()) self.assertTrue(resp.isclosed())
self.assertFalse(resp.closed)
resp.close()
self.assertTrue(resp.closed)
def test_partial_reads_no_content_length(self): def test_partial_reads_no_content_length(self):
# when no length is present, the socket should be gracefully closed when # when no length is present, the socket should be gracefully closed when
...@@ -215,6 +224,9 @@ class BasicTest(TestCase): ...@@ -215,6 +224,9 @@ class BasicTest(TestCase):
self.assertEqual(resp.read(2), b'xt') self.assertEqual(resp.read(2), b'xt')
self.assertEqual(resp.read(1), b'') self.assertEqual(resp.read(1), b'')
self.assertTrue(resp.isclosed()) self.assertTrue(resp.isclosed())
self.assertFalse(resp.closed)
resp.close()
self.assertTrue(resp.closed)
def test_partial_readintos_no_content_length(self): def test_partial_readintos_no_content_length(self):
# when no length is present, the socket should be gracefully closed when # when no length is present, the socket should be gracefully closed when
...@@ -266,6 +278,9 @@ class BasicTest(TestCase): ...@@ -266,6 +278,9 @@ class BasicTest(TestCase):
n = resp.readinto(b) n = resp.readinto(b)
self.assertEqual(n, 0) self.assertEqual(n, 0)
self.assertTrue(resp.isclosed()) self.assertTrue(resp.isclosed())
self.assertFalse(resp.closed)
resp.close()
self.assertTrue(resp.closed)
def test_host_port(self): def test_host_port(self):
# Check invalid host_port # Check invalid host_port
...@@ -493,6 +508,9 @@ class BasicTest(TestCase): ...@@ -493,6 +508,9 @@ class BasicTest(TestCase):
self.assertEqual(resp.status, 200) self.assertEqual(resp.status, 200)
self.assertEqual(resp.reason, 'OK') self.assertEqual(resp.reason, 'OK')
self.assertTrue(resp.isclosed()) self.assertTrue(resp.isclosed())
self.assertFalse(resp.closed)
resp.close()
self.assertTrue(resp.closed)
def test_readinto_chunked_head(self): def test_readinto_chunked_head(self):
chunked_start = ( chunked_start = (
...@@ -513,6 +531,9 @@ class BasicTest(TestCase): ...@@ -513,6 +531,9 @@ class BasicTest(TestCase):
self.assertEqual(resp.status, 200) self.assertEqual(resp.status, 200)
self.assertEqual(resp.reason, 'OK') self.assertEqual(resp.reason, 'OK')
self.assertTrue(resp.isclosed()) self.assertTrue(resp.isclosed())
self.assertFalse(resp.closed)
resp.close()
self.assertTrue(resp.closed)
def test_negative_content_length(self): def test_negative_content_length(self):
sock = FakeSocket( sock = FakeSocket(
...@@ -588,6 +609,9 @@ class BasicTest(TestCase): ...@@ -588,6 +609,9 @@ class BasicTest(TestCase):
resp.begin() resp.begin()
self.assertEqual(resp.read(), b'') self.assertEqual(resp.read(), b'')
self.assertTrue(resp.isclosed()) self.assertTrue(resp.isclosed())
self.assertFalse(resp.closed)
resp.close()
self.assertTrue(resp.closed)
class OfflineTest(TestCase): class OfflineTest(TestCase):
def test_responses(self): def test_responses(self):
......
...@@ -163,6 +163,9 @@ Core and Builtins ...@@ -163,6 +163,9 @@ Core and Builtins
Library Library
------- -------
- Issue #16723: httplib.HTTPResponse no longer marked closed when the connection
is automatically closed.
- Issue #16948: Fix quoted printable body encoding for non-latin1 character - Issue #16948: Fix quoted printable body encoding for non-latin1 character
sets in the email package. sets in the email package.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment