asynchat.py 10.5 KB
Newer Older
1
# -*- Mode: Python; tab-width: 4 -*-
Tim Peters's avatar
Tim Peters committed
2
#       Id: asynchat.py,v 2.26 2000/09/07 22:29:26 rushing Exp
Tim Peters's avatar
Tim Peters committed
3
#       Author: Sam Rushing <rushing@nightmare.com>
4 5 6

# ======================================================================
# Copyright 1996 by Sam Rushing
Tim Peters's avatar
Tim Peters committed
7
#
8
#                         All Rights Reserved
Tim Peters's avatar
Tim Peters committed
9
#
10 11 12 13 14 15 16 17
# Permission to use, copy, modify, and distribute this software and
# its documentation for any purpose and without fee is hereby
# granted, provided that the above copyright notice appear in all
# copies and that both that copyright notice and this permission
# notice appear in supporting documentation, and that the name of Sam
# Rushing not be used in advertising or publicity pertaining to
# distribution of the software without specific, written prior
# permission.
Tim Peters's avatar
Tim Peters committed
18
#
19 20 21 22 23 24 25 26 27
# SAM RUSHING DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
# INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN
# NO EVENT SHALL SAM RUSHING BE LIABLE FOR ANY SPECIAL, INDIRECT OR
# CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
# OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT,
# NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
# CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
# ======================================================================

28
r"""A class supporting chat-style (command/response) protocols.
29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48

This class adds support for 'chat' style protocols - where one side
sends a 'command', and the other sends a response (examples would be
the common internet protocols - smtp, nntp, ftp, etc..).

The handle_read() method looks at the input stream for the current
'terminator' (usually '\r\n' for single-line responses, '\r\n.\r\n'
for multi-line output), calling self.found_terminator() on its
receipt.

for example:
Say you build an async nntp client using this class.  At the start
of the connection, you'll have self.terminator set to '\r\n', in
order to process the single-line greeting.  Just before issuing a
'LIST' command you'll set it to '\r\n.\r\n'.  The output of the LIST
command will be accumulated (using your own 'collect_incoming_data'
method) up to the terminator, and then control will be returned to
you - by calling your self.found_terminator() method.
"""

49 50
import socket
import asyncore
51
from collections import deque
52 53

class async_chat (asyncore.dispatcher):
Tim Peters's avatar
Tim Peters committed
54 55 56 57 58 59 60 61 62 63 64 65 66 67
    """This is an abstract class.  You must derive from this class, and add
    the two methods collect_incoming_data() and found_terminator()"""

    # these are overridable defaults

    ac_in_buffer_size       = 4096
    ac_out_buffer_size      = 4096

    def __init__ (self, conn=None):
        self.ac_in_buffer = ''
        self.ac_out_buffer = ''
        self.producer_fifo = fifo()
        asyncore.dispatcher.__init__ (self, conn)

68 69
    def collect_incoming_data(self, data):
        raise NotImplementedError, "must be implemented in subclass"
Tim Peters's avatar
Tim Peters committed
70

71 72
    def found_terminator(self):
        raise NotImplementedError, "must be implemented in subclass"
Tim Peters's avatar
Tim Peters committed
73

Tim Peters's avatar
Tim Peters committed
74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103
    def set_terminator (self, term):
        "Set the input delimiter.  Can be a fixed string of any length, an integer, or None"
        self.terminator = term

    def get_terminator (self):
        return self.terminator

    # grab some more data from the socket,
    # throw it to the collector method,
    # check for the terminator,
    # if found, transition to the next state.

    def handle_read (self):

        try:
            data = self.recv (self.ac_in_buffer_size)
        except socket.error, why:
            self.handle_error()
            return

        self.ac_in_buffer = self.ac_in_buffer + data

        # Continue to search for self.terminator in self.ac_in_buffer,
        # while calling self.collect_incoming_data.  The while loop
        # is necessary because we might read several data+terminator
        # combos with a single recv(1024).

        while self.ac_in_buffer:
            lb = len(self.ac_in_buffer)
            terminator = self.get_terminator()
104
            if not terminator:
Tim Peters's avatar
Tim Peters committed
105 106 107
                # no terminator, collect it all
                self.collect_incoming_data (self.ac_in_buffer)
                self.ac_in_buffer = ''
108
            elif isinstance(terminator, int) or isinstance(terminator, long):
Tim Peters's avatar
Tim Peters committed
109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128
                # numeric terminator
                n = terminator
                if lb < n:
                    self.collect_incoming_data (self.ac_in_buffer)
                    self.ac_in_buffer = ''
                    self.terminator = self.terminator - lb
                else:
                    self.collect_incoming_data (self.ac_in_buffer[:n])
                    self.ac_in_buffer = self.ac_in_buffer[n:]
                    self.terminator = 0
                    self.found_terminator()
            else:
                # 3 cases:
                # 1) end of buffer matches terminator exactly:
                #    collect data, transition
                # 2) end of buffer matches some prefix:
                #    collect data to the prefix
                # 3) end of buffer does not match any prefix:
                #    collect data
                terminator_len = len(terminator)
129
                index = self.ac_in_buffer.find(terminator)
