Show More
treediscovery.py
187 lines
| 5.9 KiB
| text/x-python
|
PythonLexer
/ mercurial / treediscovery.py
Peter Arrenbrecht
|
r14164 | # discovery.py - protocol changeset discovery functions | ||
# | ||||
Raphaël Gomès
|
r47575 | # Copyright 2010 Olivia Mackall <olivia@selenic.com> | ||
Peter Arrenbrecht
|
r14164 | # | ||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
Gregory Szorc
|
r25987 | from __future__ import absolute_import | ||
Martin von Zweigbergk
|
r25113 | import collections | ||
Gregory Szorc
|
r25987 | |||
from .i18n import _ | ||||
Joerg Sonnenberger
|
r47771 | from .node import short | ||
Gregory Szorc
|
r25987 | from . import ( | ||
error, | ||||
Gregory Szorc
|
r38806 | pycompat, | ||
Gregory Szorc
|
r25987 | ) | ||
Peter Arrenbrecht
|
r14164 | |||
Augie Fackler
|
r43346 | |||
r46726 | def findcommonincoming(repo, remote, heads=None, force=False, audit=None): | |||
Peter Arrenbrecht
|
r14164 | """Return a tuple (common, fetch, heads) used to identify the common | ||
subset of nodes between repo and remote. | ||||
"common" is a list of (at least) the heads of the common subset. | ||||
"fetch" is a list of roots of the nodes that would be incoming, to be | ||||
supplied to changegroupsubset. | ||||
"heads" is either the supplied heads, or else the remote's heads. | ||||
""" | ||||
Pierre-Yves David
|
r20225 | knownnode = repo.changelog.hasnode | ||
Peter Arrenbrecht
|
r14164 | search = [] | ||
fetch = set() | ||||
seen = set() | ||||
seenbranch = set() | ||||
base = set() | ||||
if not heads: | ||||
Gregory Szorc
|
r37652 | with remote.commandexecutor() as e: | ||
Augie Fackler
|
r43347 | heads = e.callcommand(b'heads', {}).result() | ||
Peter Arrenbrecht
|
r14164 | |||
r46726 | if audit is not None: | |||
audit[b'total-roundtrips'] = 1 | ||||
Joerg Sonnenberger
|
r47771 | if repo.changelog.tip() == repo.nullid: | ||
base.add(repo.nullid) | ||||
if heads != [repo.nullid]: | ||||
return [repo.nullid], [repo.nullid], list(heads) | ||||
return [repo.nullid], [], heads | ||||
Peter Arrenbrecht
|
r14164 | |||
# assume we're closer to the tip than the root | ||||
# and start by examining the heads | ||||
Augie Fackler
|
r43347 | repo.ui.status(_(b"searching for changes\n")) | ||
Peter Arrenbrecht
|
r14164 | |||
unknown = [] | ||||
for h in heads: | ||||
Pierre-Yves David
|
r20225 | if not knownnode(h): | ||
Peter Arrenbrecht
|
r14164 | unknown.append(h) | ||
else: | ||||
base.add(h) | ||||
Peter Arrenbrecht
|
r14199 | if not unknown: | ||
return list(base), [], list(heads) | ||||
Peter Arrenbrecht
|
r14164 | req = set(unknown) | ||
reqcnt = 0 | ||||
Augie Fackler
|
r43347 | progress = repo.ui.makeprogress(_(b'searching'), unit=_(b'queries')) | ||
Peter Arrenbrecht
|
r14164 | |||
# search through remote branches | ||||
# a 'branch' here is a linear segment of history, with four parts: | ||||
# head, root, first parent, second parent | ||||
# (a branch always has two parents (or none) by definition) | ||||
Gregory Szorc
|
r37652 | with remote.commandexecutor() as e: | ||
Augie Fackler
|
r43347 | branches = e.callcommand(b'branches', {b'nodes': unknown}).result() | ||
Gregory Szorc
|
r37652 | |||
unknown = collections.deque(branches) | ||||
Peter Arrenbrecht
|
r14164 | while unknown: | ||
r = [] | ||||
while unknown: | ||||
Bryan O'Sullivan
|
r16803 | n = unknown.popleft() | ||
Peter Arrenbrecht
|
r14164 | if n[0] in seen: | ||
continue | ||||
Augie Fackler
|
r43347 | repo.ui.debug(b"examining %s:%s\n" % (short(n[0]), short(n[1]))) | ||
Joerg Sonnenberger
|
r47771 | if n[0] == repo.nullid: # found the end of the branch | ||
Peter Arrenbrecht
|
r14164 | pass | ||
elif n in seenbranch: | ||||
Augie Fackler
|
r43347 | repo.ui.debug(b"branch already found\n") | ||
Peter Arrenbrecht
|
r14164 | continue | ||
Augie Fackler
|
r43346 | elif n[1] and knownnode(n[1]): # do we know the base? | ||
repo.ui.debug( | ||||
Augie Fackler
|
r43347 | b"found incomplete branch %s:%s\n" | ||
Augie Fackler
|
r43346 | % (short(n[0]), short(n[1])) | ||
) | ||||
search.append(n[0:2]) # schedule branch range for scanning | ||||
Peter Arrenbrecht
|
r14164 | seenbranch.add(n) | ||
else: | ||||
if n[1] not in seen and n[1] not in fetch: | ||||
Pierre-Yves David
|
r20225 | if knownnode(n[2]) and knownnode(n[3]): | ||
Augie Fackler
|
r43347 | repo.ui.debug(b"found new changeset %s\n" % short(n[1])) | ||
Augie Fackler
|
r43346 | fetch.add(n[1]) # earliest unknown | ||
Peter Arrenbrecht
|
r14164 | for p in n[2:4]: | ||
Pierre-Yves David
|
r20225 | if knownnode(p): | ||
Augie Fackler
|
r43346 | base.add(p) # latest known | ||
Peter Arrenbrecht
|
r14164 | |||
for p in n[2:4]: | ||||
Pierre-Yves David
|
r20225 | if p not in req and not knownnode(p): | ||
Peter Arrenbrecht
|
r14164 | r.append(p) | ||
req.add(p) | ||||
seen.add(n[0]) | ||||
if r: | ||||
reqcnt += 1 | ||||
Martin von Zweigbergk
|
r38419 | progress.increment() | ||
Augie Fackler
|
r43346 | repo.ui.debug( | ||
Augie Fackler
|
r43347 | b"request %d: %s\n" % (reqcnt, b" ".join(map(short, r))) | ||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r38806 | for p in pycompat.xrange(0, len(r), 10): | ||
Gregory Szorc
|
r37652 | with remote.commandexecutor() as e: | ||
Augie Fackler
|
r43346 | branches = e.callcommand( | ||
Augie Fackler
|
r46554 | b'branches', | ||
{ | ||||
b'nodes': r[p : p + 10], | ||||
}, | ||||
Augie Fackler
|
r43346 | ).result() | ||
Gregory Szorc
|
r37652 | |||
for b in branches: | ||||
Augie Fackler
|
r43346 | repo.ui.debug( | ||
Augie Fackler
|
r43347 | b"received %s:%s\n" % (short(b[0]), short(b[1])) | ||
Augie Fackler
|
r43346 | ) | ||
Peter Arrenbrecht
|
r14164 | unknown.append(b) | ||
# do binary search on the branches we found | ||||
while search: | ||||
newsearch = [] | ||||
reqcnt += 1 | ||||
Martin von Zweigbergk
|
r38419 | progress.increment() | ||
Gregory Szorc
|
r37652 | |||
with remote.commandexecutor() as e: | ||||
Augie Fackler
|
r43347 | between = e.callcommand(b'between', {b'pairs': search}).result() | ||
Gregory Szorc
|
r37652 | |||
for n, l in zip(search, between): | ||||
Peter Arrenbrecht
|
r14164 | l.append(n[1]) | ||
p = n[0] | ||||
f = 1 | ||||
for i in l: | ||||
Augie Fackler
|
r43347 | repo.ui.debug(b"narrowing %d:%d %s\n" % (f, len(l), short(i))) | ||
Pierre-Yves David
|
r20225 | if knownnode(i): | ||
Peter Arrenbrecht
|
r14164 | if f <= 2: | ||
Augie Fackler
|
r43346 | repo.ui.debug( | ||
Augie Fackler
|
r43347 | b"found new branch changeset %s\n" % short(p) | ||
Augie Fackler
|
r43346 | ) | ||
Peter Arrenbrecht
|
r14164 | fetch.add(p) | ||
base.add(i) | ||||
else: | ||||
Augie Fackler
|
r43346 | repo.ui.debug( | ||
Augie Fackler
|
r43347 | b"narrowed branch search to %s:%s\n" | ||
Augie Fackler
|
r43346 | % (short(p), short(i)) | ||
) | ||||
Peter Arrenbrecht
|
r14164 | newsearch.append((p, i)) | ||
break | ||||
p, f = i, f * 2 | ||||
search = newsearch | ||||
# sanity check our fetch list | ||||
for f in fetch: | ||||
Pierre-Yves David
|
r20225 | if knownnode(f): | ||
Augie Fackler
|
r43347 | raise error.RepoError(_(b"already have changeset ") + short(f[:4])) | ||
Peter Arrenbrecht
|
r14164 | |||
base = list(base) | ||||
Joerg Sonnenberger
|
r47771 | if base == [repo.nullid]: | ||
Peter Arrenbrecht
|
r14164 | if force: | ||
Augie Fackler
|
r43347 | repo.ui.warn(_(b"warning: repository is unrelated\n")) | ||
Peter Arrenbrecht
|
r14164 | else: | ||
Augie Fackler
|
r43347 | raise error.Abort(_(b"repository is unrelated")) | ||
Peter Arrenbrecht
|
r14164 | |||
Augie Fackler
|
r43346 | repo.ui.debug( | ||
Augie Fackler
|
r43347 | b"found new changesets starting at " | ||
+ b" ".join([short(f) for f in fetch]) | ||||
+ b"\n" | ||||
Augie Fackler
|
r43346 | ) | ||
Peter Arrenbrecht
|
r14164 | |||
Martin von Zweigbergk
|
r38419 | progress.complete() | ||
Augie Fackler
|
r43347 | repo.ui.debug(b"%d total queries\n" % reqcnt) | ||
r46726 | if audit is not None: | |||
audit[b'total-roundtrips'] = reqcnt | ||||
Peter Arrenbrecht
|
r14164 | |||
return base, list(fetch), heads | ||||