14
15
# along with this program; if not, write to the Free Software
15
16
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17
# TODO: Could remember a bias towards whether a particular store is typically
21
Stores are the main data-storage mechanism for Bazaar-NG.
18
"""Stores are the main data-storage mechanism for Bazaar-NG.
23
20
A store is a simple write-once container indexed by a universally
21
unique ID, which is typically the SHA-1 of the content."""
23
__copyright__ = "Copyright (C) 2005 Canonical Ltd."
24
__author__ = "Martin Pool <mbp@canonical.com>"
26
import os, tempfile, types, osutils, gzip, errno
32
27
from stat import ST_SIZE
33
28
from StringIO import StringIO
35
from bzrlib.errors import BzrError, UnlistableStore
36
from bzrlib.trace import mutter
38
import bzrlib.osutils as osutils
29
from trace import mutter
41
31
######################################################################
68
58
>>> st['123123'].read()
71
TODO: Atomic add by writing to a temporary file and renaming.
73
In bzr 0.0.5 and earlier, files within the store were marked
74
readonly on disk. This is no longer done but existing stores need
61
:todo: Atomic add by writing to a temporary file and renaming.
63
:todo: Perhaps automatically transform to/from XML in a method?
64
Would just need to tell the constructor what class to
67
:todo: Even within a simple disk store like this, we could
68
gzip the files. But since many are less than one disk
69
block, that might not help a lot.
78
73
def __init__(self, basedir):
74
"""ImmutableStore constructor."""
79
75
self._basedir = basedir
81
def _path(self, entry_id):
82
if not isinstance(entry_id, basestring):
83
raise TypeError(type(entry_id))
84
if '\\' in entry_id or '/' in entry_id:
85
raise ValueError("invalid store id %r" % entry_id)
86
return os.path.join(self._basedir, entry_id)
78
return os.path.join(self._basedir, id)
88
80
def __repr__(self):
89
81
return "%s(%r)" % (self.__class__.__name__, self._basedir)
91
83
def add(self, f, fileid, compressed=True):
92
84
"""Add contents of a file into the store.
94
f -- An open file, or file-like object."""
95
# FIXME: Only works on files that will fit in memory
97
from bzrlib.atomicfile import AtomicFile
86
:param f: An open file, or file-like object."""
87
# FIXME: Only works on smallish files
88
# TODO: Can be optimized by copying at the same time as
99
90
mutter("add store entry %r" % (fileid))
100
91
if isinstance(f, types.StringTypes):
103
94
content = f.read()
105
96
p = self._path(fileid)
106
97
if os.access(p, os.F_OK) or os.access(p + '.gz', os.F_OK):
107
raise BzrError("store %r already contains id %r" % (self._basedir, fileid))
98
bailout("store %r already contains id %r" % (self._basedir, fileid))
113
af = AtomicFile(fn, 'wb')
116
gf = gzip.GzipFile(mode='wb', fileobj=af)
126
def copy_multi(self, other, ids, permit_failure=False):
127
"""Copy texts for ids from other into self.
129
If an id is present in self, it is skipped.
131
Returns (count_copied, failed), where failed is a collection of ids
132
that could not be copied.
134
pb = bzrlib.ui.ui_factory.progress_bar()
136
pb.update('preparing to copy')
137
to_copy = [id for id in ids if id not in self]
138
if isinstance(other, ImmutableStore):
139
return self.copy_multi_immutable(other, to_copy, pb,
140
permit_failure=permit_failure)
145
pb.update('copy', count, len(to_copy))
146
if not permit_failure:
147
self.add(other[id], id)
156
if not permit_failure:
157
assert count == len(to_copy)
161
def copy_multi_immutable(self, other, to_copy, pb, permit_failure=False):
166
other_p = other._path(id)
168
osutils.link_or_copy(other_p, p)
169
except (IOError, OSError), e:
170
if e.errno == errno.ENOENT:
171
if not permit_failure:
172
osutils.link_or_copy(other_p+".gz", p+".gz")
175
osutils.link_or_copy(other_p+".gz", p+".gz")
177
if e.errno == errno.ENOENT:
185
pb.update('copy', count, len(to_copy))
186
assert count == len(to_copy)
101
f = gzip.GzipFile(p + '.gz', 'wb')
102
os.chmod(p + '.gz', 0444)
191
111
def __contains__(self, fileid):
265
176
os.rmdir(self._basedir)
266
177
mutter("%r destroyed" % self)
268
def copy_all(store_from, store_to):
269
"""Copy all ids from one store to another."""
270
if not hasattr(store_from, "__iter__"):
271
raise UnlistableStore(store_from)
272
ids = [f for f in store_from]
273
store_to.copy_multi(store_from, ids)