24
23
from inventory import Inventory
25
24
from trace import mutter, note
26
25
from osutils import pumpfile, compare_files, filesize, quotefn, sha_file, \
27
joinpath, splitpath, appendpath, isdir, isfile, file_kind
26
joinpath, splitpath, appendpath, isdir, isfile, file_kind, fingerprint_file
28
27
from errors import bailout
30
29
from stat import S_ISREG, S_ISDIR, ST_MODE, ST_SIZE
74
73
doc="Inventory of this Tree")
76
75
def _check_retrieved(self, ie, f):
77
# TODO: Test this check by damaging the store?
76
fp = fingerprint_file(f)
78
79
if ie.text_size is not None:
80
if fs != ie.text_size:
80
if ie.text_size != fp['size']:
81
81
bailout("mismatched size for file %r in %r" % (ie.file_id, self._store),
82
82
["inventory expects %d bytes" % ie.text_size,
83
"file is actually %d bytes" % fs,
83
"file is actually %d bytes" % fp['size'],
84
84
"store is probably damaged/corrupt"])
88
if ie.text_sha1 != f_hash:
86
if ie.text_sha1 != fp['sha1']:
89
87
bailout("wrong SHA-1 for file %r in %r" % (ie.file_id, self._store),
90
88
["inventory expects %s" % ie.text_sha1,
91
"file is actually %s" % f_hash,
89
"file is actually %s" % fp['sha1'],
92
90
"store is probably damaged/corrupt"])
138
136
return "<%s of %s>" % (self.__class__.__name__,
141
def _rel(self, filename):
139
def abspath(self, filename):
142
140
return os.path.join(self.basedir, filename)
144
142
def has_filename(self, filename):
145
return os.path.exists(self._rel(filename))
143
return os.path.exists(self.abspath(filename))
147
145
def get_file(self, file_id):
148
146
return self.get_file_byname(self.id2path(file_id))
150
148
def get_file_byname(self, filename):
151
return file(self._rel(filename), 'rb')
149
return file(self.abspath(filename), 'rb')
153
151
def _get_store_filename(self, file_id):
154
return self._rel(self.id2path(file_id))
152
return self.abspath(self.id2path(file_id))
156
154
def has_id(self, file_id):
157
155
# files that have been deleted are excluded
158
156
if not self.inventory.has_id(file_id):
160
return os.access(self._rel(self.inventory.id2path(file_id)), os.F_OK)
158
return os.access(self.abspath(self.inventory.id2path(file_id)), os.F_OK)
162
160
def get_file_size(self, file_id):
163
161
return os.stat(self._get_store_filename(file_id))[ST_SIZE]
245
def unknowns(self, path='', dir_id=None):
246
"""Yield names of unknown files in this WorkingTree.
238
for subp in self.extras():
239
if not self.is_ignored(subp):
244
"""Yield all unknown files in this WorkingTree.
248
246
If there are any unknown directories then only the directory is
249
247
returned, not all its children. But if there are unknown files
252
250
Currently returned depth-first, sorted by name within directories.
254
for fpath, fclass, fkind, fid in self.list_files():
252
## TODO: Work from given directory downwards
254
for path, dir_entry in self.inventory.directories():
255
mutter("search for unknowns in %r" % path)
256
dirabs = self.abspath(path)
257
if not isdir(dirabs):
258
# e.g. directory deleted
262
for subf in os.listdir(dirabs):
264
and (subf not in dir_entry.children)):
269
subp = appendpath(path, subf)
259
273
def ignored_files(self):
260
for fpath, fclass, fkind, fid in self.list_files():
274
"""Yield list of PATH, IGNORE_PATTERN"""
275
for subp in self.extras():
276
pat = self.is_ignored(subp)
265
281
def get_ignore_list(self):
266
"""Return list of ignore patterns."""
282
"""Return list of ignore patterns.
284
Cached in the Tree object after the first call.
286
if hasattr(self, '_ignorelist'):
287
return self._ignorelist
289
l = bzrlib.DEFAULT_IGNORE[:]
267
290
if self.has_filename(bzrlib.IGNORE_FILENAME):
268
291
f = self.get_file_byname(bzrlib.IGNORE_FILENAME)
269
return [line.rstrip("\n\r") for line in f.readlines()]
271
return bzrlib.DEFAULT_IGNORE
292
l.extend([line.rstrip("\n\r") for line in f.readlines()])
274
297
def is_ignored(self, filename):
275
298
"""Check whether the filename matches an ignore pattern.
277
300
Patterns containing '/' need to match the whole path; others
278
match against only the last component."""
279
## TODO: Take them from a file, not hardcoded
280
## TODO: Use extended zsh-style globs maybe?
281
## TODO: Use '**' to match directories?
301
match against only the last component.
303
If the file is ignored, returns the pattern which caused it to
304
be ignored, otherwise None. So this can simply be used as a
305
boolean if desired."""
307
## TODO: Use '**' to match directories, and other extended globbing stuff from cvs/rsync.
282
309
for pat in self.get_ignore_list():
284
if fnmatch.fnmatchcase(filename, pat):
311
# as a special case, you can put ./ at the start of a pattern;
312
# this is good to match in the top-level only;
317
if fnmatch.fnmatchcase(filename, newpat):
287
320
if fnmatch.fnmatchcase(splitpath(filename)[-1], pat):