-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathcerealizer.py
620 lines (479 loc) · 22.8 KB
/
cerealizer.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
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
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
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
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
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
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
# -*- coding: utf-8 -*-
# Cerealizer
# Copyright (C) 2005-2008 Jean-Baptiste LAMY
# Copyright (C) 2008 Peter Eckersley
#
# This program is free software.
# It is available under the Python licence.
"""Cerealizer -- A secure Pickle-like module
The interface of the Cerealizer module is similar to Pickle, and it supports
__getstate__, __setstate__, __getinitargs__ and __getnewargs__.
Cerealizer supports int, long, float, bool, complex, string, unicode, tuple, list, set, frozenset,
dict, old-style and new-style class instances. C-defined types are supported but saving the C-side
data may require to write e.g. a specific Handler or a __getstate__ and __setstate__ pair.
Objects with __slots__ are supported too.
You have to register the class you want to serialize, by calling cerealizer.register(YourClass).
Cerealizer can be considered as secure AS LONG AS the following methods of 'YourClass' are secure:
- __new__
- __del__
- __getstate__
- __setstate__
- __init__ (ONLY if __getinitargs__ is used for the class)
These methods are the only one Cerealizer may call. For a higher security, Cerealizer maintains
its own reference to these method (exepted __del__ that can only be called indirectly).
Cerealizer doesn't aim at producing Human-readable files. About performances, Cerealizer is
really fast and, when powered by Psyco, it may even beat cPickle! Although Cerealizer is
implemented in less than 500 lines of pure-Python code (which is another reason for Cerealizer
to be secure, since less code means less bugs :-).
Compared to Pickle (cPickle):
- Cerealizer is secure
- Cerealizer achieves similar performances (using Psyco)
- Cerealizer requires you to declare the serializable classes
Compared to Jelly (from TwistedMatrix):
- Cerealizer is faster
- Cerealizer does a better job with object cycles, C-defined types and tuples (*)
- Cerealizer files are not Human readable
(*) Jelly handles them, but tuples and objects in a cycle are first created as _Tuple or
_Dereference objects; this works for Python classes, but not with C-defined types which
expects a precise type (e.g. tuple and not _Tuple).
IMPLEMENTATION DETAILS
GENERAL FILE FORMAT STRUCTURE
Cerealizer format is simple but quite surprising. It uses a "double flat list" format.
It looks like that :
<magic code (currently cereal1)>\\n
<number of objects>\\n
<classname of object #0>\\n
<optional data for creating object #0 (currently nothing except for tuples)>
<classname of object #1>\\n
<optional data for creating object #1 (currently nothing except for tuples)>
[...]
<data of object #0 (format depend of the type of object #0)>
<data of object #1 (format depend of the type of object #1)>
[...]
<reference to the 'root' object>
As you can see, the information for a given object is splitted in two parts, the first one
for object's class, and the second one for the object's data.
To avoid problems, the order of the objects is the following:
<list, dict, set>
<object, instance>
<tuple, sorted by depth (=max number of folded tuples)>
Objects are put after basic types (list,...), since object's __setstate__ might rely on
a list, and thus the list must be fully loaded BEFORE calling the object's __setstate__.
DATA (<data of object #n> above)
The part <data of object #n> saves the data of object #n. It may contains reference to other data
(see below, in Cerealizer references include reference to other objects but also raw data like int).
- an object is saved by : <reference to the object state (the value returned by object.__getstate__() or object.__dict__)>
e.g. 'r7\\n' (object #7 being e.g. the __dict__).
- a list or a set is saved by : <number of item>\\n
<reference to item #0>
<reference to item #1>
[...]
e.g. '3\\ni0\\ni1\\ni2\\n' for [0, 1, 2]
- a dict is saved by : <number of item>\\n
<reference to value #0>
<reference to key #0>
<reference to value #1>
<reference to key #1>
[...]
REFERENCES (<reference to XXX> above)
In Cerealizer a reference can be either a reference to another object being serialized in the
same file, or a raw value (e.g. an integer).
- an int is saved by e.g. 'i187\\n'
- a long is saved by e.g. 'l10000000000\\n'
- a float is saved by e.g. 'f1.07\\n'
- a bool is saved by 'b0' or 'b1'
- a string is saved by e.g. 's5\\nascii' (where 5 is the number of characters)
- an unicode is saved by e.g. 'u4\\nutf8' (where 4 is the number of characters)
- an object reference is saved by e.g. 'r3\\n' (where 3 means reference to object #3)
- None is saved by 'n'
"""
__alls__ = ["load", "dump", "loads", "dumps", "freeze_configuration", "register"]
VERSION = "0.6"
import logging
logger = logging.getLogger("cerealizer")
#logging.basicConfig(level=logging.INFO)
from cStringIO import StringIO
from new import instance
class EndOfFile(StandardError): pass
class NotCerealizerFileError(StandardError): pass
class NonCerealizableObjectError(StandardError): pass
def _priority_sorter(a, b): return cmp(a[0], b[0])
class Dumper(object):
def __init__(self): self.init()
def init(self):
self.objs = []
self.objs_id = set()
self.priorities_objs = [] # [(priority1, obj1), (priority2, obj2),...]
self.obj2state = {}
self.obj2newargs = {}
self.id2id = {}
self.id2obj = None
def dump(self, root_obj, s):
self.collect(root_obj)
self.priorities_objs.sort(_priority_sorter)
self.objs.extend([o for (priority, o) in self.priorities_objs])
s.write("cereal1\n%s\n" % len(self.objs))
i = 0
for obj in self.objs:
self.id2id[id(obj)] = i
i += 1
for obj in self.objs: _HANDLERS_[obj.__class__].dump_obj (obj, self, s)
for obj in self.objs: _HANDLERS_[obj.__class__].dump_data(obj, self, s)
_HANDLERS_[root_obj.__class__].dump_ref(root_obj, self, s)
self.init()
def undump(self, s):
txt = s.read(8)
if txt != "cereal1\n":
if txt == "":
raise EndOfFile("")
raise NotCerealizerFileError('Not a cerealizer file:\n"%s"' % txt)
nb = int(s.readline())
self.id2obj = [ None ] * nb # DO NOT DO self.id2obj = [comprehension list], since undump_ref may access id2obj during its construction
for i in range(nb):
classname = s.readline()
handler = _HANDLERS.get(classname)
if not handler: raise NonCerealizableObjectError("Object of class/type '%s' cannot be de-cerealized! Use cerealizer.register to extend Cerealizer support to other classes." % classname[:-1])
self.id2obj[i] = handler.undump_obj(self, s)
for obj in self.id2obj: _HANDLERS_[obj.__class__].undump_data(obj, self, s)
r = self.undump_ref(s)
self.init()
return r
def collect(self, obj):
"""Dumper.collect(OBJ) -> bool
Collects OBJ for serialization. Returns false is OBJ is already collected; else returns true."""
handler = _HANDLERS_.get(obj.__class__)
if not handler: raise NonCerealizableObjectError("Object of class/type '%s' cannot be cerealized! Use cerealizer.register to extend Cerealizer support to other classes." % obj.__class__)
handler.collect(obj, self)
def dump_ref (self, obj, s):
"""Dumper.dump_ref(OBJ, S)
Writes a reference to OBJ in file S."""
_HANDLERS_[obj.__class__].dump_ref(obj, self, s)
def undump_ref(self, s):
"""Dumper.undump_ref(S) -> obj
Reads a reference from file S."""
c = s.read(1)
if c == "i": return int (s.readline())
elif c == "f": return float(s.readline())
elif c == "s": return s.read(int(s.readline()))
elif c == "u": return s.read(int(s.readline())).decode("utf8")
elif c == "r": return self.id2obj[int(s.readline())]
elif c == "n": return None
elif c == "b": return bool(int(s.read(1)))
elif c == "l": return long(s.readline())
elif c == "c": return complex(s.readline())
raise ValueError("Unknown ref code '%s'!" % c)
def immutable_depth(self, t):
depth = 0
for i in t:
i2 = self.obj2newargs.get(id(i))
if not i2 is None: i = i2
if isinstance(i, tuple) or isinstance(i, frozenset):
x = self.immutable_depth(i)
if x > depth: depth = x
return depth + 1
class Handler(object):
"""Handler
A customized handler for serialization and deserialization.
You can subclass it to extend cerealization support to new object.
See also ObjHandler."""
def collect(self, obj, dumper):
"""Handler.collect(obj, dumper) -> bool
Collects all the objects referenced by OBJ.
For each objects ROBJ referenced by OBJ, calls collect method of the Handler for ROBJ's class,
i.e._HANDLERS_[ROBJ.__class__].collect(ROBJ, dumper).
Returns false if OBJ is already referenced (and thus no collection should occur); else returns true.
"""
i = id(obj)
if not i in dumper.objs_id:
dumper.objs.append(obj)
dumper.objs_id.add(i)
return 1
def dump_obj (self, obj, dumper, s):
"""Handler.dump_obj(obj, dumper, s)
Dumps OBJ classname in file S."""
s.write(self.classname)
def dump_data(self, obj, dumper, s):
"""Handler.dump_data(obj, dumper, s)
Dumps OBJ data in file S."""
def dump_ref (self, obj, dumper, s):
"""Handler.dump_ref(obj, dumper, s)
Write a reference to OBJ in file S.
You should not override dump_ref, since they is no corresponding 'undump_ref' that you
can override."""
s.write("r%s\n" % dumper.id2id[id(obj)])
def undump_obj(self, dumper, s):
"""Handler.undump_obj(dumper, s)
Returns a new uninitialized (=no __init__'ed) instance of the class.
If you override undump_obj, DUMPER and file S can be used to read additional data
saved by Handler.dump_obj()."""
def undump_data(self, obj, dumper, s):
"""Handler.undump_data(obj, dumper, s)
Reads the data for OBJ, from DUMPER and file S.
If you override undump_data, you should use DUMPER.undump_ref(S) to
read a reference or a basic type (=a string, an int,...)."""
class RefHandler(object):
def collect (self, obj, dumper) : pass
def dump_obj (self, obj, dumper, s): pass
def dump_data(self, obj, dumper, s): pass
class NoneHandler(RefHandler):
def dump_ref (self, obj, dumper, s): s.write("n")
class StrHandler(RefHandler):
def dump_ref (self, obj, dumper, s): s.write("s%s\n%s" % (len(obj), obj))
class UnicodeHandler(RefHandler):
def dump_ref (self, obj, dumper, s):
obj = obj.encode("utf8")
s.write("u%s\n%s" % (len(obj), obj))
class BoolHandler(RefHandler):
def dump_ref (self, obj, dumper, s): s.write("b%r" % int(obj))
class IntHandler(RefHandler):
def dump_ref (self, obj, dumper, s): s.write("i%r\n" % obj)
class LongHandler(RefHandler):
def dump_ref (self, obj, dumper, s): s.write("l%r\n" % obj)
class FloatHandler(RefHandler):
def dump_ref (self, obj, dumper, s): s.write("f%r\n" % obj)
class ComplexHandler(RefHandler):
def dump_ref (self, obj, dumper, s):
c = repr(obj)
if c.startswith("("): c = c[1:-1] # complex("(1+2j)") doesn't work
s.write("c%s\n" % c)
class TupleHandler(Handler):
classname = "tuple\n"
def collect(self, obj, dumper):
if not id(obj) in dumper.objs_id:
dumper.priorities_objs.append((dumper.immutable_depth(obj), obj))
dumper.objs_id.add(id(obj))
for i in obj: dumper.collect(i)
return 1
def dump_obj(self, obj, dumper, s):
s.write("%s%s\n" % (self.classname, len(obj)))
for i in obj: _HANDLERS_[i.__class__].dump_ref(i, dumper, s)
def undump_obj(self, dumper, s): return tuple([dumper.undump_ref(s) for i in range(int(s.readline()))])
class FrozensetHandler(TupleHandler):
classname = "frozenset\n"
def undump_obj(self, dumper, s): return frozenset([dumper.undump_ref(s) for i in range(int(s.readline()))])
class ListHandler(Handler):
classname = "list\n"
def collect(self, obj, dumper):
if Handler.collect(self, obj, dumper):
for i in obj: dumper.collect(i)
return 1
def dump_data(self, obj, dumper, s):
s.write("%s\n" % len(obj))
for i in obj: _HANDLERS_[i.__class__].dump_ref(i, dumper, s)
def undump_obj(self, dumper, s): return []
def undump_data(self, obj, dumper, s):
for i in range(int(s.readline())): obj.append(dumper.undump_ref(s))
class SetHandler(ListHandler):
classname = "set\n"
def undump_obj(self, dumper, s): return set()
def undump_data(self, obj, dumper, s):
for i in range(int(s.readline())): obj.add(dumper.undump_ref(s))
class DictHandler(Handler):
classname = "dict\n"
def collect(self, obj, dumper):
if Handler.collect(self, obj, dumper):
for i in obj.iterkeys (): dumper.collect(i) # Collect is not ordered
for i in obj.itervalues(): dumper.collect(i)
return 1
def dump_data(self, obj, dumper, s):
s.write("%s\n" % len(obj))
for k, v in obj.iteritems():
_HANDLERS_[v.__class__].dump_ref(v, dumper, s) # Value is saved fist
_HANDLERS_[k.__class__].dump_ref(k, dumper, s)
def undump_obj(self, dumper, s): return {}
def undump_data(self, obj, dumper, s):
for i in range(int(s.readline())):
obj[dumper.undump_ref(s)] = dumper.undump_ref(s) # Value is read fist
class ObjHandler(Handler):
"""ObjHandler
A Cerealizer Handler that can support any new-style class instances, old-style class instances
as well as C-defined types (although it may not save the C-side data)."""
def __init__(self, Class, classname = ""):
self.Class = Class
self.Class_new = getattr(Class, "__new__" , instance)
self.Class_getstate = getattr(Class, "__getstate__", None) # Check for and store __getstate__ and __setstate__ now
self.Class_setstate = getattr(Class, "__setstate__", None) # so we are are they are not modified in the class or the object
if classname: self.classname = "%s\n" % classname
else: self.classname = "%s.%s\n" % (Class.__module__, Class.__name__)
def collect(self, obj, dumper):
i = id(obj)
if not i in dumper.objs_id:
dumper.priorities_objs.append((-1, obj))
dumper.objs_id.add(i)
if self.Class_getstate: state = self.Class_getstate(obj)
else: state = obj.__dict__
dumper.obj2state[i] = state
dumper.collect(state)
return 1
def dump_data(self, obj, dumper, s):
i = dumper.obj2state[id(obj)]
_HANDLERS_[i.__class__].dump_ref(i, dumper, s)
def undump_obj(self, dumper, s): return self.Class_new(self.Class)
def undump_data(self, obj, dumper, s):
if self.Class_setstate: self.Class_setstate(obj, dumper.undump_ref(s))
else: obj.__dict__ = dumper.undump_ref(s)
class SlotedObjHandler(ObjHandler):
"""SlotedObjHandler
A Cerealizer Handler that can support new-style class instances with __slot__."""
def __init__(self, Class, classname = ""):
ObjHandler.__init__(self, Class, classname)
self.Class_slots = Class.__slots__
def collect(self, obj, dumper):
i = id(obj)
if not i in dumper.objs_id:
dumper.priorities_objs.append((-1, obj))
dumper.objs_id.add(i)
if self.Class_getstate: state = self.Class_getstate(obj)
else: state = dict([(slot, getattr(obj, slot, None)) for slot in self.Class_slots])
dumper.obj2state[i] = state
dumper.collect(state)
return 1
def undump_data(self, obj, dumper, s):
if self.Class_setstate: self.Class_setstate(obj, dumper.undump_ref(s))
else:
state = dumper.undump_ref(s)
for slot in self.Class_slots: setattr(obj, slot, state[slot])
class InitArgsObjHandler(ObjHandler):
"""InitArgsObjHandler
A Cerealizer Handler that can support class instances with __getinitargs__."""
def __init__(self, Class, classname = ""):
ObjHandler.__init__(self, Class, classname)
self.Class_getinitargs = Class.__getinitargs__
self.Class_init = Class.__init__
def collect(self, obj, dumper):
i = id(obj)
if not i in dumper.objs_id:
dumper.priorities_objs.append((-1, obj))
dumper.objs_id.add(i)
dumper.obj2state[i] = state = self.Class_getinitargs(obj)
dumper.collect(state)
return 1
def undump_data(self, obj, dumper, s): self.Class_init(obj, *dumper.undump_ref(s))
class NewArgsObjHandler(ObjHandler):
"""NewArgsObjHandler
A Cerealizer Handler that can support class instances with __getnewargs__."""
def __init__(self, Class, classname = ""):
ObjHandler.__init__(self, Class, classname)
self.Class_getnewargs = Class.__getnewargs__
def collect(self, obj, dumper):
i = id(obj)
if not i in dumper.objs_id:
dumper.obj2newargs[i] = newargs = self.Class_getnewargs(obj)
dumper.collect(newargs)
dumper.priorities_objs.append((dumper.immutable_depth(newargs), obj))
dumper.objs_id.add(i)
if self.Class_getstate: state = self.Class_getstate(obj)
else: state = obj.__dict__
dumper.obj2state[i] = state
dumper.collect(state)
return 1
def dump_obj (self, obj, dumper, s):
s.write(self.classname)
newargs = dumper.obj2newargs[id(obj)]
_HANDLERS_[newargs.__class__].dump_ref(newargs, dumper, s)
def undump_obj(self, dumper, s): return self.Class_new(self.Class, *dumper.undump_ref(s))
_configurable = 1
_HANDLERS = {}
_HANDLERS_ = {}
def register(Class, handler = None, classname = ""):
"""register(Class, handler = None, classname = "")
Registers CLASS as a serializable and secure class.
By calling register, YOU HAVE TO ASSUME THAT THE FOLLOWING METHODS ARE SECURE:
- CLASS.__new__
- CLASS.__del__
- CLASS.__getstate__
- CLASS.__setstate__
- CLASS.__getinitargs__
- CLASS.__init__ (only if CLASS.__getinitargs__ exists)
HANDLER is the Cerealizer Handler object that handles serialization and deserialization for Class.
If not given, Cerealizer create an instance of ObjHandler, which is suitable for old-style and
new_style Python class, and also C-defined types (although if it has some C-side data, you may
have to write a custom Handler or a __getstate__ and __setstate__ pair).
CLASSNAME is the classname used in Cerealizer files. It defaults to the full classname (module.class)
but you may choose something shorter -- as long as there is no risk of name clash."""
if not _configurable: raise StandardError("Cannot register new classes after freeze_configuration has been called!")
if "\n" in classname: raise ValueError("CLASSNAME cannot have \\n (Cerealizer automatically add a trailing \\n for performance reason)!")
if not handler:
if hasattr(Class, "__getnewargs__" ): handler = NewArgsObjHandler (Class, classname)
elif hasattr(Class, "__getinitargs__"): handler = InitArgsObjHandler(Class, classname)
elif hasattr(Class, "__slots__" ): handler = SlotedObjHandler (Class, classname)
else: handler = ObjHandler (Class, classname)
if _HANDLERS_.has_key(Class): raise ValueError("Class %s has already been registred!" % Class)
if not isinstance(handler, RefHandler):
if _HANDLERS .has_key(handler.classname): raise ValueError("A class has already been registred under the name %s!" % handler.classname[:-1])
_HANDLERS [handler.classname] = handler
if handler.__class__ is ObjHandler:
logger.info("Registring class %s as '%s'" % (Class, handler.classname[:-1]))
else:
logger.info("Registring class %s as '%s' (using %s)" % (Class, handler.classname[:-1], handler.__class__.__name__))
else:
logger.info("Registring reference '%s'" % Class)
_HANDLERS_[Class] = handler
register_class = register # For backward compatibility
def register_alias(Class, alias):
"""register_alias(Class, alias)
Registers ALIAS as an alias classname for CLASS.
Usefull for keeping backward compatibility in files: e.g. if you have renamed OldClass to
NewClass, just do:
cerealizer.register_alias(NewClass, "OldClass")
and you'll be able to open old files containing OldClass serialized."""
handler = _HANDLERS_.get(Class)
if not handler:
raise ValueError("Cannot register alias '%s' to Class %s: the class is not yet registred!" % (alias, Class))
if _HANDLERS.has_key(alias):
raise ValueError("Cannot register alias '%s' to Class %s: another class is already registred under the alias name!" % (alias, Class))
logger.info("Registring alias '%s' for %s" % (alias, Class))
_HANDLERS[alias + "\n"] = handler
def freeze_configuration():
"""freeze_configuration()
Ends Cerealizer configuration. When freeze_configuration() is called, it is no longer possible
to register classes, using register().
Calling freeze_configuration() is not mandatory, but it may enforce security, by forbidding
unexpected calls to register()."""
global _configurable
_configurable = 0
logger.info("Configuration frozen")
register(type(None), NoneHandler ())
register(str , StrHandler ())
register(unicode , UnicodeHandler ())
register(bool , BoolHandler ())
register(int , IntHandler ())
register(long , LongHandler ())
register(float , FloatHandler ())
register(complex , ComplexHandler ())
register(dict , DictHandler ())
register(list , ListHandler ())
register(set , SetHandler ())
register(tuple , TupleHandler ())
register(frozenset , FrozensetHandler())
def dump(obj, file, protocol = 0):
"""dump(obj, file, protocol = 0)
Serializes object OBJ in FILE.
FILE should be an opened file in *** binary *** mode.
PROTOCOL is unused, it exists only for compatibility with Pickle."""
Dumper().dump(obj, file)
def load(file):
"""load(file) -> obj
De-serializes an object from FILE.
FILE should be an opened file in *** binary *** mode."""
return Dumper().undump(file)
def dumps(obj, protocol = 0):
"""dumps(obj, protocol = 0) -> str
Serializes object OBJ and returns the serialized string.
PROTOCOL is unused, it exists only for compatibility with Pickle."""
s = StringIO()
Dumper().dump(obj, s)
return s.getvalue()
def loads(string):
"""loads(file) -> obj
De-serializes an object from STRING."""
return Dumper().undump(StringIO(string))
def dump_class_of_module(*modules):
"""dump_class_of_module(*modules)
Utility function; for each classes found in the given module, print the needed call to register."""
class D: pass
class O(object): pass
s = set([c for module in modules for c in module.__dict__.values() if isinstance(c, type(D)) or isinstance(c, type(O))])
l = ['cerealizer.register(%s.%s)' % (c.__module__, c.__name__) for c in s]
l.sort()
for i in l: print i