1"""Manage shelves of pickled objects.
2
3A "shelf" is a persistent, dictionary-like object.  The difference
4with dbm databases is that the values (not the keys!) in a shelf can
5be essentially arbitrary Python objects -- anything that the "pickle"
6module can handle.  This includes most class instances, recursive data
7types, and objects containing lots of shared sub-objects.  The keys
8are ordinary strings.
9
10To summarize the interface (key is a string, data is an arbitrary
11object):
12
13        import shelve
14        d = shelve.open(filename) # open, with (g)dbm filename -- no suffix
15
16        d[key] = data   # store data at key (overwrites old data if
17                        # using an existing key)
18        data = d[key]   # retrieve a COPY of the data at key (raise
19                        # KeyError if no such key) -- NOTE that this
20                        # access returns a *copy* of the entry!
21        del d[key]      # delete data stored at key (raises KeyError
22                        # if no such key)
23        flag = key in d # true if the key exists
24        list = d.keys() # a list of all existing keys (slow!)
25
26        d.close()       # close it
27
28Dependent on the implementation, closing a persistent dictionary may
29or may not be necessary to flush changes to disk.
30
31Normally, d[key] returns a COPY of the entry.  This needs care when
32mutable entries are mutated: for example, if d[key] is a list,
33        d[key].append(anitem)
34does NOT modify the entry d[key] itself, as stored in the persistent
35mapping -- it only modifies the copy, which is then immediately
36discarded, so that the append has NO effect whatsoever.  To append an
37item to d[key] in a way that will affect the persistent mapping, use:
38        data = d[key]
39        data.append(anitem)
40        d[key] = data
41
42To avoid the problem with mutable entries, you may pass the keyword
43argument writeback=True in the call to shelve.open.  When you use:
44        d = shelve.open(filename, writeback=True)
45then d keeps a cache of all entries you access, and writes them all back
46to the persistent mapping when you call d.close().  This ensures that
47such usage as d[key].append(anitem) works as intended.
48
49However, using keyword argument writeback=True may consume vast amount
50of memory for the cache, and it may make d.close() very slow, if you
51access many of d's entries after opening it in this way: d has no way to
52check which of the entries you access are mutable and/or which ones you
53actually mutate, so it must cache, and write back at close, all of the
54entries that you access.  You can call d.sync() to write back all the
55entries in the cache, and empty the cache (d.sync() also synchronizes
56the persistent dictionary on disk, if feasible).
57"""
58
59from pickle import Pickler, Unpickler
60from io import BytesIO
61
62import collections.abc
63
64__all__ = ["Shelf", "BsdDbShelf", "DbfilenameShelf", "open"]
65
66class _ClosedDict(collections.abc.MutableMapping):
67    'Marker for a closed dict.  Access attempts raise a ValueError.'
68
69    def closed(self, *args):
70        raise ValueError('invalid operation on closed shelf')
71    __iter__ = __len__ = __getitem__ = __setitem__ = __delitem__ = keys = closed
72
73    def __repr__(self):
74        return '<Closed Dictionary>'
75
76
77class Shelf(collections.abc.MutableMapping):
78    """Base class for shelf implementations.
79
80    This is initialized with a dictionary-like object.
81    See the module's __doc__ string for an overview of the interface.
82    """
83
84    def __init__(self, dict, protocol=None, writeback=False,
85                 keyencoding="utf-8"):
86        self.dict = dict
87        if protocol is None:
88            protocol = 3
89        self._protocol = protocol
90        self.writeback = writeback
91        self.cache = {}
92        self.keyencoding = keyencoding
93
94    def __iter__(self):
95        for k in self.dict.keys():
96            yield k.decode(self.keyencoding)
97
98    def __len__(self):
99        return len(self.dict)
100
101    def __contains__(self, key):
102        return key.encode(self.keyencoding) in self.dict
103
104    def get(self, key, default=None):
105        if key.encode(self.keyencoding) in self.dict:
106            return self[key]
107        return default
108
109    def __getitem__(self, key):
110        try:
111            value = self.cache[key]
112        except KeyError:
113            f = BytesIO(self.dict[key.encode(self.keyencoding)])
114            value = Unpickler(f).load()
115            if self.writeback:
116                self.cache[key] = value
117        return value
118
119    def __setitem__(self, key, value):
120        if self.writeback:
121            self.cache[key] = value
122        f = BytesIO()
123        p = Pickler(f, self._protocol)
124        p.dump(value)
125        self.dict[key.encode(self.keyencoding)] = f.getvalue()
126
127    def __delitem__(self, key):
128        del self.dict[key.encode(self.keyencoding)]
129        try:
130            del self.cache[key]
131        except KeyError:
132            pass
133
134    def __enter__(self):
135        return self
136
137    def __exit__(self, type, value, traceback):
138        self.close()
139
140    def close(self):
141        if self.dict is None:
142            return
143        try:
144            self.sync()
145            try:
146                self.dict.close()
147            except AttributeError:
148                pass
149        finally:
150            # Catch errors that may happen when close is called from __del__
151            # because CPython is in interpreter shutdown.
152            try:
153                self.dict = _ClosedDict()
154            except:
155                self.dict = None
156
157    def __del__(self):
158        if not hasattr(self, 'writeback'):
159            # __init__ didn't succeed, so don't bother closing
160            # see http://bugs.python.org/issue1339007 for details
161            return
162        self.close()
163
164    def sync(self):
165        if self.writeback and self.cache:
166            self.writeback = False
167            for key, entry in self.cache.items():
168                self[key] = entry
169            self.writeback = True
170            self.cache = {}
171        if hasattr(self.dict, 'sync'):
172            self.dict.sync()
173
174
175class BsdDbShelf(Shelf):
176    """Shelf implementation using the "BSD" db interface.
177
178    This adds methods first(), next(), previous(), last() and
179    set_location() that have no counterpart in [g]dbm databases.
180
181    The actual database must be opened using one of the "bsddb"
182    modules "open" routines (i.e. bsddb.hashopen, bsddb.btopen or
183    bsddb.rnopen) and passed to the constructor.
184
185    See the module's __doc__ string for an overview of the interface.
186    """
187
188    def __init__(self, dict, protocol=None, writeback=False,
189                 keyencoding="utf-8"):
190        Shelf.__init__(self, dict, protocol, writeback, keyencoding)
191
192    def set_location(self, key):
193        (key, value) = self.dict.set_location(key)
194        f = BytesIO(value)
195        return (key.decode(self.keyencoding), Unpickler(f).load())
196
197    def next(self):
198        (key, value) = next(self.dict)
199        f = BytesIO(value)
200        return (key.decode(self.keyencoding), Unpickler(f).load())
201
202    def previous(self):
203        (key, value) = self.dict.previous()
204        f = BytesIO(value)
205        return (key.decode(self.keyencoding), Unpickler(f).load())
206
207    def first(self):
208        (key, value) = self.dict.first()
209        f = BytesIO(value)
210        return (key.decode(self.keyencoding), Unpickler(f).load())
211
212    def last(self):
213        (key, value) = self.dict.last()
214        f = BytesIO(value)
215        return (key.decode(self.keyencoding), Unpickler(f).load())
216
217
218class DbfilenameShelf(Shelf):
219    """Shelf implementation using the "dbm" generic dbm interface.
220
221    This is initialized with the filename for the dbm database.
222    See the module's __doc__ string for an overview of the interface.
223    """
224
225    def __init__(self, filename, flag='c', protocol=None, writeback=False):
226        import dbm
227        Shelf.__init__(self, dbm.open(filename, flag), protocol, writeback)
228
229
230def open(filename, flag='c', protocol=None, writeback=False):
231    """Open a persistent dictionary for reading and writing.
232
233    The filename parameter is the base filename for the underlying
234    database.  As a side-effect, an extension may be added to the
235    filename and more than one file may be created.  The optional flag
236    parameter has the same interpretation as the flag parameter of
237    dbm.open(). The optional protocol parameter specifies the
238    version of the pickle protocol.
239
240    See the module's __doc__ string for an overview of the interface.
241    """
242
243    return DbfilenameShelf(filename, flag, protocol, writeback)
244