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
18
Stores are the main data-storage mechanism for Bazaar-NG.
18
"""Stores are the main data-storage mechanism for Bazaar-NG.
20
20
A store is a simple write-once container indexed by a universally
24
import os, tempfile, types, osutils, gzip, errno
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
25
27
from stat import ST_SIZE
26
28
from StringIO import StringIO
27
29
from trace import mutter
29
32
######################################################################
56
59
>>> st['123123'].read()
59
TODO: Atomic add by writing to a temporary file and renaming.
61
In bzr 0.0.5 and earlier, files within the store were marked
62
readonly on disk. This is no longer done but existing stores need
62
:todo: Atomic add by writing to a temporary file and renaming.
64
:todo: Perhaps automatically transform to/from XML in a method?
65
Would just need to tell the constructor what class to
68
:todo: Even within a simple disk store like this, we could
69
gzip the files. But since many are less than one disk
70
block, that might not help a lot.
66
74
def __init__(self, basedir):
75
"""ImmutableStore constructor."""
67
76
self._basedir = basedir
69
78
def _path(self, id):
70
if '\\' in id or '/' in id:
71
raise ValueError("invalid store id %r" % id)
72
79
return os.path.join(self._basedir, id)
74
81
def __repr__(self):
75
82
return "%s(%r)" % (self.__class__.__name__, self._basedir)
77
def add(self, f, fileid, compressed=True):
84
def add(self, f, fileid):
78
85
"""Add contents of a file into the store.
80
f -- An open file, or file-like object."""
81
# FIXME: Only works on files that will fit in memory
83
from bzrlib.atomicfile import AtomicFile
87
:param f: An open file, or file-like object."""
88
# FIXME: Only works on smallish files
89
# TODO: Can be optimized by copying at the same time as
85
91
mutter("add store entry %r" % (fileid))
86
92
if isinstance(f, types.StringTypes):
91
p = self._path(fileid)
92
if os.access(p, os.F_OK) or os.access(p + '.gz', os.F_OK):
93
raise BzrError("store %r already contains id %r" % (self._basedir, fileid))
99
af = AtomicFile(fn, 'wb')
102
gf = gzip.GzipFile(mode='wb', fileobj=af)
112
def copy_multi(self, other, ids):
113
"""Copy texts for ids from other into self.
115
If an id is present in self, it is skipped. A count of copied
116
ids is returned, which may be less than len(ids).
118
from bzrlib.progress import ProgressBar
120
pb.update('preparing to copy')
121
to_copy = [id for id in ids if id not in self]
122
if isinstance(other, ImmutableStore):
123
return self.copy_multi_immutable(other, to_copy, pb)
127
pb.update('copy', count, len(to_copy))
128
self.add(other[id], id)
129
assert count == len(to_copy)
134
def copy_multi_immutable(self, other, to_copy, pb):
135
from shutil import copyfile
139
other_p = other._path(id)
143
if e.errno == errno.ENOENT:
144
copyfile(other_p+".gz", p+".gz")
149
pb.update('copy', count, len(to_copy))
150
assert count == len(to_copy)
96
if fileid not in self:
97
filename = self._path(fileid)
98
f = file(filename, 'wb')
101
## os.fsync(f.fileno())
103
osutils.make_readonly(filename)
155
106
def __contains__(self, fileid):
157
p = self._path(fileid)
158
return (os.access(p, os.R_OK)
159
or os.access(p + '.gz', os.R_OK))
108
return os.access(self._path(fileid), os.R_OK)
161
# TODO: Guard against the same thing being stored twice, compressed and uncompresse
163
111
def __iter__(self):
164
for f in os.listdir(self._basedir):
166
# TODO: case-insensitive?
112
return iter(os.listdir(self._basedir))
171
114
def __len__(self):
172
115
return len(os.listdir(self._basedir))
174
117
def __getitem__(self, fileid):
175
118
"""Returns a file reading from a particular entry."""
176
p = self._path(fileid)
178
return gzip.GzipFile(p + '.gz', 'rb')
180
if e.errno == errno.ENOENT:
119
return file(self._path(fileid), 'rb')
185
121
def total_size(self):
186
"""Return (count, bytes)
188
This is the (compressed) size stored on disk, not the size of
122
"""Return (count, bytes)"""
196
total += os.stat(p)[ST_SIZE]
198
total += os.stat(p + '.gz')[ST_SIZE]
127
total += os.stat(self._path(fid))[ST_SIZE]
200
128
return count, total
130
def delete_all(self):
134
def delete(self, fileid):
135
"""Remove nominated store entry.
137
Most stores will be add-only."""
138
filename = self._path(fileid)
139
## osutils.make_writable(filename)
143
"""Remove store; only allowed if it is empty."""
144
os.rmdir(self._basedir)
145
mutter("%r destroyed" % self)