filelog.py
329 lines
| 9.6 KiB
| text/x-python
|
PythonLexer
/ mercurial / filelog.py
mpm@selenic.com
|
r1089 | # filelog.py - file history class for mercurial | ||
# | ||||
Raphaël Gomès
|
r47575 | # Copyright 2005-2007 Olivia Mackall <olivia@selenic.com> | ||
mpm@selenic.com
|
r1089 | # | ||
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. | ||
mpm@selenic.com
|
r1089 | |||
Matt Harbison
|
r52756 | from __future__ import annotations | ||
Gregory Szorc
|
r25948 | |||
Matt Harbison
|
r52715 | import typing | ||
Matt Harbison
|
r52719 | from typing import ( | ||
Iterable, | ||||
Iterator, | ||||
) | ||||
Gregory Szorc
|
r40425 | from .i18n import _ | ||
Joerg Sonnenberger
|
r47771 | from .node import nullrev | ||
Gregory Szorc
|
r25948 | from . import ( | ||
Gregory Szorc
|
r37515 | error, | ||
Pulkit Goyal
|
r43078 | revlog, | ||
) | ||||
from .interfaces import ( | ||||
Gregory Szorc
|
r37459 | repository, | ||
Pulkit Goyal
|
r43079 | util as interfaceutil, | ||
Gregory Szorc
|
r25948 | ) | ||
Augie Fackler
|
r43346 | from .utils import storageutil | ||
r47838 | from .revlogutils import ( | |||
constants as revlog_constants, | ||||
r48629 | rewrite, | |||
r47838 | ) | |||
Augie Fackler
|
r43346 | |||
mpm@selenic.com
|
r1089 | |||
Matt Harbison
|
r52715 | class FileLog: | ||
Matt Harbison
|
r52719 | _revlog: revlog.revlog | ||
nullid: bytes | ||||
_fix_issue6528: bool | ||||
r51242 | def __init__(self, opener, path, try_split=False): | |||
Augie Fackler
|
r43346 | self._revlog = revlog.revlog( | ||
r47838 | opener, | |||
# XXX should use the unencoded path | ||||
target=(revlog_constants.KIND_FILELOG, path), | ||||
r47921 | radix=b'/'.join((b'data', path)), | |||
r47838 | censorable=True, | |||
Joerg Sonnenberger
|
r49876 | canonical_parent_order=False, # see comment in revlog.py | ||
r51242 | try_split=try_split, | |||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r39819 | # Full name of the user visible file, relative to the repository root. | ||
# Used by LFS. | ||||
Gregory Szorc
|
r39892 | self._revlog.filename = path | ||
Joerg Sonnenberger
|
r47538 | self.nullid = self._revlog.nullid | ||
r48630 | opts = opener.options | |||
self._fix_issue6528 = opts.get(b'issue6528.fix-incoming', True) | ||||
Gregory Szorc
|
r37515 | |||
Matt Harbison
|
r52719 | def get_revlog(self) -> revlog.revlog: | ||
r51530 | """return an actual revlog instance if any | |||
This exist because a lot of code leverage the fact the underlying | ||||
storage is a revlog for optimization, so giving simple way to access | ||||
the revlog instance helps such code. | ||||
""" | ||||
return self._revlog | ||||
Matt Harbison
|
r52719 | def __len__(self) -> int: | ||
Gregory Szorc
|
r37515 | return len(self._revlog) | ||
Matt Harbison
|
r52719 | def __iter__(self) -> Iterator[int]: | ||
Gregory Szorc
|
r37515 | return self._revlog.__iter__() | ||
Gregory Szorc
|
r40423 | def hasnode(self, node): | ||
Joerg Sonnenberger
|
r47771 | if node in (self.nullid, nullrev): | ||
Gregory Szorc
|
r40423 | return False | ||
try: | ||||
self._revlog.rev(node) | ||||
return True | ||||
except (TypeError, ValueError, IndexError, error.LookupError): | ||||
return False | ||||
Gregory Szorc
|
r37515 | def revs(self, start=0, stop=None): | ||
return self._revlog.revs(start=start, stop=stop) | ||||
def parents(self, node): | ||||
return self._revlog.parents(node) | ||||
def parentrevs(self, rev): | ||||
return self._revlog.parentrevs(rev) | ||||
def rev(self, node): | ||||
return self._revlog.rev(node) | ||||
def node(self, rev): | ||||
return self._revlog.node(rev) | ||||
def lookup(self, node): | ||||
Augie Fackler
|
r43346 | return storageutil.fileidlookup( | ||
r47926 | self._revlog, node, self._revlog.display_id | |||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r37515 | |||
def linkrev(self, rev): | ||||
return self._revlog.linkrev(rev) | ||||
def commonancestorsheads(self, node1, node2): | ||||
return self._revlog.commonancestorsheads(node1, node2) | ||||
Gregory Szorc
|
r39819 | # Used by dagop.blockdescendants(). | ||
Gregory Szorc
|
r37515 | def descendants(self, revs): | ||
return self._revlog.descendants(revs) | ||||
def heads(self, start=None, stop=None): | ||||
return self._revlog.heads(start, stop) | ||||
Gregory Szorc
|
r39819 | # Used by hgweb, children extension. | ||
Gregory Szorc
|
r37515 | def children(self, node): | ||
return self._revlog.children(node) | ||||
def iscensored(self, rev): | ||||
return self._revlog.iscensored(rev) | ||||
r51915 | def revision(self, node): | |||
return self._revlog.revision(node) | ||||
Gregory Szorc
|
r37515 | |||
r51916 | def rawdata(self, node): | |||
return self._revlog.rawdata(node) | ||||
r42946 | ||||
Augie Fackler
|
r43346 | def emitrevisions( | ||
self, | ||||
nodes, | ||||
nodesorder=None, | ||||
revisiondata=False, | ||||
assumehaveparentrevisions=False, | ||||
deltamode=repository.CG_DELTAMODE_STD, | ||||
Raphaël Gomès
|
r47449 | sidedata_helpers=None, | ||
r50505 | debug_info=None, | |||
Augie Fackler
|
r43346 | ): | ||
Gregory Szorc
|
r39898 | return self._revlog.emitrevisions( | ||
Augie Fackler
|
r43346 | nodes, | ||
nodesorder=nodesorder, | ||||
revisiondata=revisiondata, | ||||
Gregory Szorc
|
r39898 | assumehaveparentrevisions=assumehaveparentrevisions, | ||
Augie Fackler
|
r43346 | deltamode=deltamode, | ||
Raphaël Gomès
|
r47449 | sidedata_helpers=sidedata_helpers, | ||
r50505 | debug_info=debug_info, | |||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r39898 | |||
Augie Fackler
|
r43346 | def addrevision( | ||
self, | ||||
revisiondata, | ||||
transaction, | ||||
linkrev, | ||||
p1, | ||||
p2, | ||||
node=None, | ||||
flags=revlog.REVIDX_DEFAULT_FLAGS, | ||||
cachedelta=None, | ||||
): | ||||
return self._revlog.addrevision( | ||||
revisiondata, | ||||
transaction, | ||||
linkrev, | ||||
p1, | ||||
p2, | ||||
node=node, | ||||
flags=flags, | ||||
cachedelta=cachedelta, | ||||
) | ||||
Gregory Szorc
|
r37515 | |||
Augie Fackler
|
r43346 | def addgroup( | ||
self, | ||||
deltas, | ||||
linkmapper, | ||||
transaction, | ||||
addrevisioncb=None, | ||||
Joerg Sonnenberger
|
r46373 | duplicaterevisioncb=None, | ||
Augie Fackler
|
r43346 | maybemissingparents=False, | ||
r50506 | debug_info=None, | |||
r50660 | delta_base_reuse_policy=None, | |||
Augie Fackler
|
r43346 | ): | ||
Gregory Szorc
|
r40425 | if maybemissingparents: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
_( | ||||
Augie Fackler
|
r43347 | b'revlog storage does not support missing ' | ||
b'parents write mode' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Gregory Szorc
|
r40425 | |||
r48628 | with self._revlog._writing(transaction): | |||
r48630 | if self._fix_issue6528: | |||
deltas = rewrite.filter_delta_issue6528(self._revlog, deltas) | ||||
r48629 | ||||
r48628 | return self._revlog.addgroup( | |||
deltas, | ||||
linkmapper, | ||||
transaction, | ||||
addrevisioncb=addrevisioncb, | ||||
duplicaterevisioncb=duplicaterevisioncb, | ||||
r50506 | debug_info=debug_info, | |||
r50660 | delta_base_reuse_policy=delta_base_reuse_policy, | |||
r48628 | ) | |||
Gregory Szorc
|
r37515 | |||
def getstrippoint(self, minlink): | ||||
return self._revlog.getstrippoint(minlink) | ||||
def strip(self, minlink, transaction): | ||||
return self._revlog.strip(minlink, transaction) | ||||
Gregory Szorc
|
r39814 | def censorrevision(self, tr, node, tombstone=b''): | ||
Gregory Szorc
|
r40092 | return self._revlog.censorrevision(tr, node, tombstone=tombstone) | ||
Gregory Szorc
|
r39814 | |||
Gregory Szorc
|
r37515 | def files(self): | ||
return self._revlog.files() | ||||
mpm@selenic.com
|
r1089 | def read(self, node): | ||
Gregory Szorc
|
r39916 | return storageutil.filtermetadata(self.revision(node)) | ||
mpm@selenic.com
|
r1089 | |||
def add(self, text, meta, transaction, link, p1=None, p2=None): | ||||
Augie Fackler
|
r43347 | if meta or text.startswith(b'\1\n'): | ||
Gregory Szorc
|
r39914 | text = storageutil.packmeta(meta, text) | ||
Joerg Sonnenberger
|
r47258 | rev = self.addrevision(text, transaction, link, p1, p2) | ||
return self.node(rev) | ||||
mpm@selenic.com
|
r1089 | |||
mpm@selenic.com
|
r1116 | def renamed(self, node): | ||
Gregory Szorc
|
r40041 | return storageutil.filerevisioncopied(self, node) | ||
mpm@selenic.com
|
r1116 | |||
Matt Mackall
|
r2898 | def size(self, rev): | ||
"""return the size of a given revision""" | ||||
# for revisions with renames, we have to go the slow way | ||||
node = self.node(rev) | ||||
Arseniy Alekseyev
|
r50066 | if self.iscensored(rev): | ||
return 0 | ||||
Matt Mackall
|
r2898 | if self.renamed(node): | ||
return len(self.read(node)) | ||||
Nicolas Dumazet
|
r11540 | # XXX if self.read(node).startswith("\1\n"), this returns (size+4) | ||
Joerg Sonnenberger
|
r49876 | # XXX See also basefilectx.cmp. | ||
Gregory Szorc
|
r37515 | return self._revlog.size(rev) | ||
Matt Mackall
|
r2898 | |||
Matt Mackall
|
r2887 | def cmp(self, node, text): | ||
Nicolas Dumazet
|
r11539 | """compare text with a given file revision | ||
returns True if text is different than what is stored. | ||||
""" | ||||
Gregory Szorc
|
r40043 | return not storageutil.filedataequivalent(self, node, text) | ||
Gregory Szorc
|
r37515 | |||
Matt Harbison
|
r52719 | def verifyintegrity(self, state) -> Iterable[revlog.RevLogProblem]: | ||
Gregory Szorc
|
r39878 | return self._revlog.verifyintegrity(state) | ||
Augie Fackler
|
r43346 | def storageinfo( | ||
self, | ||||
exclusivefiles=False, | ||||
sharedfiles=False, | ||||
revisionscount=False, | ||||
trackedsize=False, | ||||
storedsize=False, | ||||
): | ||||
Gregory Szorc
|
r39905 | return self._revlog.storageinfo( | ||
Augie Fackler
|
r43346 | exclusivefiles=exclusivefiles, | ||
sharedfiles=sharedfiles, | ||||
revisionscount=revisionscount, | ||||
trackedsize=trackedsize, | ||||
storedsize=storedsize, | ||||
) | ||||
Gregory Szorc
|
r39905 | |||
Gregory Szorc
|
r39819 | # Used by repo upgrade. | ||
Gregory Szorc
|
r37515 | def clone(self, tr, destrevlog, **kwargs): | ||
if not isinstance(destrevlog, filelog): | ||||
r50102 | msg = b'expected filelog to clone(), not %r' | |||
msg %= destrevlog | ||||
raise error.ProgrammingError(msg) | ||||
Gregory Szorc
|
r37515 | |||
return self._revlog.clone(tr, destrevlog._revlog, **kwargs) | ||||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r52715 | filelog = interfaceutil.implementer(repository.ifilestorage)(FileLog) | ||
if typing.TYPE_CHECKING: | ||||
filelog = FileLog | ||||
Gregory Szorc
|
r39801 | class narrowfilelog(filelog): | ||
"""Filelog variation to be used with narrow stores.""" | ||||
r51242 | def __init__(self, opener, path, narrowmatch, try_split=False): | |||
super(narrowfilelog, self).__init__(opener, path, try_split=try_split) | ||||
Gregory Szorc
|
r39801 | self._narrowmatch = narrowmatch | ||
def renamed(self, node): | ||||
res = super(narrowfilelog, self).renamed(node) | ||||
# Renames that come from outside the narrowspec are problematic | ||||
# because we may lack the base text for the rename. This can result | ||||
# in code attempting to walk the ancestry or compute a diff | ||||
# encountering a missing revision. We address this by silently | ||||
# removing rename metadata if the source file is outside the | ||||
# narrow spec. | ||||
# | ||||
# A better solution would be to see if the base revision is available, | ||||
# rather than assuming it isn't. | ||||
# | ||||
# An even better solution would be to teach all consumers of rename | ||||
# metadata that the base revision may not be available. | ||||
# | ||||
# TODO consider better ways of doing this. | ||||
if res and not self._narrowmatch(res[0]): | ||||
return None | ||||
return res | ||||
def size(self, rev): | ||||
# Because we have a custom renamed() that may lie, we need to call | ||||
# the base renamed() to report accurate results. | ||||
node = self.node(rev) | ||||
if super(narrowfilelog, self).renamed(node): | ||||
return len(self.read(node)) | ||||
else: | ||||
return super(narrowfilelog, self).size(rev) | ||||
def cmp(self, node, text): | ||||
Raphaël Gomès
|
r47280 | # We don't call `super` because narrow parents can be buggy in case of a | ||
# ambiguous dirstate. Always take the slow path until there is a better | ||||
# fix, see issue6150. | ||||
Gregory Szorc
|
r39801 | |||
Raphaël Gomès
|
r47280 | # Censored files compare against the empty file. | ||
if self.iscensored(self.rev(node)): | ||||
return text != b'' | ||||
Gregory Szorc
|
r39801 | |||
Raphaël Gomès
|
r47280 | return self.read(node) != text | ||