Show More
merge.py
1207 lines
| 44.7 KiB
| text/x-python
|
PythonLexer
/ mercurial / merge.py
Matt Mackall
|
r2775 | # merge.py - directory-level update/merge handling for Mercurial | ||
# | ||||
Thomas Arendsen Hein
|
r4635 | # Copyright 2006, 2007 Matt Mackall <mpm@selenic.com> | ||
Matt Mackall
|
r2775 | # | ||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Matt Mackall
|
r2775 | |||
Pierre-Yves David
|
r20590 | import struct | ||
Matt Mackall
|
r6518 | from node import nullid, nullrev, hex, bin | ||
Matt Mackall
|
r3891 | from i18n import _ | ||
Pierre-Yves David
|
r18985 | from mercurial import obsolete | ||
Martin von Zweigbergk
|
r22964 | import error as errormod, util, filemerge, copies, subrepo, worker | ||
Simon Heimberg
|
r8312 | import errno, os, shutil | ||
Matt Mackall
|
r6512 | |||
Pierre-Yves David
|
r20590 | _pack = struct.pack | ||
_unpack = struct.unpack | ||||
Pierre-Yves David
|
r20593 | def _droponode(data): | ||
# used for compatibility for v1 | ||||
Martin von Zweigbergk
|
r23380 | bits = data.split('\0') | ||
Pierre-Yves David
|
r20593 | bits = bits[:-2] + bits[-1:] | ||
Martin von Zweigbergk
|
r23380 | return '\0'.join(bits) | ||
Pierre-Yves David
|
r20593 | |||
Matt Mackall
|
r6512 | class mergestate(object): | ||
Pierre-Yves David
|
r20590 | '''track 3-way merge state of individual files | ||
it is stored on disk when needed. Two file are used, one with an old | ||||
format, one with a new format. Both contains similar data, but the new | ||||
format can store new kind of field. | ||||
Current new format is a list of arbitrary record of the form: | ||||
[type][length][content] | ||||
Type is a single character, length is a 4 bytes integer, content is an | ||||
Olle Lundberg
|
r20607 | arbitrary suites of bytes of length `length`. | ||
Pierre-Yves David
|
r20590 | |||
Type should be a letter. Capital letter are mandatory record, Mercurial | ||||
should abort if they are unknown. lower case record can be safely ignored. | ||||
Currently known record: | ||||
L: the node of the "local" part of the merge (hexified version) | ||||
Pierre-Yves David
|
r20591 | O: the node of the "other" part of the merge (hexified version) | ||
Pierre-Yves David
|
r20590 | F: a file to be merged entry | ||
''' | ||||
Martin von Zweigbergk
|
r23380 | statepathv1 = 'merge/state' | ||
statepathv2 = 'merge/state2' | ||||
Pierre-Yves David
|
r20651 | |||
Matt Mackall
|
r6512 | def __init__(self, repo): | ||
self._repo = repo | ||||
Peter Arrenbrecht
|
r12369 | self._dirty = False | ||
Matt Mackall
|
r6518 | self._read() | ||
Pierre-Yves David
|
r20651 | |||
Pierre-Yves David
|
r20591 | def reset(self, node=None, other=None): | ||
Matt Mackall
|
r6512 | self._state = {} | ||
Gregory Szorc
|
r21261 | self._local = None | ||
self._other = None | ||||
Matt Mackall
|
r7848 | if node: | ||
self._local = node | ||||
Pierre-Yves David
|
r20591 | self._other = other | ||
Martin von Zweigbergk
|
r23380 | shutil.rmtree(self._repo.join('merge'), True) | ||
Peter Arrenbrecht
|
r12369 | self._dirty = False | ||
Pierre-Yves David
|
r20651 | |||
Matt Mackall
|
r6518 | def _read(self): | ||
Pierre-Yves David
|
r20652 | """Analyse each record content to restore a serialized state from disk | ||
This function process "record" entry produced by the de-serialization | ||||
of on disk file. | ||||
""" | ||||
Matt Mackall
|
r6518 | self._state = {} | ||
Gregory Szorc
|
r21261 | self._local = None | ||
self._other = None | ||||
Pierre-Yves David
|
r20589 | records = self._readrecords() | ||
for rtype, record in records: | ||||
if rtype == 'L': | ||||
self._local = bin(record) | ||||
Pierre-Yves David
|
r20591 | elif rtype == 'O': | ||
self._other = bin(record) | ||||
Martin von Zweigbergk
|
r23380 | elif rtype == 'F': | ||
bits = record.split('\0') | ||||
Pierre-Yves David
|
r20589 | self._state[bits[0]] = bits[1:] | ||
elif not rtype.islower(): | ||||
FUJIWARA Katsunori
|
r20868 | raise util.Abort(_('unsupported merge state record: %s') | ||
% rtype) | ||||
Pierre-Yves David
|
r20589 | self._dirty = False | ||
Pierre-Yves David
|
r20651 | |||
Pierre-Yves David
|
r20589 | def _readrecords(self): | ||
Pierre-Yves David
|
r20652 | """Read merge state from disk and return a list of record (TYPE, data) | ||
Mads Kiilerich
|
r21024 | We read data from both v1 and v2 files and decide which one to use. | ||
Pierre-Yves David
|
r20652 | |||
Mads Kiilerich
|
r21024 | V1 has been used by version prior to 2.9.1 and contains less data than | ||
v2. We read both versions and check if no data in v2 contradicts | ||||
Pierre-Yves David
|
r20652 | v1. If there is not contradiction we can safely assume that both v1 | ||
and v2 were written at the same time and use the extract data in v2. If | ||||
there is contradiction we ignore v2 content as we assume an old version | ||||
Mads Kiilerich
|
r21024 | of Mercurial has overwritten the mergestate file and left an old v2 | ||
Pierre-Yves David
|
r20652 | file around. | ||
returns list of record [(TYPE, data), ...]""" | ||||
Pierre-Yves David
|
r20590 | v1records = self._readrecordsv1() | ||
v2records = self._readrecordsv2() | ||||
Pierre-Yves David
|
r20593 | oldv2 = set() # old format version of v2 record | ||
for rec in v2records: | ||||
if rec[0] == 'L': | ||||
oldv2.add(rec) | ||||
elif rec[0] == 'F': | ||||
# drop the onode data (not contained in v1) | ||||
oldv2.add(('F', _droponode(rec[1]))) | ||||
for rec in v1records: | ||||
if rec not in oldv2: | ||||
Pierre-Yves David
|
r20590 | # v1 file is newer than v2 file, use it | ||
Pierre-Yves David
|
r20592 | # we have to infer the "other" changeset of the merge | ||
# we cannot do better than that with v1 of the format | ||||
mctx = self._repo[None].parents()[-1] | ||||
v1records.append(('O', mctx.hex())) | ||||
Pierre-Yves David
|
r20593 | # add place holder "other" file node information | ||
# nobody is using it yet so we do no need to fetch the data | ||||
# if mctx was wrong `mctx[bits[-2]]` may fails. | ||||
for idx, r in enumerate(v1records): | ||||
if r[0] == 'F': | ||||
Martin von Zweigbergk
|
r23380 | bits = r[1].split('\0') | ||
Pierre-Yves David
|
r20593 | bits.insert(-2, '') | ||
Martin von Zweigbergk
|
r23380 | v1records[idx] = (r[0], '\0'.join(bits)) | ||
Pierre-Yves David
|
r20590 | return v1records | ||
else: | ||||
return v2records | ||||
Pierre-Yves David
|
r20651 | |||
Pierre-Yves David
|
r20590 | def _readrecordsv1(self): | ||
Pierre-Yves David
|
r20652 | """read on disk merge state for version 1 file | ||
returns list of record [(TYPE, data), ...] | ||||
Note: the "F" data from this file are one entry short | ||||
(no "other file node" entry) | ||||
""" | ||||
Pierre-Yves David
|
r20589 | records = [] | ||
Matt Mackall
|
r6518 | try: | ||
Angel Ezquerra
|
r23877 | f = self._repo.vfs(self.statepathv1) | ||
Patrick Mezard
|
r6530 | for i, l in enumerate(f): | ||
if i == 0: | ||||
Pierre-Yves David
|
r20589 | records.append(('L', l[:-1])) | ||
Patrick Mezard
|
r6530 | else: | ||
Pierre-Yves David
|
r20589 | records.append(('F', l[:-1])) | ||
Dan Villiom Podlaski Christiansen
|
r13400 | f.close() | ||
Matt Mackall
|
r6518 | except IOError, err: | ||
if err.errno != errno.ENOENT: | ||||
raise | ||||
Pierre-Yves David
|
r20589 | return records | ||
Pierre-Yves David
|
r20651 | |||
Pierre-Yves David
|
r20590 | def _readrecordsv2(self): | ||
Pierre-Yves David
|
r20652 | """read on disk merge state for version 2 file | ||
returns list of record [(TYPE, data), ...] | ||||
""" | ||||
Pierre-Yves David
|
r20590 | records = [] | ||
try: | ||||
Angel Ezquerra
|
r23877 | f = self._repo.vfs(self.statepathv2) | ||
Pierre-Yves David
|
r20590 | data = f.read() | ||
off = 0 | ||||
end = len(data) | ||||
while off < end: | ||||
rtype = data[off] | ||||
off += 1 | ||||
Olle Lundberg
|
r20607 | length = _unpack('>I', data[off:(off + 4)])[0] | ||
Pierre-Yves David
|
r20590 | off += 4 | ||
Olle Lundberg
|
r20607 | record = data[off:(off + length)] | ||
off += length | ||||
Pierre-Yves David
|
r20590 | records.append((rtype, record)) | ||
f.close() | ||||
except IOError, err: | ||||
if err.errno != errno.ENOENT: | ||||
raise | ||||
return records | ||||
Pierre-Yves David
|
r20651 | |||
Gregory Szorc
|
r21264 | def active(self): | ||
"""Whether mergestate is active. | ||||
Returns True if there appears to be mergestate. This is a rough proxy | ||||
for "is a merge in progress." | ||||
""" | ||||
# Check local variables before looking at filesystem for performance | ||||
# reasons. | ||||
return bool(self._local) or bool(self._state) or \ | ||||
Angel Ezquerra
|
r23877 | self._repo.vfs.exists(self.statepathv1) or \ | ||
self._repo.vfs.exists(self.statepathv2) | ||||
Gregory Szorc
|
r21264 | |||
Peter Arrenbrecht
|
r12369 | def commit(self): | ||
Pierre-Yves David
|
r20652 | """Write current state on disk (if necessary)""" | ||
Peter Arrenbrecht
|
r12369 | if self._dirty: | ||
Pierre-Yves David
|
r20589 | records = [] | ||
Martin von Zweigbergk
|
r23380 | records.append(('L', hex(self._local))) | ||
records.append(('O', hex(self._other))) | ||||
Peter Arrenbrecht
|
r12369 | for d, v in self._state.iteritems(): | ||
Martin von Zweigbergk
|
r23380 | records.append(('F', '\0'.join([d] + v))) | ||
Pierre-Yves David
|
r20589 | self._writerecords(records) | ||
Peter Arrenbrecht
|
r12369 | self._dirty = False | ||
Pierre-Yves David
|
r20651 | |||
Pierre-Yves David
|
r20589 | def _writerecords(self, records): | ||
Pierre-Yves David
|
r20652 | """Write current state on disk (both v1 and v2)""" | ||
Pierre-Yves David
|
r20590 | self._writerecordsv1(records) | ||
self._writerecordsv2(records) | ||||
Pierre-Yves David
|
r20651 | |||
Pierre-Yves David
|
r20590 | def _writerecordsv1(self, records): | ||
Pierre-Yves David
|
r20652 | """Write current state on disk in a version 1 file""" | ||
Angel Ezquerra
|
r23877 | f = self._repo.vfs(self.statepathv1, 'w') | ||
Pierre-Yves David
|
r20589 | irecords = iter(records) | ||
lrecords = irecords.next() | ||||
assert lrecords[0] == 'L' | ||||
Martin von Zweigbergk
|
r23380 | f.write(hex(self._local) + '\n') | ||
Pierre-Yves David
|
r20589 | for rtype, data in irecords: | ||
Martin von Zweigbergk
|
r23380 | if rtype == 'F': | ||
f.write('%s\n' % _droponode(data)) | ||||
Pierre-Yves David
|
r20589 | f.close() | ||
Pierre-Yves David
|
r20651 | |||
Pierre-Yves David
|
r20590 | def _writerecordsv2(self, records): | ||
Pierre-Yves David
|
r20652 | """Write current state on disk in a version 2 file""" | ||
Angel Ezquerra
|
r23877 | f = self._repo.vfs(self.statepathv2, 'w') | ||
Pierre-Yves David
|
r20590 | for key, data in records: | ||
assert len(key) == 1 | ||||
Martin von Zweigbergk
|
r23380 | format = '>sI%is' % len(data) | ||
Pierre-Yves David
|
r20590 | f.write(_pack(format, key, len(data), data)) | ||
f.close() | ||||
Pierre-Yves David
|
r20651 | |||
Mads Kiilerich
|
r18338 | def add(self, fcl, fco, fca, fd): | ||
Pierre-Yves David
|
r20652 | """add a new (potentially?) conflicting file the merge state | ||
fcl: file context for local, | ||||
fco: file context for remote, | ||||
fca: file context for ancestors, | ||||
fd: file path of the resulting merge. | ||||
note: also write the local version to the `.hg/merge` directory. | ||||
""" | ||||
Dirkjan Ochtman
|
r6517 | hash = util.sha1(fcl.path()).hexdigest() | ||
Angel Ezquerra
|
r23877 | self._repo.vfs.write('merge/' + hash, fcl.data()) | ||
Pierre-Yves David
|
r20593 | self._state[fd] = ['u', hash, fcl.path(), | ||
fca.path(), hex(fca.filenode()), | ||||
fco.path(), hex(fco.filenode()), | ||||
fcl.flags()] | ||||
Peter Arrenbrecht
|
r12369 | self._dirty = True | ||
Pierre-Yves David
|
r20651 | |||
Matt Mackall
|
r6512 | def __contains__(self, dfile): | ||
return dfile in self._state | ||||
Pierre-Yves David
|
r20651 | |||
Matt Mackall
|
r6512 | def __getitem__(self, dfile): | ||
Matt Mackall
|
r6518 | return self._state[dfile][0] | ||
Pierre-Yves David
|
r20651 | |||
Matt Mackall
|
r6518 | def __iter__(self): | ||
Mads Kiilerich
|
r21268 | return iter(sorted(self._state)) | ||
Pierre-Yves David
|
r20651 | |||
Bryan O'Sullivan
|
r19285 | def files(self): | ||
return self._state.keys() | ||||
Pierre-Yves David
|
r20651 | |||
Matt Mackall
|
r6512 | def mark(self, dfile, state): | ||
Matt Mackall
|
r6518 | self._state[dfile][0] = state | ||
Peter Arrenbrecht
|
r12369 | self._dirty = True | ||
Pierre-Yves David
|
r20651 | |||
Gregory Szorc
|
r21266 | def unresolved(self): | ||
"""Obtain the paths of unresolved files.""" | ||||
for f, entry in self._state.items(): | ||||
if entry[0] == 'u': | ||||
yield f | ||||
Durham Goode
|
r21524 | def resolve(self, dfile, wctx, labels=None): | ||
Pierre-Yves David
|
r20652 | """rerun merge process for file path `dfile`""" | ||
Matt Mackall
|
r6512 | if self[dfile] == 'r': | ||
return 0 | ||||
Pierre-Yves David
|
r20593 | stateentry = self._state[dfile] | ||
state, hash, lfile, afile, anode, ofile, onode, flags = stateentry | ||||
Pierre-Yves David
|
r20594 | octx = self._repo[self._other] | ||
Mads Kiilerich
|
r18338 | fcd = wctx[dfile] | ||
fco = octx[ofile] | ||||
fca = self._repo.filectx(afile, fileid=anode) | ||||
# "premerge" x flags | ||||
flo = fco.flags() | ||||
fla = fca.flags() | ||||
if 'x' in flags + flo + fla and 'l' not in flags + flo + fla: | ||||
if fca.node() == nullid: | ||||
self._repo.ui.warn(_('warning: cannot merge flags for %s\n') % | ||||
afile) | ||||
elif flags == fla: | ||||
flags = flo | ||||
# restore local | ||||
Angel Ezquerra
|
r23877 | f = self._repo.vfs('merge/' + hash) | ||
Matt Mackall
|
r6512 | self._repo.wwrite(dfile, f.read(), flags) | ||
Dan Villiom Podlaski Christiansen
|
r13400 | f.close() | ||
Durham Goode
|
r21524 | r = filemerge.filemerge(self._repo, self._local, lfile, fcd, fco, fca, | ||
labels=labels) | ||||
Matt Mackall
|
r13536 | if r is None: | ||
# no real conflict | ||||
del self._state[dfile] | ||||
Mads Kiilerich
|
r20792 | self._dirty = True | ||
Matt Mackall
|
r13536 | elif not r: | ||
Matt Mackall
|
r6512 | self.mark(dfile, 'r') | ||
return r | ||||
Matt Mackall
|
r2775 | |||
Martin von Zweigbergk
|
r23653 | def _checkunknownfile(repo, wctx, mctx, f, f2=None): | ||
if f2 is None: | ||||
f2 = f | ||||
Martin von Zweigbergk
|
r23478 | return (os.path.isfile(repo.wjoin(f)) | ||
Angel Ezquerra
|
r23879 | and repo.wvfs.audit.check(f) | ||
Matt Mackall
|
r16284 | and repo.dirstate.normalize(f) not in repo.dirstate | ||
Martin von Zweigbergk
|
r23653 | and mctx[f2].cmp(wctx[f])) | ||
Matt Mackall
|
r16093 | |||
Martin von Zweigbergk
|
r23655 | def _checkunknownfiles(repo, wctx, mctx, force, actions): | ||
""" | ||||
Considers any actions that care about the presence of conflicting unknown | ||||
files. For some actions, the result is to abort; for others, it is to | ||||
choose a different action. | ||||
""" | ||||
aborts = [] | ||||
if not force: | ||||
for f, (m, args, msg) in actions.iteritems(): | ||||
if m in ('c', 'dc'): | ||||
if _checkunknownfile(repo, wctx, mctx, f): | ||||
aborts.append(f) | ||||
elif m == 'dg': | ||||
if _checkunknownfile(repo, wctx, mctx, f, args[0]): | ||||
aborts.append(f) | ||||
for f in sorted(aborts): | ||||
repo.ui.warn(_("%s: untracked file differs\n") % f) | ||||
if aborts: | ||||
raise util.Abort(_("untracked files in working directory differ " | ||||
"from files in requested revision")) | ||||
for f, (m, args, msg) in actions.iteritems(): | ||||
if m == 'c': | ||||
actions[f] = ('g', args, msg) | ||||
elif m == 'cm': | ||||
fl2, anc = args | ||||
different = _checkunknownfile(repo, wctx, mctx, f) | ||||
if different: | ||||
actions[f] = ('m', (f, f, None, False, anc), | ||||
"remote differs from untracked local") | ||||
else: | ||||
actions[f] = ('g', (fl2,), "remote created") | ||||
Matt Mackall
|
r6269 | def _forgetremoved(wctx, mctx, branchmerge): | ||
Matt Mackall
|
r3107 | """ | ||
Forget removed files | ||||
If we're jumping between revisions (as opposed to merging), and if | ||||
neither the working directory nor the target rev has the file, | ||||
then we need to remove it from the dirstate, to prevent the | ||||
dirstate from listing the file when it is no longer in the | ||||
manifest. | ||||
Alexis S. L. Carvalho
|
r6242 | |||
If we're merging, and the other revision has removed a file | ||||
that is not present in the working directory, we need to mark it | ||||
as removed. | ||||
Matt Mackall
|
r3107 | """ | ||
Martin von Zweigbergk
|
r23640 | actions = {} | ||
m = 'f' | ||||
Mads Kiilerich
|
r21545 | if branchmerge: | ||
Martin von Zweigbergk
|
r23640 | m = 'r' | ||
Alexis S. L. Carvalho
|
r6242 | for f in wctx.deleted(): | ||
Matt Mackall
|
r6272 | if f not in mctx: | ||
Martin von Zweigbergk
|
r23640 | actions[f] = m, None, "forget deleted" | ||
Alexis S. L. Carvalho
|
r6242 | |||
if not branchmerge: | ||||
for f in wctx.removed(): | ||||
Matt Mackall
|
r6272 | if f not in mctx: | ||
Martin von Zweigbergk
|
r23640 | actions[f] = 'f', None, "forget removed" | ||
Matt Mackall
|
r3107 | |||
Martin von Zweigbergk
|
r23640 | return actions | ||
Matt Mackall
|
r3107 | |||
Mads Kiilerich
|
r20640 | def _checkcollision(repo, wmf, actions): | ||
FUJIWARA Katsunori
|
r19105 | # build provisional merged manifest up | ||
pmmf = set(wmf) | ||||
Mads Kiilerich
|
r21545 | if actions: | ||
# k, dr, e and rd are no-op | ||||
for m in 'a', 'f', 'g', 'cd', 'dc': | ||||
for f, args, msg in actions[m]: | ||||
pmmf.add(f) | ||||
for f, args, msg in actions['r']: | ||||
pmmf.discard(f) | ||||
for f, args, msg in actions['dm']: | ||||
f2, flags = args | ||||
pmmf.discard(f2) | ||||
pmmf.add(f) | ||||
for f, args, msg in actions['dg']: | ||||
pmmf.add(f) | ||||
for f, args, msg in actions['m']: | ||||
f1, f2, fa, move, anc = args | ||||
if move: | ||||
pmmf.discard(f1) | ||||
pmmf.add(f) | ||||
FUJIWARA Katsunori
|
r19105 | |||
# check case-folding collision in provisional merged manifest | ||||
foldmap = {} | ||||
for f in sorted(pmmf): | ||||
fold = util.normcase(f) | ||||
if fold in foldmap: | ||||
raise util.Abort(_("case-folding collision between %s and %s") | ||||
% (f, foldmap[fold])) | ||||
foldmap[fold] = f | ||||
Durham Goode
|
r18778 | def manifestmerge(repo, wctx, p2, pa, branchmerge, force, partial, | ||
Mads Kiilerich
|
r21080 | acceptremote, followcopies): | ||
Matt Mackall
|
r3105 | """ | ||
Alecs King
|
r11817 | Merge p1 and p2 with ancestor pa and generate merge action list | ||
Matt Mackall
|
r3315 | |||
Siddharth Agarwal
|
r18605 | branchmerge and force are as passed in to update | ||
Matt Mackall
|
r3315 | partial = function to filter file lists | ||
Durham Goode
|
r18778 | acceptremote = accept the incoming changes without prompting | ||
Matt Mackall
|
r3105 | """ | ||
Martin von Zweigbergk
|
r23526 | copy, movewithdir, diverge, renamedelete = {}, {}, {}, {} | ||
Matt Mackall
|
r8753 | |||
Siddharth Agarwal
|
r18651 | # manifests fetched in order are going to be faster, so prime the caches | ||
[x.manifest() for x in | ||||
sorted(wctx.parents() + [p2, pa], key=lambda x: x.rev())] | ||||
if followcopies: | ||||
Bryan O'Sullivan
|
r18611 | ret = copies.mergecopies(repo, wctx, p2, pa) | ||
Siddharth Agarwal
|
r18134 | copy, movewithdir, diverge, renamedelete = ret | ||
Matt Mackall
|
r8753 | |||
repo.ui.note(_("resolving manifests\n")) | ||||
Siddharth Agarwal
|
r18605 | repo.ui.debug(" branchmerge: %s, force: %s, partial: %s\n" | ||
% (bool(branchmerge), bool(force), bool(partial))) | ||||
Bryan O'Sullivan
|
r18611 | repo.ui.debug(" ancestor: %s, local: %s, remote: %s\n" % (pa, wctx, p2)) | ||
Matt Mackall
|
r8753 | |||
Bryan O'Sullivan
|
r18611 | m1, m2, ma = wctx.manifest(), p2.manifest(), pa.manifest() | ||
Matt Mackall
|
r8753 | copied = set(copy.values()) | ||
Siddharth Agarwal
|
r18134 | copied.update(movewithdir.values()) | ||
Matt Mackall
|
r3295 | |||
Matt Mackall
|
r11470 | if '.hgsubstate' in m1: | ||
Matt Mackall
|
r9783 | # check whether sub state is modified | ||
Bryan O'Sullivan
|
r18611 | for s in sorted(wctx.substate): | ||
if wctx.sub(s).dirty(): | ||||
Martin von Zweigbergk
|
r23380 | m1['.hgsubstate'] += '+' | ||
Matt Mackall
|
r9783 | break | ||
Matt Mackall
|
r3105 | # Compare manifests | ||
Martin von Zweigbergk
|
r22964 | diff = m1.diff(m2) | ||
Siddharth Agarwal
|
r18822 | |||
Martin von Zweigbergk
|
r23637 | actions = {} | ||
Martin von Zweigbergk
|
r22966 | for f, ((n1, fl1), (n2, fl2)) in diff.iteritems(): | ||
Matt Mackall
|
r3248 | if partial and not partial(f): | ||
continue | ||||
Martin von Zweigbergk
|
r23473 | if n1 and n2: # file exists on both local and remote side | ||
Martin von Zweigbergk
|
r23396 | if f not in ma: | ||
Martin von Zweigbergk
|
r23397 | fa = copy.get(f, None) | ||
if fa is not None: | ||||
Martin von Zweigbergk
|
r23637 | actions[f] = ('m', (f, f, fa, False, pa.node()), | ||
"both renamed from " + fa) | ||||
Mads Kiilerich
|
r18338 | else: | ||
Martin von Zweigbergk
|
r23637 | actions[f] = ('m', (f, f, None, False, pa.node()), | ||
"both created") | ||||
Matt Mackall
|
r16094 | else: | ||
Martin von Zweigbergk
|
r23396 | a = ma[f] | ||
fla = ma.flags(f) | ||||
Martin von Zweigbergk
|
r23395 | nol = 'l' not in fl1 + fl2 + fla | ||
if n2 == a and fl2 == fla: | ||||
Martin von Zweigbergk
|
r23637 | actions[f] = ('k' , (), "remote unchanged") | ||
Martin von Zweigbergk
|
r23395 | elif n1 == a and fl1 == fla: # local unchanged - use remote | ||
if n1 == n2: # optimization: keep local content | ||||
Martin von Zweigbergk
|
r23637 | actions[f] = ('e', (fl2,), "update permissions") | ||
Martin von Zweigbergk
|
r23395 | else: | ||
Martin von Zweigbergk
|
r23637 | actions[f] = ('g', (fl2,), "remote is newer") | ||
Martin von Zweigbergk
|
r23395 | elif nol and n2 == a: # remote only changed 'x' | ||
Martin von Zweigbergk
|
r23637 | actions[f] = ('e', (fl2,), "update permissions") | ||
Martin von Zweigbergk
|
r23395 | elif nol and n1 == a: # local only changed 'x' | ||
Martin von Zweigbergk
|
r23637 | actions[f] = ('g', (fl1,), "remote is newer") | ||
Martin von Zweigbergk
|
r23395 | else: # both changed something | ||
Martin von Zweigbergk
|
r23637 | actions[f] = ('m', (f, f, f, False, pa.node()), | ||
"versions differ") | ||||
Martin von Zweigbergk
|
r23473 | elif n1: # file exists only on local side | ||
Martin von Zweigbergk
|
r23474 | if f in copied: | ||
pass # we'll deal with it on m2 side | ||||
elif f in movewithdir: # directory rename, move local | ||||
Martin von Zweigbergk
|
r23473 | f2 = movewithdir[f] | ||
Martin von Zweigbergk
|
r23475 | if f2 in m2: | ||
Martin von Zweigbergk
|
r23637 | actions[f2] = ('m', (f, f2, None, True, pa.node()), | ||
"remote directory rename, both created") | ||||
Martin von Zweigbergk
|
r23475 | else: | ||
Martin von Zweigbergk
|
r23637 | actions[f2] = ('dm', (f, fl1), | ||
"remote directory rename - move from " + f) | ||||
Martin von Zweigbergk
|
r23473 | elif f in copy: | ||
f2 = copy[f] | ||||
Martin von Zweigbergk
|
r23637 | actions[f] = ('m', (f, f2, f2, False, pa.node()), | ||
"local copied/moved from " + f2) | ||||
Martin von Zweigbergk
|
r23473 | elif f in ma: # clean, a different, no remote | ||
if n1 != ma[f]: | ||||
if acceptremote: | ||||
Martin von Zweigbergk
|
r23637 | actions[f] = ('r', None, "remote delete") | ||
Martin von Zweigbergk
|
r23473 | else: | ||
Martin von Zweigbergk
|
r23637 | actions[f] = ('cd', None, "prompt changed/deleted") | ||
Martin von Zweigbergk
|
r23473 | elif n1[20:] == 'a': | ||
# This extra 'a' is added by working copy manifest to mark | ||||
# the file as locally added. We should forget it instead of | ||||
# deleting it. | ||||
Martin von Zweigbergk
|
r23637 | actions[f] = ('f', None, "remote deleted") | ||
Mads Kiilerich
|
r20639 | else: | ||
Martin von Zweigbergk
|
r23637 | actions[f] = ('r', None, "other deleted") | ||
Martin von Zweigbergk
|
r23473 | elif n2: # file exists only on remote side | ||
Martin von Zweigbergk
|
r23474 | if f in copied: | ||
pass # we'll deal with it on m1 side | ||||
elif f in movewithdir: | ||||
Martin von Zweigbergk
|
r23473 | f2 = movewithdir[f] | ||
Martin von Zweigbergk
|
r23476 | if f2 in m1: | ||
Martin von Zweigbergk
|
r23637 | actions[f2] = ('m', (f2, f, None, False, pa.node()), | ||
"local directory rename, both created") | ||||
Martin von Zweigbergk
|
r23476 | else: | ||
Martin von Zweigbergk
|
r23637 | actions[f2] = ('dg', (f, fl2), | ||
"local directory rename - get from " + f) | ||||
Martin von Zweigbergk
|
r23473 | elif f in copy: | ||
f2 = copy[f] | ||||
if f2 in m2: | ||||
Martin von Zweigbergk
|
r23637 | actions[f] = ('m', (f2, f, f2, False, pa.node()), | ||
"remote copied from " + f2) | ||||
Martin von Zweigbergk
|
r23473 | else: | ||
Martin von Zweigbergk
|
r23637 | actions[f] = ('m', (f2, f, f2, True, pa.node()), | ||
"remote moved from " + f2) | ||||
Martin von Zweigbergk
|
r23473 | elif f not in ma: | ||
# local unknown, remote created: the logic is described by the | ||||
# following table: | ||||
# | ||||
# force branchmerge different | action | ||||
Martin von Zweigbergk
|
r23651 | # n * * | create | ||
Martin von Zweigbergk
|
r23650 | # y n * | create | ||
# y y n | create | ||||
Martin von Zweigbergk
|
r23473 | # y y y | merge | ||
# | ||||
# Checking whether the files are different is expensive, so we | ||||
# don't do that when we can avoid it. | ||||
Martin von Zweigbergk
|
r23649 | if not force: | ||
Martin von Zweigbergk
|
r23651 | actions[f] = ('c', (fl2,), "remote created") | ||
Martin von Zweigbergk
|
r23649 | elif not branchmerge: | ||
Martin von Zweigbergk
|
r23650 | actions[f] = ('c', (fl2,), "remote created") | ||
Martin von Zweigbergk
|
r23473 | else: | ||
Martin von Zweigbergk
|
r23654 | actions[f] = ('cm', (fl2, pa.node()), | ||
"remote created, get or merge") | ||||
Martin von Zweigbergk
|
r23473 | elif n2 != ma[f]: | ||
Martin von Zweigbergk
|
r23651 | if acceptremote: | ||
actions[f] = ('c', (fl2,), "remote recreating") | ||||
Siddharth Agarwal
|
r18606 | else: | ||
Martin von Zweigbergk
|
r23651 | actions[f] = ('dc', (fl2,), "prompt deleted/changed") | ||
Martin von Zweigbergk
|
r23526 | return actions, diverge, renamedelete | ||
Matt Mackall
|
r3105 | |||
Martin von Zweigbergk
|
r23531 | def _resolvetrivial(repo, wctx, mctx, ancestor, actions): | ||
"""Resolves false conflicts where the nodeid changed but the content | ||||
remained the same.""" | ||||
Martin von Zweigbergk
|
r23639 | for f, (m, args, msg) in actions.items(): | ||
if m == 'cd' and f in ancestor and not wctx[f].cmp(ancestor[f]): | ||||
Martin von Zweigbergk
|
r23531 | # local did change but ended up with same content | ||
Martin von Zweigbergk
|
r23639 | actions[f] = 'r', None, "prompt same" | ||
elif m == 'dc' and f in ancestor and not mctx[f].cmp(ancestor[f]): | ||||
Martin von Zweigbergk
|
r23531 | # remote did change but ended up with same content | ||
Martin von Zweigbergk
|
r23639 | del actions[f] # don't get = keep local deleted | ||
Martin von Zweigbergk
|
r23531 | |||
Martin von Zweigbergk
|
r23385 | def calculateupdates(repo, wctx, mctx, ancestors, branchmerge, force, partial, | ||
acceptremote, followcopies): | ||||
"Calculate the actions needed to merge mctx into wctx using ancestors" | ||||
if len(ancestors) == 1: # default | ||||
Martin von Zweigbergk
|
r23526 | actions, diverge, renamedelete = manifestmerge( | ||
repo, wctx, mctx, ancestors[0], branchmerge, force, partial, | ||||
acceptremote, followcopies) | ||||
Martin von Zweigbergk
|
r23656 | _checkunknownfiles(repo, wctx, mctx, force, actions) | ||
Martin von Zweigbergk
|
r23385 | |||
else: # only when merge.preferancestor=* - the default | ||||
repo.ui.note( | ||||
_("note: merging %s and %s using bids from ancestors %s\n") % | ||||
(wctx, mctx, _(' and ').join(str(anc) for anc in ancestors))) | ||||
# Call for bids | ||||
fbids = {} # mapping filename to bids (action method to list af actions) | ||||
Martin von Zweigbergk
|
r23526 | diverge, renamedelete = None, None | ||
Martin von Zweigbergk
|
r23385 | for ancestor in ancestors: | ||
repo.ui.note(_('\ncalculating bids for ancestor %s\n') % ancestor) | ||||
Martin von Zweigbergk
|
r23526 | actions, diverge1, renamedelete1 = manifestmerge( | ||
repo, wctx, mctx, ancestor, branchmerge, force, partial, | ||||
acceptremote, followcopies) | ||||
Martin von Zweigbergk
|
r23656 | _checkunknownfiles(repo, wctx, mctx, force, actions) | ||
Martin von Zweigbergk
|
r23526 | if diverge is None: # and renamedelete is None. | ||
# Arbitrarily pick warnings from first iteration | ||||
diverge = diverge1 | ||||
renamedelete = renamedelete1 | ||||
Martin von Zweigbergk
|
r23638 | for f, a in sorted(actions.iteritems()): | ||
m, args, msg = a | ||||
repo.ui.debug(' %s: %s -> %s\n' % (f, msg, m)) | ||||
if f in fbids: | ||||
d = fbids[f] | ||||
if m in d: | ||||
d[m].append(a) | ||||
Martin von Zweigbergk
|
r23385 | else: | ||
Martin von Zweigbergk
|
r23638 | d[m] = [a] | ||
else: | ||||
fbids[f] = {m: [a]} | ||||
Martin von Zweigbergk
|
r23385 | |||
# Pick the best bid for each file | ||||
repo.ui.note(_('\nauction for merging merge bids\n')) | ||||
Martin von Zweigbergk
|
r23638 | actions = {} | ||
Martin von Zweigbergk
|
r23385 | for f, bids in sorted(fbids.items()): | ||
# bids is a mapping from action method to list af actions | ||||
# Consensus? | ||||
if len(bids) == 1: # all bids are the same kind of method | ||||
m, l = bids.items()[0] | ||||
if util.all(a == l[0] for a in l[1:]): # len(bids) is > 1 | ||||
repo.ui.note(" %s: consensus for %s\n" % (f, m)) | ||||
Martin von Zweigbergk
|
r23638 | actions[f] = l[0] | ||
Martin von Zweigbergk
|
r23385 | continue | ||
# If keep is an option, just do it. | ||||
if 'k' in bids: | ||||
repo.ui.note(" %s: picking 'keep' action\n" % f) | ||||
Martin von Zweigbergk
|
r23638 | actions[f] = bids['k'][0] | ||
Martin von Zweigbergk
|
r23385 | continue | ||
# If there are gets and they all agree [how could they not?], do it. | ||||
if 'g' in bids: | ||||
ga0 = bids['g'][0] | ||||
if util.all(a == ga0 for a in bids['g'][1:]): | ||||
repo.ui.note(" %s: picking 'get' action\n" % f) | ||||
Martin von Zweigbergk
|
r23638 | actions[f] = ga0 | ||
Martin von Zweigbergk
|
r23385 | continue | ||
# TODO: Consider other simple actions such as mode changes | ||||
# Handle inefficient democrazy. | ||||
repo.ui.note(_(' %s: multiple bids for merge action:\n') % f) | ||||
for m, l in sorted(bids.items()): | ||||
for _f, args, msg in l: | ||||
repo.ui.note(' %s -> %s\n' % (msg, m)) | ||||
# Pick random action. TODO: Instead, prompt user when resolving | ||||
m, l = bids.items()[0] | ||||
repo.ui.warn(_(' %s: ambiguous merge - picked %s action\n') % | ||||
(f, m)) | ||||
Martin von Zweigbergk
|
r23638 | actions[f] = l[0] | ||
Martin von Zweigbergk
|
r23385 | continue | ||
repo.ui.note(_('end of auction\n\n')) | ||||
Martin von Zweigbergk
|
r23639 | _resolvetrivial(repo, wctx, mctx, ancestors[0], actions) | ||
Martin von Zweigbergk
|
r23640 | if wctx.rev() is None: | ||
fractions = _forgetremoved(wctx, mctx, branchmerge) | ||||
actions.update(fractions) | ||||
Martin von Zweigbergk
|
r23526 | return actions, diverge, renamedelete | ||
Martin von Zweigbergk
|
r23385 | |||
Mads Kiilerich
|
r21392 | def batchremove(repo, actions): | ||
"""apply removes to the working directory | ||||
Bryan O'Sullivan
|
r18630 | |||
yields tuples for progress updates | ||||
""" | ||||
Bryan O'Sullivan
|
r18640 | verbose = repo.ui.verbose | ||
unlink = util.unlinkpath | ||||
wjoin = repo.wjoin | ||||
Angel Ezquerra
|
r23879 | audit = repo.wvfs.audit | ||
Bryan O'Sullivan
|
r18633 | i = 0 | ||
Mads Kiilerich
|
r21545 | for f, args, msg in actions: | ||
Mads Kiilerich
|
r21392 | repo.ui.debug(" %s: %s -> r\n" % (f, msg)) | ||
Mads Kiilerich
|
r21551 | if verbose: | ||
repo.ui.note(_("removing %s\n") % f) | ||||
audit(f) | ||||
try: | ||||
unlink(wjoin(f), ignoremissing=True) | ||||
except OSError, inst: | ||||
repo.ui.warn(_("update failed to remove %s: %s!\n") % | ||||
(f, inst.strerror)) | ||||
Mads Kiilerich
|
r21392 | if i == 100: | ||
yield i, f | ||||
i = 0 | ||||
i += 1 | ||||
if i > 0: | ||||
yield i, f | ||||
def batchget(repo, mctx, actions): | ||||
"""apply gets to the working directory | ||||
mctx is the context to get from | ||||
yields tuples for progress updates | ||||
""" | ||||
verbose = repo.ui.verbose | ||||
fctx = mctx.filectx | ||||
wwrite = repo.wwrite | ||||
i = 0 | ||||
Mads Kiilerich
|
r21545 | for f, args, msg in actions: | ||
Mads Kiilerich
|
r21392 | repo.ui.debug(" %s: %s -> g\n" % (f, msg)) | ||
Mads Kiilerich
|
r21551 | if verbose: | ||
repo.ui.note(_("getting %s\n") % f) | ||||
wwrite(f, fctx(f).data(), args[0]) | ||||
Bryan O'Sullivan
|
r18633 | if i == 100: | ||
yield i, f | ||||
i = 0 | ||||
i += 1 | ||||
if i > 0: | ||||
Bryan O'Sullivan
|
r18630 | yield i, f | ||
Durham Goode
|
r21524 | def applyupdates(repo, actions, wctx, mctx, overwrite, labels=None): | ||
Peter Arrenbrecht
|
r11454 | """apply the merge action list to the working directory | ||
wctx is the working copy context | ||||
mctx is the context to be merged into the working copy | ||||
Greg Ward
|
r13162 | |||
Return a tuple of counts (updated, merged, removed, unresolved) that | ||||
describes how many files were affected by the update. | ||||
Peter Arrenbrecht
|
r11454 | """ | ||
Matt Mackall
|
r3315 | |||
Matt Mackall
|
r3111 | updated, merged, removed, unresolved = 0, 0, 0, 0 | ||
Matt Mackall
|
r6512 | ms = mergestate(repo) | ||
Pierre-Yves David
|
r20591 | ms.reset(wctx.p1().node(), mctx.node()) | ||
Matt Mackall
|
r6512 | moves = [] | ||
Mads Kiilerich
|
r21545 | for m, l in actions.items(): | ||
l.sort() | ||||
Matt Mackall
|
r6512 | |||
# prescan for merges | ||||
Mads Kiilerich
|
r21545 | for f, args, msg in actions['m']: | ||
Mads Kiilerich
|
r21551 | f1, f2, fa, move, anc = args | ||
if f == '.hgsubstate': # merged internally | ||||
continue | ||||
repo.ui.debug(" preserving %s for resolve of %s\n" % (f1, f)) | ||||
fcl = wctx[f1] | ||||
fco = mctx[f2] | ||||
actx = repo[anc] | ||||
if fa in actx: | ||||
fca = actx[fa] | ||||
else: | ||||
fca = repo.filectx(f1, fileid=nullrev) | ||||
ms.add(fcl, fco, fca, f) | ||||
if f1 != f and move: | ||||
moves.append(f1) | ||||
Matt Mackall
|
r6512 | |||
Angel Ezquerra
|
r23879 | audit = repo.wvfs.audit | ||
Mads Kiilerich
|
r21390 | _updating = _('updating') | ||
_files = _('files') | ||||
progress = repo.ui.progress | ||||
Adrian Buehlmann
|
r14398 | |||
Matt Mackall
|
r6512 | # remove renamed files after safely stored | ||
for f in moves: | ||||
Martin Geisler
|
r12032 | if os.path.lexists(repo.wjoin(f)): | ||
Martin Geisler
|
r9467 | repo.ui.debug("removing %s\n" % f) | ||
Adrian Buehlmann
|
r14398 | audit(f) | ||
Mads Kiilerich
|
r18333 | util.unlinkpath(repo.wjoin(f)) | ||
Matt Mackall
|
r5042 | |||
Martin von Zweigbergk
|
r23526 | numupdates = sum(len(l) for m, l in actions.items() if m != 'k') | ||
Bryan O'Sullivan
|
r18630 | |||
FUJIWARA Katsunori
|
r23971 | def dirtysubstate(): | ||
# mark '.hgsubstate' as possibly dirty forcibly, because | ||||
# modified '.hgsubstate' is misunderstood as clean, | ||||
# when both st_size/st_mtime of '.hgsubstate' aren't changed, | ||||
# even if "submerge" fails and '.hgsubstate' is inconsistent | ||||
repo.dirstate.normallookup('.hgsubstate') | ||||
Mads Kiilerich
|
r21545 | if [a for a in actions['r'] if a[0] == '.hgsubstate']: | ||
FUJIWARA Katsunori
|
r23971 | dirtysubstate() | ||
Bryan O'Sullivan
|
r18632 | subrepo.submerge(repo, wctx, mctx, wctx, overwrite) | ||
Mads Kiilerich
|
r21390 | # remove in parallel (must come first) | ||
Bryan O'Sullivan
|
r18633 | z = 0 | ||
Mads Kiilerich
|
r21545 | prog = worker.worker(repo.ui, 0.001, batchremove, (repo,), actions['r']) | ||
FUJIWARA Katsunori
|
r19095 | for i, item in prog: | ||
z += i | ||||
Mads Kiilerich
|
r21390 | progress(_updating, z, item=item, total=numupdates, unit=_files) | ||
Mads Kiilerich
|
r21545 | removed = len(actions['r']) | ||
Mads Kiilerich
|
r21390 | |||
# get in parallel | ||||
Mads Kiilerich
|
r21545 | prog = worker.worker(repo.ui, 0.001, batchget, (repo, mctx), actions['g']) | ||
Bryan O'Sullivan
|
r18639 | for i, item in prog: | ||
Bryan O'Sullivan
|
r18633 | z += i | ||
Mads Kiilerich
|
r21390 | progress(_updating, z, item=item, total=numupdates, unit=_files) | ||
Mads Kiilerich
|
r21545 | updated = len(actions['g']) | ||
Bryan O'Sullivan
|
r18630 | |||
Mads Kiilerich
|
r21545 | if [a for a in actions['g'] if a[0] == '.hgsubstate']: | ||
FUJIWARA Katsunori
|
r23971 | dirtysubstate() | ||
Bryan O'Sullivan
|
r18632 | subrepo.submerge(repo, wctx, mctx, wctx, overwrite) | ||
Mads Kiilerich
|
r21551 | # forget (manifest only, just log it) (must come first) | ||
for f, args, msg in actions['f']: | ||||
repo.ui.debug(" %s: %s -> f\n" % (f, msg)) | ||||
z += 1 | ||||
progress(_updating, z, item=f, total=numupdates, unit=_files) | ||||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # re-add (manifest only, just log it) | ||
for f, args, msg in actions['a']: | ||||
repo.ui.debug(" %s: %s -> a\n" % (f, msg)) | ||||
z += 1 | ||||
progress(_updating, z, item=f, total=numupdates, unit=_files) | ||||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # keep (noop, just log it) | ||
for f, args, msg in actions['k']: | ||||
repo.ui.debug(" %s: %s -> k\n" % (f, msg)) | ||||
# no progress | ||||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # merge | ||
for f, args, msg in actions['m']: | ||||
repo.ui.debug(" %s: %s -> m\n" % (f, msg)) | ||||
z += 1 | ||||
progress(_updating, z, item=f, total=numupdates, unit=_files) | ||||
if f == '.hgsubstate': # subrepo states need updating | ||||
FUJIWARA Katsunori
|
r23971 | dirtysubstate() | ||
Mads Kiilerich
|
r21551 | subrepo.submerge(repo, wctx, mctx, wctx.ancestor(mctx), | ||
overwrite) | ||||
continue | ||||
audit(f) | ||||
r = ms.resolve(f, wctx, labels=labels) | ||||
if r is not None and r > 0: | ||||
unresolved += 1 | ||||
else: | ||||
if r is None: | ||||
updated += 1 | ||||
Matt Mackall
|
r3309 | else: | ||
Mads Kiilerich
|
r21551 | merged += 1 | ||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # directory rename, move local | ||
for f, args, msg in actions['dm']: | ||||
repo.ui.debug(" %s: %s -> dm\n" % (f, msg)) | ||||
z += 1 | ||||
progress(_updating, z, item=f, total=numupdates, unit=_files) | ||||
f0, flags = args | ||||
repo.ui.note(_("moving %s to %s\n") % (f0, f)) | ||||
audit(f) | ||||
repo.wwrite(f, wctx.filectx(f0).data(), flags) | ||||
util.unlinkpath(repo.wjoin(f0)) | ||||
updated += 1 | ||||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # local directory rename, get | ||
for f, args, msg in actions['dg']: | ||||
repo.ui.debug(" %s: %s -> dg\n" % (f, msg)) | ||||
z += 1 | ||||
progress(_updating, z, item=f, total=numupdates, unit=_files) | ||||
f0, flags = args | ||||
repo.ui.note(_("getting %s to %s\n") % (f0, f)) | ||||
repo.wwrite(f, mctx.filectx(f0).data(), flags) | ||||
updated += 1 | ||||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # exec | ||
for f, args, msg in actions['e']: | ||||
repo.ui.debug(" %s: %s -> e\n" % (f, msg)) | ||||
z += 1 | ||||
progress(_updating, z, item=f, total=numupdates, unit=_files) | ||||
flags, = args | ||||
audit(f) | ||||
util.setflags(repo.wjoin(f), 'l' in flags, 'x' in flags) | ||||
updated += 1 | ||||
Mads Kiilerich
|
r21391 | |||
Peter Arrenbrecht
|
r12369 | ms.commit() | ||
Bryan O'Sullivan
|
r18640 | progress(_updating, None, total=numupdates, unit=_files) | ||
Matt Mackall
|
r3111 | |||
return updated, merged, removed, unresolved | ||||
Mads Kiilerich
|
r18330 | def recordupdates(repo, actions, branchmerge): | ||
Matt Mackall
|
r3315 | "record merge actions to the dirstate" | ||
Mads Kiilerich
|
r21551 | # remove (must come first) | ||
for f, args, msg in actions['r']: | ||||
if branchmerge: | ||||
repo.dirstate.remove(f) | ||||
else: | ||||
Mads Kiilerich
|
r21389 | repo.dirstate.drop(f) | ||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # forget (must come first) | ||
for f, args, msg in actions['f']: | ||||
repo.dirstate.drop(f) | ||||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # re-add | ||
for f, args, msg in actions['a']: | ||||
if not branchmerge: | ||||
repo.dirstate.add(f) | ||||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # exec change | ||
for f, args, msg in actions['e']: | ||||
repo.dirstate.normallookup(f) | ||||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # keep | ||
for f, args, msg in actions['k']: | ||||
pass | ||||
# get | ||||
for f, args, msg in actions['g']: | ||||
if branchmerge: | ||||
repo.dirstate.otherparent(f) | ||||
else: | ||||
repo.dirstate.normal(f) | ||||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # merge | ||
for f, args, msg in actions['m']: | ||||
f1, f2, fa, move, anc = args | ||||
if branchmerge: | ||||
# We've done a branch merge, mark this file as merged | ||||
# so that we properly record the merger later | ||||
repo.dirstate.merge(f) | ||||
if f1 != f2: # copy/rename | ||||
Matt Mackall
|
r3308 | if move: | ||
Mads Kiilerich
|
r21551 | repo.dirstate.remove(f1) | ||
if f1 != f: | ||||
repo.dirstate.copy(f1, f) | ||||
else: | ||||
repo.dirstate.copy(f2, f) | ||||
else: | ||||
# We've update-merged a locally modified file, so | ||||
# we set the dirstate to emulate a normal checkout | ||||
# of that file some time in the past. Thus our | ||||
# merge will appear as a normal local file | ||||
# modification. | ||||
if f2 == f: # file not locally copied/moved | ||||
repo.dirstate.normallookup(f) | ||||
if move: | ||||
repo.dirstate.drop(f1) | ||||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # directory rename, move local | ||
for f, args, msg in actions['dm']: | ||||
f0, flag = args | ||||
if branchmerge: | ||||
repo.dirstate.add(f) | ||||
repo.dirstate.remove(f0) | ||||
repo.dirstate.copy(f0, f) | ||||
else: | ||||
repo.dirstate.normal(f) | ||||
repo.dirstate.drop(f0) | ||||
Mads Kiilerich
|
r21391 | |||
Mads Kiilerich
|
r21551 | # directory rename, get | ||
for f, args, msg in actions['dg']: | ||||
f0, flag = args | ||||
if branchmerge: | ||||
repo.dirstate.add(f) | ||||
repo.dirstate.copy(f0, f) | ||||
else: | ||||
repo.dirstate.normal(f) | ||||
Matt Mackall
|
r3111 | |||
Patrick Mezard
|
r16696 | def update(repo, node, branchmerge, force, partial, ancestor=None, | ||
Durham Goode
|
r21524 | mergeancestor=False, labels=None): | ||
Matt Mackall
|
r3315 | """ | ||
Perform a merge between the working directory and the given node | ||||
Stuart W Marks
|
r9716 | node = the node to update to, or None if unspecified | ||
Matt Mackall
|
r3315 | branchmerge = whether to merge between branches | ||
force = whether to force branch merging or file overwriting | ||||
partial = a function to filter file lists (dirstate not updated) | ||||
Durham Goode
|
r18778 | mergeancestor = whether it is merging with an ancestor. If true, | ||
we should accept the incoming changes for any prompts that occur. | ||||
If false, merging with an ancestor (fast-forward) is only allowed | ||||
between different named branches. This flag is used by rebase extension | ||||
as a temporary fix and should be avoided in general. | ||||
Stuart W Marks
|
r9716 | |||
The table below shows all the behaviors of the update command | ||||
given the -c and -C or no options, whether the working directory | ||||
is dirty, whether a revision is specified, and the relationship of | ||||
the parent rev to the target rev (linear, on the same named | ||||
branch, or on another named branch). | ||||
Adrian Buehlmann
|
r12279 | This logic is tested by test-update-branches.t. | ||
Stuart W Marks
|
r9716 | |||
-c -C dirty rev | linear same cross | ||||
n n n n | ok (1) x | ||||
Stuart W Marks
|
r9717 | n n n y | ok ok ok | ||
Siddharth Agarwal
|
r19799 | n n y n | merge (2) (2) | ||
n n y y | merge (3) (3) | ||||
Stuart W Marks
|
r9716 | n y * * | --- discard --- | ||
Siddharth Agarwal
|
r19799 | y n y * | --- (4) --- | ||
Stuart W Marks
|
r9716 | y n n * | --- ok --- | ||
Siddharth Agarwal
|
r19799 | y y * * | --- (5) --- | ||
Stuart W Marks
|
r9716 | |||
x = can't happen | ||||
* = don't-care | ||||
Siddharth Agarwal
|
r19798 | 1 = abort: not a linear update (merge or update --check to force update) | ||
Siddharth Agarwal
|
r19800 | 2 = abort: uncommitted changes (commit and merge, or update --clean to | ||
discard changes) | ||||
Siddharth Agarwal
|
r19799 | 3 = abort: uncommitted changes (commit or update --clean to discard changes) | ||
Siddharth Agarwal
|
r19801 | 4 = abort: uncommitted changes (checked in commands.py) | ||
Siddharth Agarwal
|
r19799 | 5 = incompatible options (checked in commands.py) | ||
Greg Ward
|
r13162 | |||
Return the same tuple as applyupdates(). | ||||
Matt Mackall
|
r3315 | """ | ||
Matt Mackall
|
r2815 | |||
Stuart W Marks
|
r9717 | onode = node | ||
Matt Mackall
|
r4917 | wlock = repo.wlock() | ||
Matt Mackall
|
r4915 | try: | ||
Matt Mackall
|
r6747 | wc = repo[None] | ||
Sean Farley
|
r20279 | pl = wc.parents() | ||
p1 = pl[0] | ||||
Mads Kiilerich
|
r21081 | pas = [None] | ||
Mads Kiilerich
|
r23405 | if ancestor is not None: | ||
Mads Kiilerich
|
r21081 | pas = [repo[ancestor]] | ||
Sean Farley
|
r20279 | |||
Matt Mackall
|
r4915 | if node is None: | ||
Sean Farley
|
r20278 | # Here is where we should consider bookmarks, divergent bookmarks, | ||
# foreground changesets (successors), and tip of current branch; | ||||
# but currently we are only checking the branch tips. | ||||
Matt Mackall
|
r4915 | try: | ||
Brodie Rao
|
r16719 | node = repo.branchtip(wc.branch()) | ||
Mads Kiilerich
|
r22200 | except errormod.RepoLookupError: | ||
Martin von Zweigbergk
|
r23380 | if wc.branch() == 'default': # no default branch! | ||
node = repo.lookup('tip') # update to tip | ||||
Matt Mackall
|
r5570 | else: | ||
raise util.Abort(_("branch %s not found") % wc.branch()) | ||||
Sean Farley
|
r20280 | |||
if p1.obsolete() and not p1.children(): | ||||
# allow updating to successors | ||||
successors = obsolete.successorssets(repo, p1.node()) | ||||
# behavior of certain cases is as follows, | ||||
# | ||||
# divergent changesets: update to highest rev, similar to what | ||||
# is currently done when there are more than one head | ||||
# (i.e. 'tip') | ||||
# | ||||
# replaced changesets: same as divergent except we know there | ||||
# is no conflict | ||||
# | ||||
# pruned changeset: no update is done; though, we could | ||||
# consider updating to the first non-obsolete parent, | ||||
# similar to what is current done for 'hg prune' | ||||
if successors: | ||||
# flatten the list here handles both divergent (len > 1) | ||||
# and the usual case (len = 1) | ||||
successors = [n for sub in successors for n in sub] | ||||
# get the max revision for the given successors set, | ||||
# i.e. the 'tip' of a set | ||||
Martin von Zweigbergk
|
r23380 | node = repo.revs('max(%ln)', successors).first() | ||
Mads Kiilerich
|
r21081 | pas = [p1] | ||
Sean Farley
|
r20280 | |||
Matt Mackall
|
r4915 | overwrite = force and not branchmerge | ||
Sean Farley
|
r20279 | |||
p2 = repo[node] | ||||
Mads Kiilerich
|
r21081 | if pas[0] is None: | ||
Martin von Zweigbergk
|
r23380 | if repo.ui.config('merge', 'preferancestor', '*') == '*': | ||
Mads Kiilerich
|
r21128 | cahs = repo.changelog.commonancestorsheads(p1.node(), p2.node()) | ||
pas = [repo[anc] for anc in (sorted(cahs) or [nullid])] | ||||
else: | ||||
Mads Kiilerich
|
r22179 | pas = [p1.ancestor(p2, warn=branchmerge)] | ||
Matt Mackall
|
r13874 | |||
Matt Mackall
|
r4915 | fp1, fp2, xp1, xp2 = p1.node(), p2.node(), str(p1), str(p2) | ||
Matt Mackall
|
r3314 | |||
Matt Mackall
|
r4915 | ### check phase | ||
if not overwrite and len(pl) > 1: | ||||
Martin von Zweigbergk
|
r22841 | raise util.Abort(_("outstanding uncommitted merge")) | ||
Matt Mackall
|
r6375 | if branchmerge: | ||
Mads Kiilerich
|
r21081 | if pas == [p2]: | ||
Matt Mackall
|
r11417 | raise util.Abort(_("merging with a working directory ancestor" | ||
" has no effect")) | ||||
Mads Kiilerich
|
r21081 | elif pas == [p1]: | ||
Patrick Mezard
|
r16696 | if not mergeancestor and p1.branch() == p2.branch(): | ||
Kevin Bullock
|
r15619 | raise util.Abort(_("nothing to merge"), | ||
hint=_("use 'hg update' " | ||||
"or check 'hg heads'")) | ||||
Matt Mackall
|
r6375 | if not force and (wc.files() or wc.deleted()): | ||
Siddharth Agarwal
|
r19802 | raise util.Abort(_("uncommitted changes"), | ||
Kevin Bullock
|
r15619 | hint=_("use 'hg status' to list changes")) | ||
Mads Kiilerich
|
r18364 | for s in sorted(wc.substate): | ||
FUJIWARA Katsunori
|
r24471 | wc.sub(s).bailifchanged() | ||
Oleg Stepanov
|
r13437 | |||
Matt Mackall
|
r6375 | elif not overwrite: | ||
Siddharth Agarwal
|
r19929 | if p1 == p2: # no-op update | ||
# call the hooks and exit early | ||||
repo.hook('preupdate', throw=True, parent1=xp2, parent2='') | ||||
repo.hook('update', parent1=xp2, parent2='', error=0) | ||||
return 0, 0, 0, 0 | ||||
Mads Kiilerich
|
r21081 | if pas not in ([p1], [p2]): # nonlinear | ||
Pierre-Yves David
|
r18985 | dirty = wc.dirty(missing=True) | ||
if dirty or onode is None: | ||||
# Branching is a bit strange to ensure we do the minimal | ||||
# amount of call to obsolete.background. | ||||
foreground = obsolete.foreground(repo, [p1.node()]) | ||||
# note: the <node> variable contains a random identifier | ||||
if repo[node].node() in foreground: | ||||
Mads Kiilerich
|
r21081 | pas = [p1] # allow updating to successors | ||
Siddharth Agarwal
|
r19799 | elif dirty: | ||
msg = _("uncommitted changes") | ||||
Siddharth Agarwal
|
r19800 | if onode is None: | ||
hint = _("commit and merge, or update --clean to" | ||||
" discard changes") | ||||
else: | ||||
hint = _("commit or update --clean to discard" | ||||
" changes") | ||||
Siddharth Agarwal
|
r19799 | raise util.Abort(msg, hint=hint) | ||
Pierre-Yves David
|
r18985 | else: # node is none | ||
Siddharth Agarwal
|
r19798 | msg = _("not a linear update") | ||
hint = _("merge or update --check to force update") | ||||
raise util.Abort(msg, hint=hint) | ||||
Pierre-Yves David
|
r18985 | else: | ||
# Allow jumping branches if clean and specific rev given | ||||
Mads Kiilerich
|
r21081 | pas = [p1] | ||
Matt Mackall
|
r2814 | |||
Mads Kiilerich
|
r21080 | followcopies = False | ||
if overwrite: | ||||
Mads Kiilerich
|
r21081 | pas = [wc] | ||
elif pas == [p2]: # backwards | ||||
pas = [wc.p1()] | ||||
Mads Kiilerich
|
r21080 | elif not branchmerge and not wc.dirty(missing=True): | ||
pass | ||||
Martin von Zweigbergk
|
r23380 | elif pas[0] and repo.ui.configbool('merge', 'followcopies', True): | ||
Mads Kiilerich
|
r21080 | followcopies = True | ||
Matt Mackall
|
r4915 | ### calculate phase | ||
Martin von Zweigbergk
|
r23641 | actionbyfile, diverge, renamedelete = calculateupdates( | ||
Martin von Zweigbergk
|
r23526 | repo, wc, p2, pas, branchmerge, force, partial, mergeancestor, | ||
followcopies) | ||||
Martin von Zweigbergk
|
r23641 | # Convert to dictionary-of-lists format | ||
actions = dict((m, []) for m in 'a f g cd dc r dm dg m e k'.split()) | ||||
for f, (m, args, msg) in actionbyfile.iteritems(): | ||||
if m not in actions: | ||||
actions[m] = [] | ||||
actions[m].append((f, args, msg)) | ||||
Matt Mackall
|
r2775 | |||
Martin von Zweigbergk
|
r23544 | if not util.checkcase(repo.path): | ||
# check collision between files only in p2 for clean update | ||||
if (not branchmerge and | ||||
(force or not wc.dirty(missing=True, branch=False))): | ||||
_checkcollision(repo, p2.manifest(), None) | ||||
else: | ||||
_checkcollision(repo, wc.manifest(), actions) | ||||
Martin von Zweigbergk
|
r23541 | # Prompt and create actions. TODO: Move this towards resolve phase. | ||
for f, args, msg in sorted(actions['cd']): | ||||
if repo.ui.promptchoice( | ||||
_("local changed %s which remote deleted\n" | ||||
"use (c)hanged version or (d)elete?" | ||||
"$$ &Changed $$ &Delete") % f, 0): | ||||
actions['r'].append((f, None, "prompt delete")) | ||||
else: | ||||
actions['a'].append((f, None, "prompt keep")) | ||||
del actions['cd'][:] | ||||
for f, args, msg in sorted(actions['dc']): | ||||
flags, = args | ||||
if repo.ui.promptchoice( | ||||
_("remote changed %s which local deleted\n" | ||||
"use (c)hanged version or leave (d)eleted?" | ||||
"$$ &Changed $$ &Deleted") % f, 0) == 0: | ||||
actions['g'].append((f, (flags,), "prompt recreating")) | ||||
del actions['dc'][:] | ||||
Matt Mackall
|
r4915 | ### apply phase | ||
Matt Mackall
|
r13550 | if not branchmerge: # just jump to the new rev | ||
Matt Mackall
|
r4915 | fp1, fp2, xp1, xp2 = fp2, nullid, xp2, '' | ||
if not partial: | ||||
repo.hook('preupdate', throw=True, parent1=xp1, parent2=xp2) | ||||
Matt Mackall
|
r19482 | # note that we're in the middle of an update | ||
repo.vfs.write('updatestate', p2.hex()) | ||||
Matt Mackall
|
r2775 | |||
Durham Goode
|
r21524 | stats = applyupdates(repo, actions, wc, p2, overwrite, labels=labels) | ||
Matt Mackall
|
r2899 | |||
Martin von Zweigbergk
|
r23525 | # divergent renames | ||
Martin von Zweigbergk
|
r23526 | for f, fl in sorted(diverge.iteritems()): | ||
Martin von Zweigbergk
|
r23525 | repo.ui.warn(_("note: possible conflict - %s was renamed " | ||
"multiple times to:\n") % f) | ||||
for nf in fl: | ||||
repo.ui.warn(" %s\n" % nf) | ||||
# rename and delete | ||||
Martin von Zweigbergk
|
r23526 | for f, fl in sorted(renamedelete.iteritems()): | ||
Martin von Zweigbergk
|
r23525 | repo.ui.warn(_("note: possible conflict - %s was deleted " | ||
"and renamed to:\n") % f) | ||||
for nf in fl: | ||||
repo.ui.warn(" %s\n" % nf) | ||||
Matt Mackall
|
r4915 | if not partial: | ||
Durham Goode
|
r22405 | repo.dirstate.beginparentchange() | ||
Patrick Mezard
|
r16551 | repo.setparents(fp1, fp2) | ||
Mads Kiilerich
|
r18330 | recordupdates(repo, actions, branchmerge) | ||
Matt Mackall
|
r19482 | # update completed, clear state | ||
util.unlink(repo.join('updatestate')) | ||||
Mads Kiilerich
|
r13561 | if not branchmerge: | ||
Matt Mackall
|
r4915 | repo.dirstate.setbranch(p2.branch()) | ||
Durham Goode
|
r22405 | repo.dirstate.endparentchange() | ||
Matt Mackall
|
r4915 | finally: | ||
Ronny Pfannschmidt
|
r8109 | wlock.release() | ||
Sune Foldager
|
r10492 | |||
if not partial: | ||||
Matt Harbison
|
r24881 | def updatehook(parent1=xp1, parent2=xp2, error=stats[3]): | ||
repo.hook('update', parent1=parent1, parent2=parent2, error=error) | ||||
repo._afterlock(updatehook) | ||||
Sune Foldager
|
r10492 | return stats | ||
Matt Mackall
|
r22902 | |||
def graft(repo, ctx, pctx, labels): | ||||
"""Do a graft-like merge. | ||||
This is a merge where the merge ancestor is chosen such that one | ||||
or more changesets are grafted onto the current changeset. In | ||||
addition to the merge, this fixes up the dirstate to include only | ||||
a single parent and tries to duplicate any renames/copies | ||||
appropriately. | ||||
ctx - changeset to rebase | ||||
pctx - merge base, usually ctx.p1() | ||||
labels - merge labels eg ['local', 'graft'] | ||||
""" | ||||
Durham Goode
|
r24643 | # If we're grafting a descendant onto an ancestor, be sure to pass | ||
# mergeancestor=True to update. This does two things: 1) allows the merge if | ||||
# the destination is the same as the parent of the ctx (so we can use graft | ||||
# to copy commits), and 2) informs update that the incoming changes are | ||||
# newer than the destination so it doesn't prompt about "remote changed foo | ||||
# which local deleted". | ||||
mergeancestor = repo.changelog.isancestor(repo['.'].node(), ctx.node()) | ||||
Matt Mackall
|
r22902 | |||
stats = update(repo, ctx.node(), True, True, False, pctx.node(), | ||||
Durham Goode
|
r24643 | mergeancestor=mergeancestor, labels=labels) | ||
Matt Mackall
|
r22902 | # drop the second merge parent | ||
repo.dirstate.beginparentchange() | ||||
repo.setparents(repo['.'].node(), nullid) | ||||
repo.dirstate.write() | ||||
# fix up dirstate for copies and renames | ||||
copies.duplicatecopies(repo, ctx.rev(), pctx.rev()) | ||||
repo.dirstate.endparentchange() | ||||
return stats | ||||