1 # This program is free software; you can redistribute it and/or modify
2 # it under the terms of the GNU General Public License as published by
3 # the Free Software Foundation; either version 2 of the License, or
4 # (at your option) any later version.
6 # This program is distributed in the hope that it will be useful,
7 # but WITHOUT ANY WARRANTY; without even the implied warranty of
8 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
9 # GNU Library General Public License for more details.
11 # You should have received a copy of the GNU General Public License
12 # along with this program; if not, write to the Free Software
13 # Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
15 # See the COPYING file for license information.
17 # Copyright (c) 2006, 2007, 2008 Guillaume Chazarain <guichaz@gmail.com>
22 from gsh
.console
import console_output
24 class buffered_dispatcher(asyncore
.file_dispatcher
):
25 """A dispatcher with a write buffer to allow asynchronous writers, and a
26 read buffer to permit line oriented manipulations"""
28 # 1 MiB should be enough for everybody
29 MAX_BUFFER_SIZE
= 1 * 1024 * 1024
31 def __init__(self
, fd
):
32 asyncore
.file_dispatcher
.__init
__(self
, fd
)
35 self
.write_buffer
= ''
37 def handle_error(self
):
38 """Handle the Ctrl-C or print the exception and its stack trace.
39 Returns True if it was an actual error"""
43 # I/O error, let the parent take action
46 def handle_read(self
):
47 """Some data can be read"""
49 buffer_length
= len(self
.read_buffer
)
51 while buffer_length
< buffered_dispatcher
.MAX_BUFFER_SIZE
:
53 piece
= self
.recv(4096)
55 if e
.errno
== errno
.EAGAIN
:
56 # End of the available data
58 elif e
.errno
== errno
.EIO
and new_data
:
59 # Hopefully we could read an error message before the
65 buffer_length
+= len(piece
)
67 new_data
= new_data
.replace('\r', '\n')
68 self
.read_buffer
+= new_data
72 """No need to ask data if our buffer is already full"""
73 return len(self
.read_buffer
) < buffered_dispatcher
.MAX_BUFFER_SIZE
76 """Do we have something to write?"""
77 return self
.write_buffer
!= ''
79 def dispatch_write(self
, buf
):
80 """Augment the buffer with stuff to write when possible"""
81 self
.write_buffer
+= buf
82 if len(self
.write_buffer
) > buffered_dispatcher
.MAX_BUFFER_SIZE
:
83 console_output('Buffer too big (%d) for %s\n' %
84 (len(self
.write_buffer
), str(self
)))
85 raise asyncore
.ExitNow(1)