summaryrefslogtreecommitdiffstats
path: root/Lib/asyncore.py
diff options
context:
space:
mode:
Diffstat (limited to 'Lib/asyncore.py')
-rw-r--r--Lib/asyncore.py196
1 files changed, 119 insertions, 77 deletions
diff --git a/Lib/asyncore.py b/Lib/asyncore.py
index e731784..a7a5427 100644
--- a/Lib/asyncore.py
+++ b/Lib/asyncore.py
@@ -1,5 +1,5 @@
# -*- Mode: Python; tab-width: 4 -*-
-# Id: asyncore.py,v 2.40 1999/05/27 04:08:25 rushing Exp
+# Id: asyncore.py,v 2.51 2000/09/07 22:29:26 rushing Exp
# Author: Sam Rushing <rushing@nightmare.com>
# ======================================================================
@@ -46,6 +46,7 @@ many of the difficult problems for you, making the task of building
sophisticated high-performance network servers and clients a snap.
"""
+import exceptions
import select
import socket
import string
@@ -62,70 +63,100 @@ if os.name == 'nt':
else:
from errno import EALREADY, EINPROGRESS, EWOULDBLOCK, ECONNRESET, ENOTCONN, ESHUTDOWN
-socket_map = {}
+try:
+ socket_map
+except NameError:
+ socket_map = {}
-def poll (timeout=0.0):
- if socket_map:
+class ExitNow (exceptions.Exception):
+ pass
+
+DEBUG = 0
+
+def poll (timeout=0.0, map=None):
+ global DEBUG
+ if map is None:
+ map = socket_map
+ if map:
r = []; w = []; e = []
- for s in socket_map.keys():
- if s.readable():
- r.append (s)
- if s.writable():
- w.append (s)
+ for fd, obj in map.items():
+ if obj.readable():
+ r.append (fd)
+ if obj.writable():
+ w.append (fd)
+ r,w,e = select.select (r,w,e, timeout)
- (r,w,e) = select.select (r,w,e, timeout)
+ if DEBUG:
+ print r,w,e
- for x in r:
+ for fd in r:
try:
- x.handle_read_event()
- except:
- x.handle_error()
- for x in w:
+ obj = map[fd]
+ try:
+ obj.handle_read_event()
+ except ExitNow:
+ raise ExitNow
+ except:
+ obj.handle_error()
+ except KeyError:
+ pass
+
+ for fd in w:
try:
- x.handle_write_event()
- except:
- x.handle_error()
+ obj = map[fd]
+ try:
+ obj.handle_write_event()
+ except ExitNow:
+ raise ExitNow
+ except:
+ obj.handle_error()
+ except KeyError:
+ pass
-def poll2 (timeout=0.0):
+def poll2 (timeout=0.0, map=None):
import poll
+ if map is None:
+ map=socket_map
# timeout is in milliseconds
timeout = int(timeout*1000)
- if socket_map:
- fd_map = {}
- for s in socket_map.keys():
- fd_map[s.fileno()] = s
+ if map:
l = []
- for fd, s in fd_map.items():
+ for fd, obj in map.items():
flags = 0
- if s.readable():
+ if obj.readable():
flags = poll.POLLIN
- if s.writable():
+ if obj.writable():
flags = flags | poll.POLLOUT
if flags:
l.append ((fd, flags))
r = poll.poll (l, timeout)
for fd, flags in r:
- s = fd_map[fd]
try:
- if (flags & poll.POLLIN):
- s.handle_read_event()
- if (flags & poll.POLLOUT):
- s.handle_write_event()
- if (flags & poll.POLLERR):
- s.handle_expt_event()
- except:
- s.handle_error()
-
+ obj = map[fd]
+ try:
+ if (flags & poll.POLLIN):
+ obj.handle_read_event()
+ if (flags & poll.POLLOUT):
+ obj.handle_write_event()
+ except ExitNow:
+ raise ExitNow
+ except:
+ obj.handle_error()
+ except KeyError:
+ pass
-def loop (timeout=30.0, use_poll=0):
+def loop (timeout=30.0, use_poll=0, map=None):
if use_poll:
poll_fun = poll2
else:
poll_fun = poll
- while socket_map:
- poll_fun (timeout)
+ if map is None:
+ map=socket_map
+
+ while map:
+ poll_fun (timeout, map)
class dispatcher:
debug = 0
@@ -134,9 +165,9 @@ class dispatcher:
closing = 0
addr = None
- def __init__ (self, sock=None):
+ def __init__ (self, sock=None, map=None):
if sock:
- self.set_socket (sock)
+ self.set_socket (sock, map)
# I think it should inherit this anyway
self.socket.setblocking (0)
self.connected = 1
@@ -163,27 +194,31 @@ class dispatcher:
return '<__repr__ (self) failed for object at %x (addr=%s)>' % (id(self),ar)
- def add_channel (self):
- if __debug__:
- self.log ('adding channel %s' % self)
- socket_map [self] = 1
+ def add_channel (self, map=None):
+ #self.log_info ('adding channel %s' % self)
+ if map is None:
+ map=socket_map
+ map [self._fileno] = self
- def del_channel (self):
- if socket_map.has_key (self):
- if __debug__:
- self.log ('closing channel %d:%s' % (self.fileno(), self))
- del socket_map [self]
+ def del_channel (self, map=None):
+ fd = self._fileno
+ if map is None:
+ map=socket_map
+ if map.has_key (fd):
+ #self.log_info ('closing channel %d:%s' % (fd, self))
+ del map [fd]
def create_socket (self, family, type):
self.family_and_type = family, type
self.socket = socket.socket (family, type)
self.socket.setblocking(0)
+ self._fileno = self.socket.fileno()
self.add_channel()
- def set_socket (self, socket):
- # This is done so we can be called safely from __init__
- self.__dict__['socket'] = socket
- self.add_channel()
+ def set_socket (self, sock, map=None):
+ self.__dict__['socket'] = sock
+ self._fileno = sock.fileno()
+ self.add_channel (map)
def set_reuse_addr (self):
# try to re-use a server port if possible
@@ -284,12 +319,19 @@ class dispatcher:
# cheap inheritance, used to pass all other attribute
# references to the underlying socket object.
- # NOTE: this may be removed soon for performance reasons.
def __getattr__ (self, attr):
return getattr (self.socket, attr)
+ # log and log_info maybe overriden to provide more sophisitcated
+ # logging and warning methods. In general, log is for 'hit' logging
+ # and 'log_info' is for informational, warning and error logging.
+
def log (self, message):
- print 'log:', message
+ sys.stderr.write ('log: %s\n' % str(message))
+
+ def log_info (self, message, type='info'):
+ if __debug__ or type != 'info':
+ print '%s: %s' % (type, message)
def handle_read_event (self):
if self.accepting:
@@ -324,39 +366,34 @@ class dispatcher:
except:
self_repr = '<__repr__ (self) failed for object at %0x>' % id(self)
- print (
+ self.log_info (
'uncaptured python exception, closing channel %s (%s:%s %s)' % (
self_repr,
t,
v,
tbinfo
- )
+ ),
+ 'error'
)
self.close()
def handle_expt (self):
- if __debug__:
- self.log ('unhandled exception')
+ self.log_info ('unhandled exception', 'warning')
def handle_read (self):
- if __debug__:
- self.log ('unhandled read event')
+ self.log_info ('unhandled read event', 'warning')
def handle_write (self):
- if __debug__:
- self.log ('unhandled write event')
+ self.log_info ('unhandled write event', 'warning')
def handle_connect (self):
- if __debug__:
- self.log ('unhandled connect event')
+ self.log_info ('unhandled connect event', 'warning')
def handle_accept (self):
- if __debug__:
- self.log ('unhandled accept event')
+ self.log_info ('unhandled accept event', 'warning')
def handle_close (self):
- if __debug__:
- self.log ('unhandled close event')
+ self.log_info ('unhandled close event', 'warning')
self.close()
# ---------------------------------------------------------------------------
@@ -382,7 +419,7 @@ class dispatcher_with_send (dispatcher):
def send (self, data):
if self.debug:
- self.log ('sending %s' % repr(data))
+ self.log_info ('sending %s' % repr(data))
self.out_buffer = self.out_buffer + data
self.initiate_send()
@@ -396,7 +433,7 @@ def compact_traceback ():
while 1:
tbinfo.append ((
tb.tb_frame.f_code.co_filename,
- tb.tb_frame.f_code.co_name,
+ tb.tb_frame.f_code.co_name,
str(tb.tb_lineno)
))
tb = tb.tb_next
@@ -416,11 +453,12 @@ def compact_traceback ():
) + ']'
return (file, function, line), t, v, info
-def close_all ():
- global socket_map
- for x in socket_map.keys():
+def close_all (map=None):
+ if map is None:
+ map=socket_map
+ for x in map.values():
x.socket.close()
- socket_map.clear()
+ map.clear()
# Asynchronous File I/O:
#
@@ -449,9 +487,12 @@ if os.name == 'posix':
def recv (self, *args):
return apply (os.read, (self.fd,)+args)
- def write (self, *args):
+ def send (self, *args):
return apply (os.write, (self.fd,)+args)
+ read = recv
+ write = send
+
def close (self):
return os.close (self.fd)
@@ -469,6 +510,7 @@ if os.name == 'posix':
self.set_file (fd)
def set_file (self, fd):
+ self._fileno = fd
self.socket = file_wrapper (fd)
self.add_channel()