29
29
WorkingTree.open(dir).
32
# TODO: Give the workingtree sole responsibility for the working inventory;
33
# remove the variable and references to it from the branch. This may require
34
# updating the commit code so as to update the inventory within the working
35
# copy, and making sure there's only one WorkingTree for any directory on disk.
36
# At the moment they may alias the inventory and have old copies of it in
37
# memory. (Now done? -- mbp 20060309)
39
33
from cStringIO import StringIO
102
95
from bzrlib.filters import filtered_input_file
103
96
from bzrlib.trace import mutter, note
104
97
from bzrlib.transport.local import LocalTransport
105
from bzrlib.progress import DummyProgress, ProgressPhase
106
98
from bzrlib.revision import CURRENT_REVISION
107
99
from bzrlib.rio import RioReader, rio_file, Stanza
108
100
from bzrlib.symbol_versioning import (
343
350
path = osutils.getcwd()
344
351
control, relpath = bzrdir.BzrDir.open_containing(path)
346
352
return control.open_workingtree(), relpath
355
def open_containing_paths(file_list, default_directory='.',
356
canonicalize=True, apply_view=True):
357
"""Open the WorkingTree that contains a set of paths.
359
Fail if the paths given are not all in a single tree.
361
This is used for the many command-line interfaces that take a list of
362
any number of files and that require they all be in the same tree.
364
# recommended replacement for builtins.internal_tree_files
365
if file_list is None or len(file_list) == 0:
366
tree = WorkingTree.open_containing(default_directory)[0]
367
# XXX: doesn't really belong here, and seems to have the strange
368
# side effect of making it return a bunch of files, not the whole
369
# tree -- mbp 20100716
370
if tree.supports_views() and apply_view:
371
view_files = tree.views.lookup_view()
373
file_list = view_files
374
view_str = views.view_display_str(view_files)
375
note("Ignoring files outside view. View is %s" % view_str)
376
return tree, file_list
377
tree = WorkingTree.open_containing(file_list[0])[0]
378
return tree, tree.safe_relpath_files(file_list, canonicalize,
379
apply_view=apply_view)
381
def safe_relpath_files(self, file_list, canonicalize=True, apply_view=True):
382
"""Convert file_list into a list of relpaths in tree.
384
:param self: A tree to operate on.
385
:param file_list: A list of user provided paths or None.
386
:param apply_view: if True and a view is set, apply it or check that
387
specified files are within it
388
:return: A list of relative paths.
389
:raises errors.PathNotChild: When a provided path is in a different self
392
if file_list is None:
394
if self.supports_views() and apply_view:
395
view_files = self.views.lookup_view()
399
# self.relpath exists as a "thunk" to osutils, but canonical_relpath
400
# doesn't - fix that up here before we enter the loop.
402
fixer = lambda p: osutils.canonical_relpath(self.basedir, p)
405
for filename in file_list:
406
relpath = fixer(osutils.dereference_path(filename))
407
if view_files and not osutils.is_inside_any(view_files, relpath):
408
raise errors.FileOutsideView(filename, view_files)
409
new_list.append(relpath)
349
413
def open_downlevel(path=None):
350
414
"""Open an unsupported working tree.
364
428
return True, None
366
430
return True, tree
367
transport = get_transport(location)
368
iterator = bzrdir.BzrDir.find_bzrdirs(transport, evaluate=evaluate,
431
t = transport.get_transport(location)
432
iterator = bzrdir.BzrDir.find_bzrdirs(t, evaluate=evaluate,
369
433
list_current=list_current)
370
return [t for t in iterator if t is not None]
434
return [tr for tr in iterator if tr is not None]
372
436
# should be deprecated - this is slow and in any case treating them as a
373
437
# container is (we now know) bad style -- mbp 20070302
458
522
return (file_obj, stat_value)
460
524
def get_file_text(self, file_id, path=None, filtered=True):
461
return self.get_file(file_id, path=path, filtered=filtered).read()
525
my_file = self.get_file(file_id, path=path, filtered=filtered)
527
return my_file.read()
463
531
def get_file_byname(self, filename, filtered=True):
464
532
path = self.abspath(filename)
519
587
# Now we have the parents of this content
520
588
annotator = self.branch.repository.texts.get_annotator()
521
text = self.get_file(file_id).read()
589
text = self.get_file_text(file_id)
522
590
this_key =(file_id, default_revision)
523
591
annotator.add_special_text(this_key, file_parent_keys, text)
524
592
annotations = [(key[-1], line)
635
705
def _is_executable_from_path_and_stat_from_basis(self, path, stat_result):
636
706
file_id = self.path2id(path)
708
# For unversioned files on win32, we just assume they are not
637
711
return self._inventory[file_id].executable
639
713
def _is_executable_from_path_and_stat_from_stat(self, path, stat_result):
750
824
kind = _mapper(stat_result.st_mode)
751
825
if kind == 'file':
752
size = stat_result.st_size
753
# try for a stat cache lookup
754
executable = self._is_executable_from_path_and_stat(path, stat_result)
755
return (kind, size, executable, self._sha_from_stat(
826
return self._file_content_summary(path, stat_result)
757
827
elif kind == 'directory':
758
828
# perhaps it looks like a plain directory, but it's really a
767
837
return (kind, None, None, None)
839
def _file_content_summary(self, path, stat_result):
840
size = stat_result.st_size
841
executable = self._is_executable_from_path_and_stat(path, stat_result)
842
# try for a stat cache lookup
843
return ('file', size, executable, self._sha_from_stat(
769
846
def _check_parents_for_ghosts(self, revision_ids, allow_leftmost_as_ghost):
770
847
"""Common ghost checking functionality from set_parent_*.
892
969
@needs_write_lock # because merge pulls data into the branch.
893
970
def merge_from_branch(self, branch, to_revision=None, from_revision=None,
971
merge_type=None, force=False):
895
972
"""Merge from a branch into this working tree.
897
974
:param branch: The branch to merge from.
901
978
branch.last_revision().
903
980
from bzrlib.merge import Merger, Merge3Merger
904
pb = ui.ui_factory.nested_progress_bar()
906
merger = Merger(self.branch, this_tree=self, pb=pb)
907
merger.pp = ProgressPhase("Merge phase", 5, pb)
908
merger.pp.next_phase()
909
# check that there are no
911
merger.check_basis(check_clean=True, require_commits=False)
912
if to_revision is None:
913
to_revision = _mod_revision.ensure_null(branch.last_revision())
914
merger.other_rev_id = to_revision
915
if _mod_revision.is_null(merger.other_rev_id):
916
raise errors.NoCommits(branch)
917
self.branch.fetch(branch, last_revision=merger.other_rev_id)
918
merger.other_basis = merger.other_rev_id
919
merger.other_tree = self.branch.repository.revision_tree(
921
merger.other_branch = branch
922
merger.pp.next_phase()
923
if from_revision is None:
926
merger.set_base_revision(from_revision, branch)
927
if merger.base_rev_id == merger.other_rev_id:
928
raise errors.PointlessMerge
929
merger.backup_files = False
930
if merge_type is None:
931
merger.merge_type = Merge3Merger
933
merger.merge_type = merge_type
934
merger.set_interesting_files(None)
935
merger.show_base = False
936
merger.reprocess = False
937
conflicts = merger.do_merge()
981
merger = Merger(self.branch, this_tree=self)
982
# check that there are no local alterations
983
if not force and self.has_changes():
984
raise errors.UncommittedChanges(self)
985
if to_revision is None:
986
to_revision = _mod_revision.ensure_null(branch.last_revision())
987
merger.other_rev_id = to_revision
988
if _mod_revision.is_null(merger.other_rev_id):
989
raise errors.NoCommits(branch)
990
self.branch.fetch(branch, last_revision=merger.other_rev_id)
991
merger.other_basis = merger.other_rev_id
992
merger.other_tree = self.branch.repository.revision_tree(
994
merger.other_branch = branch
995
if from_revision is None:
998
merger.set_base_revision(from_revision, branch)
999
if merger.base_rev_id == merger.other_rev_id:
1000
raise errors.PointlessMerge
1001
merger.backup_files = False
1002
if merge_type is None:
1003
merger.merge_type = Merge3Merger
1005
merger.merge_type = merge_type
1006
merger.set_interesting_files(None)
1007
merger.show_base = False
1008
merger.reprocess = False
1009
conflicts = merger.do_merge()
1010
merger.set_pending()
941
1011
return conflicts
943
1013
@needs_read_lock
1090
1160
tree_transport = self.bzrdir.root_transport.clone(sub_path)
1091
1161
if tree_transport.base != branch_transport.base:
1092
1162
tree_bzrdir = format.initialize_on_transport(tree_transport)
1093
branch.BranchReferenceFormat().initialize(tree_bzrdir, new_branch)
1163
branch.BranchReferenceFormat().initialize(tree_bzrdir,
1164
target_branch=new_branch)
1095
1166
tree_bzrdir = branch_bzrdir
1096
1167
wt = tree_bzrdir.create_workingtree(_mod_revision.NULL_REVISION)
1134
1205
This does not include files that have been deleted in this
1135
1206
tree. Skips the control directory.
1137
:param include_root: if True, do not return an entry for the root
1208
:param include_root: if True, return an entry for the root
1138
1209
:param from_dir: start from this directory or None for the root
1139
1210
:param recursive: whether to recurse into subdirectories or not
1195
1266
# absolute path
1196
1267
fap = from_dir_abspath + '/' + f
1198
f_ie = inv.get_child(from_dir_id, f)
1269
dir_ie = inv[from_dir_id]
1270
if dir_ie.kind == 'directory':
1271
f_ie = dir_ie.children.get(f)
1201
1276
elif self.is_ignored(fp[1:]):
1204
# we may not have found this file, because of a unicode issue
1279
# we may not have found this file, because of a unicode
1280
# issue, or because the directory was actually a symlink.
1205
1281
f_norm, can_access = osutils.normalized_filename(f)
1206
1282
if f == f_norm or not can_access:
1207
1283
# No change, so treat this file normally
1291
1367
# check for deprecated use of signature
1292
1368
if to_dir is None:
1293
to_dir = kwargs.get('to_name', None)
1295
raise TypeError('You must supply a target directory')
1297
symbol_versioning.warn('The parameter to_name was deprecated'
1298
' in version 0.13. Use to_dir instead',
1369
raise TypeError('You must supply a target directory')
1301
1370
# check destination directory
1302
1371
if isinstance(from_paths, basestring):
1303
1372
raise ValueError()
1594
1663
@needs_write_lock
1595
1664
def pull(self, source, overwrite=False, stop_revision=None,
1596
1665
change_reporter=None, possible_transports=None, local=False):
1597
top_pb = ui.ui_factory.nested_progress_bar()
1598
1666
source.lock_read()
1600
pp = ProgressPhase("Pull phase", 2, top_pb)
1602
1668
old_revision_info = self.branch.last_revision_info()
1603
1669
basis_tree = self.basis_tree()
1604
1670
count = self.branch.pull(source, overwrite, stop_revision,
1617
1681
new_basis_tree,
1619
1683
this_tree=self,
1621
1685
change_reporter=change_reporter)
1622
if (basis_tree.inventory.root is None and
1623
new_basis_tree.inventory.root is not None):
1624
self.set_root_id(new_basis_tree.get_root_id())
1686
basis_root_id = basis_tree.get_root_id()
1687
new_root_id = new_basis_tree.get_root_id()
1688
if basis_root_id != new_root_id:
1689
self.set_root_id(new_root_id)
1627
1691
basis_tree.unlock()
1628
1692
# TODO - dedup parents list with things merged by pull ?
1629
1693
# reuse the revisiontree we merged against to set the new
1733
1796
r"""Check whether the filename matches an ignore pattern.
1735
1798
Patterns containing '/' or '\' need to match the whole path;
1736
others match against only the last component.
1799
others match against only the last component. Patterns starting
1800
with '!' are ignore exceptions. Exceptions take precedence
1801
over regular patterns and cause the filename to not be ignored.
1738
1803
If the file is ignored, returns the pattern which caused it to
1739
1804
be ignored, otherwise None. So this can simply be used as a
1740
1805
boolean if desired."""
1741
1806
if getattr(self, '_ignoreglobster', None) is None:
1742
self._ignoreglobster = globbing.Globster(self.get_ignore_list())
1807
self._ignoreglobster = globbing.ExceptionGlobster(self.get_ignore_list())
1743
1808
return self._ignoreglobster.match(filename)
1745
1810
def kind(self, file_id):
1795
1860
raise errors.ObjectNotLocked(self)
1797
1862
def lock_read(self):
1798
"""See Branch.lock_read, and WorkingTree.unlock."""
1863
"""Lock the tree for reading.
1865
This also locks the branch, and can be unlocked via self.unlock().
1867
:return: A bzrlib.lock.LogicalLockResult.
1799
1869
if not self.is_locked():
1800
1870
self._reset_data()
1801
1871
self.branch.lock_read()
1803
return self._control_files.lock_read()
1873
self._control_files.lock_read()
1874
return LogicalLockResult(self.unlock)
1805
1876
self.branch.unlock()
1808
1879
def lock_tree_write(self):
1809
"""See MutableTree.lock_tree_write, and WorkingTree.unlock."""
1880
"""See MutableTree.lock_tree_write, and WorkingTree.unlock.
1882
:return: A bzrlib.lock.LogicalLockResult.
1810
1884
if not self.is_locked():
1811
1885
self._reset_data()
1812
1886
self.branch.lock_read()
1814
return self._control_files.lock_write()
1888
self._control_files.lock_write()
1889
return LogicalLockResult(self.unlock)
1816
1891
self.branch.unlock()
1819
1894
def lock_write(self):
1820
"""See MutableTree.lock_write, and WorkingTree.unlock."""
1895
"""See MutableTree.lock_write, and WorkingTree.unlock.
1897
:return: A bzrlib.lock.LogicalLockResult.
1821
1899
if not self.is_locked():
1822
1900
self._reset_data()
1823
1901
self.branch.lock_write()
1825
return self._control_files.lock_write()
1903
self._control_files.lock_write()
1904
return LogicalLockResult(self.unlock)
1827
1906
self.branch.unlock()
1889
1972
# revision_id is set. We must check for this full string, because a
1890
1973
# root node id can legitimately look like 'revision_id' but cannot
1891
1974
# contain a '"'.
1892
xml = self.branch.repository.get_inventory_xml(new_revision)
1975
xml = self.branch.repository._get_inventory_xml(new_revision)
1893
1976
firstline = xml.split('\n', 1)[0]
1894
1977
if (not 'revision_id="' in firstline or
1895
1978
'format="7"' not in firstline):
1896
inv = self.branch.repository.deserialise_inventory(
1979
inv = self.branch.repository._serializer.read_inventory_from_string(
1898
1981
xml = self._create_basis_xml_from_inventory(new_revision, inv)
1899
1982
self._write_basis_inventory(xml)
1900
1983
except (errors.NoSuchRevision, errors.RevisionNotPresent):
1940
2027
new_files=set()
1941
2028
unknown_nested_files=set()
2030
to_file = sys.stdout
1943
2032
def recurse_directory_to_add_files(directory):
1944
2033
# Recurse directory and add all files
1945
2034
# so we can check if they have changed.
1946
for parent_info, file_infos in\
1947
self.walkdirs(directory):
2035
for parent_info, file_infos in self.walkdirs(directory):
1948
2036
for relpath, basename, kind, lstat, fileid, kind in file_infos:
1949
2037
# Is it versioned or ignored?
1950
2038
if self.path2id(relpath) or self.is_ignored(relpath):
1985
2073
# ... but not ignored
1986
2074
has_changed_files = True
1988
elif content_change and (kind[1] is not None):
1989
# Versioned and changed, but not deleted
2076
elif (content_change and (kind[1] is not None) and
2077
osutils.is_inside_any(files, path[1])):
2078
# Versioned and changed, but not deleted, and still
2079
# in one of the dirs to be deleted.
1990
2080
has_changed_files = True
2014
2104
new_status = 'I'
2016
2106
new_status = '?'
2017
textui.show_status(new_status, self.kind(fid), f,
2107
# XXX: Really should be a more abstract reporter interface
2108
kind_ch = osutils.kind_marker(self.kind(fid))
2109
to_file.write(new_status + ' ' + f + kind_ch + '\n')
2019
2110
# Unversion file
2020
2111
inv_delta.append((f, None, fid, None))
2021
2112
message = "removed %s" % (f,)
2173
2264
raise NotImplementedError(self.unlock)
2175
def update(self, change_reporter=None, possible_transports=None):
2268
def update(self, change_reporter=None, possible_transports=None,
2269
revision=None, old_tip=_marker):
2176
2270
"""Update a working tree along its branch.
2178
2272
This will update the branch if its bound too, which means we have
2196
2290
- Merge current state -> basis tree of the master w.r.t. the old tree
2198
2292
- Do a 'normal' merge of the old branch basis if it is relevant.
2294
:param revision: The target revision to update to. Must be in the
2296
:param old_tip: If branch.update() has already been run, the value it
2297
returned (old tip of the branch or None). _marker is used
2200
2300
if self.branch.get_bound_location() is not None:
2201
2301
self.lock_write()
2202
update_branch = True
2302
update_branch = (old_tip is self._marker)
2204
2304
self.lock_tree_write()
2205
2305
update_branch = False
2207
2307
if update_branch:
2208
2308
old_tip = self.branch.update(possible_transports)
2211
return self._update_tree(old_tip, change_reporter)
2310
if old_tip is self._marker:
2312
return self._update_tree(old_tip, change_reporter, revision)
2215
2316
@needs_tree_write_lock
2216
def _update_tree(self, old_tip=None, change_reporter=None):
2317
def _update_tree(self, old_tip=None, change_reporter=None, revision=None):
2217
2318
"""Update a tree to the master branch.
2219
2320
:param old_tip: if supplied, the previous tip revision the branch,
2229
2330
# We MUST save it even if an error occurs, because otherwise the users
2230
2331
# local work is unreferenced and will appear to have been lost.
2234
2335
last_rev = self.get_parent_ids()[0]
2235
2336
except IndexError:
2236
2337
last_rev = _mod_revision.NULL_REVISION
2237
if last_rev != _mod_revision.ensure_null(self.branch.last_revision()):
2238
# merge tree state up to new branch tip.
2338
if revision is None:
2339
revision = self.branch.last_revision()
2341
old_tip = old_tip or _mod_revision.NULL_REVISION
2343
if not _mod_revision.is_null(old_tip) and old_tip != last_rev:
2344
# the branch we are bound to was updated
2345
# merge those changes in first
2346
base_tree = self.basis_tree()
2347
other_tree = self.branch.repository.revision_tree(old_tip)
2348
nb_conflicts = merge.merge_inner(self.branch, other_tree,
2349
base_tree, this_tree=self,
2350
change_reporter=change_reporter)
2352
self.add_parent_tree((old_tip, other_tree))
2353
trace.note('Rerun update after fixing the conflicts.')
2356
if last_rev != _mod_revision.ensure_null(revision):
2357
# the working tree is up to date with the branch
2358
# we can merge the specified revision from master
2359
to_tree = self.branch.repository.revision_tree(revision)
2360
to_root_id = to_tree.get_root_id()
2239
2362
basis = self.basis_tree()
2240
2363
basis.lock_read()
2242
to_tree = self.branch.basis_tree()
2243
if basis.inventory.root is None:
2244
self.set_root_id(to_tree.get_root_id())
2365
if (basis.inventory.root is None
2366
or basis.inventory.root.file_id != to_root_id):
2367
self.set_root_id(to_root_id)
2246
result += merge.merge_inner(
2251
change_reporter=change_reporter)
2372
# determine the branch point
2373
graph = self.branch.repository.get_graph()
2374
base_rev_id = graph.find_unique_lca(self.branch.last_revision(),
2376
base_tree = self.branch.repository.revision_tree(base_rev_id)
2378
nb_conflicts = merge.merge_inner(self.branch, to_tree, base_tree,
2380
change_reporter=change_reporter)
2381
self.set_last_revision(revision)
2254
2382
# TODO - dedup parents list with things merged by pull ?
2255
2383
# reuse the tree we've updated to to set the basis:
2256
parent_trees = [(self.branch.last_revision(), to_tree)]
2384
parent_trees = [(revision, to_tree)]
2257
2385
merges = self.get_parent_ids()[1:]
2258
2386
# Ideally we ask the tree for the trees here, that way the working
2259
2387
# tree can decide whether to give us the entire tree or give us a
2263
2391
for parent in merges:
2264
2392
parent_trees.append(
2265
2393
(parent, self.branch.repository.revision_tree(parent)))
2266
if (old_tip is not None and not _mod_revision.is_null(old_tip)):
2394
if not _mod_revision.is_null(old_tip):
2267
2395
parent_trees.append(
2268
2396
(old_tip, self.branch.repository.revision_tree(old_tip)))
2269
2397
self.set_parent_trees(parent_trees)
2270
2398
last_rev = parent_trees[0][0]
2272
# the working tree had the same last-revision as the master
2273
# branch did. We may still have pivot local work from the local
2274
# branch into old_tip:
2275
if (old_tip is not None and not _mod_revision.is_null(old_tip)):
2276
self.add_parent_tree_id(old_tip)
2277
if (old_tip is not None and not _mod_revision.is_null(old_tip)
2278
and old_tip != last_rev):
2279
# our last revision was not the prior branch last revision
2280
# and we have converted that last revision to a pending merge.
2281
# base is somewhere between the branch tip now
2282
# and the now pending merge
2284
# Since we just modified the working tree and inventory, flush out
2285
# the current state, before we modify it again.
2286
# TODO: jam 20070214 WorkingTree3 doesn't require this, dirstate
2287
# requires it only because TreeTransform directly munges the
2288
# inventory and calls tree._write_inventory(). Ultimately we
2289
# should be able to remove this extra flush.
2291
graph = self.branch.repository.get_graph()
2292
base_rev_id = graph.find_unique_lca(self.branch.last_revision(),
2294
base_tree = self.branch.repository.revision_tree(base_rev_id)
2295
other_tree = self.branch.repository.revision_tree(old_tip)
2296
result += merge.merge_inner(
2301
change_reporter=change_reporter)
2304
2401
def _write_hashcache_if_dirty(self):
2305
2402
"""Write out the hashcache if it is dirty."""
2760
2860
"""Return the format for the working tree object in a_bzrdir."""
2762
2862
transport = a_bzrdir.get_workingtree_transport(None)
2763
format_string = transport.get("format").read()
2863
format_string = transport.get_bytes("format")
2764
2864
return klass._formats[format_string]
2765
2865
except errors.NoSuchFile:
2766
2866
raise errors.NoWorkingTree(base=transport.base)
3030
3130
return self.get_format_string()
3033
__default_format = WorkingTreeFormat4()
3133
__default_format = WorkingTreeFormat6()
3034
3134
WorkingTreeFormat.register_format(__default_format)
3035
WorkingTreeFormat.register_format(WorkingTreeFormat6())
3036
3135
WorkingTreeFormat.register_format(WorkingTreeFormat5())
3136
WorkingTreeFormat.register_format(WorkingTreeFormat4())
3037
3137
WorkingTreeFormat.register_format(WorkingTreeFormat3())
3038
3138
WorkingTreeFormat.set_default_format(__default_format)
3039
3139
# formats which have no format string are not discoverable