upgrade.py
418 lines
| 14.6 KiB
| text/x-python
|
PythonLexer
/ mercurial / upgrade.py
Pierre-Yves David
|
r31894 | # upgrade.py - functions for in place upgrade of Mercurial repository | ||
Pierre-Yves David
|
r31864 | # | ||
Pierre-Yves David
|
r31895 | # Copyright (c) 2016-present, Gregory Szorc | ||
Pierre-Yves David
|
r31864 | # | ||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
from .i18n import _ | ||||
from . import ( | ||||
error, | ||||
Boris Feld
|
r35344 | hg, | ||
Pierre-Yves David
|
r31893 | localrepo, | ||
Pulkit Goyal
|
r46852 | lock as lockmod, | ||
Yuya Nishihara
|
r38183 | pycompat, | ||
Pulkit Goyal
|
r46852 | requirements as requirementsmod, | ||
scmutil, | ||||
r46661 | ) | |||
from .upgrade_utils import ( | ||||
r46662 | actions as upgrade_actions, | |||
r50087 | auto_upgrade, | |||
r46661 | engine as upgrade_engine, | |||
Pierre-Yves David
|
r31864 | ) | ||
Pulkit Goyal
|
r46852 | from .utils import ( | ||
stringutil, | ||||
) | ||||
r50087 | may_auto_upgrade = auto_upgrade.may_auto_upgrade | |||
r46662 | allformatvariant = upgrade_actions.allformatvariant | |||
Augie Fackler
|
r43346 | |||
def upgraderepo( | ||||
ui, | ||||
repo, | ||||
run=False, | ||||
optimize=None, | ||||
backup=True, | ||||
manifest=None, | ||||
changelog=None, | ||||
r46612 | filelogs=None, | |||
Augie Fackler
|
r43346 | ): | ||
Pierre-Yves David
|
r31864 | """Upgrade a repository in place.""" | ||
Boris Feld
|
r41120 | if optimize is None: | ||
Matt Harbison
|
r49303 | optimize = set() | ||
Pierre-Yves David
|
r31864 | repo = repo.unfiltered() | ||
r49274 | specified_revlogs = {} | |||
if changelog is not None: | ||||
specified_revlogs[upgrade_engine.UPGRADE_CHANGELOG] = changelog | ||||
if manifest is not None: | ||||
specified_revlogs[upgrade_engine.UPGRADE_MANIFEST] = manifest | ||||
if filelogs is not None: | ||||
specified_revlogs[upgrade_engine.UPGRADE_FILELOGS] = filelogs | ||||
r43098 | ||||
Pierre-Yves David
|
r31864 | # Ensure the repository can be upgraded. | ||
r46663 | upgrade_actions.check_source_requirements(repo) | |||
Pierre-Yves David
|
r31864 | |||
r46666 | default_options = localrepo.defaultcreateopts(repo.ui) | |||
newreqs = localrepo.newreporequirements(repo.ui, default_options) | ||||
r46662 | newreqs.update(upgrade_actions.preservedrequirements(repo)) | |||
Pierre-Yves David
|
r31864 | |||
r46666 | upgrade_actions.check_requirements_changes(repo, newreqs) | |||
Pierre-Yves David
|
r31864 | |||
# Find and validate all improvements that can be made. | ||||
r46662 | alloptimizations = upgrade_actions.findoptimizations(repo) | |||
Pierre-Yves David
|
r31864 | |||
Pierre-Yves David
|
r31899 | # Apply and Validate arguments. | ||
optimizations = [] | ||||
for o in alloptimizations: | ||||
if o.name in optimize: | ||||
optimizations.append(o) | ||||
optimize.discard(o.name) | ||||
Augie Fackler
|
r43346 | if optimize: # anything left is unknown | ||
raise error.Abort( | ||||
Augie Fackler
|
r43347 | _(b'unknown optimization action requested: %s') | ||
% b', '.join(sorted(optimize)), | ||||
Martin von Zweigbergk
|
r43387 | hint=_(b'run without arguments to see valid optimizations'), | ||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r31864 | |||
Pulkit Goyal
|
r46822 | format_upgrades = upgrade_actions.find_format_upgrades(repo) | ||
Pulkit Goyal
|
r46827 | up_actions = upgrade_actions.determine_upgrade_actions( | ||
Pulkit Goyal
|
r46826 | repo, format_upgrades, optimizations, repo.requirements, newreqs | ||
Augie Fackler
|
r43346 | ) | ||
Pulkit Goyal
|
r46830 | removed_actions = upgrade_actions.find_format_downgrades(repo) | ||
Pierre-Yves David
|
r31864 | |||
r49273 | # check if we need to touch revlog and if so, which ones | |||
r49275 | touched_revlogs = set() | |||
overwrite_msg = _(b'warning: ignoring %14s, as upgrade is changing: %s\n') | ||||
r49276 | select_msg = _(b'note: selecting %s for processing to change: %s\n') | |||
msg_issued = 0 | ||||
r49275 | ||||
FL = upgrade_engine.UPGRADE_FILELOGS | ||||
MN = upgrade_engine.UPGRADE_MANIFEST | ||||
CL = upgrade_engine.UPGRADE_CHANGELOG | ||||
r49277 | if optimizations: | |||
if any(specified_revlogs.values()): | ||||
# we have some limitation on revlogs to be recloned | ||||
for rl, enabled in specified_revlogs.items(): | ||||
if enabled: | ||||
touched_revlogs.add(rl) | ||||
else: | ||||
touched_revlogs = set(upgrade_engine.UPGRADE_ALL_REVLOGS) | ||||
for rl, enabled in specified_revlogs.items(): | ||||
if not enabled: | ||||
touched_revlogs.discard(rl) | ||||
r49628 | if repo.shared(): | |||
unsafe_actions = set() | ||||
unsafe_actions.update(up_actions) | ||||
unsafe_actions.update(removed_actions) | ||||
unsafe_actions.update(optimizations) | ||||
unsafe_actions = [ | ||||
a for a in unsafe_actions if not a.compatible_with_share | ||||
] | ||||
unsafe_actions.sort(key=lambda a: a.name) | ||||
if unsafe_actions: | ||||
m = _(b'cannot use these actions on a share repository: %s') | ||||
h = _(b'upgrade the main repository directly') | ||||
actions = b', '.join(a.name for a in unsafe_actions) | ||||
m %= actions | ||||
raise error.Abort(m, hint=h) | ||||
r49275 | for action in sorted(up_actions + removed_actions, key=lambda a: a.name): | |||
# optimisation does not "requires anything, they just needs it. | ||||
if action.type != upgrade_actions.FORMAT_VARIANT: | ||||
continue | ||||
r43100 | ||||
r49275 | if action.touches_filelogs and FL not in touched_revlogs: | |||
if FL in specified_revlogs: | ||||
if not specified_revlogs[FL]: | ||||
msg = overwrite_msg % (b'--no-filelogs', action.name) | ||||
ui.warn(msg) | ||||
r49276 | msg_issued = 2 | |||
else: | ||||
msg = select_msg % (b'all-filelogs', action.name) | ||||
ui.status(msg) | ||||
if not ui.quiet: | ||||
msg_issued = 1 | ||||
r49275 | touched_revlogs.add(FL) | |||
r49276 | ||||
r49275 | if action.touches_manifests and MN not in touched_revlogs: | |||
if MN in specified_revlogs: | ||||
if not specified_revlogs[MN]: | ||||
msg = overwrite_msg % (b'--no-manifest', action.name) | ||||
ui.warn(msg) | ||||
r49276 | msg_issued = 2 | |||
else: | ||||
msg = select_msg % (b'all-manifestlogs', action.name) | ||||
ui.status(msg) | ||||
if not ui.quiet: | ||||
msg_issued = 1 | ||||
r49275 | touched_revlogs.add(MN) | |||
r49276 | ||||
r49275 | if action.touches_changelog and CL not in touched_revlogs: | |||
if CL in specified_revlogs: | ||||
if not specified_revlogs[CL]: | ||||
msg = overwrite_msg % (b'--no-changelog', action.name) | ||||
ui.warn(msg) | ||||
msg_issued = True | ||||
r49276 | else: | |||
msg = select_msg % (b'changelog', action.name) | ||||
ui.status(msg) | ||||
if not ui.quiet: | ||||
msg_issued = 1 | ||||
r49275 | touched_revlogs.add(CL) | |||
r49276 | if msg_issued >= 2: | |||
r49275 | ui.warn((b"\n")) | |||
r49276 | elif msg_issued >= 1: | |||
ui.status((b"\n")) | ||||
r43100 | ||||
r46673 | upgrade_op = upgrade_actions.UpgradeOperation( | |||
Pulkit Goyal
|
r46804 | ui, | ||
r46673 | newreqs, | |||
Pulkit Goyal
|
r46807 | repo.requirements, | ||
Pulkit Goyal
|
r46827 | up_actions, | ||
Pulkit Goyal
|
r46830 | removed_actions, | ||
r49277 | touched_revlogs, | |||
Pulkit Goyal
|
r47092 | backup, | ||
r46673 | ) | |||
Pierre-Yves David
|
r31864 | if not run: | ||
fromconfig = [] | ||||
Pierre-Yves David
|
r31904 | onlydefault = [] | ||
Pierre-Yves David
|
r31864 | |||
Pulkit Goyal
|
r46822 | for d in format_upgrades: | ||
Pierre-Yves David
|
r32031 | if d.fromconfig(repo): | ||
Pierre-Yves David
|
r31901 | fromconfig.append(d) | ||
Pierre-Yves David
|
r32031 | elif d.default: | ||
Pierre-Yves David
|
r31904 | onlydefault.append(d) | ||
Pierre-Yves David
|
r31864 | |||
Pierre-Yves David
|
r31904 | if fromconfig or onlydefault: | ||
Pierre-Yves David
|
r31864 | |||
if fromconfig: | ||||
r45302 | ui.status( | |||
Augie Fackler
|
r43346 | _( | ||
Augie Fackler
|
r43347 | b'repository lacks features recommended by ' | ||
b'current config options:\n\n' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Pierre-Yves David
|
r31864 | for i in fromconfig: | ||
r45302 | ui.status(b'%s\n %s\n\n' % (i.name, i.description)) | |||
Pierre-Yves David
|
r31864 | |||
if onlydefault: | ||||
r45302 | ui.status( | |||
Augie Fackler
|
r43346 | _( | ||
Augie Fackler
|
r43347 | b'repository lacks features used by the default ' | ||
b'config options:\n\n' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Pierre-Yves David
|
r31864 | for i in onlydefault: | ||
r45302 | ui.status(b'%s\n %s\n\n' % (i.name, i.description)) | |||
Pierre-Yves David
|
r31864 | |||
r45302 | ui.status(b'\n') | |||
Pierre-Yves David
|
r31864 | else: | ||
Pulkit Goyal
|
r46822 | ui.status(_(b'(no format upgrades found in existing repository)\n')) | ||
Pierre-Yves David
|
r31864 | |||
r45302 | ui.status( | |||
Augie Fackler
|
r43346 | _( | ||
Augie Fackler
|
r43347 | b'performing an upgrade with "--run" will make the following ' | ||
b'changes:\n\n' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Pierre-Yves David
|
r31864 | |||
Pulkit Goyal
|
r46807 | upgrade_op.print_requirements() | ||
Pulkit Goyal
|
r46806 | upgrade_op.print_optimisations() | ||
Pulkit Goyal
|
r46805 | upgrade_op.print_upgrade_actions() | ||
Pulkit Goyal
|
r46804 | upgrade_op.print_affected_revlogs() | ||
Pierre-Yves David
|
r31864 | |||
Pulkit Goyal
|
r46808 | if upgrade_op.unused_optimizations: | ||
r45302 | ui.status( | |||
Augie Fackler
|
r43346 | _( | ||
Augie Fackler
|
r43347 | b'additional optimizations are available by specifying ' | ||
b'"--optimize <name>":\n\n' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Pulkit Goyal
|
r46808 | upgrade_op.print_unused_optimizations() | ||
Pierre-Yves David
|
r31864 | return | ||
Pulkit Goyal
|
r46848 | if not (upgrade_op.upgrade_actions or upgrade_op.removed_actions): | ||
ui.status(_(b'nothing to do\n')) | ||||
return | ||||
Pierre-Yves David
|
r31864 | # Else we're in the run=true case. | ||
Augie Fackler
|
r43347 | ui.write(_(b'upgrade will perform the following actions:\n\n')) | ||
Pulkit Goyal
|
r46807 | upgrade_op.print_requirements() | ||
Pulkit Goyal
|
r46806 | upgrade_op.print_optimisations() | ||
Pulkit Goyal
|
r46805 | upgrade_op.print_upgrade_actions() | ||
Pulkit Goyal
|
r46804 | upgrade_op.print_affected_revlogs() | ||
Pierre-Yves David
|
r31864 | |||
r45302 | ui.status(_(b'beginning upgrade...\n')) | |||
Jun Wu
|
r33438 | with repo.wlock(), repo.lock(): | ||
r45302 | ui.status(_(b'repository locked and read-only\n')) | |||
Jun Wu
|
r33438 | # Our strategy for upgrading the repository is to create a new, | ||
# temporary repository, write data to it, then do a swap of the | ||||
# data. There are less heavyweight ways to do this, but it is easier | ||||
# to create a new repo object than to instantiate all the components | ||||
# (like the store) separately. | ||||
Augie Fackler
|
r43347 | tmppath = pycompat.mkdtemp(prefix=b'upgrade.', dir=repo.path) | ||
Jun Wu
|
r33438 | backuppath = None | ||
try: | ||||
r45302 | ui.status( | |||
Augie Fackler
|
r43346 | _( | ||
Pulkit Goyal
|
r46839 | b'creating temporary repository to stage upgraded ' | ||
Augie Fackler
|
r43347 | b'data: %s\n' | ||
Augie Fackler
|
r43346 | ) | ||
% tmppath | ||||
) | ||||
Boris Feld
|
r35343 | |||
Yuya Nishihara
|
r35380 | # clone ui without using ui.copy because repo.ui is protected | ||
repoui = repo.ui.__class__(repo.ui) | ||||
dstrepo = hg.repository(repoui, path=tmppath, create=True) | ||||
Pierre-Yves David
|
r31864 | |||
Jun Wu
|
r33438 | with dstrepo.wlock(), dstrepo.lock(): | ||
r46661 | backuppath = upgrade_engine.upgrade( | |||
r46673 | ui, repo, dstrepo, upgrade_op | |||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r31864 | |||
Jun Wu
|
r33438 | finally: | ||
r45302 | ui.status(_(b'removing temporary repository %s\n') % tmppath) | |||
Jun Wu
|
r33438 | repo.vfs.rmtree(tmppath, forcibly=True) | ||
Pierre-Yves David
|
r31864 | |||
r45302 | if backuppath and not ui.quiet: | |||
Augie Fackler
|
r43346 | ui.warn( | ||
Augie Fackler
|
r43347 | _(b'copy of old repository backed up at %s\n') % backuppath | ||
Augie Fackler
|
r43346 | ) | ||
ui.warn( | ||||
_( | ||||
Augie Fackler
|
r43347 | b'the old repository will not be deleted; remove ' | ||
b'it to free up disk space once the upgraded ' | ||||
b'repository is verified\n' | ||||
Augie Fackler
|
r43346 | ) | ||
) | ||||
Pulkit Goyal
|
r46617 | |||
Pulkit Goyal
|
r46830 | upgrade_op.print_post_op_messages() | ||
Pulkit Goyal
|
r46852 | |||
Pulkit Goyal
|
r47050 | def upgrade_share_to_safe( | ||
Pulkit Goyal
|
r47051 | ui, | ||
hgvfs, | ||||
storevfs, | ||||
current_requirements, | ||||
mismatch_config, | ||||
mismatch_warn, | ||||
Pulkit Goyal
|
r47050 | ): | ||
Pulkit Goyal
|
r46852 | """Upgrades a share to use share-safe mechanism""" | ||
wlock = None | ||||
Pulkit Goyal
|
r47047 | store_requirements = localrepo._readrequires(storevfs, False) | ||
Pulkit Goyal
|
r47048 | original_crequirements = current_requirements.copy() | ||
Pulkit Goyal
|
r47047 | # after upgrade, store requires will be shared, so lets find | ||
# the requirements which are not present in store and | ||||
# write them to share's .hg/requires | ||||
diffrequires = current_requirements - store_requirements | ||||
# add share-safe requirement as it will mark the share as share-safe | ||||
diffrequires.add(requirementsmod.SHARESAFE_REQUIREMENT) | ||||
current_requirements.add(requirementsmod.SHARESAFE_REQUIREMENT) | ||||
Pulkit Goyal
|
r47050 | # in `allow` case, we don't try to upgrade, we just respect the source | ||
# state, update requirements and continue | ||||
if mismatch_config == b'allow': | ||||
return | ||||
Pulkit Goyal
|
r46852 | try: | ||
wlock = lockmod.trylock(ui, hgvfs, b'wlock', 0, 0) | ||||
Pulkit Goyal
|
r47048 | # some process might change the requirement in between, re-read | ||
# and update current_requirements | ||||
locked_requirements = localrepo._readrequires(hgvfs, True) | ||||
if locked_requirements != original_crequirements: | ||||
removed = current_requirements - locked_requirements | ||||
# update current_requirements in place because it's passed | ||||
# as reference | ||||
current_requirements -= removed | ||||
current_requirements |= locked_requirements | ||||
diffrequires = current_requirements - store_requirements | ||||
# add share-safe requirement as it will mark the share as share-safe | ||||
diffrequires.add(requirementsmod.SHARESAFE_REQUIREMENT) | ||||
current_requirements.add(requirementsmod.SHARESAFE_REQUIREMENT) | ||||
Pulkit Goyal
|
r46852 | scmutil.writerequires(hgvfs, diffrequires) | ||
ui.warn(_(b'repository upgraded to use share-safe mode\n')) | ||||
except error.LockError as e: | ||||
r47077 | hint = _( | |||
Matt Harbison
|
r49280 | b"see `hg help config.format.use-share-safe` for more information" | ||
r47077 | ) | |||
Pulkit Goyal
|
r47050 | if mismatch_config == b'upgrade-abort': | ||
Pulkit Goyal
|
r46856 | raise error.Abort( | ||
_(b'failed to upgrade share, got error: %s') | ||||
r47077 | % stringutil.forcebytestr(e.strerror), | |||
hint=hint, | ||||
Pulkit Goyal
|
r46856 | ) | ||
Pulkit Goyal
|
r47051 | elif mismatch_warn: | ||
Pulkit Goyal
|
r46855 | ui.warn( | ||
_(b'failed to upgrade share, got error: %s\n') | ||||
r47077 | % stringutil.forcebytestr(e.strerror), | |||
hint=hint, | ||||
Pulkit Goyal
|
r46855 | ) | ||
Pulkit Goyal
|
r46852 | finally: | ||
if wlock: | ||||
wlock.release() | ||||
Pulkit Goyal
|
r46853 | |||
def downgrade_share_to_non_safe( | ||||
ui, | ||||
hgvfs, | ||||
sharedvfs, | ||||
current_requirements, | ||||
Pulkit Goyal
|
r47050 | mismatch_config, | ||
Pulkit Goyal
|
r47051 | mismatch_warn, | ||
Pulkit Goyal
|
r46853 | ): | ||
"""Downgrades a share which use share-safe to not use it""" | ||||
wlock = None | ||||
Pulkit Goyal
|
r47047 | source_requirements = localrepo._readrequires(sharedvfs, True) | ||
Pulkit Goyal
|
r47048 | original_crequirements = current_requirements.copy() | ||
Pulkit Goyal
|
r47047 | # we cannot be 100% sure on which requirements were present in store when | ||
# the source supported share-safe. However, we do know that working | ||||
# directory requirements were not there. Hence we remove them | ||||
source_requirements -= requirementsmod.WORKING_DIR_REQUIREMENTS | ||||
current_requirements |= source_requirements | ||||
current_requirements.remove(requirementsmod.SHARESAFE_REQUIREMENT) | ||||
Pulkit Goyal
|
r47050 | if mismatch_config == b'allow': | ||
return | ||||
Pulkit Goyal
|
r47047 | |||
Pulkit Goyal
|
r46853 | try: | ||
wlock = lockmod.trylock(ui, hgvfs, b'wlock', 0, 0) | ||||
Pulkit Goyal
|
r47048 | # some process might change the requirement in between, re-read | ||
# and update current_requirements | ||||
locked_requirements = localrepo._readrequires(hgvfs, True) | ||||
if locked_requirements != original_crequirements: | ||||
removed = current_requirements - locked_requirements | ||||
# update current_requirements in place because it's passed | ||||
# as reference | ||||
current_requirements -= removed | ||||
current_requirements |= locked_requirements | ||||
current_requirements |= source_requirements | ||||
current_requirements -= set(requirementsmod.SHARESAFE_REQUIREMENT) | ||||
Pulkit Goyal
|
r46853 | scmutil.writerequires(hgvfs, current_requirements) | ||
ui.warn(_(b'repository downgraded to not use share-safe mode\n')) | ||||
except error.LockError as e: | ||||
r47077 | hint = _( | |||
Matt Harbison
|
r49280 | b"see `hg help config.format.use-share-safe` for more information" | ||
r47077 | ) | |||
Pulkit Goyal
|
r47050 | # If upgrade-abort is set, abort when upgrade fails, else let the | ||
# process continue as `upgrade-allow` is set | ||||
if mismatch_config == b'downgrade-abort': | ||||
raise error.Abort( | ||||
_(b'failed to downgrade share, got error: %s') | ||||
r47077 | % stringutil.forcebytestr(e.strerror), | |||
hint=hint, | ||||
Pulkit Goyal
|
r47050 | ) | ||
Pulkit Goyal
|
r47051 | elif mismatch_warn: | ||
ui.warn( | ||||
_(b'failed to downgrade share, got error: %s\n') | ||||
r47077 | % stringutil.forcebytestr(e.strerror), | |||
hint=hint, | ||||
Pulkit Goyal
|
r47051 | ) | ||
Pulkit Goyal
|
r46853 | finally: | ||
if wlock: | ||||
wlock.release() | ||||