19
19
# TODO: 'bzr resolve' should accept a directory name and work from that
22
# TODO: bzr revert should resolve; even when reverting the whole tree
23
# or particular directories
24
from bzrlib.lazy_import import lazy_import
25
lazy_import(globals(), """
35
from bzrlib.option import Option
38
CONFLICT_SUFFIXES = ('.THIS', '.BASE', '.OTHER')
41
class cmd_conflicts(commands.Command):
29
from bzrlib.branch import Branch
30
from bzrlib.errors import BzrCommandError, NotConflicted
31
from bzrlib.commands import register_command
32
from bzrlib.workingtree import CONFLICT_SUFFIXES
34
class cmd_conflicts(bzrlib.commands.Command):
42
35
"""List files with conflicts.
44
Merge will do its best to combine the changes in two branches, but there
45
are some kinds of problems only a human can fix. When it encounters those,
46
it will mark a conflict. A conflict means that you need to fix something,
47
before you should commit.
49
Conflicts normally are listed as short, human-readable messages. If --text
50
is supplied, the pathnames of files with text conflicts are listed,
51
instead. (This is useful for editing all files with text conflicts.)
53
Use bzr resolve when you have fixed a problem.
36
(conflicts are determined by the presence of .BASE .TREE, and .OTHER
57
takes_options = [Option('text', help='list text conflicts by pathname')]
59
def run(self, text=False):
60
from bzrlib.workingtree import WorkingTree
61
wt = WorkingTree.open_containing(u'.')[0]
62
for conflict in wt.conflicts():
64
if conflict.typestring != 'text conflict':
66
self.outf.write(conflict.path + '\n')
68
self.outf.write(str(conflict) + '\n')
71
class cmd_resolve(commands.Command):
40
for path in Branch.open_containing('.')[0].working_tree().iter_conflicts():
43
class cmd_resolve(bzrlib.commands.Command):
72
44
"""Mark a conflict as resolved.
74
Merge will do its best to combine the changes in two branches, but there
75
are some kinds of problems only a human can fix. When it encounters those,
76
it will mark a conflict. A conflict means that you need to fix something,
77
before you should commit.
79
Once you have fixed a problem, use "bzr resolve FILE.." to mark
80
individual files as fixed, or "bzr resolve --all" to mark all conflicts as
83
See also bzr conflicts.
85
46
aliases = ['resolved']
86
47
takes_args = ['file*']
87
takes_options = [Option('all', help='Resolve all conflicts in this tree')]
48
takes_options = ['all']
88
49
def run(self, file_list=None, all=False):
89
from bzrlib.workingtree import WorkingTree
92
raise errors.BzrCommandError("If --all is specified,"
93
" no FILE may be provided")
94
tree = WorkingTree.open_containing('.')[0]
98
raise errors.BzrCommandError("command 'resolve' needs one or"
99
" more FILE, or --all")
100
tree = WorkingTree.open_containing(file_list[0])[0]
101
to_resolve = [tree.relpath(p) for p in file_list]
102
resolve(tree, to_resolve)
105
def resolve(tree, paths=None, ignore_misses=False):
106
tree.lock_tree_write()
108
tree_conflicts = tree.conflicts()
110
new_conflicts = ConflictList()
111
selected_conflicts = tree_conflicts
113
new_conflicts, selected_conflicts = \
114
tree_conflicts.select_conflicts(tree, paths, ignore_misses)
116
tree.set_conflicts(new_conflicts)
117
except errors.UnsupportedOperation:
119
selected_conflicts.remove_files(tree)
52
raise BzrCommandError(
53
"command 'resolve' needs one or more FILE, or --all")
54
tree = Branch.open_containing('.')[0].working_tree()
55
file_list = list(tree.abspath(f) for f in tree.iter_conflicts())
58
raise BzrCommandError(
59
"If --all is specified, no FILE may be provided")
60
for filename in file_list:
62
for suffix in CONFLICT_SUFFIXES:
64
os.unlink(filename+suffix)
66
if e.errno != errno.ENOENT:
70
if failures == len(CONFLICT_SUFFIXES):
71
if not os.path.exists(filename):
72
print "%s does not exist" % filename
74
print "%s is not conflicted" % filename
124
76
def restore(filename):
146
98
if e.errno != errno.ENOENT:
148
100
if not conflicted:
149
raise errors.NotConflicted(filename)
152
class ConflictList(object):
153
"""List of conflicts.
155
Typically obtained from WorkingTree.conflicts()
157
Can be instantiated from stanzas or from Conflict subclasses.
160
def __init__(self, conflicts=None):
161
object.__init__(self)
162
if conflicts is None:
165
self.__list = conflicts
168
return len(self.__list) == 0
171
return len(self.__list)
174
return iter(self.__list)
176
def __getitem__(self, key):
177
return self.__list[key]
179
def append(self, conflict):
180
return self.__list.append(conflict)
182
def __eq__(self, other_list):
183
return list(self) == list(other_list)
185
def __ne__(self, other_list):
186
return not (self == other_list)
189
return "ConflictList(%r)" % self.__list
192
def from_stanzas(stanzas):
193
"""Produce a new ConflictList from an iterable of stanzas"""
194
conflicts = ConflictList()
195
for stanza in stanzas:
196
conflicts.append(Conflict.factory(**stanza.as_dict()))
199
def to_stanzas(self):
200
"""Generator of stanzas"""
201
for conflict in self:
202
yield conflict.as_stanza()
204
def to_strings(self):
205
"""Generate strings for the provided conflicts"""
206
for conflict in self:
209
def remove_files(self, tree):
210
"""Remove the THIS, BASE and OTHER files for listed conflicts"""
211
for conflict in self:
212
if not conflict.has_files:
214
for suffix in CONFLICT_SUFFIXES:
216
osutils.delete_any(tree.abspath(conflict.path+suffix))
218
if e.errno != errno.ENOENT:
221
def select_conflicts(self, tree, paths, ignore_misses=False):
222
"""Select the conflicts associated with paths in a tree.
224
File-ids are also used for this.
225
:return: a pair of ConflictLists: (not_selected, selected)
227
path_set = set(paths)
229
selected_paths = set()
230
new_conflicts = ConflictList()
231
selected_conflicts = ConflictList()
233
file_id = tree.path2id(path)
234
if file_id is not None:
237
for conflict in self:
239
for key in ('path', 'conflict_path'):
240
cpath = getattr(conflict, key, None)
243
if cpath in path_set:
245
selected_paths.add(cpath)
246
for key in ('file_id', 'conflict_file_id'):
247
cfile_id = getattr(conflict, key, None)
251
cpath = ids[cfile_id]
255
selected_paths.add(cpath)
257
selected_conflicts.append(conflict)
259
new_conflicts.append(conflict)
260
if ignore_misses is not True:
261
for path in [p for p in paths if p not in selected_paths]:
262
if not os.path.exists(tree.abspath(path)):
263
print "%s does not exist" % path
265
print "%s is not conflicted" % path
266
return new_conflicts, selected_conflicts
269
class Conflict(object):
270
"""Base class for all types of conflict"""
274
def __init__(self, path, file_id=None):
276
self.file_id = file_id
279
s = rio.Stanza(type=self.typestring, path=self.path)
280
if self.file_id is not None:
281
s.add('file_id', self.file_id)
285
return [type(self), self.path, self.file_id]
287
def __cmp__(self, other):
288
if getattr(other, "_cmp_list", None) is None:
290
return cmp(self._cmp_list(), other._cmp_list())
293
return hash((type(self), self.path, self.file_id))
295
def __eq__(self, other):
296
return self.__cmp__(other) == 0
298
def __ne__(self, other):
299
return not self.__eq__(other)
302
return self.format % self.__dict__
305
rdict = dict(self.__dict__)
306
rdict['class'] = self.__class__.__name__
307
return self.rformat % rdict
310
def factory(type, **kwargs):
312
return ctype[type](**kwargs)
315
def sort_key(conflict):
316
if conflict.path is not None:
317
return conflict.path, conflict.typestring
318
elif getattr(conflict, "conflict_path", None) is not None:
319
return conflict.conflict_path, conflict.typestring
321
return None, conflict.typestring
324
class PathConflict(Conflict):
325
"""A conflict was encountered merging file paths"""
327
typestring = 'path conflict'
329
format = 'Path conflict: %(path)s / %(conflict_path)s'
331
rformat = '%(class)s(%(path)r, %(conflict_path)r, %(file_id)r)'
332
def __init__(self, path, conflict_path=None, file_id=None):
333
Conflict.__init__(self, path, file_id)
334
self.conflict_path = conflict_path
337
s = Conflict.as_stanza(self)
338
if self.conflict_path is not None:
339
s.add('conflict_path', self.conflict_path)
343
class ContentsConflict(PathConflict):
344
"""The files are of different types, or not present"""
348
typestring = 'contents conflict'
350
format = 'Contents conflict in %(path)s'
353
class TextConflict(PathConflict):
354
"""The merge algorithm could not resolve all differences encountered."""
358
typestring = 'text conflict'
360
format = 'Text conflict in %(path)s'
363
class HandledConflict(Conflict):
364
"""A path problem that has been provisionally resolved.
365
This is intended to be a base class.
368
rformat = "%(class)s(%(action)r, %(path)r, %(file_id)r)"
370
def __init__(self, action, path, file_id=None):
371
Conflict.__init__(self, path, file_id)
375
return Conflict._cmp_list(self) + [self.action]
378
s = Conflict.as_stanza(self)
379
s.add('action', self.action)
383
class HandledPathConflict(HandledConflict):
384
"""A provisionally-resolved path problem involving two paths.
385
This is intended to be a base class.
388
rformat = "%(class)s(%(action)r, %(path)r, %(conflict_path)r,"\
389
" %(file_id)r, %(conflict_file_id)r)"
391
def __init__(self, action, path, conflict_path, file_id=None,
392
conflict_file_id=None):
393
HandledConflict.__init__(self, action, path, file_id)
394
self.conflict_path = conflict_path
395
self.conflict_file_id = conflict_file_id
398
return HandledConflict._cmp_list(self) + [self.conflict_path,
399
self.conflict_file_id]
402
s = HandledConflict.as_stanza(self)
403
s.add('conflict_path', self.conflict_path)
404
if self.conflict_file_id is not None:
405
s.add('conflict_file_id', self.conflict_file_id)
410
class DuplicateID(HandledPathConflict):
411
"""Two files want the same file_id."""
413
typestring = 'duplicate id'
415
format = 'Conflict adding id to %(conflict_path)s. %(action)s %(path)s.'
418
class DuplicateEntry(HandledPathConflict):
419
"""Two directory entries want to have the same name."""
421
typestring = 'duplicate'
423
format = 'Conflict adding file %(conflict_path)s. %(action)s %(path)s.'
426
class ParentLoop(HandledPathConflict):
427
"""An attempt to create an infinitely-looping directory structure.
428
This is rare, but can be produced like so:
437
typestring = 'parent loop'
439
format = 'Conflict moving %(conflict_path)s into %(path)s. %(action)s.'
442
class UnversionedParent(HandledConflict):
443
"""An attempt to version an file whose parent directory is not versioned.
444
Typically, the result of a merge where one tree unversioned the directory
445
and the other added a versioned file to it.
448
typestring = 'unversioned parent'
450
format = 'Conflict because %(path)s is not versioned, but has versioned'\
451
' children. %(action)s.'
454
class MissingParent(HandledConflict):
455
"""An attempt to add files to a directory that is not present.
456
Typically, the result of a merge where THIS deleted the directory and
457
the OTHER added a file to it.
458
See also: DeletingParent (same situation, reversed THIS and OTHER)
461
typestring = 'missing parent'
463
format = 'Conflict adding files to %(path)s. %(action)s.'
466
class DeletingParent(HandledConflict):
467
"""An attempt to add files to a directory that is not present.
468
Typically, the result of a merge where one OTHER deleted the directory and
469
the THIS added a file to it.
472
typestring = 'deleting parent'
474
format = "Conflict: can't delete %(path)s because it is not empty. "\
481
def register_types(*conflict_types):
482
"""Register a Conflict subclass for serialization purposes"""
484
for conflict_type in conflict_types:
485
ctype[conflict_type.typestring] = conflict_type
488
register_types(ContentsConflict, TextConflict, PathConflict, DuplicateID,
489
DuplicateEntry, ParentLoop, UnversionedParent, MissingParent,
101
raise NotConflicted(filename)