debian/control: Add Build-Depends for `dh-python'.
[mLib-python] / pkbuf.pyx
1 ### -*-pyrex-*-
2 ###
3 ### Packet buffering
4 ###
5 ### (c) 2005 Straylight/Edgeware
6 ###
7
8 ###----- Licensing notice ---------------------------------------------------
9 ###
10 ### This file is part of the Python interface to mLib.
11 ###
12 ### mLib/Python is free software; you can redistribute it and/or modify
13 ### it under the terms of the GNU General Public License as published by
14 ### the Free Software Foundation; either version 2 of the License, or
15 ### (at your option) any later version.
16 ###
17 ### mLib/Python is distributed in the hope that it will be useful,
18 ### but WITHOUT ANY WARRANTY; without even the implied warranty of
19 ### MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 ### GNU General Public License for more details.
21 ###
22 ### You should have received a copy of the GNU General Public License
23 ### along with mLib/Python; if not, write to the Free Software Foundation,
24 ### Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
25
26 cdef class PacketBuffer:
27 """
28 PacketBuffer([packetproc = None], [eofproc = None])
29
30 Split an incoming stream into packets.
31 """
32 cdef pkbuf pk
33 cdef _packet
34 cdef _eof
35 def __cinit__(me, packetproc = None, eofproc = None, *hunoz, **hukairz):
36 pkbuf_init(&me.pk, _pkfunc, <void *>me)
37 me._packet = _checkcallable(packetproc, 'packet proc')
38 me._eof = _checkcallable(eofproc, 'eof proc')
39 def __dealloc__(me):
40 pkbuf_destroy(&me.pk)
41 property activep:
42 """PK.activep -> BOOL: is the buffer still active?"""
43 def __get__(me):
44 return _tobool(me.pk.f & PKBUF_ENABLE)
45 property want:
46 """PK.want -> INT: size of next packet to return"""
47 def __get__(me):
48 return me.pk.want
49 def __set__(me, want):
50 if want <= 0:
51 raise TypeError, 'want must be positive'
52 pkbuf_want(&me.pk, pk)
53 property packetproc:
54 """PK.packetproc -> FUNC: call FUNC(PACKET) on each packet"""
55 def __get__(me):
56 return me._packet
57 def __set__(me, proc):
58 me._line = _checkcallable(proc, 'packet proc')
59 def __del__(me):
60 me._line = None
61 property eofproc:
62 """PK.eofproc -> FUNC: call FUNC() at end-of-file"""
63 def __get__(me):
64 return me._eof
65 def __set__(me, proc):
66 me._eof = _checkcallable(proc, 'eof proc')
67 def __del__(me):
68 me._eof = None
69 def enable(me):
70 """PK.enable(): enable the buffer, allowing packets to be emitted"""
71 if me.pk.f & PKBUF_ENABLE:
72 raise ValueError, 'already enabled'
73 me.pk.f = me.pk.f | PKBUF_ENABLE
74 me.enabled()
75 return me
76 def disable(me):
77 """PK.disable(): disable the buffer, suspending packet emission"""
78 if not (me.pk.f & PKBUF_ENABLE):
79 raise ValueError, 'already disabled'
80 me.pk.f = me.pk.f & ~PKBUF_ENABLE
81 me.disabled()
82 return me
83 def close(me):
84 """PK.close(): report the end of the input stream"""
85 if not (me.pk.f & PKBUF_ENABLE):
86 raise ValueError, 'buffer disabled'
87 pkbuf_close(&me.pk)
88 return me
89 property free:
90 """PK.free -> INT: amount of space remaining in buffer"""
91 def __get__(me):
92 cdef unsigned char *p
93 return pkbuf_free(&me.pk, &p)
94 def flush(me, str):
95 """PK.flush(STR) -> insert STR into the buffer and emit packets"""
96 cdef Py_ssize_t len
97 cdef unsigned char *p
98 cdef unsigned char *q
99 cdef size_t n
100 PyObject_AsReadBuffer(str, <cvp *>&p, &len)
101 while len > 0:
102 n = pkbuf_free(&me.pk, &q)
103 if n > len:
104 n = len
105 memcpy(q, p, n)
106 p = p + n
107 len = len - n
108 if not (me.pk.f & PKBUF_ENABLE):
109 break
110 pkbuf_flush(&me.pk, q, n)
111 return PyString_FromStringAndSize(<char *>p, len)
112 def enabled(me):
113 """PK.enabled(): called when buffer is enabled"""
114 pass
115 def disabled(me):
116 """PK.disabled(): called when buffer is disabled"""
117 pass
118 def packet(me, pk):
119 """PK.packet(PACKET): called for each completed packet"""
120 return _maybecall(me._packet, (pk,))
121 def eof(me):
122 """PK.eof(): called at end-of-file"""
123 return _maybecall(me._eof, ())
124
125 cdef void _pkfunc(unsigned char *p, size_t n, pkbuf *pk,
126 size_t *keep, void *arg):
127 cdef PacketBuffer pb
128 cdef void *rp
129 cdef Py_ssize_t rn
130 pb = <PacketBuffer>arg
131 if p is NULL:
132 pb.eof()
133 else:
134 r = pb.packet(PyString_FromStringAndSize(<char *>p, n))
135 if r is not None:
136 PyObject_AsReadBuffer(r, <cvp *>&rp, &rn)
137 if rn > n:
138 raise ValueError, 'remaining buffer too large'
139 if rn:
140 memcpy(p + n - rn, rp, rn)
141 keep[0] = rn
142
143 ###----- That's all, folks --------------------------------------------------