194
202
(branch.base is not cross checked, because for remote branches that
195
203
would be meaningless).
205
self._format = _format
206
self.bzrdir = _bzrdir
208
# not created via open etc.
209
warn("WorkingTree() is deprecated as of bzr version 0.8. "
210
"Please use bzrdir.open_workingtree or WorkingTree.open().",
213
wt = WorkingTree.open(basedir)
214
self.branch = wt.branch
215
self.basedir = wt.basedir
216
self._control_files = wt._control_files
217
self._hashcache = wt._hashcache
218
self._set_inventory(wt._inventory)
219
self._format = wt._format
220
self.bzrdir = wt.bzrdir
197
221
from bzrlib.hashcache import HashCache
198
222
from bzrlib.trace import note, mutter
199
223
assert isinstance(basedir, basestring), \
200
224
"base directory %r is not a string" % basedir
201
225
basedir = safe_unicode(basedir)
202
mutter("openeing working tree %r", basedir)
204
branch = Branch.open(basedir)
205
assert isinstance(branch, Branch), \
206
"branch %r is not a Branch" % branch
226
mutter("opening working tree %r", basedir)
227
if deprecated_passed(branch):
229
warn("WorkingTree(..., branch=XXX) is deprecated as of bzr 0.8."
230
" Please use bzrdir.open_workingtree() or WorkingTree.open().",
236
self.branch = self.bzrdir.open_branch()
237
assert isinstance(self.branch, Branch), \
238
"branch %r is not a Branch" % self.branch
208
239
self.basedir = realpath(basedir)
209
240
# if branch is at our basedir and is a format 6 or less
210
if (isinstance(self.branch._branch_format,
211
(BzrBranchFormat4, BzrBranchFormat5, BzrBranchFormat6))
212
# might be able to share control object
213
and self.branch.base.split('/')[-2] == self.basedir.split('/')[-1]):
241
if isinstance(self._format, WorkingTreeFormat2):
242
# share control object
214
243
self._control_files = self.branch.control_files
215
244
elif _control_files is not None:
216
245
assert False, "not done yet"
217
246
# self._control_files = _control_files
248
# only ready for format 3
249
assert isinstance(self._format, WorkingTreeFormat3)
219
250
self._control_files = LockableFiles(
220
get_transport(self.basedir).clone(bzrlib.BZRDIR), 'branch-lock')
251
self.bzrdir.get_workingtree_transport(None),
222
254
# update the whole cache up front and write to disk if anything changed;
223
255
# in the future we might want to do this more selectively
861
946
def kind(self, file_id):
862
947
return file_kind(self.id2abspath(file_id))
950
def last_revision(self):
951
"""Return the last revision id of this working tree.
953
In early branch formats this was == the branch last_revision,
954
but that cannot be relied upon - for working tree operations,
955
always use tree.last_revision().
957
return self.branch.last_revision()
864
959
def lock_read(self):
865
960
"""See Branch.lock_read, and WorkingTree.unlock."""
866
return self.branch.lock_read()
961
self.branch.lock_read()
963
return self._control_files.lock_read()
868
968
def lock_write(self):
869
969
"""See Branch.lock_write, and WorkingTree.unlock."""
870
return self.branch.lock_write()
970
self.branch.lock_write()
972
return self._control_files.lock_write()
872
977
def _basis_inventory_name(self, revision_id):
873
978
return 'basis-inventory.%s' % revision_id
875
981
def set_last_revision(self, new_revision, old_revision=None):
876
if old_revision is not None:
878
path = self._basis_inventory_name(old_revision)
879
path = self.branch.control_files._escape(path)
880
self.branch.control_files._transport.delete(path)
982
"""Change the last revision in the working tree."""
983
self._remove_old_basis(old_revision)
984
if self._change_last_revision(new_revision):
985
self._cache_basis_inventory(new_revision)
987
def _change_last_revision(self, new_revision):
988
"""Template method part of set_last_revision to perform the change."""
989
if new_revision is None:
990
self.branch.set_revision_history([])
992
# current format is locked in with the branch
993
revision_history = self.branch.revision_history()
995
position = revision_history.index(new_revision)
997
raise errors.NoSuchRevision(self.branch, new_revision)
998
self.branch.set_revision_history(revision_history[:position + 1])
1001
def _cache_basis_inventory(self, new_revision):
1002
"""Cache new_revision as the basis inventory."""
884
1004
xml = self.branch.repository.get_inventory_xml(new_revision)
885
1005
path = self._basis_inventory_name(new_revision)
886
self.branch.control_files.put_utf8(path, xml)
1006
self._control_files.put_utf8(path, xml)
887
1007
except WeaveRevisionNotPresent:
1010
def _remove_old_basis(self, old_revision):
1011
"""Remove the old basis inventory 'old_revision'."""
1012
if old_revision is not None:
1014
path = self._basis_inventory_name(old_revision)
1015
path = self._control_files._escape(path)
1016
self._control_files._transport.delete(path)
890
1020
def read_basis_inventory(self, revision_id):
891
1021
"""Read the cached basis inventory."""
892
1022
path = self._basis_inventory_name(revision_id)
893
return self.branch.control_files.get_utf8(path).read()
1023
return self._control_files.get_utf8(path).read()
895
1025
@needs_read_lock
896
1026
def read_working_inventory(self):
897
1027
"""Read the working inventory."""
898
1028
# ElementTree does its own conversion from UTF-8, so open in
900
return bzrlib.xml5.serializer_v5.read_inventory(
1030
result = bzrlib.xml5.serializer_v5.read_inventory(
901
1031
self._control_files.get('inventory'))
1032
self._set_inventory(result)
903
1035
@needs_write_lock
904
1036
def remove(self, files, verbose=False):
1018
1181
self._control_files.put('inventory', sio)
1019
1182
self._set_inventory(inv)
1020
1183
mutter('wrote working inventory')
1186
class WorkingTree3(WorkingTree):
1187
"""This is the Format 3 working tree.
1189
This differs from the base WorkingTree by:
1190
- having its own file lock
1191
- having its own last-revision property.
1195
def last_revision(self):
1196
"""See WorkingTree.last_revision."""
1198
return self._control_files.get_utf8('last-revision').read()
1202
def _change_last_revision(self, revision_id):
1203
"""See WorkingTree._change_last_revision."""
1204
if revision_id is None or revision_id == NULL_REVISION:
1206
self._control_files._transport.delete('last-revision')
1207
except errors.NoSuchFile:
1212
self.branch.revision_history().index(revision_id)
1214
raise errors.NoSuchRevision(self.branch, revision_id)
1215
self._control_files.put_utf8('last-revision', revision_id)
1023
1219
CONFLICT_SUFFIXES = ('.THIS', '.BASE', '.OTHER')
1024
1220
def get_conflicted_stem(path):
1025
1221
for suffix in CONFLICT_SUFFIXES:
1026
1222
if path.endswith(suffix):
1027
1223
return path[:-len(suffix)]
1225
@deprecated_function(zero_eight)
1226
def is_control_file(filename):
1227
"""See WorkingTree.is_control_filename(filename)."""
1228
## FIXME: better check
1229
filename = normpath(filename)
1230
while filename != '':
1231
head, tail = os.path.split(filename)
1232
## mutter('check %r for control file' % ((head, tail),))
1235
if filename == head:
1241
class WorkingTreeFormat(object):
1242
"""An encapsulation of the initialization and open routines for a format.
1244
Formats provide three things:
1245
* An initialization routine,
1249
Formats are placed in an dict by their format string for reference
1250
during workingtree opening. Its not required that these be instances, they
1251
can be classes themselves with class methods - it simply depends on
1252
whether state is needed for a given format or not.
1254
Once a format is deprecated, just deprecate the initialize and open
1255
methods on the format class. Do not deprecate the object, as the
1256
object will be created every time regardless.
1259
_default_format = None
1260
"""The default format used for new trees."""
1263
"""The known formats."""
1266
def find_format(klass, a_bzrdir):
1267
"""Return the format for the working tree object in a_bzrdir."""
1269
transport = a_bzrdir.get_workingtree_transport(None)
1270
format_string = transport.get("format").read()
1271
return klass._formats[format_string]
1273
raise errors.NoWorkingTree(base=transport.base)
1275
raise errors.UnknownFormatError(format_string)
1278
def get_default_format(klass):
1279
"""Return the current default format."""
1280
return klass._default_format
1282
def get_format_string(self):
1283
"""Return the ASCII format string that identifies this format."""
1284
raise NotImplementedError(self.get_format_string)
1286
def is_supported(self):
1287
"""Is this format supported?
1289
Supported formats can be initialized and opened.
1290
Unsupported formats may not support initialization or committing or
1291
some other features depending on the reason for not being supported.
1296
def register_format(klass, format):
1297
klass._formats[format.get_format_string()] = format
1300
def set_default_format(klass, format):
1301
klass._default_format = format
1304
def unregister_format(klass, format):
1305
assert klass._formats[format.get_format_string()] is format
1306
del klass._formats[format.get_format_string()]
1310
class WorkingTreeFormat2(WorkingTreeFormat):
1311
"""The second working tree format.
1313
This format modified the hash cache from the format 1 hash cache.
1316
def initialize(self, a_bzrdir, revision_id=None):
1317
"""See WorkingTreeFormat.initialize()."""
1318
if not isinstance(a_bzrdir.transport, LocalTransport):
1319
raise errors.NotLocalUrl(a_bzrdir.transport.base)
1320
branch = a_bzrdir.open_branch()
1321
if revision_id is not None:
1324
revision_history = branch.revision_history()
1326
position = revision_history.index(revision_id)
1328
raise errors.NoSuchRevision(branch, revision_id)
1329
branch.set_revision_history(revision_history[:position + 1])
1332
revision = branch.last_revision()
1333
basis_tree = branch.repository.revision_tree(revision)
1334
inv = basis_tree.inventory
1335
wt = WorkingTree(a_bzrdir.root_transport.base,
1341
wt._write_inventory(inv)
1342
wt.set_root_id(inv.root.file_id)
1343
wt.set_last_revision(revision)
1344
wt.set_pending_merges([])
1349
super(WorkingTreeFormat2, self).__init__()
1350
self._matchingbzrdir = bzrdir.BzrDirFormat6()
1352
def open(self, a_bzrdir, _found=False):
1353
"""Return the WorkingTree object for a_bzrdir
1355
_found is a private parameter, do not use it. It is used to indicate
1356
if format probing has already been done.
1359
# we are being called directly and must probe.
1360
raise NotImplementedError
1361
if not isinstance(a_bzrdir.transport, LocalTransport):
1362
raise errors.NotLocalUrl(a_bzrdir.transport.base)
1363
return WorkingTree(a_bzrdir.root_transport.base,
1369
class WorkingTreeFormat3(WorkingTreeFormat):
1370
"""The second working tree format updated to record a format marker.
1372
This format modified the hash cache from the format 1 hash cache.
1375
def get_format_string(self):
1376
"""See WorkingTreeFormat.get_format_string()."""
1377
return "Bazaar-NG Working Tree format 3"
1379
def initialize(self, a_bzrdir, revision_id=None):
1380
"""See WorkingTreeFormat.initialize().
1382
revision_id allows creating a working tree at a differnet
1383
revision than the branch is at.
1385
if not isinstance(a_bzrdir.transport, LocalTransport):
1386
raise errors.NotLocalUrl(a_bzrdir.transport.base)
1387
transport = a_bzrdir.get_workingtree_transport(self)
1388
control_files = LockableFiles(transport, 'lock')
1389
control_files.put_utf8('format', self.get_format_string())
1390
branch = a_bzrdir.open_branch()
1391
if revision_id is None:
1392
revision_id = branch.last_revision()
1393
new_basis_tree = branch.repository.revision_tree(revision_id)
1394
inv = new_basis_tree.inventory
1395
wt = WorkingTree3(a_bzrdir.root_transport.base,
1401
wt._write_inventory(inv)
1402
wt.set_root_id(inv.root.file_id)
1403
wt.set_last_revision(revision_id)
1404
wt.set_pending_merges([])
1409
super(WorkingTreeFormat3, self).__init__()
1410
self._matchingbzrdir = bzrdir.BzrDirMetaFormat1()
1412
def open(self, a_bzrdir, _found=False):
1413
"""Return the WorkingTree object for a_bzrdir
1415
_found is a private parameter, do not use it. It is used to indicate
1416
if format probing has already been done.
1419
# we are being called directly and must probe.
1420
raise NotImplementedError
1421
if not isinstance(a_bzrdir.transport, LocalTransport):
1422
raise errors.NotLocalUrl(a_bzrdir.transport.base)
1423
return WorkingTree3(a_bzrdir.root_transport.base,
1429
return self.get_format_string()
1432
# formats which have no format string are not discoverable
1433
# and not independently creatable, so are not registered.
1434
__default_format = WorkingTreeFormat3()
1435
WorkingTreeFormat.register_format(__default_format)
1436
WorkingTreeFormat.set_default_format(__default_format)
1437
_legacy_formats = [WorkingTreeFormat2(),
1441
class WorkingTreeTestProviderAdapter(object):
1442
"""A tool to generate a suite testing multiple workingtree formats at once.
1444
This is done by copying the test once for each transport and injecting
1445
the transport_server, transport_readonly_server, and workingtree_format
1446
classes into each copy. Each copy is also given a new id() to make it
1450
def __init__(self, transport_server, transport_readonly_server, formats):
1451
self._transport_server = transport_server
1452
self._transport_readonly_server = transport_readonly_server
1453
self._formats = formats
1455
def adapt(self, test):
1456
from bzrlib.tests import TestSuite
1457
result = TestSuite()
1458
for workingtree_format, bzrdir_format in self._formats:
1459
new_test = deepcopy(test)
1460
new_test.transport_server = self._transport_server
1461
new_test.transport_readonly_server = self._transport_readonly_server
1462
new_test.bzrdir_format = bzrdir_format
1463
new_test.workingtree_format = workingtree_format
1464
def make_new_test_id():
1465
new_id = "%s(%s)" % (new_test.id(), workingtree_format.__class__.__name__)
1466
return lambda: new_id
1467
new_test.id = make_new_test_id()
1468
result.addTest(new_test)