##// END OF EJS Templates
run-test: keep track of the "real hg" binary before we replace it with a proxy...
run-test: keep track of the "real hg" binary before we replace it with a proxy When using `--rhg` or `--chg`, the `_hgcommand` attribute get replaced because we actually install them. This make it hard to refer to the installed hg in _installhg. So we keep track of it in a new attribute that we will use in the next few patches.

File last commit:

r53324:8c509a70 default
r53410:7fd3a930 default
Show More
command.py
193 lines | 5.6 KiB | text/x-python | PythonLexer
config: move edition code in its own module...
r53314 # Gather code related to command dealing with configuration.
from __future__ import annotations
import os
config: extra the code showing config in its own module too...
r53316 from typing import Any, Collection, Dict, Optional
config: move edition code in its own module...
r53314
from ..i18n import _
from .. import (
cmdutil,
error,
config: extra the code showing config in its own module too...
r53316 formatter,
pycompat,
config: move edition code in its own module...
r53314 requirements,
ui as uimod,
util,
)
config: gather constant and type into the `__init__.py`...
r53322 from . import (
ConfigLevelT,
EDIT_LEVELS,
LEVEL_SHARED,
config: gather the path to edit through rcutil...
r53324 NO_REPO_EDIT_LEVELS,
config: gather constant and type into the `__init__.py`...
r53322 rcutil,
)
config: move edition code in its own module...
r53314
EDIT_FLAG = 'edit'
def find_edit_level(
config: gather constant and type into the `__init__.py`...
r53322 ui: uimod.ui,
repo,
opts: Dict[str, Any],
config: move edition code in its own module...
r53314 ) -> Optional[ConfigLevelT]:
"""return the level we should edit, if any.
Parse the command option to detect when an edit is requested, and if so the
configuration level we should edit.
"""
if opts.get(EDIT_FLAG) or any(opts.get(o) for o in EDIT_LEVELS):
cmdutil.check_at_most_one_arg(opts, *EDIT_LEVELS)
for level in EDIT_LEVELS:
if opts.get(level):
return level
return EDIT_LEVELS[0]
return None
def edit_config(ui: uimod.ui, repo, level: ConfigLevelT) -> None:
"""let the user edit configuration file for the given level"""
config: gather the path to edit through rcutil...
r53324 # validate input
if repo is None and level not in NO_REPO_EDIT_LEVELS:
msg = b"can't use --%s outside a repository" % pycompat.bytestr(level)
raise error.InputError(_(msg))
if level == LEVEL_SHARED:
config: move edition code in its own module...
r53314 if not repo.shared():
config: gather the path to edit through rcutil...
r53324 msg = _(b"repository is not shared; can't use --shared")
raise error.InputError(msg)
config: move edition code in its own module...
r53314 if requirements.SHARESAFE_REQUIREMENT not in repo.requirements:
raise error.InputError(
_(
b"share safe feature not enabled; "
b"unable to edit shared source repository config"
)
)
config: gather the path to edit through rcutil...
r53324
# find rc files paths
repo_path = None
if repo is not None:
repo_path = repo.root
all_rcs = rcutil.all_rc_components(repo_path)
rc_by_level = {}
for lvl, rc_type, values in all_rcs:
if rc_type != b'path':
continue
rc_by_level.setdefault(lvl, []).append(values)
if level not in rc_by_level:
config: move edition code in its own module...
r53314 msg = 'unknown config level: %s' % level
raise error.ProgrammingError(msg)
config: gather the path to edit through rcutil...
r53324 paths = rc_by_level[level]
config: move edition code in its own module...
r53314 for f in paths:
if os.path.exists(f):
break
else:
config: gather the path to edit through rcutil...
r53324 samplehgrc = uimod.samplehgrcs.get(level)
config: move edition code in its own module...
r53314
f = paths[0]
config: gather the path to edit through rcutil...
r53324 if samplehgrc is not None:
util.writefile(f, util.tonativeeol(samplehgrc))
config: move edition code in its own module...
r53314
editor = ui.geteditor()
ui.system(
b"%s \"%s\"" % (editor, f),
onerr=error.InputError,
errprefix=_(b"edit failed"),
blockedtag=b'config_edit',
)
config: move "component display" in the new module...
r53315
def show_component(ui: uimod.ui, repo) -> None:
"""show the component used to build the config
XXX this skip over various source and ignore the repository config, so it
XXX is probably useless old code.
"""
config: include the component level when returning them...
r53323 for _lvl, t, f in rcutil.rccomponents():
config: move "component display" in the new module...
r53315 if t == b'path':
ui.debug(b'read config from: %s\n' % f)
elif t == b'resource':
ui.debug(b'read config from: resource:%s.%s\n' % (f[0], f[1]))
elif t == b'items':
# Don't print anything for 'items'.
pass
else:
raise error.ProgrammingError(b'unknown rctype: %s' % t)
config: extra the code showing config in its own module too...
r53316
def show_config(
ui: uimod.ui,
repo,
value_filters: Collection[bytes],
formatter_options: dict,
untrusted: bool = False,
all_known: bool = False,
show_source: bool = False,
) -> bool:
"""Display config value to the user
The display is done using a dedicated `formatter` object.
:value_filters:
if non-empty filter the display value according to these filters. If
the filter does not match any value, the function return False. True
otherwise.
:formatter_option:
options passed to the formatter
:untrusted:
When set, use untrusted value instead of ignoring them
:all_known:
Display all known config item, not just the one with an explicit value.
:show_source:
Show where each value has been defined.
"""
fm = ui.formatter(b'config', formatter_options)
selsections = selentries = []
filtered = False
if value_filters:
selsections = [v for v in value_filters if b'.' not in v]
selentries = [v for v in value_filters if b'.' in v]
filtered = True
uniquesel = len(selentries) == 1 and not selsections
selsections = set(selsections)
selentries = set(selentries)
matched = False
entries = ui.walkconfig(untrusted=untrusted, all_known=all_known)
for section, name, value in entries:
source = ui.configsource(section, name, untrusted)
value = pycompat.bytestr(value)
defaultvalue = ui.configdefault(section, name)
if fm.isplain():
source = source or b'none'
value = value.replace(b'\n', b'\\n')
entryname = section + b'.' + name
if filtered and not (section in selsections or entryname in selentries):
continue
fm.startitem()
fm.condwrite(show_source, b'source', b'%s: ', source)
if uniquesel:
fm.data(name=entryname)
fm.write(b'value', b'%s\n', value)
else:
fm.write(b'name value', b'%s=%s\n', entryname, value)
if formatter.isprintable(defaultvalue):
fm.data(defaultvalue=defaultvalue)
elif isinstance(defaultvalue, list) and all(
formatter.isprintable(e) for e in defaultvalue
):
fm.data(defaultvalue=fm.formatlist(defaultvalue, name=b'value'))
# TODO: no idea how to process unsupported defaultvalue types
matched = True
fm.end()
return matched