mirror of
https://github.com/msgpack/msgpack-c.git
synced 2025-03-19 21:18:23 +01:00
Add object_hook
option to unpack and default
option to pack.
(see simplejson for how to use).
This commit is contained in:
parent
2af7df5865
commit
70982e204c
@ -20,6 +20,9 @@ cdef extern from "Python.h":
|
|||||||
cdef bint PyFloat_Check(object o)
|
cdef bint PyFloat_Check(object o)
|
||||||
cdef bint PyBytes_Check(object o)
|
cdef bint PyBytes_Check(object o)
|
||||||
cdef bint PyUnicode_Check(object o)
|
cdef bint PyUnicode_Check(object o)
|
||||||
|
cdef bint PyCallable_Check(object o)
|
||||||
|
cdef void Py_INCREF(object o)
|
||||||
|
cdef void Py_DECREF(object o)
|
||||||
|
|
||||||
cdef extern from "stdlib.h":
|
cdef extern from "stdlib.h":
|
||||||
void* malloc(size_t)
|
void* malloc(size_t)
|
||||||
@ -60,6 +63,7 @@ cdef class Packer(object):
|
|||||||
astream.write(packer.pack(b))
|
astream.write(packer.pack(b))
|
||||||
"""
|
"""
|
||||||
cdef msgpack_packer pk
|
cdef msgpack_packer pk
|
||||||
|
cdef object default
|
||||||
|
|
||||||
def __cinit__(self):
|
def __cinit__(self):
|
||||||
cdef int buf_size = 1024*1024
|
cdef int buf_size = 1024*1024
|
||||||
@ -67,6 +71,12 @@ cdef class Packer(object):
|
|||||||
self.pk.buf_size = buf_size
|
self.pk.buf_size = buf_size
|
||||||
self.pk.length = 0
|
self.pk.length = 0
|
||||||
|
|
||||||
|
def __init__(self, default=None):
|
||||||
|
if default is not None:
|
||||||
|
if not PyCallable_Check(default):
|
||||||
|
raise TypeError("default must be a callable.")
|
||||||
|
self.default = default
|
||||||
|
|
||||||
def __dealloc__(self):
|
def __dealloc__(self):
|
||||||
free(self.pk.buf);
|
free(self.pk.buf);
|
||||||
|
|
||||||
@ -126,9 +136,18 @@ cdef class Packer(object):
|
|||||||
for v in o:
|
for v in o:
|
||||||
ret = self._pack(v)
|
ret = self._pack(v)
|
||||||
if ret != 0: break
|
if ret != 0: break
|
||||||
|
elif self.default is not None:
|
||||||
|
o = self.default(o)
|
||||||
|
d = o
|
||||||
|
ret = msgpack_pack_map(&self.pk, len(d))
|
||||||
|
if ret == 0:
|
||||||
|
for k,v in d.items():
|
||||||
|
ret = self._pack(k)
|
||||||
|
if ret != 0: break
|
||||||
|
ret = self._pack(v)
|
||||||
|
if ret != 0: break
|
||||||
else:
|
else:
|
||||||
# TODO: Serialize with defalt() like simplejson.
|
raise TypeError("can't serialize %r" % (o,))
|
||||||
raise TypeError, "can't serialize %r" % (o,)
|
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
def pack(self, object obj):
|
def pack(self, object obj):
|
||||||
@ -141,14 +160,14 @@ cdef class Packer(object):
|
|||||||
return buf
|
return buf
|
||||||
|
|
||||||
|
|
||||||
def pack(object o, object stream):
|
def pack(object o, object stream, default=None):
|
||||||
"""pack an object `o` and write it to stream)."""
|
"""pack an object `o` and write it to stream)."""
|
||||||
packer = Packer()
|
packer = Packer(default)
|
||||||
stream.write(packer.pack(o))
|
stream.write(packer.pack(o))
|
||||||
|
|
||||||
def packb(object o):
|
def packb(object o, default=None):
|
||||||
"""pack o and return packed bytes."""
|
"""pack o and return packed bytes."""
|
||||||
packer = Packer()
|
packer = Packer(default=default)
|
||||||
return packer.pack(o)
|
return packer.pack(o)
|
||||||
|
|
||||||
packs = packb
|
packs = packb
|
||||||
@ -156,6 +175,7 @@ packs = packb
|
|||||||
cdef extern from "unpack.h":
|
cdef extern from "unpack.h":
|
||||||
ctypedef struct msgpack_user:
|
ctypedef struct msgpack_user:
|
||||||
int use_list
|
int use_list
|
||||||
|
PyObject* object_hook
|
||||||
|
|
||||||
ctypedef struct template_context:
|
ctypedef struct template_context:
|
||||||
msgpack_user user
|
msgpack_user user
|
||||||
@ -170,7 +190,7 @@ cdef extern from "unpack.h":
|
|||||||
object template_data(template_context* ctx)
|
object template_data(template_context* ctx)
|
||||||
|
|
||||||
|
|
||||||
def unpackb(bytes packed_bytes):
|
def unpackb(bytes packed_bytes, object object_hook=None):
|
||||||
"""Unpack packed_bytes to object. Returns an unpacked object."""
|
"""Unpack packed_bytes to object. Returns an unpacked object."""
|
||||||
cdef const_char_ptr p = packed_bytes
|
cdef const_char_ptr p = packed_bytes
|
||||||
cdef template_context ctx
|
cdef template_context ctx
|
||||||
@ -178,7 +198,16 @@ def unpackb(bytes packed_bytes):
|
|||||||
cdef int ret
|
cdef int ret
|
||||||
template_init(&ctx)
|
template_init(&ctx)
|
||||||
ctx.user.use_list = 0
|
ctx.user.use_list = 0
|
||||||
|
ctx.user.object_hook = NULL
|
||||||
|
if object_hook is not None:
|
||||||
|
if not PyCallable_Check(object_hook):
|
||||||
|
raise TypeError("object_hook must be a callable.")
|
||||||
|
Py_INCREF(object_hook)
|
||||||
|
ctx.user.object_hook = <PyObject*>object_hook
|
||||||
ret = template_execute(&ctx, p, len(packed_bytes), &off)
|
ret = template_execute(&ctx, p, len(packed_bytes), &off)
|
||||||
|
if object_hook is not None:
|
||||||
|
pass
|
||||||
|
#Py_DECREF(object_hook)
|
||||||
if ret == 1:
|
if ret == 1:
|
||||||
return template_data(&ctx)
|
return template_data(&ctx)
|
||||||
else:
|
else:
|
||||||
@ -186,10 +215,10 @@ def unpackb(bytes packed_bytes):
|
|||||||
|
|
||||||
unpacks = unpackb
|
unpacks = unpackb
|
||||||
|
|
||||||
def unpack(object stream):
|
def unpack(object stream, object object_hook=None):
|
||||||
"""unpack an object from stream."""
|
"""unpack an object from stream."""
|
||||||
packed = stream.read()
|
packed = stream.read()
|
||||||
return unpackb(packed)
|
return unpackb(packed, object_hook=object_hook)
|
||||||
|
|
||||||
cdef class UnpackIterator(object):
|
cdef class UnpackIterator(object):
|
||||||
cdef object unpacker
|
cdef object unpacker
|
||||||
@ -234,6 +263,7 @@ cdef class Unpacker(object):
|
|||||||
cdef int read_size
|
cdef int read_size
|
||||||
cdef object waiting_bytes
|
cdef object waiting_bytes
|
||||||
cdef bint use_list
|
cdef bint use_list
|
||||||
|
cdef object object_hook
|
||||||
|
|
||||||
def __cinit__(self):
|
def __cinit__(self):
|
||||||
self.buf = NULL
|
self.buf = NULL
|
||||||
@ -242,7 +272,8 @@ cdef class Unpacker(object):
|
|||||||
if self.buf:
|
if self.buf:
|
||||||
free(self.buf);
|
free(self.buf);
|
||||||
|
|
||||||
def __init__(self, file_like=None, int read_size=0, bint use_list=0):
|
def __init__(self, file_like=None, int read_size=0, bint use_list=0,
|
||||||
|
object object_hook=None):
|
||||||
if read_size == 0:
|
if read_size == 0:
|
||||||
read_size = 1024*1024
|
read_size = 1024*1024
|
||||||
self.use_list = use_list
|
self.use_list = use_list
|
||||||
@ -255,6 +286,11 @@ cdef class Unpacker(object):
|
|||||||
self.buf_tail = 0
|
self.buf_tail = 0
|
||||||
template_init(&self.ctx)
|
template_init(&self.ctx)
|
||||||
self.ctx.user.use_list = use_list
|
self.ctx.user.use_list = use_list
|
||||||
|
self.ctx.user.object_hook = <PyObject*>NULL
|
||||||
|
if object_hook is not None:
|
||||||
|
if not PyCallable_Check(object_hook):
|
||||||
|
raise TypeError("object_hook must be a callable.")
|
||||||
|
self.ctx.user.object_hook = <PyObject*>object_hook
|
||||||
|
|
||||||
def feed(self, bytes next_bytes):
|
def feed(self, bytes next_bytes):
|
||||||
self.waiting_bytes.append(next_bytes)
|
self.waiting_bytes.append(next_bytes)
|
||||||
|
@ -21,6 +21,7 @@
|
|||||||
|
|
||||||
typedef struct unpack_user {
|
typedef struct unpack_user {
|
||||||
int use_list;
|
int use_list;
|
||||||
|
PyObject *object_hook;
|
||||||
} unpack_user;
|
} unpack_user;
|
||||||
|
|
||||||
|
|
||||||
@ -172,6 +173,19 @@ static inline int template_callback_map_item(unpack_user* u, msgpack_unpack_obje
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//static inline int template_callback_map_end(unpack_user* u, msgpack_unpack_object* c)
|
||||||
|
int template_callback_map_end(unpack_user* u, msgpack_unpack_object* c)
|
||||||
|
{
|
||||||
|
if (u->object_hook) {
|
||||||
|
PyObject *arglist = Py_BuildValue("(O)", *c);
|
||||||
|
Py_INCREF(*c);
|
||||||
|
*c = PyEval_CallObject(u->object_hook, arglist);
|
||||||
|
Py_DECREF(arglist);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
static inline int template_callback_raw(unpack_user* u, const char* b, const char* p, unsigned int l, msgpack_unpack_object* o)
|
static inline int template_callback_raw(unpack_user* u, const char* b, const char* p, unsigned int l, msgpack_unpack_object* o)
|
||||||
{
|
{
|
||||||
PyObject *py;
|
PyObject *py;
|
||||||
|
@ -317,6 +317,7 @@ _push:
|
|||||||
case CT_MAP_VALUE:
|
case CT_MAP_VALUE:
|
||||||
if(msgpack_unpack_callback(_map_item)(user, &c->obj, c->map_key, obj) < 0) { goto _failed; }
|
if(msgpack_unpack_callback(_map_item)(user, &c->obj, c->map_key, obj) < 0) { goto _failed; }
|
||||||
if(--c->count == 0) {
|
if(--c->count == 0) {
|
||||||
|
msgpack_unpack_callback(_map_end)(user, &c->obj);
|
||||||
obj = c->obj;
|
obj = c->obj;
|
||||||
--top;
|
--top;
|
||||||
/*printf("stack pop %d\n", top);*/
|
/*printf("stack pop %d\n", top);*/
|
||||||
|
31
python/test/test_obj.py
Normal file
31
python/test/test_obj.py
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
# coding: utf-8
|
||||||
|
|
||||||
|
from nose import main
|
||||||
|
from nose.tools import *
|
||||||
|
|
||||||
|
from msgpack import packs, unpacks
|
||||||
|
|
||||||
|
def _decode_complex(obj):
|
||||||
|
if '__complex__' in obj:
|
||||||
|
return complex(obj['real'], obj['imag'])
|
||||||
|
return obj
|
||||||
|
|
||||||
|
def _encode_complex(obj):
|
||||||
|
if isinstance(obj, complex):
|
||||||
|
return {'__complex__': True, 'real': 1, 'imag': 2}
|
||||||
|
return obj
|
||||||
|
|
||||||
|
def test_encode_hook():
|
||||||
|
packed = packs([3, 1+2j], default=_encode_complex)
|
||||||
|
unpacked = unpacks(packed)
|
||||||
|
eq_(unpacked[1], {'__complex__': True, 'real': 1, 'imag': 2})
|
||||||
|
|
||||||
|
def test_decode_hook():
|
||||||
|
packed = packs([3, {'__complex__': True, 'real': 1, 'imag': 2}])
|
||||||
|
unpacked = unpacks(packed, object_hook=_decode_complex)
|
||||||
|
eq_(unpacked[1], 1+2j)
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
#main()
|
||||||
|
test_decode_hook()
|
Loading…
x
Reference in New Issue
Block a user