Commit 8f579d78 authored by Jason Madden's avatar Jason Madden

Document the way BaseServer closes the client socket. See #594. [skip ci]

parent 88663a4a
...@@ -62,6 +62,16 @@ Unreleased ...@@ -62,6 +62,16 @@ Unreleased
- ``gevent.pool.Group.imap`` and ``imap_unordered`` now accept - ``gevent.pool.Group.imap`` and ``imap_unordered`` now accept
multiple iterables like ``itertools.imap``. Issue #565 reported by multiple iterables like ``itertools.imap``. Issue #565 reported by
Thomas Steinacher. Thomas Steinacher.
- Potentially breaking change: ``gevent.baseserver.BaseServer`` and
its subclass ``gevent.server.StreamServer`` now deterministically
close the client socket when the request handler returns.
Previously, the socket was left at the mercies of the garbage
collector; under CPython 2.x this meant when the last reference went
away, which was usually, but not necessarily, when the request
handler returned, but under PyPy it was some arbitrary point in the
future and under CPython 3.x a ResourceWarning could be generated.
This was undocumented behaviour, and the client socket could be kept
open after the request handler returned either accidentally or intentionally.
Release 1.0.2 Release 1.0.2
============= =============
......
...@@ -7,6 +7,7 @@ API reference ...@@ -7,6 +7,7 @@ API reference
networking networking
synchronization synchronization
gevent.pool gevent.pool
gevent.threadpool
servers servers
gevent.local gevent.local
gevent.monkey gevent.monkey
......
...@@ -51,8 +51,7 @@ More examples are available in the `code repository`_: ...@@ -51,8 +51,7 @@ More examples are available in the `code repository`_:
.. toctree:: .. toctree::
gevent.baseserver
gevent.server gevent.server
gevent.pywsgi gevent.pywsgi
gevent.wsgi gevent.wsgi
...@@ -12,23 +12,39 @@ __all__ = ['BaseServer'] ...@@ -12,23 +12,39 @@ __all__ = ['BaseServer']
class BaseServer(object): class BaseServer(object):
"""An abstract base class that implements some common functionality for the servers in gevent. """
An abstract base class that implements some common functionality for the servers in gevent.
*listener* can either be an address that the server should bind on or a :class:`gevent.socket.socket`
instance that is already bound (and put into listening mode in case of TCP socket). :param listener: Either be an address that the server should bind
on or a :class:`gevent.socket.socket` instance that is already
*spawn*, if provided, is called to create a new greenlet to run the handler. By default, :func:`gevent.spawn` is used. bound (and put into listening mode in case of TCP socket).
Possible values for *spawn*: :keyword handle: If given, the request handler. The request
handler can be defined in a few ways. Most commonly,
* a :class:`gevent.pool.Pool` instance -- *handle* will be executed subclasses will implement a ``handle`` method as an
using :meth:`Pool.spawn` method only if the pool is not full. instance method. Alternatively, a function can be passed
as the ``handle`` argument to the constructor. In either
case, the handler can later be changed by calling
:meth:`set_handle`.
When the request handler returns, the socket used for the
request will be closed. (New in gevent 1.1a1.)
:keyword spawn: If provided, is called to create a new
greenlet to run the handler. By default,
:func:`gevent.spawn` is used (meaning there is no
artificial limit on the number of concurrent requests). Possible values for *spawn*:
- a :class:`gevent.pool.Pool` instance -- ``handle`` will be executed
using :meth:`gevent.pool.Pool.spawn` only if the pool is not full.
While it is full, all the connection are dropped; While it is full, all the connection are dropped;
* :func:`gevent.spawn_raw` -- *handle* will be executed in a raw - :func:`gevent.spawn_raw` -- ``handle`` will be executed in a raw
greenlet which have a little less overhead then :class:`gevent.Greenlet` instances spawned by default; greenlet which have a little less overhead then :class:`gevent.Greenlet` instances spawned by default;
* ``None`` -- *handle* will be executed right away, in the :class:`Hub` greenlet. - ``None`` -- ``handle`` will be executed right away, in the :class:`Hub` greenlet.
*handle* cannot use any blocking functions as it means switching to the :class:`Hub`. ``handle`` cannot use any blocking functions as it means switching to the :class:`Hub`.
* an integer -- a shortcut for ``gevent.pool.Pool(integer)`` - an integer -- a shortcut for ``gevent.pool.Pool(integer)``
""" """
# the number of seconds to sleep in case there was an error in accept() call # the number of seconds to sleep in case there was an error in accept() call
# for consecutive errors the delay will double until it reaches max_delay # for consecutive errors the delay will double until it reaches max_delay
......
...@@ -41,6 +41,8 @@ class StreamServer(BaseServer): ...@@ -41,6 +41,8 @@ class StreamServer(BaseServer):
The delay starts with :attr:`min_delay` and doubles with each successive error until it reaches :attr:`max_delay`. The delay starts with :attr:`min_delay` and doubles with each successive error until it reaches :attr:`max_delay`.
A successful :func:`accept` resets the delay to :attr:`min_delay` again. A successful :func:`accept` resets the delay to :attr:`min_delay` again.
See :class:`BaseServer` for information on defining the *handle* function and important restrictions on it.
""" """
# the default backlog to use if none was provided in __init__ # the default backlog to use if none was provided in __init__
backlog = 256 backlog = 256
......
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