@@ -106,6 +106,7 @@ cdef class Packer:
106
106
cdef object _default
107
107
cdef object _berrors
108
108
cdef const char * unicode_errors
109
+ cdef size_t exports # number of exported buffers
109
110
cdef bint strict_types
110
111
cdef bint use_float
111
112
cdef bint autoreset
@@ -117,10 +118,16 @@ cdef class Packer:
117
118
raise MemoryError (" Unable to allocate internal buffer." )
118
119
self .pk.buf_size = buf_size
119
120
self .pk.length = 0
121
+ self .exports = 0
120
122
121
123
def __dealloc__ (self ):
122
124
PyMem_Free(self .pk.buf)
123
125
self .pk.buf = NULL
126
+ assert self .exports == 0
127
+
128
+ cdef _check_exports(self ):
129
+ if self .exports > 0 :
130
+ raise BufferError(" Existing exports of data: Packer cannot be changed" )
124
131
125
132
def __init__ (self , *, default = None ,
126
133
bint use_single_float = False , bint autoreset = True , bint use_bin_type = True ,
@@ -149,16 +156,16 @@ cdef class Packer:
149
156
cdef unsigned long ulval
150
157
cdef const char * rawval
151
158
cdef Py_ssize_t L
152
- cdef bool strict_types = self .strict_types
153
159
cdef Py_buffer view
160
+ cdef bint strict = self .strict_types
154
161
155
162
if o is None :
156
163
msgpack_pack_nil(& self .pk)
157
164
elif o is True :
158
165
msgpack_pack_true(& self .pk)
159
166
elif o is False :
160
167
msgpack_pack_false(& self .pk)
161
- elif PyLong_CheckExact(o) if strict_types else PyLong_Check(o):
168
+ elif PyLong_CheckExact(o) if strict else PyLong_Check(o):
162
169
try :
163
170
if o > 0 :
164
171
ullval = o
@@ -171,19 +178,19 @@ cdef class Packer:
171
178
return - 2
172
179
else :
173
180
raise OverflowError (" Integer value out of range" )
174
- elif PyFloat_CheckExact(o) if strict_types else PyFloat_Check(o):
181
+ elif PyFloat_CheckExact(o) if strict else PyFloat_Check(o):
175
182
if self .use_float:
176
183
msgpack_pack_float(& self .pk, < float > o)
177
184
else :
178
185
msgpack_pack_double(& self .pk, < double > o)
179
- elif PyBytesLike_CheckExact(o) if strict_types else PyBytesLike_Check(o):
186
+ elif PyBytesLike_CheckExact(o) if strict else PyBytesLike_Check(o):
180
187
L = Py_SIZE(o)
181
188
if L > ITEM_LIMIT:
182
189
PyErr_Format(ValueError , b" %.200s object is too large" , Py_TYPE(o).tp_name)
183
190
rawval = o
184
191
msgpack_pack_bin(& self .pk, L)
185
192
msgpack_pack_raw_body(& self .pk, rawval, L)
186
- elif PyUnicode_CheckExact(o) if strict_types else PyUnicode_Check(o):
193
+ elif PyUnicode_CheckExact(o) if strict else PyUnicode_Check(o):
187
194
if self .unicode_errors == NULL :
188
195
rawval = PyUnicode_AsUTF8AndSize(o, & L)
189
196
if L > ITEM_LIMIT:
@@ -196,15 +203,15 @@ cdef class Packer:
196
203
rawval = o
197
204
msgpack_pack_raw(& self .pk, L)
198
205
msgpack_pack_raw_body(& self .pk, rawval, L)
199
- elif PyDict_CheckExact(o) if strict_types else PyDict_Check(o):
206
+ elif PyDict_CheckExact(o) if strict else PyDict_Check(o):
200
207
L = len (o)
201
208
if L > ITEM_LIMIT:
202
209
raise ValueError (" dict is too large" )
203
210
msgpack_pack_map(& self .pk, L)
204
211
for k, v in o.items():
205
212
self ._pack(k, nest_limit)
206
213
self ._pack(v, nest_limit)
207
- elif type (o) is ExtType if strict_types else isinstance (o, ExtType):
214
+ elif type (o) is ExtType if strict else isinstance (o, ExtType):
208
215
# This should be before Tuple because ExtType is namedtuple.
209
216
rawval = o.data
210
217
L = len (o.data)
@@ -216,7 +223,7 @@ cdef class Packer:
216
223
llval = o.seconds
217
224
ulval = o.nanoseconds
218
225
msgpack_pack_timestamp(& self .pk, llval, ulval)
219
- elif PyList_CheckExact(o) if strict_types else (PyTuple_Check(o) or PyList_Check(o)):
226
+ elif PyList_CheckExact(o) if strict else (PyTuple_Check(o) or PyList_Check(o)):
220
227
L = Py_SIZE(o)
221
228
if L > ITEM_LIMIT:
222
229
raise ValueError (" list is too large" )
@@ -264,6 +271,7 @@ cdef class Packer:
264
271
265
272
def pack (self , object obj ):
266
273
cdef int ret
274
+ self ._check_exports()
267
275
try :
268
276
ret = self ._pack(obj, DEFAULT_RECURSE_LIMIT)
269
277
except :
@@ -277,21 +285,26 @@ cdef class Packer:
277
285
return buf
278
286
279
287
def pack_ext_type (self , typecode , data ):
288
+ self ._check_exports()
289
+ if len (data) > ITEM_LIMIT:
290
+ raise ValueError (" ext data too large" )
280
291
msgpack_pack_ext(& self .pk, typecode, len (data))
281
292
msgpack_pack_raw_body(& self .pk, data, len (data))
282
293
283
294
def pack_array_header (self , long long size ):
295
+ self ._check_exports()
284
296
if size > ITEM_LIMIT:
285
- raise ValueError
297
+ raise ValueError ( " array too large " )
286
298
msgpack_pack_array(& self .pk, size)
287
299
if self .autoreset:
288
300
buf = PyBytes_FromStringAndSize(self .pk.buf, self .pk.length)
289
301
self .pk.length = 0
290
302
return buf
291
303
292
304
def pack_map_header (self , long long size ):
305
+ self ._check_exports()
293
306
if size > ITEM_LIMIT:
294
- raise ValueError
307
+ raise ValueError ( " map too learge " )
295
308
msgpack_pack_map(& self .pk, size)
296
309
if self .autoreset:
297
310
buf = PyBytes_FromStringAndSize(self .pk.buf, self .pk.length)
@@ -305,7 +318,11 @@ cdef class Packer:
305
318
*pairs* should be a sequence of pairs.
306
319
(`len(pairs)` and `for k, v in pairs:` should be supported.)
307
320
"""
308
- msgpack_pack_map(& self .pk, len (pairs))
321
+ self ._check_exports()
322
+ size = len (pairs)
323
+ if size > ITEM_LIMIT:
324
+ raise ValueError (" map too large" )
325
+ msgpack_pack_map(& self .pk, size)
309
326
for k, v in pairs:
310
327
self ._pack(k)
311
328
self ._pack(v)
@@ -319,18 +336,23 @@ cdef class Packer:
319
336
320
337
This method is useful only when autoreset=False.
321
338
"""
339
+ self ._check_exports()
322
340
self .pk.length = 0
323
341
324
342
def bytes (self ):
325
343
""" Return internal buffer contents as bytes object"""
326
344
return PyBytes_FromStringAndSize(self .pk.buf, self .pk.length)
327
345
328
346
def getbuffer (self ):
329
- """ Return view of internal buffer."""
347
+ """ Return memoryview of internal buffer.
348
+
349
+ Note: Packer now supports buffer protocol. You can use memoryview(packer).
350
+ """
330
351
return memoryview(self )
331
352
332
353
def __getbuffer__ (self , Py_buffer *buffer , int flags ):
333
354
PyBuffer_FillInfo(buffer , self , self .pk.buf, self .pk.length, 1 , flags)
355
+ self .exports += 1
334
356
335
357
def __releasebuffer__ (self , Py_buffer *buffer ):
336
- pass
358
+ self .exports -= 1
0 commit comments