Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gevent
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
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
gevent
Commits
aee4d51e
Commit
aee4d51e
authored
Feb 08, 2016
by
Jason Madden
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fix #289. Avoid OverflowError in Py2 for large WSGI uploads.
parent
4a3cc9f0
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
68 additions
and
3 deletions
+68
-3
changelog.rst
changelog.rst
+5
-0
gevent/pywsgi.py
gevent/pywsgi.py
+26
-1
greentest/test__pywsgi.py
greentest/test__pywsgi.py
+37
-2
No files found.
changelog.rst
View file @
aee4d51e
...
@@ -10,6 +10,11 @@
...
@@ -10,6 +10,11 @@
- Python 2: Using the blocking API at import time when multiple
- Python 2: Using the blocking API at import time when multiple
greenlets are also importing should not lead to ``LoopExit``.
greenlets are also importing should not lead to ``LoopExit``.
Reported in :issue:`798` by Garrett Heel.
Reported in :issue:`798` by Garrett Heel.
- Python 2: Don't raise ``OverflowError`` when using the ``readline``
method of the WSGI input stream without a size hint or with a large
size hint when the client is uploading a large amount of data.
Reported in :issue:`289` by ggjjlldd, with contributions by Nathan
Hoad.
1.1rc3 (Jan 04, 2016)
1.1rc3 (Jan 04, 2016)
=====================
=====================
...
...
gevent/pywsgi.py
View file @
aee4d51e
...
@@ -145,6 +145,7 @@ class Input(object):
...
@@ -145,6 +145,7 @@ class Input(object):
# Either Content-Length or "Transfer-Encoding: chunked" must be present in a request with a body
# Either Content-Length or "Transfer-Encoding: chunked" must be present in a request with a body
# if it was chunked, then this function would have not been called
# if it was chunked, then this function would have not been called
return
b''
return
b''
self
.
_send_100_continue
()
self
.
_send_100_continue
()
left
=
content_length
-
self
.
position
left
=
content_length
-
self
.
position
if
length
is
None
:
if
length
is
None
:
...
@@ -153,7 +154,31 @@ class Input(object):
...
@@ -153,7 +154,31 @@ class Input(object):
length
=
left
length
=
left
if
not
length
:
if
not
length
:
return
b''
return
b''
# On Python 2, self.rfile is usually socket.makefile(), which
# uses cStringIO.StringIO. If *length* is greater than the C
# sizeof(int) (typically 32 bits), parsing the argument to
# readline raises OverflowError. StringIO.read(), OTOH, uses
# PySize_t, typically a long (64 bits). In a bare readline()
# case, because the header lines we're trying to read with
# readline are typically expected to be small, we can correct
# that failure by simply doing a smaller call to readline and
# appending; failures in read we let propagate.
try
:
read
=
reader
(
length
)
read
=
reader
(
length
)
except
OverflowError
:
if
not
use_readline
:
# Expecting to read more than 64 bits of data. Ouch!
raise
# We could loop on calls to smaller readline(), appending them
# until we actually get a newline. For uses in this module,
# we expect the actual length to be small, but WSGI applications
# are allowed to pass in an arbitrary length. (This loop isn't optimal,
# but even client applications *probably* have short lines.)
read
=
b''
while
len
(
read
)
<
length
and
not
read
.
endswith
(
b'
\
n
'
):
read
+=
reader
(
MAX_REQUEST_LINE
)
self
.
position
+=
len
(
read
)
self
.
position
+=
len
(
read
)
if
len
(
read
)
<
length
:
if
len
(
read
)
<
length
:
if
(
use_readline
and
not
read
.
endswith
(
b"
\
n
"
))
or
not
use_readline
:
if
(
use_readline
and
not
read
.
endswith
(
b"
\
n
"
))
or
not
use_readline
:
...
...
greentest/test__pywsgi.py
View file @
aee4d51e
...
@@ -29,7 +29,11 @@ except ImportError:
...
@@ -29,7 +29,11 @@ except ImportError:
import
os
import
os
import
sys
import
sys
try
:
try
:
from
StringIO
import
StringIO
# On Python 2, we want the C-optimized version if
# available; it has different corner-case behaviour than
# the Python implementation, and it used by socket.makefile
# by default.
from
cStringIO
import
StringIO
except
ImportError
:
except
ImportError
:
from
io
import
BytesIO
as
StringIO
from
io
import
BytesIO
as
StringIO
import
weakref
import
weakref
...
@@ -42,7 +46,7 @@ except ImportError:
...
@@ -42,7 +46,7 @@ except ImportError:
import
greentest
import
greentest
import
gevent
import
gevent
from
gevent.hub
import
PY3
from
gevent.hub
import
PY3
,
PYPY
from
gevent
import
socket
from
gevent
import
socket
from
gevent
import
pywsgi
from
gevent
import
pywsgi
from
gevent.pywsgi
import
Input
from
gevent.pywsgi
import
Input
...
@@ -1432,6 +1436,37 @@ class TestInputRaw(greentest.BaseTestCase):
...
@@ -1432,6 +1436,37 @@ class TestInputRaw(greentest.BaseTestCase):
i
=
self
.
make_input
(
"2
\
r
\
n
1"
,
chunked_input
=
True
)
i
=
self
.
make_input
(
"2
\
r
\
n
1"
,
chunked_input
=
True
)
self
.
assertRaises
(
IOError
,
i
.
readline
)
self
.
assertRaises
(
IOError
,
i
.
readline
)
def
test_32bit_overflow
(
self
):
# https://github.com/gevent/gevent/issues/289
# Should not raise an OverflowError on Python 2
data
=
b'asdf
\
n
ghij
\
n
'
long_data
=
b'a'
*
(
pywsgi
.
MAX_REQUEST_LINE
+
10
)
long_data
+=
b'
\
n
'
data
=
data
+
long_data
partial_data
=
b'qjk
\
n
'
# Note terminating \n
n
=
25
*
1000000000
if
hasattr
(
n
,
'bit_length'
):
self
.
assertEqual
(
n
.
bit_length
(),
35
)
if
not
PY3
and
not
PYPY
:
# Make sure we have the impl we think we do
self
.
assertRaises
(
OverflowError
,
StringIO
(
data
).
readline
,
n
)
i
=
self
.
make_input
(
data
,
content_length
=
n
)
# No size hint, but we have too large a content_length to fit
self
.
assertEqual
(
i
.
readline
(),
b'asdf
\
n
'
)
# Large size hint
self
.
assertEqual
(
i
.
readline
(
n
),
b'ghij
\
n
'
)
self
.
assertEqual
(
i
.
readline
(
n
),
long_data
)
# Now again with the real content length, assuring we can't read past it
i
=
self
.
make_input
(
data
+
partial_data
,
content_length
=
len
(
data
)
+
1
)
self
.
assertEqual
(
i
.
readline
(),
b'asdf
\
n
'
)
self
.
assertEqual
(
i
.
readline
(
n
),
b'ghij
\
n
'
)
self
.
assertEqual
(
i
.
readline
(
n
),
long_data
)
# Now we've reached content_length so we shouldn't be able to
# read anymore except the one byte remaining
self
.
assertEqual
(
i
.
readline
(
n
),
b'q'
)
class
Test414
(
TestCase
):
class
Test414
(
TestCase
):
...
...
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