1
from merge_core import merge_flex, ApplyMerge3, BackupBeforeChange
1
from merge_core import merge_flex
2
2
from changeset import generate_changeset, ExceptionConflictHandler
3
from changeset import Inventory, Diff3Merge
3
from changeset import Inventory
4
4
from bzrlib import find_branch
5
5
import bzrlib.osutils
6
6
from bzrlib.errors import BzrCommandError
207
200
raise UnrelatedBranches()
208
201
base_revision = ['.', base_revno]
209
202
base_branch, base_tree = get_tree(base_revision, tempdir, "base")
210
if file_list is None:
211
interesting_ids = None
213
interesting_ids = set()
214
this_tree = this_branch.working_tree()
215
for fname in file_list:
216
path = this_branch.relpath(fname)
218
for tree in (this_tree, base_tree.tree, other_tree.tree):
219
file_id = tree.inventory.path2id(path)
220
if file_id is not None:
221
interesting_ids.add(file_id)
224
raise BzrCommandError("%s is not a source file in any"
226
203
merge_inner(this_branch, other_tree, base_tree, tempdir,
227
ignore_zero=ignore_zero, backup_files=backup_files,
228
merge_type=merge_type, interesting_ids=interesting_ids)
204
ignore_zero=ignore_zero)
230
206
shutil.rmtree(tempdir)
233
def set_interesting(inventory_a, inventory_b, interesting_ids):
234
"""Mark files whose ids are in interesting_ids as interesting
236
for inventory in (inventory_a, inventory_b):
237
for path, source_file in inventory.iteritems():
238
source_file.interesting = source_file.id in interesting_ids
241
def generate_cset_optimized(tree_a, tree_b, inventory_a, inventory_b,
242
interesting_ids=None):
243
"""Generate a changeset. If interesting_ids is supplied, only changes
244
to those files will be shown. Metadata changes are stripped.
209
def generate_cset_optimized(tree_a, tree_b, inventory_a, inventory_b):
210
"""Generate a changeset, using the text_id to mark really-changed files.
211
This permits blazing comparisons when text_ids are present. It also
212
disables metadata comparison for files with identical texts.
246
if interesting_ids is not None:
247
set_interesting(inventory_a, inventory_b, interesting_ids)
214
for file_id in tree_a.tree.inventory:
215
if file_id not in tree_b.tree.inventory:
217
entry_a = tree_a.tree.inventory[file_id]
218
entry_b = tree_b.tree.inventory[file_id]
219
if (entry_a.kind, entry_b.kind) != ("file", "file"):
221
if None in (entry_a.text_id, entry_b.text_id):
223
if entry_a.text_id != entry_b.text_id:
225
inventory_a[abspath(tree_a.tree, file_id)].interesting = False
226
inventory_b[abspath(tree_b.tree, file_id)].interesting = False
248
227
cset = generate_changeset(tree_a, tree_b, inventory_a, inventory_b)
249
228
for entry in cset.entries.itervalues():
250
229
entry.metadata_change = None
254
233
def merge_inner(this_branch, other_tree, base_tree, tempdir,
255
ignore_zero=False, merge_type=ApplyMerge3, backup_files=False,
256
interesting_ids=None):
258
def merge_factory(base_file, other_file):
259
contents_change = merge_type(base_file, other_file)
261
contents_change = BackupBeforeChange(contents_change)
262
return contents_change
264
def generate_cset(tree_a, tree_b, inventory_a, inventory_b):
265
return generate_cset_optimized(tree_a, tree_b, inventory_a, inventory_b,
268
235
this_tree = get_tree((this_branch.base, None), tempdir, "this")[1]
270
237
def get_inventory(tree):
271
238
return tree.inventory
273
240
inv_changes = merge_flex(this_tree, base_tree, other_tree,
274
generate_cset, get_inventory,
241
generate_cset_optimized, get_inventory,
275
242
MergeConflictHandler(base_tree.root,
276
ignore_zero=ignore_zero),
277
merge_factory=merge_factory)
243
ignore_zero=ignore_zero))
280
246
for id, path in inv_changes.iteritems():
285
251
assert path.startswith('./')
287
253
adjust_ids.append((path, id))
288
if len(adjust_ids) > 0:
289
this_branch.set_inventory(regen_inventory(this_branch, this_tree.root,
254
this_branch.set_inventory(regen_inventory(this_branch, this_tree.root, adjust_ids))
293
257
def regen_inventory(this_branch, root, new_entries):