Tim Peters's avatar
Tim Peters committed
130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174
                if index != -1:
                    # we found the terminator
                    if index > 0:
                        # don't bother reporting the empty string (source of subtle bugs)
                        self.collect_incoming_data (self.ac_in_buffer[:index])
                    self.ac_in_buffer = self.ac_in_buffer[index+terminator_len:]
                    # This does the Right Thing if the terminator is changed here.
                    self.found_terminator()
                else:
                    # check for a prefix of the terminator
                    index = find_prefix_at_end (self.ac_in_buffer, terminator)
                    if index:
                        if index != lb:
                            # we found a prefix, collect up to the prefix
                            self.collect_incoming_data (self.ac_in_buffer[:-index])
                            self.ac_in_buffer = self.ac_in_buffer[-index:]
                        break
                    else:
                        # no prefix, collect it all
                        self.collect_incoming_data (self.ac_in_buffer)
                        self.ac_in_buffer = ''

    def handle_write (self):
        self.initiate_send ()

    def handle_close (self):
        self.close()

    def push (self, data):
        self.producer_fifo.push (simple_producer (data))
        self.initiate_send()

    def push_with_producer (self, producer):
        self.producer_fifo.push (producer)
        self.initiate_send()

    def readable (self):
        "predicate for inclusion in the readable for select()"
        return (len(self.ac_in_buffer) <= self.ac_in_buffer_size)

    def writable (self):
        "predicate for inclusion in the writable for select()"
        # return len(self.ac_out_buffer) or len(self.producer_fifo) or (not self.connected)
        # this is about twice as fast, though not as clear.
        return not (
175
                (self.ac_out_buffer == '') and
Tim Peters's avatar
Tim Peters committed
176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196
                self.producer_fifo.is_empty() and
                self.connected
                )

    def close_when_done (self):
        "automatically close this channel once the outgoing queue is empty"
        self.producer_fifo.push (None)

    # refill the outgoing buffer by calling the more() method
    # of the first producer in the queue
    def refill_buffer (self):
        while 1:
            if len(self.producer_fifo):
                p = self.producer_fifo.first()
                # a 'None' in the producer fifo is a sentinel,
                # telling us to close the channel.
                if p is None:
                    if not self.ac_out_buffer:
                        self.producer_fifo.pop()
                        self.close()
                    return
197
                elif isinstance(p, str):
Tim Peters's avatar
Tim Peters committed
198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232
                    self.producer_fifo.pop()
                    self.ac_out_buffer = self.ac_out_buffer + p
                    return
                data = p.more()
                if data:
                    self.ac_out_buffer = self.ac_out_buffer + data
                    return
                else:
                    self.producer_fifo.pop()
            else:
                return

    def initiate_send (self):
        obs = self.ac_out_buffer_size
        # try to refill the buffer
        if (len (self.ac_out_buffer) < obs):
            self.refill_buffer()

        if self.ac_out_buffer and self.connected:
            # try to send the buffer
            try:
                num_sent = self.send (self.ac_out_buffer[:obs])
                if num_sent:
                    self.ac_out_buffer = self.ac_out_buffer[num_sent:]

            except socket.error, why:
                self.handle_error()
                return

    def discard_buffers (self):
        # Emergencies only!
        self.ac_in_buffer = ''
        self.ac_out_buffer = ''
        while self.producer_fifo:
            self.producer_fifo.pop()
233

234

235
class simple_producer:
Guido van Rossum's avatar
Guido van Rossum committed
236

Tim Peters's avatar
Tim Peters committed
237 238 239
    def __init__ (self, data, buffer_size=512):
        self.data = data
        self.buffer_size = buffer_size
240

Tim Peters's avatar
Tim Peters committed
241 242 243 244 245 246 247 248 249
    def more (self):
        if len (self.data) > self.buffer_size:
            result = self.data[:self.buffer_size]
            self.data = self.data[self.buffer_size:]
            return result
        else:
            result = self.data
            self.data = ''
            return result
250 251

class fifo:
Tim Peters's avatar
Tim Peters committed
252 253
    def __init__ (self, list=None):
        if not list:
254
            self.list = deque()
Tim Peters's avatar
Tim Peters committed
255
        else:
256
            self.list = deque(list)
Tim Peters's avatar
Tim Peters committed
257 258 259 260 261

    def __len__ (self):
        return len(self.list)

    def is_empty (self):
262
        return not self.list
Tim Peters's avatar
Tim Peters committed
263 264

    def first (self):
265
        return self.list[0]
Tim Peters's avatar
Tim Peters committed
266 267

    def push (self, data):
268
        self.list.append(data)
Tim Peters's avatar
Tim Peters committed
269 270 271

    def pop (self):
        if self.list:
272
            return (1, self.list.popleft())
Tim Peters's avatar
Tim Peters committed
273 274
        else:
            return (0, None)
275 276 277 278 279 280 281

# Given 'haystack', see if any prefix of 'needle' is at its end.  This
# assumes an exact match has already been checked.  Return the number of
# characters matched.
# for example:
# f_p_a_e ("qwerty\r", "\r\n") => 1
# f_p_a_e ("qwertydkjf", "\r\n") => 0
282
# f_p_a_e ("qwerty\r\n", "\r\n") => <undefined>
283 284

# this could maybe be made faster with a computed regex?
285
# [answer: no; circa Python-2.0, Jan 2001]
286 287
# new python:   28961/s
# old python:   18307/s
288 289
# re:        12820/s
# regex:     14035/s
290 291

def find_prefix_at_end (haystack, needle):
Tim Peters's avatar
Tim Peters committed
292 293 294 295
    l = len(needle) - 1
    while l and not haystack.endswith(needle[:l]):
        l -= 1
    return l