21
20
from sets import Set
22
21
import os.path, os, fnmatch
23
from osutils import pumpfile, compare_files, filesize, quotefn, sha_file, \
24
joinpath, splitpath, appendpath, isdir, isfile, file_kind, fingerprint_file
26
from stat import S_ISREG, S_ISDIR, ST_MODE, ST_SIZE
24
28
from inventory import Inventory
25
29
from trace import mutter, note
26
from osutils import pumpfile, compare_files, filesize, quotefn, sha_file, \
27
joinpath, splitpath, appendpath, isdir, isfile, file_kind
28
30
from errors import bailout
30
from stat import S_ISREG, S_ISDIR, ST_MODE, ST_SIZE
74
75
doc="Inventory of this Tree")
76
77
def _check_retrieved(self, ie, f):
77
# TODO: Test this check by damaging the store?
78
if ie.text_size is not None:
80
if fs != ie.text_size:
78
fp = fingerprint_file(f)
81
if ie.text_size != None:
82
if ie.text_size != fp['size']:
81
83
bailout("mismatched size for file %r in %r" % (ie.file_id, self._store),
82
84
["inventory expects %d bytes" % ie.text_size,
83
"file is actually %d bytes" % fs,
85
"file is actually %d bytes" % fp['size'],
84
86
"store is probably damaged/corrupt"])
88
if ie.text_sha1 != f_hash:
88
if ie.text_sha1 != fp['sha1']:
89
89
bailout("wrong SHA-1 for file %r in %r" % (ie.file_id, self._store),
90
90
["inventory expects %s" % ie.text_sha1,
91
"file is actually %s" % f_hash,
91
"file is actually %s" % fp['sha1'],
92
92
"store is probably damaged/corrupt"])
95
def export(self, dest):
95
def print_file(self, fileid):
96
"""Print file with id `fileid` to stdout."""
98
pumpfile(self.get_file(fileid), sys.stdout)
101
def export(self, dest):
96
102
"""Export this tree to a new directory.
98
104
`dest` should not exist, and will be created holding the
99
105
contents of this tree.
101
:todo: To handle subdirectories we need to create the
107
TODO: To handle subdirectories we need to create the
102
108
directories first.
104
110
:note: If the export fails, the destination directory will be
138
144
return "<%s of %s>" % (self.__class__.__name__,
141
def _rel(self, filename):
147
def abspath(self, filename):
142
148
return os.path.join(self.basedir, filename)
144
150
def has_filename(self, filename):
145
return os.path.exists(self._rel(filename))
151
return os.path.exists(self.abspath(filename))
147
153
def get_file(self, file_id):
148
154
return self.get_file_byname(self.id2path(file_id))
150
156
def get_file_byname(self, filename):
151
return file(self._rel(filename), 'rb')
157
return file(self.abspath(filename), 'rb')
153
159
def _get_store_filename(self, file_id):
154
return self._rel(self.id2path(file_id))
160
## XXX: badly named; this isn't in the store at all
161
return self.abspath(self.id2path(file_id))
156
163
def has_id(self, file_id):
157
164
# files that have been deleted are excluded
158
165
if not self.inventory.has_id(file_id):
160
return os.access(self._rel(self.inventory.id2path(file_id)), os.F_OK)
167
return os.access(self.abspath(self.inventory.id2path(file_id)), os.F_OK)
162
169
def get_file_size(self, file_id):
163
170
return os.stat(self._get_store_filename(file_id))[ST_SIZE]
198
196
inv = self.inventory
200
def descend(from_dir, from_dir_id, dp):
198
def descend(from_dir_relpath, from_dir_id, dp):
201
199
ls = os.listdir(dp)
202
## TODO: If we find a subdirectory with its own .bzr
203
## directory, then that is a separate tree and we
204
## should exclude it.
204
205
if bzrlib.BZRDIR == f:
207
208
# path within tree
208
fp = appendpath(from_dir, f)
209
fp = appendpath(from_dir_relpath, f)
211
212
fap = appendpath(dp, f)
237
238
for ff in descend(fp, f_ie.file_id, fap):
240
for f in descend('', None, self.basedir):
241
for f in descend('', inv.root.file_id, self.basedir):
245
def unknowns(self, path='', dir_id=None):
246
"""Yield names of unknown files in this WorkingTree.
247
for subp in self.extras():
248
if not self.is_ignored(subp):
253
"""Yield all unknown files in this WorkingTree.
248
255
If there are any unknown directories then only the directory is
249
256
returned, not all its children. But if there are unknown files
252
259
Currently returned depth-first, sorted by name within directories.
254
for fpath, fclass, fkind, fid in self.list_files():
261
## TODO: Work from given directory downwards
263
for path, dir_entry in self.inventory.directories():
264
mutter("search for unknowns in %r" % path)
265
dirabs = self.abspath(path)
266
if not isdir(dirabs):
267
# e.g. directory deleted
271
for subf in os.listdir(dirabs):
273
and (subf not in dir_entry.children)):
278
subp = appendpath(path, subf)
259
282
def ignored_files(self):
260
for fpath, fclass, fkind, fid in self.list_files():
283
"""Yield list of PATH, IGNORE_PATTERN"""
284
for subp in self.extras():
285
pat = self.is_ignored(subp)
265
290
def get_ignore_list(self):
266
"""Return list of ignore patterns."""
291
"""Return list of ignore patterns.
293
Cached in the Tree object after the first call.
295
if hasattr(self, '_ignorelist'):
296
return self._ignorelist
298
l = bzrlib.DEFAULT_IGNORE[:]
267
299
if self.has_filename(bzrlib.IGNORE_FILENAME):
268
300
f = self.get_file_byname(bzrlib.IGNORE_FILENAME)
269
return [line.rstrip("\n\r") for line in f.readlines()]
271
return bzrlib.DEFAULT_IGNORE
301
l.extend([line.rstrip("\n\r") for line in f.readlines()])
274
306
def is_ignored(self, filename):
275
"""Check whether the filename matches an ignore pattern.
277
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?
307
r"""Check whether the filename matches an ignore pattern.
309
Patterns containing '/' or '\' need to match the whole path;
310
others match against only the last component.
312
If the file is ignored, returns the pattern which caused it to
313
be ignored, otherwise None. So this can simply be used as a
314
boolean if desired."""
316
# TODO: Use '**' to match directories, and other extended
317
# globbing stuff from cvs/rsync.
319
# XXX: fnmatch is actually not quite what we want: it's only
320
# approximately the same as real Unix fnmatch, and doesn't
321
# treat dotfiles correctly and allows * to match /.
322
# Eventually it should be replaced with something more
282
325
for pat in self.get_ignore_list():
284
if fnmatch.fnmatchcase(filename, pat):
326
if '/' in pat or '\\' in pat:
328
# as a special case, you can put ./ at the start of a
329
# pattern; this is good to match in the top-level
332
if (pat[:2] == './') or (pat[:2] == '.\\'):
336
if fnmatch.fnmatchcase(filename, newpat):
287
339
if fnmatch.fnmatchcase(splitpath(filename)[-1], pat):