repoview.py
332 lines
| 11.7 KiB
| text/x-python
|
PythonLexer
/ mercurial / repoview.py
Pierre-Yves David
|
r18100 | # repoview.py - Filtered view of a localrepo object | ||
# | ||||
# Copyright 2012 Pierre-Yves David <pierre-yves.david@ens-lyon.org> | ||||
# Logilab SA <contact@logilab.fr> | ||||
# | ||||
# 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
|
r25972 | from __future__ import absolute_import | ||
Pierre-Yves David
|
r18100 | import copy | ||
Yuya Nishihara
|
r35248 | import weakref | ||
Gregory Szorc
|
r25972 | |||
from .node import nullrev | ||||
from . import ( | ||||
obsolete, | ||||
phases, | ||||
Yuya Nishihara
|
r35249 | pycompat, | ||
Gregory Szorc
|
r25972 | tags as tagsmod, | ||
r42417 | util, | |||
) | ||||
Augie Fackler
|
r43346 | from .utils import repoviewutil | ||
Pierre-Yves David
|
r18242 | |||
Pierre-Yves David
|
r18293 | def hideablerevs(repo): | ||
Pierre-Yves David
|
r28780 | """Revision candidates to be hidden | ||
This is a standalone function to allow extensions to wrap it. | ||||
Pierre-Yves David
|
r18293 | |||
Pierre-Yves David
|
r28780 | Because we use the set of immutable changesets as a fallback subset in | ||
r42314 | branchmap (see mercurial.utils.repoviewutils.subsettable), you cannot set | |||
"public" changesets as "hideable". Doing so would break multiple code | ||||
assertions and lead to crashes.""" | ||||
Augie Fackler
|
r43347 | obsoletes = obsolete.getrevs(repo, b'obsolete') | ||
Boris Feld
|
r39333 | internals = repo._phasecache.getrevset(repo, phases.localhiddenphases) | ||
internals = frozenset(internals) | ||||
return obsoletes | internals | ||||
Pierre-Yves David
|
r18293 | |||
Augie Fackler
|
r43346 | |||
Martin von Zweigbergk
|
r32580 | def pinnedrevs(repo): | ||
Martin von Zweigbergk
|
r32579 | """revisions blocking hidden changesets from being filtered | ||
r32479 | """ | |||
r32426 | ||||
cl = repo.changelog | ||||
Martin von Zweigbergk
|
r32580 | pinned = set() | ||
pinned.update([par.rev() for par in repo[None].parents()]) | ||||
pinned.update([cl.rev(bm) for bm in repo._bookmarks.values()]) | ||||
r32426 | ||||
tags = {} | ||||
tagsmod.readlocaltags(repo.ui, repo, tags, {}) | ||||
if tags: | ||||
rev, nodemap = cl.rev, cl.nodemap | ||||
Martin von Zweigbergk
|
r32580 | pinned.update(rev(t[0]) for t in tags.values() if t[0] in nodemap) | ||
return pinned | ||||
r32426 | ||||
r32476 | ||||
Martin von Zweigbergk
|
r32582 | def _revealancestors(pfunc, hidden, revs): | ||
"""reveals contiguous chains of hidden ancestors of 'revs' by removing them | ||||
from 'hidden' | ||||
r32474 | ||||
- pfunc(r): a funtion returning parent of 'r', | ||||
Martin von Zweigbergk
|
r32581 | - hidden: the (preliminary) hidden revisions, to be updated | ||
r32474 | - revs: iterable of revnum, | |||
Martin von Zweigbergk
|
r32585 | (Ancestors are revealed exclusively, i.e. the elements in 'revs' are | ||
*not* revealed) | ||||
r32474 | """ | |||
stack = list(revs) | ||||
while stack: | ||||
for p in pfunc(stack.pop()): | ||||
Martin von Zweigbergk
|
r32582 | if p != nullrev and p in hidden: | ||
Martin von Zweigbergk
|
r32581 | hidden.remove(p) | ||
r32474 | stack.append(p) | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r35509 | def computehidden(repo, visibilityexceptions=None): | ||
Pierre-Yves David
|
r18242 | """compute the set of hidden revision to filter | ||
During most operation hidden should be filtered.""" | ||||
assert not repo.changelog.filteredrevs | ||||
David Soria Parra
|
r22151 | |||
r32478 | hidden = hideablerevs(repo) | |||
if hidden: | ||||
Martin von Zweigbergk
|
r32586 | hidden = set(hidden - pinnedrevs(repo)) | ||
Pulkit Goyal
|
r35509 | if visibilityexceptions: | ||
hidden -= visibilityexceptions | ||||
r32478 | pfunc = repo.changelog.parentrevs | |||
Boris Feld
|
r38174 | mutable = repo._phasecache.getrevset(repo, phases.mutablephases) | ||
r32478 | ||||
Martin von Zweigbergk
|
r32587 | visible = mutable - hidden | ||
_revealancestors(pfunc, hidden, visible) | ||||
r32478 | return frozenset(hidden) | |||
Pierre-Yves David
|
r18242 | |||
Augie Fackler
|
r43346 | |||
r42295 | def computesecret(repo, visibilityexceptions=None): | |||
"""compute the set of revision that can never be exposed through hgweb | ||||
Changeset in the secret phase (or above) should stay unaccessible.""" | ||||
assert not repo.changelog.filteredrevs | ||||
secrets = repo._phasecache.getrevset(repo, phases.remotehiddenphases) | ||||
return frozenset(secrets) | ||||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r35509 | def computeunserved(repo, visibilityexceptions=None): | ||
Pierre-Yves David
|
r18102 | """compute the set of revision that should be filtered when used a server | ||
Secret and hidden changeset should not pretend to be here.""" | ||||
assert not repo.changelog.filteredrevs | ||||
# fast path in simple case to avoid impact of non optimised code | ||||
Augie Fackler
|
r43347 | hiddens = filterrevs(repo, b'visible') | ||
secrets = filterrevs(repo, b'served.hidden') | ||||
r42294 | if secrets: | |||
r42295 | return frozenset(hiddens | secrets) | |||
Pierre-Yves David
|
r18273 | else: | ||
return hiddens | ||||
Pierre-Yves David
|
r18100 | |||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r35509 | def computemutable(repo, visibilityexceptions=None): | ||
Pierre-Yves David
|
r18245 | assert not repo.changelog.filteredrevs | ||
# fast check to avoid revset call on huge repo | ||||
Augie Fackler
|
r25149 | if any(repo._phasecache.phaseroots[1:]): | ||
Pierre-Yves David
|
r18274 | getphase = repo._phasecache.phase | ||
Augie Fackler
|
r43347 | maymutable = filterrevs(repo, b'base') | ||
Pierre-Yves David
|
r18274 | return frozenset(r for r in maymutable if getphase(repo, r)) | ||
Pierre-Yves David
|
r18245 | return frozenset() | ||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r35509 | def computeimpactable(repo, visibilityexceptions=None): | ||
Pierre-Yves David
|
r18246 | """Everything impactable by mutable revision | ||
Pierre-Yves David
|
r18462 | The immutable filter still have some chance to get invalidated. This will | ||
Pierre-Yves David
|
r18246 | happen when: | ||
- you garbage collect hidden changeset, | ||||
- public phase is moved backward, | ||||
- something is changed in the filtering (this could be fixed) | ||||
This filter out any mutable changeset and any public changeset that may be | ||||
impacted by something happening to a mutable revision. | ||||
This is achieved by filtered everything with a revision number egal or | ||||
higher than the first mutable changeset is filtered.""" | ||||
assert not repo.changelog.filteredrevs | ||||
cl = repo.changelog | ||||
firstmutable = len(cl) | ||||
for roots in repo._phasecache.phaseroots[1:]: | ||||
if roots: | ||||
firstmutable = min(firstmutable, min(cl.rev(r) for r in roots)) | ||||
Pierre-Yves David
|
r18443 | # protect from nullrev root | ||
firstmutable = max(0, firstmutable) | ||||
Gregory Szorc
|
r38806 | return frozenset(pycompat.xrange(firstmutable, len(cl))) | ||
Pierre-Yves David
|
r18246 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r18100 | # function to compute filtered set | ||
Pierre-Yves David
|
r20196 | # | ||
Mads Kiilerich
|
r20549 | # When adding a new filter you MUST update the table at: | ||
r42314 | # mercurial.utils.repoviewutil.subsettable | |||
Pierre-Yves David
|
r20196 | # Otherwise your filter will have to recompute all its branches cache | ||
# from scratch (very slow). | ||||
Augie Fackler
|
r43346 | filtertable = { | ||
Augie Fackler
|
r43347 | b'visible': computehidden, | ||
b'visible-hidden': computehidden, | ||||
b'served.hidden': computesecret, | ||||
b'served': computeunserved, | ||||
b'immutable': computemutable, | ||||
b'base': computeimpactable, | ||||
Augie Fackler
|
r43346 | } | ||
Pierre-Yves David
|
r18100 | |||
r42417 | _basefiltername = list(filtertable) | |||
Augie Fackler
|
r43346 | |||
r42417 | def extrafilter(ui): | |||
"""initialize extra filter and return its id | ||||
If extra filtering is configured, we make sure the associated filtered view | ||||
are declared and return the associated id. | ||||
""" | ||||
Augie Fackler
|
r43347 | frevs = ui.config(b'experimental', b'extra-filter-revs') | ||
r42417 | if frevs is None: | |||
return None | ||||
Augie Fackler
|
r43347 | fid = pycompat.sysbytes(util.DIGESTS[b'sha1'](frevs).hexdigest())[:12] | ||
r42417 | ||||
Augie Fackler
|
r43347 | combine = lambda fname: fname + b'%' + fid | ||
r42417 | ||||
subsettable = repoviewutil.subsettable | ||||
Augie Fackler
|
r43347 | if combine(b'base') not in filtertable: | ||
r42417 | for name in _basefiltername: | |||
Augie Fackler
|
r43346 | |||
r42417 | def extrafilteredrevs(repo, *args, **kwargs): | |||
baserevs = filtertable[name](repo, *args, **kwargs) | ||||
extrarevs = frozenset(repo.revs(frevs)) | ||||
return baserevs | extrarevs | ||||
Augie Fackler
|
r43346 | |||
r42417 | filtertable[combine(name)] = extrafilteredrevs | |||
if name in subsettable: | ||||
subsettable[combine(name)] = combine(subsettable[name]) | ||||
return fid | ||||
Augie Fackler
|
r43346 | |||
Pulkit Goyal
|
r35509 | def filterrevs(repo, filtername, visibilityexceptions=None): | ||
"""returns set of filtered revision for this filter name | ||||
visibilityexceptions is a set of revs which must are exceptions for | ||||
hidden-state and must be visible. They are dynamic and hence we should not | ||||
cache it's result""" | ||||
Pierre-Yves David
|
r18101 | if filtername not in repo.filteredrevcache: | ||
func = filtertable[filtername] | ||||
Pulkit Goyal
|
r35509 | if visibilityexceptions: | ||
return func(repo.unfiltered, visibilityexceptions) | ||||
Pierre-Yves David
|
r18101 | repo.filteredrevcache[filtername] = func(repo.unfiltered()) | ||
return repo.filteredrevcache[filtername] | ||||
Pierre-Yves David
|
r18100 | |||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r18100 | class repoview(object): | ||
"""Provide a read/write view of a repo through a filtered changelog | ||||
This object is used to access a filtered version of a repository without | ||||
altering the original repository object itself. We can not alter the | ||||
original object for two main reasons: | ||||
- It prevents the use of a repo with multiple filters at the same time. In | ||||
particular when multiple threads are involved. | ||||
- It makes scope of the filtering harder to control. | ||||
This object behaves very closely to the original repository. All attribute | ||||
operations are done on the original repository: | ||||
- An access to `repoview.someattr` actually returns `repo.someattr`, | ||||
- A write to `repoview.someattr` actually sets value of `repo.someattr`, | ||||
- A deletion of `repoview.someattr` actually drops `someattr` | ||||
from `repo.__dict__`. | ||||
The only exception is the `changelog` property. It is overridden to return | ||||
a (surface) copy of `repo.changelog` with some revisions filtered. The | ||||
`filtername` attribute of the view control the revisions that need to be | ||||
filtered. (the fact the changelog is copied is an implementation detail). | ||||
Unlike attributes, this object intercepts all method calls. This means that | ||||
all methods are run on the `repoview` object with the filtered `changelog` | ||||
property. For this purpose the simple `repoview` class must be mixed with | ||||
the actual class of the repository. This ensures that the resulting | ||||
`repoview` object have the very same methods than the repo object. This | ||||
leads to the property below. | ||||
repoview.method() --> repo.__class__.method(repoview) | ||||
The inheritance has to be done dynamically because `repo` can be of any | ||||
Mads Kiilerich
|
r18644 | subclasses of `localrepo`. Eg: `bundlerepo` or `statichttprepo`. | ||
Pierre-Yves David
|
r18100 | """ | ||
Pulkit Goyal
|
r35508 | def __init__(self, repo, filtername, visibilityexceptions=None): | ||
Pulkit Goyal
|
r31221 | object.__setattr__(self, r'_unfilteredrepo', repo) | ||
object.__setattr__(self, r'filtername', filtername) | ||||
object.__setattr__(self, r'_clcachekey', None) | ||||
object.__setattr__(self, r'_clcache', None) | ||||
Pulkit Goyal
|
r35508 | # revs which are exceptions and must not be hidden | ||
Augie Fackler
|
r43346 | object.__setattr__(self, r'_visibilityexceptions', visibilityexceptions) | ||
Pierre-Yves David
|
r18100 | |||
Mads Kiilerich
|
r18644 | # not a propertycache on purpose we shall implement a proper cache later | ||
Pierre-Yves David
|
r18100 | @property | ||
def changelog(self): | ||||
"""return a filtered version of the changeset | ||||
this changelog must not be used for writing""" | ||||
# some cache may be implemented later | ||||
Pierre-Yves David
|
r18445 | unfi = self._unfilteredrepo | ||
unfichangelog = unfi.changelog | ||||
Pierre-Yves David
|
r27258 | # bypass call to changelog.method | ||
unfiindex = unfichangelog.index | ||||
Martin von Zweigbergk
|
r38887 | unfilen = len(unfiindex) | ||
Pierre-Yves David
|
r27258 | unfinode = unfiindex[unfilen - 1][7] | ||
Pulkit Goyal
|
r35509 | revs = filterrevs(unfi, self.filtername, self._visibilityexceptions) | ||
Pierre-Yves David
|
r18445 | cl = self._clcache | ||
Pierre-Yves David
|
r27258 | newkey = (unfilen, unfinode, hash(revs), unfichangelog._delayed) | ||
FUJIWARA Katsunori
|
r28265 | # if cl.index is not unfiindex, unfi.changelog would be | ||
# recreated, and our clcache refers to garbage object | ||||
Augie Fackler
|
r43346 | if cl is not None and ( | ||
cl.index is not unfiindex or newkey != self._clcachekey | ||||
): | ||||
Pierre-Yves David
|
r27258 | cl = None | ||
Pierre-Yves David
|
r18445 | # could have been made None by the previous if | ||
if cl is None: | ||||
cl = copy.copy(unfichangelog) | ||||
cl.filteredrevs = revs | ||||
Augie Fackler
|
r31358 | object.__setattr__(self, r'_clcache', cl) | ||
object.__setattr__(self, r'_clcachekey', newkey) | ||||
Pierre-Yves David
|
r18100 | return cl | ||
def unfiltered(self): | ||||
"""Return an unfiltered version of a repo""" | ||||
return self._unfilteredrepo | ||||
Pulkit Goyal
|
r35508 | def filtered(self, name, visibilityexceptions=None): | ||
Pierre-Yves David
|
r18100 | """Return a filtered version of a repository""" | ||
Pulkit Goyal
|
r35508 | if name == self.filtername and not visibilityexceptions: | ||
Pierre-Yves David
|
r18100 | return self | ||
Pulkit Goyal
|
r35508 | return self.unfiltered().filtered(name, visibilityexceptions) | ||
Pierre-Yves David
|
r18100 | |||
Yuya Nishihara
|
r35249 | def __repr__(self): | ||
Augie Fackler
|
r43346 | return r'<%s:%s %r>' % ( | ||
self.__class__.__name__, | ||||
pycompat.sysstr(self.filtername), | ||||
self.unfiltered(), | ||||
) | ||||
Yuya Nishihara
|
r35249 | |||
Pierre-Yves David
|
r18100 | # everything access are forwarded to the proxied repo | ||
def __getattr__(self, attr): | ||||
return getattr(self._unfilteredrepo, attr) | ||||
def __setattr__(self, attr, value): | ||||
return setattr(self._unfilteredrepo, attr, value) | ||||
def __delattr__(self, attr): | ||||
return delattr(self._unfilteredrepo, attr) | ||||
Yuya Nishihara
|
r35248 | |||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r35248 | # Python <3.4 easily leaks types via __mro__. See | ||
# https://bugs.python.org/issue17950. We cache dynamically created types | ||||
# so they won't be leaked on every invocation of repo.filtered(). | ||||
_filteredrepotypes = weakref.WeakKeyDictionary() | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r35248 | def newtype(base): | ||
"""Create a new type with the repoview mixin and the given base class""" | ||||
if base not in _filteredrepotypes: | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r35248 | class filteredrepo(repoview, base): | ||
pass | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r35248 | _filteredrepotypes[base] = filteredrepo | ||
return _filteredrepotypes[base] | ||||