remotenames.py
446 lines
| 13.8 KiB
| text/x-python
|
PythonLexer
/ hgext / remotenames.py
Pulkit Goyal
|
r36077 | # remotenames.py - extension to display remotenames | ||
# | ||||
# Copyright 2017 Augie Fackler <raf@durin42.com> | ||||
# Copyright 2017 Sean Farley <sean@farley.io> | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
Pulkit Goyal
|
r37835 | """ showing remotebookmarks and remotebranches in UI (EXPERIMENTAL) | ||
Pulkit Goyal
|
r36079 | |||
By default both remotebookmarks and remotebranches are turned on. Config knob to | ||||
control the individually are as follows. | ||||
Config options to tweak the default behaviour: | ||||
remotenames.bookmarks | ||||
Pulkit Goyal
|
r37107 | Boolean value to enable or disable showing of remotebookmarks (default: True) | ||
Pulkit Goyal
|
r36079 | |||
remotenames.branches | ||||
Pulkit Goyal
|
r37107 | Boolean value to enable or disable showing of remotebranches (default: True) | ||
remotenames.hoistedpeer | ||||
Name of the peer whose remotebookmarks should be hoisted into the top-level | ||||
namespace (default: 'default') | ||||
Pulkit Goyal
|
r36079 | """ | ||
Pulkit Goyal
|
r36077 | |||
from __future__ import absolute_import | ||||
Pulkit Goyal
|
r36167 | from mercurial.i18n import _ | ||
Augie Fackler
|
r43346 | from mercurial.node import bin | ||
Pulkit Goyal
|
r36077 | from mercurial import ( | ||
Pulkit Goyal
|
r37108 | bookmarks, | ||
Yuya Nishihara
|
r40105 | error, | ||
Pulkit Goyal
|
r37108 | extensions, | ||
Pulkit Goyal
|
r36077 | logexchange, | ||
Pulkit Goyal
|
r36079 | namespaces, | ||
Augie Fackler
|
r36974 | pycompat, | ||
Pulkit Goyal
|
r36079 | registrar, | ||
Pulkit Goyal
|
r36167 | revsetlang, | ||
smartset, | ||||
Yuya Nishihara
|
r36939 | templateutil, | ||
Yuya Nishihara
|
r40102 | util, | ||
Pulkit Goyal
|
r36077 | ) | ||
Augie Fackler
|
r43346 | from mercurial.utils import stringutil | ||
Pulkit Goyal
|
r40095 | |||
Augie Fackler
|
r36974 | if pycompat.ispy3: | ||
import collections.abc | ||||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r36974 | mutablemapping = collections.abc.MutableMapping | ||
else: | ||||
import collections | ||||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r36974 | mutablemapping = collections.MutableMapping | ||
Pulkit Goyal
|
r36077 | # Note for extension authors: ONLY specify testedwith = 'ships-with-hg-core' for | ||
# extensions which SHIP WITH MERCURIAL. Non-mainline extensions should | ||||
# be specifying the version(s) of Mercurial they are tested with, or | ||||
# leave the attribute unspecified. | ||||
Augie Fackler
|
r43347 | testedwith = b'ships-with-hg-core' | ||
Pulkit Goyal
|
r36077 | |||
Pulkit Goyal
|
r36079 | configtable = {} | ||
configitem = registrar.configitem(configtable) | ||||
Pulkit Goyal
|
r36080 | templatekeyword = registrar.templatekeyword() | ||
Pulkit Goyal
|
r36167 | revsetpredicate = registrar.revsetpredicate() | ||
Pulkit Goyal
|
r36079 | |||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'remotenames', | ||
b'bookmarks', | ||||
default=True, | ||||
Pulkit Goyal
|
r36079 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'remotenames', | ||
b'branches', | ||||
default=True, | ||||
Pulkit Goyal
|
r36079 | ) | ||
Augie Fackler
|
r43346 | configitem( | ||
Augie Fackler
|
r46554 | b'remotenames', | ||
b'hoistedpeer', | ||||
default=b'default', | ||||
Pulkit Goyal
|
r37107 | ) | ||
Pulkit Goyal
|
r36079 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r36974 | class lazyremotenamedict(mutablemapping): | ||
Pulkit Goyal
|
r36078 | """ | ||
Read-only dict-like Class to lazily resolve remotename entries | ||||
We are doing that because remotenames startup was slow. | ||||
We lazily read the remotenames file once to figure out the potential entries | ||||
and store them in self.potentialentries. Then when asked to resolve an | ||||
entry, if it is not in self.potentialentries, then it isn't there, if it | ||||
is in self.potentialentries we resolve it and store the result in | ||||
self.cache. We cannot be lazy is when asked all the entries (keys). | ||||
""" | ||||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r36078 | def __init__(self, kind, repo): | ||
self.cache = {} | ||||
self.potentialentries = {} | ||||
Augie Fackler
|
r43346 | self._kind = kind # bookmarks or branches | ||
Pulkit Goyal
|
r36078 | self._repo = repo | ||
self.loaded = False | ||||
def _load(self): | ||||
""" Read the remotenames file, store entries matching selected kind """ | ||||
self.loaded = True | ||||
repo = self._repo | ||||
Augie Fackler
|
r43346 | for node, rpath, rname in logexchange.readremotenamefile( | ||
repo, self._kind | ||||
): | ||||
Augie Fackler
|
r43347 | name = rpath + b'/' + rname | ||
Pulkit Goyal
|
r36078 | self.potentialentries[name] = (node, rpath, name) | ||
def _resolvedata(self, potentialentry): | ||||
""" Check that the node for potentialentry exists and return it """ | ||||
if not potentialentry in self.potentialentries: | ||||
return None | ||||
node, remote, name = self.potentialentries[potentialentry] | ||||
repo = self._repo | ||||
binnode = bin(node) | ||||
# if the node doesn't exist, skip it | ||||
try: | ||||
repo.changelog.rev(binnode) | ||||
except LookupError: | ||||
return None | ||||
# Skip closed branches | ||||
Augie Fackler
|
r43347 | if self._kind == b'branches' and repo[binnode].closesbranch(): | ||
Pulkit Goyal
|
r36078 | return None | ||
return [binnode] | ||||
def __getitem__(self, key): | ||||
if not self.loaded: | ||||
self._load() | ||||
val = self._fetchandcache(key) | ||||
if val is not None: | ||||
return val | ||||
else: | ||||
raise KeyError() | ||||
Augie Fackler
|
r36266 | def __iter__(self): | ||
return iter(self.potentialentries) | ||||
def __len__(self): | ||||
return len(self.potentialentries) | ||||
def __setitem__(self): | ||||
raise NotImplementedError | ||||
def __delitem__(self): | ||||
raise NotImplementedError | ||||
Pulkit Goyal
|
r36078 | def _fetchandcache(self, key): | ||
if key in self.cache: | ||||
return self.cache[key] | ||||
val = self._resolvedata(key) | ||||
if val is not None: | ||||
self.cache[key] = val | ||||
return val | ||||
else: | ||||
return None | ||||
def keys(self): | ||||
""" Get a list of bookmark or branch names """ | ||||
if not self.loaded: | ||||
self._load() | ||||
return self.potentialentries.keys() | ||||
def iteritems(self): | ||||
""" Iterate over (name, node) tuples """ | ||||
if not self.loaded: | ||||
self._load() | ||||
Gregory Szorc
|
r43375 | for k, vtup in pycompat.iteritems(self.potentialentries): | ||
Pulkit Goyal
|
r36078 | yield (k, [bin(vtup[0])]) | ||
Martin von Zweigbergk
|
r42809 | items = iteritems | ||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r36481 | class remotenames(object): | ||
Pulkit Goyal
|
r36077 | """ | ||
This class encapsulates all the remotenames state. It also contains | ||||
Pulkit Goyal
|
r36078 | methods to access that state in convenient ways. Remotenames are lazy | ||
loaded. Whenever client code needs to ensure the freshest copy of | ||||
remotenames, use the `clearnames` method to force an eventual load. | ||||
Pulkit Goyal
|
r36077 | """ | ||
def __init__(self, repo, *args): | ||||
self._repo = repo | ||||
Pulkit Goyal
|
r36078 | self.clearnames() | ||
Pulkit Goyal
|
r36077 | |||
def clearnames(self): | ||||
""" Clear all remote names state """ | ||||
Augie Fackler
|
r43347 | self.bookmarks = lazyremotenamedict(b"bookmarks", self._repo) | ||
self.branches = lazyremotenamedict(b"branches", self._repo) | ||||
Pulkit Goyal
|
r36077 | self._invalidatecache() | ||
def _invalidatecache(self): | ||||
self._nodetobmarks = None | ||||
self._nodetobranch = None | ||||
Pulkit Goyal
|
r37107 | self._hoisttonodes = None | ||
self._nodetohoists = None | ||||
Pulkit Goyal
|
r36077 | |||
def bmarktonodes(self): | ||||
Pulkit Goyal
|
r36481 | return self.bookmarks | ||
Pulkit Goyal
|
r36077 | |||
def nodetobmarks(self): | ||||
if not self._nodetobmarks: | ||||
bmarktonodes = self.bmarktonodes() | ||||
self._nodetobmarks = {} | ||||
Gregory Szorc
|
r43375 | for name, node in pycompat.iteritems(bmarktonodes): | ||
Pulkit Goyal
|
r36077 | self._nodetobmarks.setdefault(node[0], []).append(name) | ||
return self._nodetobmarks | ||||
def branchtonodes(self): | ||||
Pulkit Goyal
|
r36481 | return self.branches | ||
Pulkit Goyal
|
r36077 | |||
def nodetobranch(self): | ||||
if not self._nodetobranch: | ||||
branchtonodes = self.branchtonodes() | ||||
self._nodetobranch = {} | ||||
Gregory Szorc
|
r43375 | for name, nodes in pycompat.iteritems(branchtonodes): | ||
Pulkit Goyal
|
r36077 | for node in nodes: | ||
self._nodetobranch.setdefault(node, []).append(name) | ||||
return self._nodetobranch | ||||
Pulkit Goyal
|
r36079 | |||
Pulkit Goyal
|
r37107 | def hoisttonodes(self, hoist): | ||
if not self._hoisttonodes: | ||||
marktonodes = self.bmarktonodes() | ||||
self._hoisttonodes = {} | ||||
Augie Fackler
|
r43347 | hoist += b'/' | ||
Gregory Szorc
|
r43375 | for name, node in pycompat.iteritems(marktonodes): | ||
Pulkit Goyal
|
r37107 | if name.startswith(hoist): | ||
Augie Fackler
|
r43346 | name = name[len(hoist) :] | ||
Pulkit Goyal
|
r37107 | self._hoisttonodes[name] = node | ||
return self._hoisttonodes | ||||
def nodetohoists(self, hoist): | ||||
if not self._nodetohoists: | ||||
marktonodes = self.bmarktonodes() | ||||
self._nodetohoists = {} | ||||
Augie Fackler
|
r43347 | hoist += b'/' | ||
Gregory Szorc
|
r43375 | for name, node in pycompat.iteritems(marktonodes): | ||
Pulkit Goyal
|
r37107 | if name.startswith(hoist): | ||
Augie Fackler
|
r43346 | name = name[len(hoist) :] | ||
Pulkit Goyal
|
r37107 | self._nodetohoists.setdefault(node[0], []).append(name) | ||
return self._nodetohoists | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r39782 | def wrapprintbookmarks(orig, ui, repo, fm, bmarks): | ||
Augie Fackler
|
r43347 | if b'remotebookmarks' not in repo.names: | ||
Pulkit Goyal
|
r37108 | return | ||
Augie Fackler
|
r43347 | ns = repo.names[b'remotebookmarks'] | ||
Pulkit Goyal
|
r37108 | |||
for name in ns.listnames(repo): | ||||
nodes = ns.nodes(repo, name) | ||||
if not nodes: | ||||
continue | ||||
node = nodes[0] | ||||
Augie Fackler
|
r43347 | bmarks[name] = (node, b' ', b'') | ||
Pulkit Goyal
|
r37108 | |||
Yuya Nishihara
|
r39782 | return orig(ui, repo, fm, bmarks) | ||
Pulkit Goyal
|
r37108 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r37108 | def extsetup(ui): | ||
Augie Fackler
|
r43347 | extensions.wrapfunction(bookmarks, b'_printbookmarks', wrapprintbookmarks) | ||
Pulkit Goyal
|
r37108 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r36079 | def reposetup(ui, repo): | ||
Pulkit Goyal
|
r38003 | |||
# set the config option to store remotenames | ||||
Augie Fackler
|
r43347 | repo.ui.setconfig(b'experimental', b'remotenames', True, b'remotenames-ext') | ||
Pulkit Goyal
|
r38003 | |||
Pulkit Goyal
|
r36079 | if not repo.local(): | ||
return | ||||
repo._remotenames = remotenames(repo) | ||||
ns = namespaces.namespace | ||||
Augie Fackler
|
r43347 | if ui.configbool(b'remotenames', b'bookmarks'): | ||
Pulkit Goyal
|
r36079 | remotebookmarkns = ns( | ||
Augie Fackler
|
r43347 | b'remotebookmarks', | ||
templatename=b'remotebookmarks', | ||||
colorname=b'remotebookmark', | ||||
logfmt=b'remote bookmark: %s\n', | ||||
Pulkit Goyal
|
r36079 | listnames=lambda repo: repo._remotenames.bmarktonodes().keys(), | ||
Augie Fackler
|
r43346 | namemap=lambda repo, name: repo._remotenames.bmarktonodes().get( | ||
name, [] | ||||
), | ||||
nodemap=lambda repo, node: repo._remotenames.nodetobmarks().get( | ||||
node, [] | ||||
), | ||||
) | ||||
Pulkit Goyal
|
r36079 | repo.names.addnamespace(remotebookmarkns) | ||
Pulkit Goyal
|
r37107 | # hoisting only works if there are remote bookmarks | ||
Augie Fackler
|
r43347 | hoist = ui.config(b'remotenames', b'hoistedpeer') | ||
Pulkit Goyal
|
r37107 | if hoist: | ||
hoistednamens = ns( | ||||
Augie Fackler
|
r43347 | b'hoistednames', | ||
templatename=b'hoistednames', | ||||
colorname=b'hoistedname', | ||||
logfmt=b'hoisted name: %s\n', | ||||
Augie Fackler
|
r43346 | listnames=lambda repo: repo._remotenames.hoisttonodes( | ||
hoist | ||||
).keys(), | ||||
namemap=lambda repo, name: repo._remotenames.hoisttonodes( | ||||
hoist | ||||
).get(name, []), | ||||
nodemap=lambda repo, node: repo._remotenames.nodetohoists( | ||||
hoist | ||||
).get(node, []), | ||||
) | ||||
Pulkit Goyal
|
r37107 | repo.names.addnamespace(hoistednamens) | ||
Augie Fackler
|
r43347 | if ui.configbool(b'remotenames', b'branches'): | ||
Pulkit Goyal
|
r36079 | remotebranchns = ns( | ||
Augie Fackler
|
r43347 | b'remotebranches', | ||
templatename=b'remotebranches', | ||||
colorname=b'remotebranch', | ||||
logfmt=b'remote branch: %s\n', | ||||
Augie Fackler
|
r43346 | listnames=lambda repo: repo._remotenames.branchtonodes().keys(), | ||
namemap=lambda repo, name: repo._remotenames.branchtonodes().get( | ||||
name, [] | ||||
), | ||||
nodemap=lambda repo, node: repo._remotenames.nodetobranch().get( | ||||
node, [] | ||||
), | ||||
) | ||||
Pulkit Goyal
|
r36079 | repo.names.addnamespace(remotebranchns) | ||
Pulkit Goyal
|
r36080 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43347 | @templatekeyword(b'remotenames', requires={b'repo', b'ctx'}) | ||
Yuya Nishihara
|
r36609 | def remotenameskw(context, mapping): | ||
Yuya Nishihara
|
r36458 | """List of strings. Remote names associated with the changeset.""" | ||
Augie Fackler
|
r43347 | repo = context.resource(mapping, b'repo') | ||
ctx = context.resource(mapping, b'ctx') | ||||
Pulkit Goyal
|
r36080 | |||
remotenames = [] | ||||
Augie Fackler
|
r43347 | if b'remotebookmarks' in repo.names: | ||
remotenames = repo.names[b'remotebookmarks'].names(repo, ctx.node()) | ||||
Pulkit Goyal
|
r36080 | |||
Augie Fackler
|
r43347 | if b'remotebranches' in repo.names: | ||
remotenames += repo.names[b'remotebranches'].names(repo, ctx.node()) | ||||
Pulkit Goyal
|
r36080 | |||
Augie Fackler
|
r43346 | return templateutil.compatlist( | ||
Augie Fackler
|
r43347 | context, mapping, b'remotename', remotenames, plural=b'remotenames' | ||
Augie Fackler
|
r43346 | ) | ||
Pulkit Goyal
|
r36080 | |||
Augie Fackler
|
r43347 | @templatekeyword(b'remotebookmarks', requires={b'repo', b'ctx'}) | ||
Yuya Nishihara
|
r36609 | def remotebookmarkskw(context, mapping): | ||
Yuya Nishihara
|
r36458 | """List of strings. Remote bookmarks associated with the changeset.""" | ||
Augie Fackler
|
r43347 | repo = context.resource(mapping, b'repo') | ||
ctx = context.resource(mapping, b'ctx') | ||||
Pulkit Goyal
|
r36080 | |||
remotebmarks = [] | ||||
Augie Fackler
|
r43347 | if b'remotebookmarks' in repo.names: | ||
remotebmarks = repo.names[b'remotebookmarks'].names(repo, ctx.node()) | ||||
Pulkit Goyal
|
r36080 | |||
Augie Fackler
|
r43346 | return templateutil.compatlist( | ||
context, | ||||
mapping, | ||||
Augie Fackler
|
r43347 | b'remotebookmark', | ||
Augie Fackler
|
r43346 | remotebmarks, | ||
Augie Fackler
|
r43347 | plural=b'remotebookmarks', | ||
Augie Fackler
|
r43346 | ) | ||
Pulkit Goyal
|
r36080 | |||
Augie Fackler
|
r43347 | @templatekeyword(b'remotebranches', requires={b'repo', b'ctx'}) | ||
Yuya Nishihara
|
r36609 | def remotebrancheskw(context, mapping): | ||
Yuya Nishihara
|
r36458 | """List of strings. Remote branches associated with the changeset.""" | ||
Augie Fackler
|
r43347 | repo = context.resource(mapping, b'repo') | ||
ctx = context.resource(mapping, b'ctx') | ||||
Pulkit Goyal
|
r36080 | |||
remotebranches = [] | ||||
Augie Fackler
|
r43347 | if b'remotebranches' in repo.names: | ||
remotebranches = repo.names[b'remotebranches'].names(repo, ctx.node()) | ||||
Pulkit Goyal
|
r36080 | |||
Augie Fackler
|
r43346 | return templateutil.compatlist( | ||
context, | ||||
mapping, | ||||
Augie Fackler
|
r43347 | b'remotebranch', | ||
Augie Fackler
|
r43346 | remotebranches, | ||
Augie Fackler
|
r43347 | plural=b'remotebranches', | ||
Augie Fackler
|
r43346 | ) | ||
Pulkit Goyal
|
r36167 | |||
Yuya Nishihara
|
r40101 | def _revsetutil(repo, subset, x, rtypes): | ||
Pulkit Goyal
|
r36167 | """utility function to return a set of revs based on the rtypes""" | ||
Augie Fackler
|
r43347 | args = revsetlang.getargs(x, 0, 1, _(b'only one argument accepted')) | ||
Yuya Nishihara
|
r40101 | if args: | ||
kind, pattern, matcher = stringutil.stringmatcher( | ||||
Augie Fackler
|
r43347 | revsetlang.getstring(args[0], _(b'argument must be a string')) | ||
Augie Fackler
|
r43346 | ) | ||
Yuya Nishihara
|
r40101 | else: | ||
Yuya Nishihara
|
r40105 | kind = pattern = None | ||
Yuya Nishihara
|
r40102 | matcher = util.always | ||
Pulkit Goyal
|
r36167 | |||
Yuya Nishihara
|
r40103 | nodes = set() | ||
Pulkit Goyal
|
r36167 | cl = repo.changelog | ||
for rtype in rtypes: | ||||
if rtype in repo.names: | ||||
ns = repo.names[rtype] | ||||
for name in ns.listnames(repo): | ||||
Augie Fackler
|
r40096 | if not matcher(name): | ||
continue | ||||
Yuya Nishihara
|
r40103 | nodes.update(ns.nodes(repo, name)) | ||
Augie Fackler
|
r43347 | if kind == b'literal' and not nodes: | ||
Augie Fackler
|
r43346 | raise error.RepoLookupError( | ||
Augie Fackler
|
r43347 | _(b"remote name '%s' does not exist") % pattern | ||
Augie Fackler
|
r43346 | ) | ||
Pulkit Goyal
|
r36167 | |||
Yuya Nishihara
|
r40103 | revs = (cl.rev(n) for n in nodes if cl.hasnode(n)) | ||
Yuya Nishihara
|
r40104 | return subset & smartset.baseset(revs) | ||
Pulkit Goyal
|
r36167 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43347 | @revsetpredicate(b'remotenames([name])') | ||
Pulkit Goyal
|
r36167 | def remotenamesrevset(repo, subset, x): | ||
Augie Fackler
|
r40096 | """All changesets which have a remotename on them. If `name` is | ||
specified, only remotenames of matching remote paths are considered. | ||||
Pulkit Goyal
|
r40095 | |||
Pattern matching is supported for `name`. See :hg:`help revisions.patterns`. | ||||
""" | ||||
Augie Fackler
|
r43347 | return _revsetutil(repo, subset, x, (b'remotebookmarks', b'remotebranches')) | ||
Pulkit Goyal
|
r40095 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43347 | @revsetpredicate(b'remotebranches([name])') | ||
Pulkit Goyal
|
r36167 | def remotebranchesrevset(repo, subset, x): | ||
Augie Fackler
|
r40096 | """All changesets which are branch heads on remotes. If `name` is | ||
specified, only remotenames of matching remote paths are considered. | ||||
Pulkit Goyal
|
r40095 | |||
Pattern matching is supported for `name`. See :hg:`help revisions.patterns`. | ||||
""" | ||||
Augie Fackler
|
r43347 | return _revsetutil(repo, subset, x, (b'remotebranches',)) | ||
Pulkit Goyal
|
r36167 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r43347 | @revsetpredicate(b'remotebookmarks([name])') | ||
Pulkit Goyal
|
r36167 | def remotebmarksrevset(repo, subset, x): | ||
Augie Fackler
|
r40096 | """All changesets which have bookmarks on remotes. If `name` is | ||
specified, only remotenames of matching remote paths are considered. | ||||
Pulkit Goyal
|
r40095 | |||
Pattern matching is supported for `name`. See :hg:`help revisions.patterns`. | ||||
""" | ||||
Augie Fackler
|
r43347 | return _revsetutil(repo, subset, x, (b'remotebookmarks',)) | ||