blob: 3068b4147b77b858a627212b5e9090f32b8d2ba2 [file] [log] [blame]
Guido van Rossume467be61997-12-05 19:42:42 +00001"""create portable serialized representations of Python objects.
Guido van Rossuma48061a1995-01-10 00:31:14 +00002
Guido van Rossume467be61997-12-05 19:42:42 +00003See module cPickle for a (much) faster implementation.
4See module copy_reg for a mechanism for registering custom picklers.
Guido van Rossuma48061a1995-01-10 00:31:14 +00005
Guido van Rossume467be61997-12-05 19:42:42 +00006Classes:
Guido van Rossuma48061a1995-01-10 00:31:14 +00007
Guido van Rossume467be61997-12-05 19:42:42 +00008 Pickler
9 Unpickler
Guido van Rossuma48061a1995-01-10 00:31:14 +000010
Guido van Rossume467be61997-12-05 19:42:42 +000011Functions:
Guido van Rossuma48061a1995-01-10 00:31:14 +000012
Guido van Rossume467be61997-12-05 19:42:42 +000013 dump(object, file)
14 dumps(object) -> string
15 load(file) -> object
16 loads(string) -> object
Guido van Rossuma48061a1995-01-10 00:31:14 +000017
Guido van Rossume467be61997-12-05 19:42:42 +000018Misc variables:
Guido van Rossuma48061a1995-01-10 00:31:14 +000019
Guido van Rossume467be61997-12-05 19:42:42 +000020 __ version__
21 format_version
22 compatible_formats
Guido van Rossuma48061a1995-01-10 00:31:14 +000023
Guido van Rossuma48061a1995-01-10 00:31:14 +000024"""
25
Guido van Rossume467be61997-12-05 19:42:42 +000026__version__ = "1.9" # Code version
Guido van Rossuma48061a1995-01-10 00:31:14 +000027
28from types import *
Guido van Rossum4fb5b281997-09-12 20:07:24 +000029from copy_reg import dispatch_table, safe_constructors
Guido van Rossumb72cf2d1997-04-09 17:32:51 +000030import string, marshal
Guido van Rossuma48061a1995-01-10 00:31:14 +000031
Guido van Rossumb72cf2d1997-04-09 17:32:51 +000032format_version = "1.2" # File format version we write
33compatible_formats = ["1.0", "1.1"] # Old format versions we can read
34
35mdumps = marshal.dumps
36mloads = marshal.loads
Guido van Rossum0c891ce1995-03-14 15:09:05 +000037
Guido van Rossum7849da81995-03-09 14:08:35 +000038PicklingError = "pickle.PicklingError"
Guido van Rossumb72cf2d1997-04-09 17:32:51 +000039UnpicklingError = "pickle.UnpicklingError"
Guido van Rossum7849da81995-03-09 14:08:35 +000040
Guido van Rossumb72cf2d1997-04-09 17:32:51 +000041MARK = '('
42STOP = '.'
43POP = '0'
44POP_MARK = '1'
45DUP = '2'
46FLOAT = 'F'
47INT = 'I'
48BININT = 'J'
49BININT1 = 'K'
50LONG = 'L'
51BININT2 = 'M'
52NONE = 'N'
53PERSID = 'P'
54BINPERSID = 'Q'
55REDUCE = 'R'
56STRING = 'S'
57BINSTRING = 'T'
58SHORT_BINSTRING = 'U'
59APPEND = 'a'
60BUILD = 'b'
61GLOBAL = 'c'
62DICT = 'd'
63EMPTY_DICT = '}'
64APPENDS = 'e'
65GET = 'g'
66BINGET = 'h'
67INST = 'i'
68LONG_BINGET = 'j'
69LIST = 'l'
70EMPTY_LIST = ']'
71OBJ = 'o'
72PUT = 'p'
73BINPUT = 'q'
74LONG_BINPUT = 'r'
75SETITEM = 's'
76TUPLE = 't'
77EMPTY_TUPLE = ')'
78SETITEMS = 'u'
Guido van Rossuma48061a1995-01-10 00:31:14 +000079
80class Pickler:
81
Guido van Rossumb72cf2d1997-04-09 17:32:51 +000082 def __init__(self, file, bin = 0):
83 self.write = file.write
84 self.memo = {}
85 self.bin = bin
Guido van Rossuma48061a1995-01-10 00:31:14 +000086
Guido van Rossumb72cf2d1997-04-09 17:32:51 +000087 def dump(self, object):
88 self.save(object)
89 self.write(STOP)
Guido van Rossuma48061a1995-01-10 00:31:14 +000090
Guido van Rossumb72cf2d1997-04-09 17:32:51 +000091 def dump_special(self, callable, args, state = None):
92 if (type(args) is not TupleType):
93 raise PicklingError, "Second argument to dump_special " \
94 "must be a tuple"
Guido van Rossuma48061a1995-01-10 00:31:14 +000095
Guido van Rossumb72cf2d1997-04-09 17:32:51 +000096 self.save_reduce(callable, args, state)
97 self.write(STOP)
Guido van Rossuma48061a1995-01-10 00:31:14 +000098
Guido van Rossumb72cf2d1997-04-09 17:32:51 +000099 def put(self, i):
100 if (self.bin):
101 s = mdumps(i)[1:]
102 if (i < 256):
103 return BINPUT + s[0]
Guido van Rossuma48061a1995-01-10 00:31:14 +0000104
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000105 return LONG_BINPUT + s
Guido van Rossuma48061a1995-01-10 00:31:14 +0000106
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000107 return PUT + `i` + '\n'
Guido van Rossuma48061a1995-01-10 00:31:14 +0000108
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000109 def get(self, i):
110 if (self.bin):
111 s = mdumps(i)[1:]
Guido van Rossuma48061a1995-01-10 00:31:14 +0000112
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000113 if (i < 256):
114 return BINGET + s[0]
Guido van Rossuma48061a1995-01-10 00:31:14 +0000115
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000116 return LONG_BINGET + s
Guido van Rossuma48061a1995-01-10 00:31:14 +0000117
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000118 return GET + `i` + '\n'
119
120 def save(self, object, pers_save = 0):
121 memo = self.memo
Guido van Rossuma48061a1995-01-10 00:31:14 +0000122
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000123 if (not pers_save):
124 pid = self.persistent_id(object)
125 if (pid is not None):
126 self.save_pers(pid)
127 return
Guido van Rossuma48061a1995-01-10 00:31:14 +0000128
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000129 d = id(object)
130
131 t = type(object)
Guido van Rossuma48061a1995-01-10 00:31:14 +0000132
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000133 if ((t is TupleType) and (len(object) == 0)):
134 if (self.bin):
135 self.save_empty_tuple(object)
136 else:
137 self.save_tuple(object)
138 return
Guido van Rossuma48061a1995-01-10 00:31:14 +0000139
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000140 if memo.has_key(d):
141 self.write(self.get(memo[d][0]))
142 return
143
144 try:
145 f = self.dispatch[t]
146 except KeyError:
147 pid = self.inst_persistent_id(object)
148 if pid is not None:
149 self.save_pers(pid)
150 return
151
152 try:
153 reduce = dispatch_table[t]
154 except KeyError:
155 try:
156 reduce = object.__reduce__
157 except AttributeError:
158 raise PicklingError, \
159 "can't pickle %s objects" % `t.__name__`
160 else:
161 tup = reduce()
162 else:
163 tup = reduce(object)
164
165 if (type(tup) is not TupleType):
166 raise PicklingError, "Value returned by %s must be a " \
167 "tuple" % reduce
168
169 l = len(tup)
170
171 if ((l != 2) and (l != 3)):
172 raise PicklingError, "tuple returned by %s must contain " \
173 "only two or three elements" % reduce
174
175 callable = tup[0]
176 arg_tup = tup[1]
177
178 if (l > 2):
179 state = tup[2]
180 else:
181 state = None
182
183 if (type(arg_tup) is not TupleType):
184 raise PicklingError, "Second element of tuple returned " \
185 "by %s must be a tuple" % reduce
186
187 self.save_reduce(callable, arg_tup, state)
Guido van Rossum4fb5b281997-09-12 20:07:24 +0000188 memo_len = len(memo)
189 self.write(self.put(memo_len))
190 memo[d] = (memo_len, object)
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000191 return
192
193 f(self, object)
194
195 def persistent_id(self, object):
196 return None
197
198 def inst_persistent_id(self, object):
199 return None
200
201 def save_pers(self, pid):
202 if (not self.bin):
203 self.write(PERSID + str(pid) + '\n')
204 else:
205 self.save(pid, 1)
206 self.write(BINPERSID)
207
208 def save_reduce(self, callable, arg_tup, state = None):
209 write = self.write
210 save = self.save
211
212 save(callable)
213 save(arg_tup)
214 write(REDUCE)
215
216 if (state is not None):
217 save(state)
218 write(BUILD)
219
220 dispatch = {}
221
222 def save_none(self, object):
223 self.write(NONE)
224 dispatch[NoneType] = save_none
225
226 def save_int(self, object):
227 if (self.bin):
228 i = mdumps(object)[1:]
229 if (i[-2:] == '\000\000'):
230 if (i[-3] == '\000'):
231 self.write(BININT1 + i[:-3])
232 return
233
234 self.write(BININT2 + i[:-2])
235 return
236
237 self.write(BININT + i)
238 else:
239 self.write(INT + `object` + '\n')
240 dispatch[IntType] = save_int
241
242 def save_long(self, object):
243 self.write(LONG + `object` + '\n')
244 dispatch[LongType] = save_long
245
246 def save_float(self, object):
247 self.write(FLOAT + `object` + '\n')
248 dispatch[FloatType] = save_float
249
250 def save_string(self, object):
251 d = id(object)
252 memo = self.memo
253
254 if (self.bin):
255 l = len(object)
256 s = mdumps(l)[1:]
257 if (l < 256):
258 self.write(SHORT_BINSTRING + s[0] + object)
259 else:
260 self.write(BINSTRING + s + object)
261 else:
262 self.write(STRING + `object` + '\n')
263
264 memo_len = len(memo)
265 self.write(self.put(memo_len))
266 memo[d] = (memo_len, object)
267 dispatch[StringType] = save_string
268
269 def save_tuple(self, object):
270
271 write = self.write
272 save = self.save
273 memo = self.memo
274
275 d = id(object)
276
277 write(MARK)
278
279 for element in object:
280 save(element)
281
282 if (len(object) and memo.has_key(d)):
283 if (self.bin):
284 write(POP_MARK + self.get(memo[d][0]))
285 return
286
287 write(POP * (len(object) + 1) + self.get(mem[d][0]))
288 return
289
290 memo_len = len(memo)
291 self.write(TUPLE + self.put(memo_len))
292 memo[d] = (memo_len, object)
293 dispatch[TupleType] = save_tuple
294
295 def save_empty_tuple(self, object):
296 self.write(EMPTY_TUPLE)
297
298 def save_list(self, object):
299 d = id(object)
300
301 write = self.write
302 save = self.save
303 memo = self.memo
304
305 if (self.bin):
306 write(EMPTY_LIST)
307 else:
308 write(MARK + LIST)
309
310 memo_len = len(memo)
311 write(self.put(memo_len))
312 memo[d] = (memo_len, object)
313
314 using_appends = (self.bin and (len(object) > 1))
315
316 if (using_appends):
317 write(MARK)
318
319 for element in object:
320 save(element)
321
322 if (not using_appends):
323 write(APPEND)
324
325 if (using_appends):
326 write(APPENDS)
327 dispatch[ListType] = save_list
328
329 def save_dict(self, object):
330 d = id(object)
331
332 write = self.write
333 save = self.save
334 memo = self.memo
335
336 if (self.bin):
337 write(EMPTY_DICT)
338 else:
339 write(MARK + DICT)
340
341 memo_len = len(memo)
342 self.write(self.put(memo_len))
343 memo[d] = (memo_len, object)
344
345 using_setitems = (self.bin and (len(object) > 1))
346
347 if (using_setitems):
348 write(MARK)
349
350 items = object.items()
351 for key, value in items:
352 save(key)
353 save(value)
354
355 if (not using_setitems):
356 write(SETITEM)
357
358 if (using_setitems):
359 write(SETITEMS)
360
361 dispatch[DictionaryType] = save_dict
362
363 def save_inst(self, object):
364 d = id(object)
365 cls = object.__class__
366
367 memo = self.memo
368 write = self.write
369 save = self.save
370
371 if hasattr(object, '__getinitargs__'):
372 args = object.__getinitargs__()
373 len(args) # XXX Assert it's a sequence
Guido van Rossum5ed5c4c1997-09-03 00:23:54 +0000374 _keep_alive(args, memo)
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000375 else:
376 args = ()
377
378 write(MARK)
379
380 if (self.bin):
381 save(cls)
382
383 for arg in args:
384 save(arg)
385
386 memo_len = len(memo)
387 if (self.bin):
388 write(OBJ + self.put(memo_len))
389 else:
Guido van Rossum4fb5b281997-09-12 20:07:24 +0000390 write(INST + cls.__module__ + '\n' + cls.__name__ + '\n' +
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000391 self.put(memo_len))
392
393 memo[d] = (memo_len, object)
394
395 try:
396 getstate = object.__getstate__
397 except AttributeError:
398 stuff = object.__dict__
399 else:
400 stuff = getstate()
Guido van Rossum5ed5c4c1997-09-03 00:23:54 +0000401 _keep_alive(stuff, memo)
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000402 save(stuff)
403 write(BUILD)
404 dispatch[InstanceType] = save_inst
405
406 def save_global(self, object, name = None):
407 write = self.write
408 memo = self.memo
409
410 if (name is None):
411 name = object.__name__
412
Guido van Rossum4fb5b281997-09-12 20:07:24 +0000413 try:
414 module = object.__module__
415 except AttributeError:
416 module = whichmodule(object, name)
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000417
418 memo_len = len(memo)
419 write(GLOBAL + module + '\n' + name + '\n' +
420 self.put(memo_len))
421 memo[id(object)] = (memo_len, object)
422 dispatch[ClassType] = save_global
423 dispatch[FunctionType] = save_global
424 dispatch[BuiltinFunctionType] = save_global
Guido van Rossum0c891ce1995-03-14 15:09:05 +0000425
Guido van Rossuma48061a1995-01-10 00:31:14 +0000426
Guido van Rossum5ed5c4c1997-09-03 00:23:54 +0000427def _keep_alive(x, memo):
428 """Keeps a reference to the object x in the memo.
429
430 Because we remember objects by their id, we have
431 to assure that possibly temporary objects are kept
432 alive by referencing them.
433 We store a reference at the id of the memo, which should
434 normally not be used unless someone tries to deepcopy
435 the memo itself...
436 """
437 try:
438 memo[id(memo)].append(x)
439 except KeyError:
440 # aha, this is the first one :-)
441 memo[id(memo)]=[x]
442
443
Guido van Rossuma48061a1995-01-10 00:31:14 +0000444classmap = {}
445
Guido van Rossum4fb5b281997-09-12 20:07:24 +0000446# This is no longer used to find classes, but still for functions
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000447def whichmodule(cls, clsname):
448 """Figure out the module in which a class occurs.
Guido van Rossuma48061a1995-01-10 00:31:14 +0000449
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000450 Search sys.modules for the module.
451 Cache in classmap.
452 Return a module name.
453 If the class cannot be found, return __main__.
454 """
455 if classmap.has_key(cls):
456 return classmap[cls]
457 import sys
458
459 for name, module in sys.modules.items():
Guido van Rossum8be9a111997-04-25 19:52:29 +0000460 if name != '__main__' and \
461 hasattr(module, clsname) and \
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000462 getattr(module, clsname) is cls:
463 break
464 else:
465 name = '__main__'
466 classmap[cls] = name
467 return name
Guido van Rossuma48061a1995-01-10 00:31:14 +0000468
469
470class Unpickler:
471
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000472 def __init__(self, file):
473 self.readline = file.readline
474 self.read = file.read
475 self.memo = {}
Guido van Rossuma48061a1995-01-10 00:31:14 +0000476
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000477 def load(self):
478 self.mark = ['spam'] # Any new unique object
479 self.stack = []
480 self.append = self.stack.append
481 read = self.read
482 dispatch = self.dispatch
483 try:
484 while 1:
485 key = read(1)
486 dispatch[key](self)
487 except STOP, value:
488 return value
489
490 def marker(self):
491 stack = self.stack
492 mark = self.mark
493 k = len(stack)-1
494 while stack[k] is not mark: k = k-1
495 return k
496
497 dispatch = {}
498
499 def load_eof(self):
500 raise EOFError
501 dispatch[''] = load_eof
502
503 def load_persid(self):
504 pid = self.readline()[:-1]
505 self.append(self.persistent_load(pid))
506 dispatch[PERSID] = load_persid
507
508 def load_binpersid(self):
509 stack = self.stack
510
511 pid = stack[-1]
512 del stack[-1]
513
514 self.append(self.persistent_load(pid))
515 dispatch[BINPERSID] = load_binpersid
516
517 def load_none(self):
518 self.append(None)
519 dispatch[NONE] = load_none
520
521 def load_int(self):
522 self.append(string.atoi(self.readline()[:-1], 0))
523 dispatch[INT] = load_int
524
525 def load_binint(self):
526 self.append(mloads('i' + self.read(4)))
527 dispatch[BININT] = load_binint
528
529 def load_binint1(self):
530 self.append(mloads('i' + self.read(1) + '\000\000\000'))
531 dispatch[BININT1] = load_binint1
532
533 def load_binint2(self):
534 self.append(mloads('i' + self.read(2) + '\000\000'))
535 dispatch[BININT2] = load_binint2
536
537 def load_long(self):
538 self.append(string.atol(self.readline()[:-1], 0))
539 dispatch[LONG] = load_long
540
541 def load_float(self):
542 self.append(string.atof(self.readline()[:-1]))
543 dispatch[FLOAT] = load_float
544
545 def load_string(self):
546 self.append(eval(self.readline()[:-1],
547 {'__builtins__': {}})) # Let's be careful
548 dispatch[STRING] = load_string
549
550 def load_binstring(self):
551 len = mloads('i' + self.read(4))
552 self.append(self.read(len))
553 dispatch[BINSTRING] = load_binstring
554
555 def load_short_binstring(self):
556 len = mloads('i' + self.read(1) + '\000\000\000')
557 self.append(self.read(len))
558 dispatch[SHORT_BINSTRING] = load_short_binstring
559
560 def load_tuple(self):
561 k = self.marker()
562 self.stack[k:] = [tuple(self.stack[k+1:])]
563 dispatch[TUPLE] = load_tuple
564
565 def load_empty_tuple(self):
566 self.stack.append(())
567 dispatch[EMPTY_TUPLE] = load_empty_tuple
568
569 def load_empty_list(self):
570 self.stack.append([])
571 dispatch[EMPTY_LIST] = load_empty_list
572
573 def load_empty_dictionary(self):
574 self.stack.append({})
575 dispatch[EMPTY_DICT] = load_empty_dictionary
576
577 def load_list(self):
578 k = self.marker()
579 self.stack[k:] = [self.stack[k+1:]]
580 dispatch[LIST] = load_list
581
582 def load_dict(self):
583 k = self.marker()
584 d = {}
585 items = self.stack[k+1:]
586 for i in range(0, len(items), 2):
587 key = items[i]
588 value = items[i+1]
589 d[key] = value
590 self.stack[k:] = [d]
591 dispatch[DICT] = load_dict
592
593 def load_inst(self):
594 k = self.marker()
595 args = tuple(self.stack[k+1:])
596 del self.stack[k:]
597 module = self.readline()[:-1]
598 name = self.readline()[:-1]
599 klass = self.find_class(module, name)
Guido van Rossume467be61997-12-05 19:42:42 +0000600 if (not args and type(klass) is ClassType and
601 not hasattr(klass, "__getinitargs__")):
602 value = _EmptyClass()
603 value.__class__ = klass
604 else:
605 value = apply(klass, args)
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000606 self.append(value)
607 dispatch[INST] = load_inst
608
609 def load_obj(self):
610 stack = self.stack
611 k = self.marker()
612 klass = stack[k + 1]
613 del stack[k + 1]
614 args = tuple(stack[k + 1:])
615 del stack[k:]
Guido van Rossume467be61997-12-05 19:42:42 +0000616 if (not args and type(klass) is ClassType and
617 not hasattr(klass, "__getinitargs__")):
618 value = _EmptyClass()
619 value.__class__ = klass
620 else:
621 value = apply(klass, args)
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000622 self.append(value)
623 dispatch[OBJ] = load_obj
624
625 def load_global(self):
626 module = self.readline()[:-1]
627 name = self.readline()[:-1]
628 klass = self.find_class(module, name)
629 self.append(klass)
630 dispatch[GLOBAL] = load_global
631
632 def find_class(self, module, name):
633 env = {}
634
635 try:
636 exec 'from %s import %s' % (module, name) in env
637 except ImportError:
638 raise SystemError, \
639 "Failed to import class %s from module %s" % \
640 (name, module)
641 klass = env[name]
642 return klass
643
644 def load_reduce(self):
645 stack = self.stack
646
647 callable = stack[-2]
648 arg_tup = stack[-1]
649 del stack[-2:]
650
651 if (type(callable) is not ClassType):
652 if (not safe_constructors.has_key(callable)):
Guido van Rossuma48061a1995-01-10 00:31:14 +0000653 try:
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000654 safe = callable.__safe_for_unpickling__
655 except AttributeError:
656 safe = None
Guido van Rossuma48061a1995-01-10 00:31:14 +0000657
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000658 if (not safe):
659 raise UnpicklingError, "%s is not safe for " \
660 "unpickling" % callable
Guido van Rossuma48061a1995-01-10 00:31:14 +0000661
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000662 value = apply(callable, arg_tup)
663 self.append(value)
664 dispatch[REDUCE] = load_reduce
Guido van Rossuma48061a1995-01-10 00:31:14 +0000665
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000666 def load_pop(self):
667 del self.stack[-1]
668 dispatch[POP] = load_pop
Guido van Rossum7b5430f1995-03-04 22:25:21 +0000669
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000670 def load_pop_mark(self):
671 k = self.marker()
672 del self.stack[k:]
673 dispatch[POP_MARK] = load_pop_mark
Guido van Rossuma48061a1995-01-10 00:31:14 +0000674
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000675 def load_dup(self):
676 self.append(stack[-1])
677 dispatch[DUP] = load_dup
Guido van Rossuma48061a1995-01-10 00:31:14 +0000678
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000679 def load_get(self):
680 self.append(self.memo[self.readline()[:-1]])
681 dispatch[GET] = load_get
Guido van Rossum78536471996-04-12 13:36:27 +0000682
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000683 def load_binget(self):
684 i = mloads('i' + self.read(1) + '\000\000\000')
685 self.append(self.memo[`i`])
686 dispatch[BINGET] = load_binget
Guido van Rossum78536471996-04-12 13:36:27 +0000687
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000688 def load_long_binget(self):
689 i = mloads('i' + self.read(4))
690 self.append(self.memo[`i`])
691 dispatch[LONG_BINGET] = load_long_binget
Guido van Rossum78536471996-04-12 13:36:27 +0000692
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000693 def load_put(self):
694 self.memo[self.readline()[:-1]] = self.stack[-1]
695 dispatch[PUT] = load_put
Guido van Rossuma48061a1995-01-10 00:31:14 +0000696
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000697 def load_binput(self):
698 i = mloads('i' + self.read(1) + '\000\000\000')
699 self.memo[`i`] = self.stack[-1]
700 dispatch[BINPUT] = load_binput
Guido van Rossuma48061a1995-01-10 00:31:14 +0000701
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000702 def load_long_binput(self):
703 i = mloads('i' + self.read(4))
704 self.memo[`i`] = self.stack[-1]
705 dispatch[LONG_BINPUT] = load_long_binput
Guido van Rossuma48061a1995-01-10 00:31:14 +0000706
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000707 def load_append(self):
708 stack = self.stack
709 value = stack[-1]
710 del stack[-1]
711 list = stack[-1]
712 list.append(value)
713 dispatch[APPEND] = load_append
Guido van Rossuma48061a1995-01-10 00:31:14 +0000714
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000715 def load_appends(self):
716 stack = self.stack
717 mark = self.marker()
718 list = stack[mark - 1]
719 for i in range(mark + 1, len(stack)):
720 list.append(stack[i])
Guido van Rossum0c891ce1995-03-14 15:09:05 +0000721
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000722 del stack[mark:]
723 dispatch[APPENDS] = load_appends
724
725 def load_setitem(self):
726 stack = self.stack
727 value = stack[-1]
728 key = stack[-2]
729 del stack[-2:]
730 dict = stack[-1]
731 dict[key] = value
732 dispatch[SETITEM] = load_setitem
Guido van Rossum0c891ce1995-03-14 15:09:05 +0000733
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000734 def load_setitems(self):
735 stack = self.stack
736 mark = self.marker()
737 dict = stack[mark - 1]
738 for i in range(mark + 1, len(stack), 2):
739 dict[stack[i]] = stack[i + 1]
Guido van Rossuma48061a1995-01-10 00:31:14 +0000740
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000741 del stack[mark:]
742 dispatch[SETITEMS] = load_setitems
Guido van Rossuma48061a1995-01-10 00:31:14 +0000743
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000744 def load_build(self):
745 stack = self.stack
746 value = stack[-1]
747 del stack[-1]
748 inst = stack[-1]
749 try:
750 setstate = inst.__setstate__
751 except AttributeError:
Guido van Rossumd6ead321997-09-08 02:08:11 +0000752 inst.__dict__.update(value)
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000753 else:
754 setstate(value)
755 dispatch[BUILD] = load_build
Guido van Rossuma48061a1995-01-10 00:31:14 +0000756
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000757 def load_mark(self):
758 self.append(self.mark)
759 dispatch[MARK] = load_mark
Guido van Rossuma48061a1995-01-10 00:31:14 +0000760
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000761 def load_stop(self):
762 value = self.stack[-1]
763 del self.stack[-1]
764 raise STOP, value
765 dispatch[STOP] = load_stop
Guido van Rossuma48061a1995-01-10 00:31:14 +0000766
Guido van Rossume467be61997-12-05 19:42:42 +0000767# Helper class for load_inst/load_obj
768
769class _EmptyClass:
770 pass
Guido van Rossuma48061a1995-01-10 00:31:14 +0000771
Guido van Rossum0c891ce1995-03-14 15:09:05 +0000772# Shorthands
773
Guido van Rossumc7c5e691996-07-22 22:26:07 +0000774from StringIO import StringIO
775
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000776def dump(object, file, bin = 0):
777 Pickler(file, bin).dump(object)
Guido van Rossum0c891ce1995-03-14 15:09:05 +0000778
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000779def dumps(object, bin = 0):
780 file = StringIO()
781 Pickler(file, bin).dump(object)
782 return file.getvalue()
Guido van Rossum0c891ce1995-03-14 15:09:05 +0000783
784def load(file):
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000785 return Unpickler(file).load()
Guido van Rossum0c891ce1995-03-14 15:09:05 +0000786
787def loads(str):
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000788 file = StringIO(str)
789 return Unpickler(file).load()
Guido van Rossum0c891ce1995-03-14 15:09:05 +0000790
791
792# The rest is used for testing only
793
Guido van Rossuma48061a1995-01-10 00:31:14 +0000794class C:
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000795 def __cmp__(self, other):
796 return cmp(self.__dict__, other.__dict__)
Guido van Rossuma48061a1995-01-10 00:31:14 +0000797
798def test():
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000799 fn = 'out'
800 c = C()
801 c.foo = 1
802 c.bar = 2
803 x = [0, 1, 2, 3]
804 y = ('abc', 'abc', c, c)
805 x.append(y)
806 x.append(y)
807 x.append(5)
808 f = open(fn, 'w')
809 F = Pickler(f)
810 F.dump(x)
811 f.close()
812 f = open(fn, 'r')
813 U = Unpickler(f)
814 x2 = U.load()
815 print x
816 print x2
817 print x == x2
818 print map(id, x)
819 print map(id, x2)
820 print F.memo
821 print U.memo
Guido van Rossuma48061a1995-01-10 00:31:14 +0000822
823if __name__ == '__main__':
Guido van Rossumb72cf2d1997-04-09 17:32:51 +0000824 test()