[Python-bugs-list] [ python-Bugs-417845 ] Python 2.1: SocketServer.ThreadingMixIn
noreply@sourceforge.net
noreply@sourceforge.net
Wed, 18 Jul 2001 06:56:31 -0700
Bugs item #417845, was opened at 2001-04-21 08:28
You can respond by visiting:
http://sourceforge.net/tracker/?func=detail&atid=105470&aid=417845&group_id=5470
Category: Python Library
Group: None
Status: Open
Resolution: Fixed
Priority: 3
Submitted By: Nobody/Anonymous (nobody)
Assigned to: Guido van Rossum (gvanrossum)
Summary: Python 2.1: SocketServer.ThreadingMixIn
Initial Comment:
SocketServer.ThreadingMixIn does not work properly
since it tries to close the socket of a request two
times.
Workaround for using SocketServer.ThreadingMixIn under
Python 2.1:
class MyThreadingHTTPServer(
SocketServer.ThreadingMixIn,
MyHTTPServer
):
def close_request(self, request):
pass
----------------------------------------------------------------------
>Comment By: Guido van Rossum (gvanrossum)
Date: 2001-07-18 06:56
Message:
Logged In: YES
user_id=6380
You can close a socket more than once, so it's easy to write
code that works under either assumption.
Since calling a different method is a feature change, we
really can't make this change in 2.1.1.
----------------------------------------------------------------------
Comment By: Stian Soiland (stain)
Date: 2001-07-18 06:36
Message:
Logged In: YES
user_id=25921
By waiting for 2.2 for this simple patch even more future
code would break. The programmer would need to write
seperate code for 2.0, 2.1 and 2.2. Why should we want
that?
It should not take much time reviewing these 6 lines
of change. Anyone else care to comment?
(This message quoted in posting on comp.lang.python)
----------------------------------------------------------------------
Comment By: Guido van Rossum (gvanrossum)
Date: 2001-07-17 07:30
Message:
Logged In: YES
user_id=6380
I have absolutely no time to review this right now.
This won't be fixed in 2.1.x; I'll look into making that
change for 2.2 at some point.
----------------------------------------------------------------------
Comment By: Stian Soiland (stain)
Date: 2001-07-13 14:51
Message:
Logged In: YES
user_id=25921
I don't think it's a good idea to let closing the request be
the handler's responsibility. It's better to let the ThreadingMixIn
do it:
stain@zoidberg:~$ diff -u SocketServer-cvs-1.24.2.1.py SocketServer.py
--- SocketServer-cvs-1.24.2.1.py Fri Jul 13 23:20:26 2001
+++ SocketServer.py Fri Jul 13 23:34:02 2001
@@ -451,8 +451,10 @@
def process_request(self, request, client_address):
"""Start a new thread to process the request."""
import threading
- t = threading.Thread(target = self.finish_request,
- args = (request, client_address))
+ # Call the BaseServers process_request to close the
+ # socket after finally_request.
+ t = threading.Thread(target = BaseServer.process_request,
+ args = (self, request, client_address))
t.start()
In my eyes this is the best, making the overriden process_request
behaving like the original process_request. This would make
the difference between Threading-servers and other servers
smaller.
If it looks bad with BaseServer-references inside
ThreadingMixIn, what about a method of ThreadingMixIn
named __process_request with the same code as
BaseServer.process_request?
--
Stian Soiland
----------------------------------------------------------------------
Comment By: Gustavo Niemeyer (niemeyer)
Date: 2001-07-11 14:45
Message:
Logged In: YES
user_id=7887
Ohh, I'm sorry. I've read comments the other way around. The
last comment seemed to be the one at bottom (as usual), then
it looked like it was not definetly fixed. I'll checkout the
fix from cvs.
Thanks!
----------------------------------------------------------------------
Comment By: Guido van Rossum (gvanrossum)
Date: 2001-07-11 14:22
Message:
Logged In: YES
user_id=6380
Gustavo, yes it's fixed in CVS (just read on and you;ll see
my announcement). We're going to release a bugfix update
release for Python 2.1, Python 2.1.1 soon, and it will be in
that release too.
----------------------------------------------------------------------
Comment By: Gustavo Niemeyer (niemeyer)
Date: 2001-07-11 14:19
Message:
Logged In: YES
user_id=7887
Just wondering, is this bug fixed somewhere (cvs, etc)??
Python 2.1 + SOAP.py seem to have this problem, when using
the ThreadingTCPServer.
----------------------------------------------------------------------
Comment By: Guido van Rossum (gvanrossum)
Date: 2001-07-10 06:59
Message:
Logged In: YES
user_id=6380
Yes, the thread is responsible for calling
self.close_request() in self.finish_request(). (Although if
you just rely on reference counting, it should go away all
by itself too.)
I don't read the newsgroup frequently, but you can try email
if you have more problems.
----------------------------------------------------------------------
Comment By: Nobody/Anonymous (nobody)
Date: 2001-07-10 05:47
Message:
Logged In: NO
I tried the fix and it seems to work. I'm wondering if the
thread is responsible for calling self.close_request() in
self.finish_request?
Note:
Can we discuss it in news:comp.lang.python? I hate that web
interface with Cookies! That's the reason why I was too lazy
to login when submitting this bug report.
----------------------------------------------------------------------
Comment By: Guido van Rossum (gvanrossum)
Date: 2001-07-10 04:59
Message:
Logged In: YES
user_id=6380
I've developed a proper fix for this, I hope. Check out the
latest SocketServer.py from the CVS tree.
----------------------------------------------------------------------
Comment By: Xavier Lagraula (xlagraula)
Date: 2001-07-02 09:05
Message:
Logged In: YES
user_id=198402
There is still a problem with your solution, Mr glchapman:
you make a copy to ensure that the server won't close the
copy of the socket that is used in the child thread. Right.
But WHO will close this very copy then?
The thread still has to close it if one wants to
be "clean". So as long as the child thread must close its
copy in the threading case, why not make it a thumb rule?
Why should we bother about closing anything in the main
thread? I still think the solution that I provide is best:
make the request handler always responsible for closing the
socket from which the request comes.
----------------------------------------------------------------------
Comment By: Greg Chapman (glchapman)
Date: 2001-06-29 08:49
Message:
Logged In: YES
user_id=86307
Since the request socket object is only a lightweight
wrapper around the real socket, why not simply pass
request.dup() to the new thread? Then the server's call to
close_request affects only its copy of request, not the
copy being used in the thread. For example, the following
change to ThreadingMixIn fixed this bug for me in a (very
simple) test program:
def process_request(self, request, client_address):
"""Start a new thread to process the request."""
import threading
t = threading.Thread(target = self.finish_request,
args = (request.dup(),
client_address))
t.start()
----------------------------------------------------------------------
Comment By: Xavier Lagraula (xlagraula)
Date: 2001-05-28 05:20
Message:
Logged In: YES
user_id=198402
I have now started a project here concerning a SOCKS proxy
written in python (PySocks). It is aimed mostly at people
who use a windows box to share their internet connection
and uses the threading server from the SocketServer module.
So it becomes VERY important to me to know of what will be
done about this bug, in the next release/patch of Python
library.
Could Mr Guido Vanrossum tell us about it? As for now I am
forced to provide my patched version of SocketServer.py
with my releases, what is not quite satisfactory.
SocketServer is provided in the python distribution, so I'd
rather tell "there is a patch for python..."
Well... In fact I forgot to put it in my first release, but
I'll correct this this evening :)
----------------------------------------------------------------------
Comment By: Xavier Lagraula (xlagraula)
Date: 2001-05-20 12:06
Message:
Logged In: YES
user_id=198402
Well I was wrong. We do need a "try" block to ensure the
request is always correctly closed:
def finish_request(self, request, client_address):
"""Finish one request by instantiating
RequestHandlerClass."""
try:
self.RequestHandlerClass(request,
client_address, self)
finally:
self.close_request(request)
This works better.
----------------------------------------------------------------------
Comment By: Xavier Lagraula (xlagraula)
Date: 2001-05-13 09:09
Message:
Logged In: YES
user_id=198402
I forgot to tell: I can not test if it does not break the
forking server. I only have a windows platform available
for now, and forking doesn't work in the python/win32
environment for now as far as I know.
----------------------------------------------------------------------
Comment By: Xavier Lagraula (xlagraula)
Date: 2001-05-13 08:51
Message:
Logged In: YES
user_id=198402
What I propose can be applied without any compatibility
issue.
I have tried something that seems to work, at least under
windows (but it does need to be more fully tested though).
Only 2 small modifications are required:
-1- In BaseServer, modification of the last line of
handle_request:
<PRE>
def handle_request(self):
"""Handle one request, possibly blocking."""
#import time
try:
request, client_address = self.get_request()
except socket.error:
return
if self.verify_request(request, client_address):
try:
print 'handle 1'
self.process_request(request,
client_address)
print 'handle 2'
except:
self.handle_error(request, client_address)
self.close_request(request)
</PRE>
Note that only the indentation of the last has been
modified so that the close_request is executed only if an
exception occur.
Still we need to close the request after it had been
processed, so here comes the second modification:
-2- Still in BaseServer:
<PRE>
def finish_request(self, request, client_address):
"""Finish one request by instantiating
RequestHandlerClass."""
self.RequestHandlerClass(request, client_address,
self)
self.close_request(request)
</PRE>
There is already a try/except block in handle_request, so I
thought it was not mandatory here to ensure the request was
always closed.
Oh... I don't know how <PRE> tags are supported by
the bugtracking system of sourceforge, so the samples I
give may not appear as I want. This is quite a problem with
python :/
----------------------------------------------------------------------
Comment By: Luke Kenneth Casson Leighton (lkcl)
Date: 2001-05-12 15:46
Message:
Logged In: YES
user_id=80200
hi there mr xlagraula,
yes, it would be a lot simpler... _if_ it wasn't
for the fact that this code is likely to already
be quite extensively used. a possible 'upgrade'
path could be done by providing a... RequestHandler2
class, or some-such.
it would be neater to do this, or similar:
for t in self.thread_list:
t.join(timeout=0.1)
which would join all threads, or you do if
stopped(), close_request. i looked into
threads a bit more: join has a timeout,
and there is a stopped-detection function.
easy :)
----------------------------------------------------------------------
Comment By: Xavier Lagraula (xlagraula)
Date: 2001-05-12 14:18
Message:
Logged In: YES
user_id=198402
Another solution could be to modify the behaviour of the
server so that it would be the responsibility of
the "child" thread/process to close the socket (except for
the forking/threading error cases). Wouldn't it be simplier
than child process tracking and thread tracking?
----------------------------------------------------------------------
Comment By: Luke Kenneth Casson Leighton (lkcl)
Date: 2001-05-04 04:17
Message:
Logged In: YES
user_id=80200
okay. the forkingmixin code does a fork, records how many
children there are, and waits for one of them to exit,
before
proceeding - in particular, before proceeding to close
the request, etc.
... so why is not something similar done in ThreadingMixIn?
this kinda- tells me that thread-tracking is really needed,
in a similar way to that in forkingmixin.
----------------------------------------------------------------------
Comment By: Gregory P. Smith (greg)
Date: 2001-05-03 16:26
Message:
Logged In: YES
user_id=413
Just a note of another casualty of this bug: I had to add
the mentioned dummy close_request method hack to our own
ThreadingMixIn class in mojo nation (in the sourceforge
mojonation project's evil module, see the
common/MojoNationHTTPServer.py file). Without it, python
2.1 would always raise an exception in the request handler
as soon as it tried to call self.connection.makefile()
because self.connection had apparently already been closed!
(its fd was always -1)
----------------------------------------------------------------------
Comment By: Luke Kenneth Casson Leighton (lkcl)
Date: 2001-05-02 01:41
Message:
Logged In: YES
user_id=80200
hi there mr jrielh,
thank you very much for the details. what i am having a
little difficulty with is, what's the difference between
this and python 2.0 SocketServer.py?
more specifically, i'm looking at python 2.0 SocketServer.py
and, whilst i'm not a Threads expert, i see a t.start() but
no t.join(). i've been looking at the Queue example code in
the test method of threads.py, and start() is called on
every thread, followed by join() on every thread. join
waits for the thread to finish, yes?
so... if that's the case, then python 2.0 SocketServer.py
should suffer from
exactly the same behaviour, yes? unless python behaves
ever-so-slightly differently (timing issues?) when you have
an extra base class like this,
with the consequence that close_request() is more likely to
be called
before ThreadingMixIn.process_request().
?
----------------------------------------------------------------------
Comment By: Jon Riehl (jriehl)
Date: 2001-05-01 14:20
Message:
Logged In: YES
user_id=22448
This is related to bug #419873.
The problem is not specifically in the ThreadingMixin
specifically, but where BaseServer calls close_request()
after calling process_request(). In the threading mixin,
process_request() spins the thread and returns, causing the
request socket to be invalidated while the thread is still
running. The fix given above will keep the socket valid
while the thread is running, but may cause the socket to not
close properly (my threads generally close the socket when
they are done anyway.)
----------------------------------------------------------------------
Comment By: Luke Kenneth Casson Leighton (lkcl)
Date: 2001-04-26 05:41
Message:
Logged In: YES
user_id=80200
follow-up. i took a look at the differences between
SocketServer.py
in 2.0 and 2.1. there is one small change by guido to the
ThreadingMixIn.process_request() function that calls
self.server_close() instead of explicitly calling
self.socket.close(),
where TCPServer.server_close() calls self.socket.close().
if mr anonymous (hi!) has over-ridden server_close() and
explicitly closes the *request* socket, then of course the
socket will get closed twice.
the rest of the code-mods is a straightforward code-shuffle
moving code from TCPServer into BaseServer: from
examining the diff, i really don't see how bypassing
close_request(), as shown above with the Workaround
in the original bug-report, will help: that will in fact
cause the request _never_ to be closed!
the rest of this report is part of an email exchange with
guido, quoted here:
"the bug-report doesn't state whether python 2.0 worked and
2.1 didn't:
it also doesn't give enough info.
for all we know, he's calling close_request() himself or
request.close()
directly somewhere in his code, and hasn't told anybody,
which
is why he has to over-ride close_request() and tell it to do
nothing.
or he's closing the socket in the HandlerClass, in finish(),
or
something.
we just don't know.
either that, or his HandlerClass creates a socket once and
only
once, with the result that close_request() closes the one
socket, and he's _completely_ stuffed, then :)"
----------------------------------------------------------------------
Comment By: Luke Kenneth Casson Leighton (lkcl)
Date: 2001-04-26 04:11
Message:
Logged In: YES
user_id=80200
hi there,
i'm the person who wrote the BaseServer class. guido
contacted me about it: could you please send me or post here
a working test example that demonstrates the problem.
i assume, but you do not state, that you have tested your
MyHTTPServer with python 2.0, please let us know, here, if
that is a correct assumption.
thanks!
luke
----------------------------------------------------------------------
You can respond by visiting:
http://sourceforge.net/tracker/?func=detail&atid=105470&aid=417845&group_id=5470