Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
cpython
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
cpython
Commits
b6c86fd8
Commit
b6c86fd8
authored
Feb 06, 2013
by
Serhiy Storchaka
Browse files
Options
Browse Files
Download
Plain Diff
Issue #16723: httplib.HTTPResponse no longer marked closed when the connection
is automatically closed.
parents
e201e9d5
b5b9c8cd
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
48 additions
and
17 deletions
+48
-17
Lib/http/client.py
Lib/http/client.py
+21
-17
Lib/test/test_httplib.py
Lib/test/test_httplib.py
+24
-0
Misc/NEWS
Misc/NEWS
+3
-0
No files found.
Lib/http/client.py
View file @
b6c86fd8
...
@@ -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
...
...
Lib/test/test_httplib.py
View file @
b6c86fd8
...
@@ -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
\
n
Text"
body
=
"HTTP/1.1 400.100 Not Ok
\
r
\
n
\
r
\
n
Text"
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
):
...
...
Misc/NEWS
View file @
b6c86fd8
...
@@ -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.
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment