ui.py
2345 lines
| 83.1 KiB
| text/x-python
|
PythonLexer
/ mercurial / ui.py
mpm@selenic.com
|
r207 | # ui.py - user interface bits for mercurial | ||
# | ||||
Raphaël Gomès
|
r47575 | # Copyright 2005-2007 Olivia Mackall <olivia@selenic.com> | ||
mpm@selenic.com
|
r207 | # | ||
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
|
r207 | |||
Matt Harbison
|
r52756 | from __future__ import annotations | ||
Gregory Szorc
|
r25989 | |||
Simon Farnsworth
|
r30976 | import collections | ||
Kostia Balytskyi
|
r30480 | import contextlib | ||
Joerg Sonnenberger
|
r45564 | import datetime | ||
Gregory Szorc
|
r25989 | import errno | ||
Pierre-Yves David
|
r25629 | import inspect | ||
Gregory Szorc
|
r25989 | import os | ||
Matt Mackall
|
r27392 | import re | ||
Augie Fackler
|
r30992 | import signal | ||
Gregory Szorc
|
r25989 | import socket | ||
Augie Fackler
|
r30992 | import subprocess | ||
Gregory Szorc
|
r25989 | import sys | ||
import traceback | ||||
r52178 | import typing | |||
Gregory Szorc
|
r25989 | |||
Matt Harbison
|
r50689 | from typing import ( | ||
Matt Harbison
|
r50692 | Any, | ||
Callable, | ||||
Matt Harbison
|
r50690 | Dict, | ||
List, | ||||
Matt Harbison
|
r50692 | NoReturn, | ||
Matt Harbison
|
r50689 | Optional, | ||
Matt Harbison
|
r50690 | Tuple, | ||
Matt Harbison
|
r50692 | Type, | ||
TypeVar, | ||||
Matt Harbison
|
r50690 | Union, | ||
cast, | ||||
Matt Harbison
|
r50694 | overload, | ||
Matt Harbison
|
r50689 | ) | ||
Gregory Szorc
|
r25989 | from .i18n import _ | ||
from .node import hex | ||||
from . import ( | ||||
Pierre-Yves David
|
r31087 | color, | ||
Gregory Szorc
|
r25989 | config, | ||
r32984 | configitems, | |||
Pulkit Goyal
|
r30277 | encoding, | ||
Gregory Szorc
|
r25989 | error, | ||
Raphaël Gomès
|
r51658 | extensions, | ||
Gregory Szorc
|
r25989 | formatter, | ||
Yuya Nishihara
|
r41030 | loggingutil, | ||
Gregory Szorc
|
r25989 | progress, | ||
Pulkit Goyal
|
r30519 | pycompat, | ||
Gregory Szorc
|
r25989 | scmutil, | ||
util, | ||||
) | ||||
r53313 | from .configuration import rcutil | |||
Yuya Nishihara
|
r37102 | from .utils import ( | ||
dateutil, | ||||
Yuya Nishihara
|
r37137 | procutil, | ||
Matt Harbison
|
r44482 | resourceutil, | ||
Yuya Nishihara
|
r37102 | stringutil, | ||
r47668 | urlutil, | |||
Yuya Nishihara
|
r37102 | ) | ||
mpm@selenic.com
|
r207 | |||
Matt Harbison
|
r50692 | _ConfigItems = Dict[Tuple[bytes, bytes], object] # {(section, name) : value} | ||
Matt Harbison
|
r50690 | # The **opts args of the various write() methods can be basically anything, but | ||
# there's no way to express it as "anything but str". So type it to be the | ||||
# handful of known types that are used. | ||||
_MsgOpts = Union[bytes, bool, List["_PromptChoice"]] | ||||
_PromptChoice = Tuple[bytes, bytes] | ||||
Matt Harbison
|
r50692 | _Tui = TypeVar('_Tui', bound="ui") | ||
Matt Harbison
|
r50690 | |||
liscju
|
r29378 | urlreq = util.urlreq | ||
Simon Farnsworth
|
r30979 | # for use with str.translate(None, _keepalnum), to keep just alphanumerics | ||
Matt Harbison
|
r50692 | _keepalnum: bytes = b''.join( | ||
Augie Fackler
|
r43346 | c for c in map(pycompat.bytechr, range(256)) if not c.isalnum() | ||
) | ||||
Simon Farnsworth
|
r30979 | |||
Augie Fackler
|
r32872 | # The config knobs that will be altered (if unset) by ui.tweakdefaults. | ||
Matt Harbison
|
r50692 | tweakrc: bytes = b""" | ||
Augie Fackler
|
r32872 | [ui] | ||
# The rollback command is dangerous. As a rule, don't use it. | ||||
rollback = False | ||||
Martin von Zweigbergk
|
r35066 | # Make `hg status` report copy information | ||
statuscopies = yes | ||||
Augie Fackler
|
r35307 | # Prefer curses UIs when available. Revert to plain-text with `text`. | ||
interface = curses | ||||
Martin von Zweigbergk
|
r41634 | # Make compatible commands emit cwd-relative paths by default. | ||
relative-paths = yes | ||||
Augie Fackler
|
r32872 | |||
[commands] | ||||
Yuya Nishihara
|
r38674 | # Grep working directory by default. | ||
grep.all-files = True | ||||
Augie Fackler
|
r34708 | # Refuse to perform an `hg update` that would cause a file content merge | ||
update.check = noconflict | ||||
Pulkit Goyal
|
r36926 | # Show conflicts information in `hg status` | ||
status.verbose = True | ||||
Valentin Gatien-Baron
|
r42764 | # Make `hg resolve` with no action (like `-m`) fail instead of re-merging. | ||
resolve.explicit-re-merge = True | ||||
Augie Fackler
|
r32872 | |||
[diff] | ||||
git = 1 | ||||
Augie Fackler
|
r35308 | showfunc = 1 | ||
Augie Fackler
|
r38642 | word-diff = 1 | ||
Augie Fackler
|
r32872 | """ | ||
Matt Harbison
|
r50692 | samplehgrcs: Dict[bytes, bytes] = { | ||
Augie Fackler
|
r43347 | b'user': b"""# example user config (see 'hg help config' for more info) | ||
Matt Mackall
|
r22419 | [ui] | ||
# name and email, e.g. | ||||
# username = Jane Doe <jdoe@example.com> | ||||
username = | ||||
Augie Fackler
|
r34569 | # We recommend enabling tweakdefaults to get slight improvements to | ||
# the UI over time. Make sure to set HGPLAIN in the environment when | ||||
# writing scripts! | ||||
# tweakdefaults = True | ||||
Pierre-Yves David
|
r32094 | # uncomment to disable color in command output | ||
Pierre-Yves David
|
r32095 | # (see 'hg help color' for details) | ||
Pierre-Yves David
|
r32094 | # color = never | ||
Pierre-Yves David
|
r31124 | |||
Pierre-Yves David
|
r32101 | # uncomment to disable command output pagination | ||
# (see 'hg help pager' for details) | ||||
Pierre-Yves David
|
r32105 | # paginate = never | ||
Pierre-Yves David
|
r32101 | |||
Matt Mackall
|
r22419 | [extensions] | ||
Jordi Gutiérrez Hermoso
|
r42165 | # uncomment the lines below to enable some popular extensions | ||
timeless
|
r29978 | # (see 'hg help extensions' for more info) | ||
Matt Mackall
|
r22419 | # | ||
Jordi Gutiérrez Hermoso
|
r42166 | # histedit = | ||
# rebase = | ||||
# uncommit = | ||||
Pierre-Yves David
|
r32098 | """, | ||
Augie Fackler
|
r43347 | b'cloned': b"""# example repository config (see 'hg help config' for more info) | ||
Jordi Gutiérrez Hermoso
|
r22837 | [paths] | ||
default = %s | ||||
# path aliases to other clones of this repo in URLs or filesystem paths | ||||
timeless
|
r29978 | # (see 'hg help config.paths' for more info) | ||
Jordi Gutiérrez Hermoso
|
r22837 | # | ||
Rishabh Madan
|
r31064 | # default:pushurl = ssh://jdoe@example.net/hg/jdoes-fork | ||
# my-fork = ssh://jdoe@example.net/hg/jdoes-fork | ||||
# my-clone = /home/jdoe/jdoes-clone | ||||
Jordi Gutiérrez Hermoso
|
r22837 | |||
[ui] | ||||
# name and email (local to this repository, optional), e.g. | ||||
# username = Jane Doe <jdoe@example.com> | ||||
""", | ||||
Augie Fackler
|
r43347 | b'local': b"""# example repository config (see 'hg help config' for more info) | ||
Jordi Gutiérrez Hermoso
|
r22836 | [paths] | ||
# path aliases to other clones of this repo in URLs or filesystem paths | ||||
timeless
|
r29978 | # (see 'hg help config.paths' for more info) | ||
Jordi Gutiérrez Hermoso
|
r22836 | # | ||
Rishabh Madan
|
r31064 | # default = http://example.com/hg/example-repo | ||
# default:pushurl = ssh://jdoe@example.net/hg/jdoes-fork | ||||
# my-fork = ssh://jdoe@example.net/hg/jdoes-fork | ||||
# my-clone = /home/jdoe/jdoes-clone | ||||
Jordi Gutiérrez Hermoso
|
r22836 | |||
[ui] | ||||
# name and email (local to this repository, optional), e.g. | ||||
# username = Jane Doe <jdoe@example.com> | ||||
Matt Mackall
|
r22419 | """, | ||
Augie Fackler
|
r43347 | b'global': b"""# example system-wide hg config (see 'hg help config' for more info) | ||
Matt Mackall
|
r22419 | |||
Pierre-Yves David
|
r31124 | [ui] | ||
Pierre-Yves David
|
r32094 | # uncomment to disable color in command output | ||
Pierre-Yves David
|
r32095 | # (see 'hg help color' for details) | ||
Pierre-Yves David
|
r32094 | # color = never | ||
Pierre-Yves David
|
r31124 | |||
Pierre-Yves David
|
r32101 | # uncomment to disable command output pagination | ||
# (see 'hg help pager' for details) | ||||
Pierre-Yves David
|
r32105 | # paginate = never | ||
Pierre-Yves David
|
r32101 | |||
Matt Mackall
|
r22419 | [extensions] | ||
Jordi Gutiérrez Hermoso
|
r42165 | # uncomment the lines below to enable some popular extensions | ||
timeless
|
r29978 | # (see 'hg help extensions' for more info) | ||
Matt Mackall
|
r22419 | # | ||
# blackbox = | ||||
Pierre-Yves David
|
r32097 | # churn = | ||
Pierre-Yves David
|
r32098 | """, | ||
Matt Mackall
|
r22419 | } | ||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r34483 | def _maybestrurl(maybebytes): | ||
Yuya Nishihara
|
r38594 | return pycompat.rapply(pycompat.strurl, maybebytes) | ||
Augie Fackler
|
r34483 | |||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r34483 | def _maybebytesurl(maybestr): | ||
Yuya Nishihara
|
r38594 | return pycompat.rapply(pycompat.bytesurl, maybestr) | ||
Kyle Lippincott
|
r30945 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49801 | class httppasswordmgrdbproxy: | ||
Kyle Lippincott
|
r30945 | """Delays loading urllib2 until it's needed.""" | ||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r50692 | def __init__(self) -> None: | ||
Kyle Lippincott
|
r30945 | self._mgr = None | ||
def _get_mgr(self): | ||||
if self._mgr is None: | ||||
self._mgr = urlreq.httppasswordmgrwithdefaultrealm() | ||||
return self._mgr | ||||
Augie Fackler
|
r34427 | def add_password(self, realm, uris, user, passwd): | ||
Augie Fackler
|
r34483 | return self._get_mgr().add_password( | ||
Augie Fackler
|
r43346 | _maybestrurl(realm), | ||
_maybestrurl(uris), | ||||
_maybestrurl(user), | ||||
_maybestrurl(passwd), | ||||
) | ||||
Kyle Lippincott
|
r30945 | |||
Augie Fackler
|
r34427 | def find_user_password(self, realm, uri): | ||
Yuya Nishihara
|
r35918 | mgr = self._get_mgr() | ||
Augie Fackler
|
r43346 | return _maybebytesurl( | ||
mgr.find_user_password(_maybestrurl(realm), _maybestrurl(uri)) | ||||
) | ||||
Kyle Lippincott
|
r30945 | |||
Matt Harbison
|
r50692 | def _catchterm(*args) -> NoReturn: | ||
Augie Fackler
|
r30992 | raise error.SignalInterrupt | ||
Kyle Lippincott
|
r30945 | |||
Augie Fackler
|
r43346 | |||
r32958 | # unique object used to detect no default value has been provided when | |||
# retrieving configuration value. | ||||
_unset = object() | ||||
Saurabh Singh
|
r34883 | # _reqexithandlers: callbacks run at the end of a request | ||
Matt Harbison
|
r50692 | _reqexithandlers: List = [] | ||
Saurabh Singh
|
r34883 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49801 | class ui: | ||
Matt Harbison
|
r50692 | def __init__(self, src: Optional["ui"] = None) -> None: | ||
Yuya Nishihara
|
r30559 | """Create a fresh new ui object if no src given | ||
Use uimod.ui.load() to create a ui which knows global and user configs. | ||||
In most cases, you should use ui.copy() to create a copy of an existing | ||||
ui object. | ||||
""" | ||||
Pierre-Yves David
|
r21132 | # _buffers: used for temporary capture of output | ||
Matt Mackall
|
r8202 | self._buffers = [] | ||
Gregory Szorc
|
r27106 | # 3-tuple describing how each buffer in the stack behaves. | ||
# Values are (capture stderr, capture subprocesses, apply labels). | ||||
Pierre-Yves David
|
r21132 | self._bufferstates = [] | ||
Gregory Szorc
|
r27106 | # When a buffer is active, defines whether we are expanding labels. | ||
# This exists to prevent an extra list lookup. | ||||
self._bufferapplylabels = None | ||||
Bryan O'Sullivan
|
r9851 | self.quiet = self.verbose = self.debugflag = self.tracebackflag = False | ||
Matt Mackall
|
r8208 | self._reportuntrusted = True | ||
r32984 | self._knownconfig = configitems.coreitems | |||
Augie Fackler
|
r43346 | self._ocfg = config.config() # overlay | ||
self._tcfg = config.config() # trusted | ||||
self._ucfg = config.config() # untrusted | ||||
Martin Geisler
|
r8478 | self._trustusers = set() | ||
self._trustgroups = set() | ||||
Idan Kamara
|
r17048 | self.callhooks = True | ||
r48046 | # hold the root to use for each [paths] entry | |||
self._path_to_root = {} | ||||
Gregory Szorc
|
r29109 | # Insecure server connections requested. | ||
self.insecureconnections = False | ||||
Simon Farnsworth
|
r30976 | # Blocked time | ||
self.logblockedtimes = False | ||||
Pierre-Yves David
|
r31106 | # color mode: see mercurial/color.py for possible value | ||
self._colormode = None | ||||
Pierre-Yves David
|
r31113 | self._terminfoparams = {} | ||
Pierre-Yves David
|
r31115 | self._styles = {} | ||
Augie Fackler
|
r38545 | self._uninterruptible = False | ||
Joerg Sonnenberger
|
r45564 | self.showtimestamp = False | ||
Matt Mackall
|
r8136 | |||
Matt Mackall
|
r8190 | if src: | ||
Yuya Nishihara
|
r40579 | self._fout = src._fout | ||
self._ferr = src._ferr | ||||
self._fin = src._fin | ||||
Yuya Nishihara
|
r40623 | self._fmsg = src._fmsg | ||
Yuya Nishihara
|
r40580 | self._fmsgout = src._fmsgout | ||
self._fmsgerr = src._fmsgerr | ||||
Yuya Nishihara
|
r39875 | self._finoutredirected = src._finoutredirected | ||
Yuya Nishihara
|
r40761 | self._loggers = src._loggers.copy() | ||
Augie Fackler
|
r30992 | self.pageractive = src.pageractive | ||
Augie Fackler
|
r31026 | self._disablepager = src._disablepager | ||
Augie Fackler
|
r32872 | self._tweaked = src._tweaked | ||
Idan Kamara
|
r14612 | |||
Matt Mackall
|
r8203 | self._tcfg = src._tcfg.copy() | ||
self._ucfg = src._ucfg.copy() | ||||
self._ocfg = src._ocfg.copy() | ||||
Matt Mackall
|
r8201 | self._trustusers = src._trustusers.copy() | ||
self._trustgroups = src._trustgroups.copy() | ||||
Sune Foldager
|
r9887 | self.environ = src.environ | ||
Idan Kamara
|
r17048 | self.callhooks = src.callhooks | ||
r48046 | self._path_to_root = src._path_to_root | |||
Gregory Szorc
|
r29109 | self.insecureconnections = src.insecureconnections | ||
Pierre-Yves David
|
r31106 | self._colormode = src._colormode | ||
Pierre-Yves David
|
r31113 | self._terminfoparams = src._terminfoparams.copy() | ||
Pierre-Yves David
|
r31115 | self._styles = src._styles.copy() | ||
Pierre-Yves David
|
r31106 | |||
Matt Mackall
|
r8143 | self.fixconfig() | ||
liscju
|
r29378 | |||
self.httppasswordmgrdb = src.httppasswordmgrdb | ||||
Simon Farnsworth
|
r30976 | self._blockedtimes = src._blockedtimes | ||
Matt Mackall
|
r8143 | else: | ||
Yuya Nishihara
|
r40579 | self._fout = procutil.stdout | ||
self._ferr = procutil.stderr | ||||
self._fin = procutil.stdin | ||||
Yuya Nishihara
|
r40623 | self._fmsg = None | ||
Yuya Nishihara
|
r40580 | self._fmsgout = self.fout # configurable | ||
self._fmsgerr = self.ferr # configurable | ||||
Yuya Nishihara
|
r39875 | self._finoutredirected = False | ||
Yuya Nishihara
|
r40761 | self._loggers = {} | ||
Augie Fackler
|
r30992 | self.pageractive = False | ||
Augie Fackler
|
r31026 | self._disablepager = False | ||
Augie Fackler
|
r32872 | self._tweaked = False | ||
Idan Kamara
|
r14612 | |||
Sune Foldager
|
r9887 | # shared read-only environment | ||
Pulkit Goyal
|
r30637 | self.environ = encoding.environ | ||
Dirkjan Ochtman
|
r8222 | |||
Kyle Lippincott
|
r30945 | self.httppasswordmgrdb = httppasswordmgrdbproxy() | ||
Simon Farnsworth
|
r30976 | self._blockedtimes = collections.defaultdict(int) | ||
liscju
|
r29378 | |||
Augie Fackler
|
r43347 | allowed = self.configlist(b'experimental', b'exportableenviron') | ||
if b'*' in allowed: | ||||
Matt Harbison
|
r30832 | self._exportableenviron = self.environ | ||
else: | ||||
self._exportableenviron = {} | ||||
for k in allowed: | ||||
if k in self.environ: | ||||
self._exportableenviron[k] = self.environ[k] | ||||
Matt Harbison
|
r50692 | def _new_source(self) -> None: | ||
r47367 | self._ocfg.new_source() | |||
self._tcfg.new_source() | ||||
self._ucfg.new_source() | ||||
Yuya Nishihara
|
r30559 | @classmethod | ||
Matt Harbison
|
r50692 | def load(cls: Type[_Tui]) -> _Tui: | ||
Yuya Nishihara
|
r30559 | """Create a ui and load global and user configs""" | ||
u = cls() | ||||
Jun Wu
|
r31685 | # we always trust global config files and environment variables | ||
r53323 | for _lvl, t, f in rcutil.rccomponents(): | |||
Augie Fackler
|
r43347 | if t == b'path': | ||
Jun Wu
|
r31683 | u.readconfig(f, trust=True) | ||
Matt Harbison
|
r44483 | elif t == b'resource': | ||
u.read_resource_config(f, trust=True) | ||||
Augie Fackler
|
r43347 | elif t == b'items': | ||
r47367 | u._new_source() | |||
Jun Wu
|
r31685 | sections = set() | ||
for section, name, value, source in f: | ||||
# do not set u._ocfg | ||||
# XXX clean this up once immutable config object is a thing | ||||
u._tcfg.set(section, name, value, source) | ||||
u._ucfg.set(section, name, value, source) | ||||
sections.add(section) | ||||
for section in sections: | ||||
u.fixconfig(section=section) | ||||
Jun Wu
|
r31683 | else: | ||
Augie Fackler
|
r43347 | raise error.ProgrammingError(b'unknown rctype: %s' % t) | ||
Augie Fackler
|
r32872 | u._maybetweakdefaults() | ||
r47367 | u._new_source() # anything after that is a different level | |||
Yuya Nishihara
|
r30559 | return u | ||
Matt Harbison
|
r50692 | def _maybetweakdefaults(self) -> None: | ||
Augie Fackler
|
r43347 | if not self.configbool(b'ui', b'tweakdefaults'): | ||
Augie Fackler
|
r32872 | return | ||
Augie Fackler
|
r43347 | if self._tweaked or self.plain(b'tweakdefaults'): | ||
Augie Fackler
|
r32872 | return | ||
# Note: it is SUPER IMPORTANT that you set self._tweaked to | ||||
# True *before* any calls to setconfig(), otherwise you'll get | ||||
# infinite recursion between setconfig and this method. | ||||
# | ||||
# TODO: We should extract an inner method in setconfig() to | ||||
# avoid this weirdness. | ||||
self._tweaked = True | ||||
tmpcfg = config.config() | ||||
Augie Fackler
|
r43347 | tmpcfg.parse(b'<tweakdefaults>', tweakrc) | ||
Augie Fackler
|
r32872 | for section in tmpcfg: | ||
for name, value in tmpcfg.items(section): | ||||
if not self.hasconfig(section, name): | ||||
Augie Fackler
|
r43347 | self.setconfig(section, name, value, b"<tweakdefaults>") | ||
Augie Fackler
|
r32872 | |||
Matt Harbison
|
r50692 | def copy(self: _Tui) -> _Tui: | ||
Ronny Pfannschmidt
|
r8220 | return self.__class__(self) | ||
Thomas Arendsen Hein
|
r1839 | |||
Matt Harbison
|
r50692 | def resetstate(self) -> None: | ||
Yuya Nishihara
|
r29366 | """Clear internal state that shouldn't persist across commands""" | ||
if self._progbar: | ||||
self._progbar.resetstate() # reset last-print time of progress bar | ||||
Kyle Lippincott
|
r30945 | self.httppasswordmgrdb = httppasswordmgrdbproxy() | ||
Yuya Nishihara
|
r29366 | |||
Simon Farnsworth
|
r30976 | @contextlib.contextmanager | ||
Matt Harbison
|
r50692 | def timeblockedsection(self, key: bytes): | ||
Simon Farnsworth
|
r30978 | # this is open-coded below - search for timeblockedsection to find them | ||
Simon Farnsworth
|
r30976 | starttime = util.timer() | ||
try: | ||||
yield | ||||
finally: | ||||
Augie Fackler
|
r43347 | self._blockedtimes[key + b'_blocked'] += ( | ||
Augie Fackler
|
r43346 | util.timer() - starttime | ||
) * 1000 | ||||
Yuya Nishihara
|
r29366 | |||
Augie Fackler
|
r38545 | @contextlib.contextmanager | ||
Kyle Lippincott
|
r41106 | def uninterruptible(self): | ||
Augie Fackler
|
r38545 | """Mark an operation as unsafe. | ||
Most operations on a repository are safe to interrupt, but a | ||||
few are risky (for example repair.strip). This context manager | ||||
lets you advise Mercurial that something risky is happening so | ||||
that control-C etc can be blocked if desired. | ||||
""" | ||||
Augie Fackler
|
r43347 | enabled = self.configbool(b'experimental', b'nointerrupt') | ||
Augie Fackler
|
r43346 | if enabled and self.configbool( | ||
Augie Fackler
|
r43347 | b'experimental', b'nointerrupt-interactiveonly' | ||
Augie Fackler
|
r43346 | ): | ||
Augie Fackler
|
r38545 | enabled = self.interactive() | ||
if self._uninterruptible or not enabled: | ||||
# if nointerrupt support is turned off, the process isn't | ||||
Kyle Lippincott
|
r41106 | # interactive, or we're already in an uninterruptible | ||
Augie Fackler
|
r38545 | # block, do nothing. | ||
yield | ||||
return | ||||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r38545 | def warn(): | ||
Augie Fackler
|
r43347 | self.warn(_(b"shutting down cleanly\n")) | ||
Augie Fackler
|
r38545 | self.warn( | ||
Augie Fackler
|
r43347 | _(b"press ^C again to terminate immediately (dangerous)\n") | ||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r38545 | return True | ||
Augie Fackler
|
r43346 | |||
Kyle Lippincott
|
r41106 | with procutil.uninterruptible(warn): | ||
Augie Fackler
|
r38545 | try: | ||
self._uninterruptible = True | ||||
yield | ||||
finally: | ||||
self._uninterruptible = False | ||||
Matt Harbison
|
r50692 | def formatter(self, topic: bytes, opts): | ||
Yuya Nishihara
|
r32573 | return formatter.formatter(self, self, topic, opts) | ||
Matt Mackall
|
r16135 | |||
Matt Harbison
|
r50692 | def _trusted(self, fp, f: bytes) -> bool: | ||
Alexis S. L. Carvalho
|
r3677 | st = util.fstat(fp) | ||
Martin Geisler
|
r8657 | if util.isowner(st): | ||
Alexis S. L. Carvalho
|
r3677 | return True | ||
Matt Mackall
|
r8141 | |||
Matt Mackall
|
r8201 | tusers, tgroups = self._trustusers, self._trustgroups | ||
Augie Fackler
|
r43347 | if b'*' in tusers or b'*' in tgroups: | ||
Matt Mackall
|
r8141 | return True | ||
user = util.username(st.st_uid) | ||||
group = util.groupname(st.st_gid) | ||||
if user in tusers or group in tgroups or user == util.username(): | ||||
return True | ||||
Matt Mackall
|
r8204 | if self._reportuntrusted: | ||
Augie Fackler
|
r43346 | self.warn( | ||
Augie Fackler
|
r43347 | _( | ||
b'not trusting file %s from untrusted ' | ||||
b'user %s, group %s\n' | ||||
) | ||||
Augie Fackler
|
r43346 | % (f, user, group) | ||
) | ||||
Matt Mackall
|
r8141 | return False | ||
Alexis S. L. Carvalho
|
r3551 | |||
Matt Harbison
|
r44482 | def read_resource_config( | ||
self, name, root=None, trust=False, sections=None, remap=None | ||||
Matt Harbison
|
r50692 | ) -> None: | ||
Matt Harbison
|
r44482 | try: | ||
fp = resourceutil.open_resource(name[0], name[1]) | ||||
except IOError: | ||||
if not sections: # ignore unless we were looking for something | ||||
return | ||||
raise | ||||
self._readconfig( | ||||
b'resource:%s.%s' % name, fp, root, trust, sections, remap | ||||
) | ||||
Augie Fackler
|
r43346 | def readconfig( | ||
self, filename, root=None, trust=False, sections=None, remap=None | ||||
Matt Harbison
|
r50692 | ) -> None: | ||
Matt Mackall
|
r8142 | try: | ||
Augie Fackler
|
r43906 | fp = open(filename, 'rb') | ||
Matt Mackall
|
r8142 | except IOError: | ||
Augie Fackler
|
r43346 | if not sections: # ignore unless we were looking for something | ||
Matt Mackall
|
r8142 | return | ||
raise | ||||
Matt Mackall
|
r8139 | |||
Matt Harbison
|
r44482 | self._readconfig(filename, fp, root, trust, sections, remap) | ||
def _readconfig( | ||||
self, filename, fp, root=None, trust=False, sections=None, remap=None | ||||
Matt Harbison
|
r50692 | ) -> None: | ||
Martin von Zweigbergk
|
r44362 | with fp: | ||
cfg = config.config() | ||||
trusted = sections or trust or self._trusted(fp, filename) | ||||
Alexis S. L. Carvalho
|
r3552 | |||
Martin von Zweigbergk
|
r44362 | try: | ||
cfg.read(filename, fp, sections=sections, remap=remap) | ||||
Martin von Zweigbergk
|
r46506 | except error.ConfigError as inst: | ||
Martin von Zweigbergk
|
r44362 | if trusted: | ||
raise | ||||
Martin von Zweigbergk
|
r46501 | self.warn( | ||
_(b'ignored %s: %s\n') % (inst.location, inst.message) | ||||
) | ||||
Alexis S. L. Carvalho
|
r3552 | |||
Matt Harbison
|
r44585 | self._applyconfig(cfg, trusted, root) | ||
Matt Harbison
|
r50692 | def applyconfig( | ||
self, configitems: _ConfigItems, source=b"", root=None | ||||
) -> None: | ||||
Matt Harbison
|
r44585 | """Add configitems from a non-file source. Unlike with ``setconfig()``, | ||
they can be overridden by subsequent config file reads. The items are | ||||
in the same format as ``configoverride()``, namely a dict of the | ||||
following structures: {(section, name) : value} | ||||
Typically this is used by extensions that inject themselves into the | ||||
config file load procedure by monkeypatching ``localrepo.loadhgrc()``. | ||||
""" | ||||
cfg = config.config() | ||||
for (section, name), value in configitems.items(): | ||||
cfg.set(section, name, value, source) | ||||
self._applyconfig(cfg, True, root) | ||||
Matt Harbison
|
r50692 | def _applyconfig(self, cfg, trusted, root) -> None: | ||
Brodie Rao
|
r10455 | if self.plain(): | ||
Augie Fackler
|
r43346 | for k in ( | ||
Augie Fackler
|
r43347 | b'debug', | ||
b'fallbackencoding', | ||||
b'quiet', | ||||
b'slash', | ||||
b'logtemplate', | ||||
b'message-output', | ||||
b'statuscopies', | ||||
b'style', | ||||
b'traceback', | ||||
b'verbose', | ||||
Augie Fackler
|
r43346 | ): | ||
Augie Fackler
|
r43347 | if k in cfg[b'ui']: | ||
del cfg[b'ui'][k] | ||||
for k, v in cfg.items(b'defaults'): | ||||
del cfg[b'defaults'][k] | ||||
for k, v in cfg.items(b'commands'): | ||||
del cfg[b'commands'][k] | ||||
Martin von Zweigbergk
|
r46350 | for k, v in cfg.items(b'command-templates'): | ||
del cfg[b'command-templates'][k] | ||||
"Yann E. MORIN"
|
r14373 | # Don't remove aliases from the configuration if in the exceptionlist | ||
Augie Fackler
|
r43347 | if self.plain(b'alias'): | ||
for k, v in cfg.items(b'alias'): | ||||
del cfg[b'alias'][k] | ||||
if self.plain(b'revsetalias'): | ||||
for k, v in cfg.items(b'revsetalias'): | ||||
del cfg[b'revsetalias'][k] | ||||
if self.plain(b'templatealias'): | ||||
for k, v in cfg.items(b'templatealias'): | ||||
del cfg[b'templatealias'][k] | ||||
Brodie Rao
|
r10455 | |||
Matt Mackall
|
r8142 | if trusted: | ||
Matt Mackall
|
r8203 | self._tcfg.update(cfg) | ||
self._tcfg.update(self._ocfg) | ||||
self._ucfg.update(cfg) | ||||
self._ucfg.update(self._ocfg) | ||||
Matt Mackall
|
r8139 | |||
Alexis S. L. Carvalho
|
r3347 | if root is None: | ||
Augie Fackler
|
r43347 | root = os.path.expanduser(b'~') | ||
Alexis S. L. Carvalho
|
r3347 | self.fixconfig(root=root) | ||
Alexis S. L. Carvalho
|
r3014 | |||
Matt Harbison
|
r50692 | def fixconfig(self, root=None, section=None) -> None: | ||
Augie Fackler
|
r43347 | if section in (None, b'paths'): | ||
Nicolas Dumazet
|
r12764 | # expand vars and ~ | ||
# translate paths relative to root (or home) into absolute paths | ||||
Matt Harbison
|
r39843 | root = root or encoding.getcwd() | ||
Nicolas Dumazet
|
r12764 | for c in self._tcfg, self._ucfg, self._ocfg: | ||
Augie Fackler
|
r43347 | for n, p in c.items(b'paths'): | ||
r48046 | old_p = p | |||
s = self.configsource(b'paths', n) or b'none' | ||||
root_key = (n, p, s) | ||||
if root_key not in self._path_to_root: | ||||
self._path_to_root[root_key] = root | ||||
Gregory Szorc
|
r29412 | # Ignore sub-options. | ||
Augie Fackler
|
r43347 | if b':' in n: | ||
Gregory Szorc
|
r29412 | continue | ||
Nicolas Dumazet
|
r12764 | if not p: | ||
continue | ||||
Augie Fackler
|
r43347 | if b'%%' in p: | ||
r48046 | if s is None: | |||
s = 'none' | ||||
Augie Fackler
|
r43346 | self.warn( | ||
Augie Fackler
|
r43347 | _(b"(deprecated '%%' in path %s=%s from %s)\n") | ||
Augie Fackler
|
r43346 | % (n, p, s) | ||
) | ||||
Augie Fackler
|
r43347 | p = p.replace(b'%%', b'%') | ||
r48046 | if p != old_p: | |||
c.alter(b"paths", n, p) | ||||
Alexis S. L. Carvalho
|
r3347 | |||
Augie Fackler
|
r43347 | if section in (None, b'ui'): | ||
Nicolas Dumazet
|
r12764 | # update ui options | ||
Yuya Nishihara
|
r40580 | self._fmsgout, self._fmsgerr = _selectmsgdests(self) | ||
Augie Fackler
|
r43347 | self.debugflag = self.configbool(b'ui', b'debug') | ||
self.verbose = self.debugflag or self.configbool(b'ui', b'verbose') | ||||
self.quiet = not self.debugflag and self.configbool(b'ui', b'quiet') | ||||
Nicolas Dumazet
|
r12764 | if self.verbose and self.quiet: | ||
self.quiet = self.verbose = False | ||||
Augie Fackler
|
r43346 | self._reportuntrusted = self.debugflag or self.configbool( | ||
Augie Fackler
|
r43347 | b"ui", b"report_untrusted" | ||
Augie Fackler
|
r43346 | ) | ||
Joerg Sonnenberger
|
r45564 | self.showtimestamp = self.configbool(b'ui', b'timestamp-output') | ||
Augie Fackler
|
r43347 | self.tracebackflag = self.configbool(b'ui', b'traceback') | ||
self.logblockedtimes = self.configbool(b'ui', b'logblockedtimes') | ||||
Alexis S. L. Carvalho
|
r3350 | |||
Augie Fackler
|
r43347 | if section in (None, b'trusted'): | ||
Nicolas Dumazet
|
r12764 | # update trust information | ||
Augie Fackler
|
r43347 | self._trustusers.update(self.configlist(b'trusted', b'users')) | ||
self._trustgroups.update(self.configlist(b'trusted', b'groups')) | ||||
Alexis S. L. Carvalho
|
r3551 | |||
Yuya Nishihara
|
r41030 | if section in (None, b'devel', b'ui') and self.debugflag: | ||
tracked = set() | ||||
if self.configbool(b'devel', b'debug.extensions'): | ||||
tracked.add(b'extension') | ||||
if tracked: | ||||
logger = loggingutil.fileobjectlogger(self._ferr, tracked) | ||||
self.setlogger(b'debug', logger) | ||||
Pierre-Yves David
|
r15919 | def backupconfig(self, section, item): | ||
Augie Fackler
|
r43346 | return ( | ||
self._ocfg.backup(section, item), | ||||
self._tcfg.backup(section, item), | ||||
self._ucfg.backup(section, item), | ||||
) | ||||
Matt Harbison
|
r50692 | def restoreconfig(self, data) -> None: | ||
Pierre-Yves David
|
r15919 | self._ocfg.restore(data[0]) | ||
self._tcfg.restore(data[1]) | ||||
self._ucfg.restore(data[2]) | ||||
Matt Harbison
|
r50692 | def setconfig(self, section, name, value, source=b'') -> None: | ||
Mads Kiilerich
|
r20787 | for cfg in (self._ocfg, self._tcfg, self._ucfg): | ||
Mads Kiilerich
|
r20788 | cfg.set(section, name, value, source) | ||
Nicolas Dumazet
|
r12764 | self.fixconfig(section=section) | ||
Augie Fackler
|
r32872 | self._maybetweakdefaults() | ||
mpm@selenic.com
|
r960 | |||
Matt Mackall
|
r8199 | def _data(self, untrusted): | ||
Matt Mackall
|
r8203 | return untrusted and self._ucfg or self._tcfg | ||
Alexis S. L. Carvalho
|
r3552 | |||
Matt Mackall
|
r8182 | def configsource(self, section, name, untrusted=False): | ||
Yuya Nishihara
|
r30618 | return self._data(untrusted).source(section, name) | ||
Matt Mackall
|
r8182 | |||
r32958 | def config(self, section, name, default=_unset, untrusted=False): | |||
r33058 | """return the plain string version of a config""" | |||
Augie Fackler
|
r43346 | value = self._config( | ||
section, name, default=default, untrusted=untrusted | ||||
) | ||||
r33058 | if value is _unset: | |||
return None | ||||
return value | ||||
def _config(self, section, name, default=_unset, untrusted=False): | ||||
Yuya Nishihara
|
r34949 | value = itemdefault = default | ||
David Demelier
|
r33329 | item = self._knownconfig.get(section, {}).get(name) | ||
alternates = [(section, name)] | ||||
Raphaël Gomès
|
r51658 | if item is not None and item.in_core_extension is not None: | ||
# Only return the default for an in-core extension item if said | ||||
# extension is enabled | ||||
if item.in_core_extension in extensions.extensions(self): | ||||
item = None | ||||
David Demelier
|
r33329 | if item is not None: | ||
alternates.extend(item.alias) | ||||
Yuya Nishihara
|
r34949 | if callable(item.default): | ||
itemdefault = item.default() | ||||
else: | ||||
itemdefault = item.default | ||||
Boris Feld
|
r34859 | else: | ||
Augie Fackler
|
r43347 | msg = b"accessing unregistered config item: '%s.%s'" | ||
Boris Feld
|
r34859 | msg %= (section, name) | ||
Augie Fackler
|
r43347 | self.develwarn(msg, 2, b'warn-config-unknown') | ||
David Demelier
|
r33329 | |||
if default is _unset: | ||||
if item is None: | ||||
value = default | ||||
Boris Feld
|
r33471 | elif item.default is configitems.dynamicdefault: | ||
value = None | ||||
Augie Fackler
|
r43347 | msg = b"config item requires an explicit default value: '%s.%s'" | ||
Boris Feld
|
r33471 | msg %= (section, name) | ||
Augie Fackler
|
r43347 | self.develwarn(msg, 2, b'warn-config-default') | ||
David Demelier
|
r33329 | else: | ||
Yuya Nishihara
|
r34949 | value = itemdefault | ||
Augie Fackler
|
r43346 | elif ( | ||
item is not None | ||||
and item.default is not configitems.dynamicdefault | ||||
and default != itemdefault | ||||
): | ||||
msg = ( | ||||
Augie Fackler
|
r43347 | b"specifying a mismatched default value for a registered " | ||
b"config item: '%s.%s' '%s'" | ||||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r36143 | msg %= (section, name, pycompat.bytestr(default)) | ||
Augie Fackler
|
r43347 | self.develwarn(msg, 2, b'warn-config-default') | ||
r32987 | ||||
r47368 | candidates = [] | |||
config = self._data(untrusted) | ||||
David Demelier
|
r33329 | for s, n in alternates: | ||
r47368 | candidate = config.get(s, n, None) | |||
r33058 | if candidate is not None: | |||
r47368 | candidates.append((s, n, candidate)) | |||
if candidates: | ||||
def level(x): | ||||
return config.level(x[0], x[1]) | ||||
value = max(candidates, key=level)[2] | ||||
Matt Mackall
|
r15035 | |||
Matt Mackall
|
r8204 | if self.debugflag and not untrusted and self._reportuntrusted: | ||
David Demelier
|
r33329 | for s, n in alternates: | ||
uvalue = self._ucfg.get(s, n) | ||||
Augie Fackler
|
r19536 | if uvalue is not None and uvalue != value: | ||
Augie Fackler
|
r43346 | self.debug( | ||
Augie Fackler
|
r43347 | b"ignoring untrusted configuration option " | ||
b"%s.%s = %s\n" % (s, n, uvalue) | ||||
Augie Fackler
|
r43346 | ) | ||
Alexis S. L. Carvalho
|
r3552 | return value | ||
Alexis S. L. Carvalho
|
r3341 | |||
r46893 | def config_default(self, section, name): | |||
"""return the default value for a config option | ||||
The default is returned "raw", for example if it is a callable, the | ||||
callable was not called. | ||||
""" | ||||
item = self._knownconfig.get(section, {}).get(name) | ||||
if item is None: | ||||
raise KeyError((section, name)) | ||||
return item.default | ||||
r32967 | def configsuboptions(self, section, name, default=_unset, untrusted=False): | |||
Gregory Szorc
|
r27252 | """Get a config option and all sub-options. | ||
Some config options have sub-options that are declared with the | ||||
format "key:opt = value". This method is used to return the main | ||||
option and all its declared sub-options. | ||||
Returns a 2-tuple of ``(option, sub-options)``, where `sub-options`` | ||||
is a dict of defined sub-options where keys and values are strings. | ||||
""" | ||||
r32966 | main = self.config(section, name, default, untrusted=untrusted) | |||
Gregory Szorc
|
r27252 | data = self._data(untrusted) | ||
sub = {} | ||||
Augie Fackler
|
r43347 | prefix = b'%s:' % name | ||
Gregory Szorc
|
r27252 | for k, v in data.items(section): | ||
if k.startswith(prefix): | ||||
Augie Fackler
|
r43346 | sub[k[len(prefix) :]] = v | ||
Gregory Szorc
|
r27252 | |||
if self.debugflag and not untrusted and self._reportuntrusted: | ||||
for k, v in sub.items(): | ||||
Augie Fackler
|
r43347 | uvalue = self._ucfg.get(section, b'%s:%s' % (name, k)) | ||
Gregory Szorc
|
r27252 | if uvalue is not None and uvalue != v: | ||
Augie Fackler
|
r43346 | self.debug( | ||
Augie Fackler
|
r43347 | b'ignoring untrusted configuration option ' | ||
b'%s:%s.%s = %s\n' % (section, name, k, uvalue) | ||||
Augie Fackler
|
r43346 | ) | ||
Gregory Szorc
|
r27252 | |||
return main, sub | ||||
r32965 | def configpath(self, section, name, default=_unset, untrusted=False): | |||
Matt Harbison
|
r44226 | """get a path config item, expanded relative to repo root or config | ||
file""" | ||||
Matt Mackall
|
r13238 | v = self.config(section, name, default, untrusted) | ||
Simon Heimberg
|
r14923 | if v is None: | ||
return None | ||||
Augie Fackler
|
r43347 | if not os.path.isabs(v) or b"://" not in v: | ||
Matt Mackall
|
r13238 | src = self.configsource(section, name, untrusted) | ||
Augie Fackler
|
r43347 | if b':' in src: | ||
base = os.path.dirname(src.rsplit(b':')[0]) | ||||
Matt Mackall
|
r13238 | v = os.path.join(base, os.path.expanduser(v)) | ||
return v | ||||
r32959 | def configbool(self, section, name, default=_unset, untrusted=False): | |||
Sune Foldager
|
r14171 | """parse a configuration element as a boolean | ||
Yuya Nishihara
|
r34133 | >>> u = ui(); s = b'foo' | ||
>>> u.setconfig(s, b'true', b'yes') | ||||
>>> u.configbool(s, b'true') | ||||
Sune Foldager
|
r14171 | True | ||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'false', b'no') | ||
>>> u.configbool(s, b'false') | ||||
Sune Foldager
|
r14171 | False | ||
Yuya Nishihara
|
r34133 | >>> u.configbool(s, b'unknown') | ||
Sune Foldager
|
r14171 | False | ||
Yuya Nishihara
|
r34133 | >>> u.configbool(s, b'unknown', True) | ||
Sune Foldager
|
r14171 | True | ||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'invalid', b'somevalue') | ||
>>> u.configbool(s, b'invalid') | ||||
Sune Foldager
|
r14171 | Traceback (most recent call last): | ||
... | ||||
ConfigError: foo.invalid is not a boolean ('somevalue') | ||||
""" | ||||
r33059 | v = self._config(section, name, default, untrusted=untrusted) | |||
Martin Geisler
|
r8527 | if v is None: | ||
r33059 | return v | |||
if v is _unset: | ||||
r32959 | if default is _unset: | |||
return False | ||||
Matt Mackall
|
r8144 | return default | ||
Dirkjan Ochtman
|
r10243 | if isinstance(v, bool): | ||
return v | ||||
Yuya Nishihara
|
r37102 | b = stringutil.parsebool(v) | ||
Augie Fackler
|
r12087 | if b is None: | ||
Augie Fackler
|
r43346 | raise error.ConfigError( | ||
Augie Fackler
|
r43347 | _(b"%s.%s is not a boolean ('%s')") % (section, name, v) | ||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r12087 | return b | ||
Alexis S. L. Carvalho
|
r3552 | |||
Augie Fackler
|
r43346 | def configwith( | ||
self, convert, section, name, default=_unset, desc=None, untrusted=False | ||||
): | ||||
Bryan O'Sullivan
|
r30926 | """parse a configuration element with a conversion function | ||
Yuya Nishihara
|
r34133 | >>> u = ui(); s = b'foo' | ||
>>> u.setconfig(s, b'float1', b'42') | ||||
>>> u.configwith(float, s, b'float1') | ||||
Bryan O'Sullivan
|
r30926 | 42.0 | ||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'float2', b'-4.25') | ||
>>> u.configwith(float, s, b'float2') | ||||
Jun Wu
|
r30932 | -4.25 | ||
Yuya Nishihara
|
r34133 | >>> u.configwith(float, s, b'unknown', 7) | ||
r32960 | 7.0 | |||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'invalid', b'somevalue') | ||
>>> u.configwith(float, s, b'invalid') | ||||
Bryan O'Sullivan
|
r30926 | Traceback (most recent call last): | ||
... | ||||
ConfigError: foo.invalid is not a valid float ('somevalue') | ||||
Yuya Nishihara
|
r34133 | >>> u.configwith(float, s, b'invalid', desc=b'womble') | ||
Bryan O'Sullivan
|
r30926 | Traceback (most recent call last): | ||
... | ||||
ConfigError: foo.invalid is not a valid womble ('somevalue') | ||||
""" | ||||
r32960 | v = self.config(section, name, default, untrusted) | |||
Bryan O'Sullivan
|
r30926 | if v is None: | ||
Augie Fackler
|
r43346 | return v # do not attempt to convert None | ||
Bryan O'Sullivan
|
r30926 | try: | ||
return convert(v) | ||||
Boris Feld
|
r32462 | except (ValueError, error.ParseError): | ||
Bryan O'Sullivan
|
r30926 | if desc is None: | ||
Yuya Nishihara
|
r34205 | desc = pycompat.sysbytes(convert.__name__) | ||
Augie Fackler
|
r43346 | raise error.ConfigError( | ||
Augie Fackler
|
r43347 | _(b"%s.%s is not a valid %s ('%s')") % (section, name, desc, v) | ||
Augie Fackler
|
r43346 | ) | ||
Bryan O'Sullivan
|
r30926 | |||
r32961 | def configint(self, section, name, default=_unset, untrusted=False): | |||
Sune Foldager
|
r14171 | """parse a configuration element as an integer | ||
Yuya Nishihara
|
r34133 | >>> u = ui(); s = b'foo' | ||
>>> u.setconfig(s, b'int1', b'42') | ||||
>>> u.configint(s, b'int1') | ||||
Sune Foldager
|
r14171 | 42 | ||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'int2', b'-42') | ||
>>> u.configint(s, b'int2') | ||||
Sune Foldager
|
r14171 | -42 | ||
Yuya Nishihara
|
r34133 | >>> u.configint(s, b'unknown', 7) | ||
Sune Foldager
|
r14171 | 7 | ||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'invalid', b'somevalue') | ||
>>> u.configint(s, b'invalid') | ||||
Sune Foldager
|
r14171 | Traceback (most recent call last): | ||
... | ||||
Bryan O'Sullivan
|
r30927 | ConfigError: foo.invalid is not a valid integer ('somevalue') | ||
Sune Foldager
|
r14171 | """ | ||
Augie Fackler
|
r43346 | return self.configwith( | ||
Augie Fackler
|
r43347 | int, section, name, default, b'integer', untrusted | ||
Augie Fackler
|
r43346 | ) | ||
Sune Foldager
|
r14171 | |||
r32962 | def configbytes(self, section, name, default=_unset, untrusted=False): | |||
Bryan O'Sullivan
|
r19065 | """parse a configuration element as a quantity in bytes | ||
Units can be specified as b (bytes), k or kb (kilobytes), m or | ||||
mb (megabytes), g or gb (gigabytes). | ||||
Yuya Nishihara
|
r34133 | >>> u = ui(); s = b'foo' | ||
>>> u.setconfig(s, b'val1', b'42') | ||||
>>> u.configbytes(s, b'val1') | ||||
Bryan O'Sullivan
|
r19065 | 42 | ||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'val2', b'42.5 kb') | ||
>>> u.configbytes(s, b'val2') | ||||
Bryan O'Sullivan
|
r19065 | 43520 | ||
Yuya Nishihara
|
r34133 | >>> u.configbytes(s, b'unknown', b'7 MB') | ||
Bryan O'Sullivan
|
r19065 | 7340032 | ||
Yuya Nishihara
|
r34133 | >>> u.setconfig(s, b'invalid', b'somevalue') | ||
>>> u.configbytes(s, b'invalid') | ||||
Bryan O'Sullivan
|
r19065 | Traceback (most recent call last): | ||
... | ||||
ConfigError: foo.invalid is not a byte quantity ('somevalue') | ||||
""" | ||||
r33060 | value = self._config(section, name, default, untrusted) | |||
if value is _unset: | ||||
r32962 | if default is _unset: | |||
default = 0 | ||||
Bryan O'Sullivan
|
r19195 | value = default | ||
Augie Fackler
|
r33586 | if not isinstance(value, bytes): | ||
r32962 | return value | |||
Bryan O'Sullivan
|
r19065 | try: | ||
Bryan O'Sullivan
|
r19195 | return util.sizetoint(value) | ||
except error.ParseError: | ||||
Augie Fackler
|
r43346 | raise error.ConfigError( | ||
Augie Fackler
|
r43347 | _(b"%s.%s is not a byte quantity ('%s')") | ||
Augie Fackler
|
r43346 | % (section, name, value) | ||
) | ||||
Bryan O'Sullivan
|
r19065 | |||
r32963 | def configlist(self, section, name, default=_unset, untrusted=False): | |||
Sune Foldager
|
r14171 | """parse a configuration element as a list of comma/space separated | ||
strings | ||||
Yuya Nishihara
|
r34133 | >>> u = ui(); s = b'foo' | ||
>>> u.setconfig(s, b'list1', b'this,is "a small" ,test') | ||||
>>> u.configlist(s, b'list1') | ||||
Sune Foldager
|
r14171 | ['this', 'is', 'a small', 'test'] | ||
Augie Fackler
|
r34958 | >>> u.setconfig(s, b'list2', b'this, is "a small" , test ') | ||
>>> u.configlist(s, b'list2') | ||||
['this', 'is', 'a small', 'test'] | ||||
Sune Foldager
|
r14171 | """ | ||
Jun Wu
|
r31481 | # default is not always a list | ||
Augie Fackler
|
r43346 | v = self.configwith( | ||
r47960 | stringutil.parselist, section, name, default, b'list', untrusted | |||
Augie Fackler
|
r43346 | ) | ||
r32963 | if isinstance(v, bytes): | |||
r47960 | return stringutil.parselist(v) | |||
r32963 | elif v is None: | |||
return [] | ||||
return v | ||||
Thomas Arendsen Hein
|
r2499 | |||
r32964 | def configdate(self, section, name, default=_unset, untrusted=False): | |||
Boris Feld
|
r32408 | """parse a configuration element as a tuple of ints | ||
Yuya Nishihara
|
r34133 | >>> u = ui(); s = b'foo' | ||
>>> u.setconfig(s, b'date', b'0 0') | ||||
>>> u.configdate(s, b'date') | ||||
Boris Feld
|
r32408 | (0, 0) | ||
""" | ||||
if self.config(section, name, default, untrusted): | ||||
Augie Fackler
|
r43346 | return self.configwith( | ||
Augie Fackler
|
r43347 | dateutil.parsedate, section, name, default, b'date', untrusted | ||
Augie Fackler
|
r43346 | ) | ||
r32964 | if default is _unset: | |||
return None | ||||
Boris Feld
|
r32408 | return default | ||
Navaneeth Suresh
|
r42899 | def configdefault(self, section, name): | ||
"""returns the default value of the config item""" | ||||
item = self._knownconfig.get(section, {}).get(name) | ||||
itemdefault = None | ||||
if item is not None: | ||||
if callable(item.default): | ||||
itemdefault = item.default() | ||||
else: | ||||
itemdefault = item.default | ||||
return itemdefault | ||||
Bryan O'Sullivan
|
r27696 | def hasconfig(self, section, name, untrusted=False): | ||
return self._data(untrusted).hasitem(section, name) | ||||
Bryan O'Sullivan
|
r4487 | def has_section(self, section, untrusted=False): | ||
Vadim Gelfer
|
r2343 | '''tell whether section exists in config.''' | ||
Matt Mackall
|
r8199 | return section in self._data(untrusted) | ||
Alexis S. L. Carvalho
|
r3552 | |||
Gregory Szorc
|
r27253 | def configitems(self, section, untrusted=False, ignoresub=False): | ||
Matt Mackall
|
r8199 | items = self._data(untrusted).items(section) | ||
Gregory Szorc
|
r27253 | if ignoresub: | ||
Augie Fackler
|
r43347 | items = [i for i in items if b':' not in i[0]] | ||
Matt Mackall
|
r8204 | if self.debugflag and not untrusted and self._reportuntrusted: | ||
Dirkjan Ochtman
|
r8222 | for k, v in self._ucfg.items(section): | ||
Matt Mackall
|
r8203 | if self._tcfg.get(section, k) != v: | ||
Augie Fackler
|
r43346 | self.debug( | ||
Augie Fackler
|
r43347 | b"ignoring untrusted configuration option " | ||
b"%s.%s = %s\n" % (section, k, v) | ||||
Augie Fackler
|
r43346 | ) | ||
Matt Mackall
|
r8144 | return items | ||
mpm@selenic.com
|
r285 | |||
r48208 | def walkconfig(self, untrusted=False, all_known=False): | |||
defined = self._walk_config(untrusted) | ||||
if not all_known: | ||||
for d in defined: | ||||
yield d | ||||
return | ||||
known = self._walk_known() | ||||
current_defined = next(defined, None) | ||||
current_known = next(known, None) | ||||
while current_defined is not None or current_known is not None: | ||||
if current_defined is None: | ||||
yield current_known | ||||
current_known = next(known, None) | ||||
elif current_known is None: | ||||
yield current_defined | ||||
current_defined = next(defined, None) | ||||
elif current_known[0:2] == current_defined[0:2]: | ||||
yield current_defined | ||||
current_defined = next(defined, None) | ||||
current_known = next(known, None) | ||||
elif current_known[0:2] < current_defined[0:2]: | ||||
yield current_known | ||||
current_known = next(known, None) | ||||
else: | ||||
yield current_defined | ||||
current_defined = next(defined, None) | ||||
def _walk_known(self): | ||||
for section, items in sorted(self._knownconfig.items()): | ||||
for k, i in sorted(items.items()): | ||||
# We don't have a way to display generic well, so skip them | ||||
if i.generic: | ||||
continue | ||||
if callable(i.default): | ||||
default = i.default() | ||||
elif i.default is configitems.dynamicdefault: | ||||
default = b'<DYNAMIC>' | ||||
else: | ||||
default = i.default | ||||
yield section, i.name, default | ||||
def _walk_config(self, untrusted): | ||||
Matt Mackall
|
r8203 | cfg = self._data(untrusted) | ||
for section in cfg.sections(): | ||||
Alexis S. L. Carvalho
|
r3552 | for name, value in self.configitems(section, untrusted): | ||
Martin Geisler
|
r13576 | yield section, name, value | ||
Bryan O'Sullivan
|
r1028 | |||
Matt Harbison
|
r50692 | def plain(self, feature: Optional[bytes] = None) -> bool: | ||
Augie Fackler
|
r46554 | """is plain mode active? | ||
Dan Villiom Podlaski Christiansen
|
r11325 | |||
Brodie Rao
|
r13849 | Plain mode means that all configuration variables which affect | ||
the behavior and output of Mercurial should be | ||||
ignored. Additionally, the output should be stable, | ||||
reproducible and suitable for use in scripts or applications. | ||||
The only way to trigger plain mode is by setting either the | ||||
`HGPLAIN' or `HGPLAINEXCEPT' environment variables. | ||||
Dan Villiom Podlaski Christiansen
|
r11325 | |||
"Yann E. MORIN"
|
r14372 | The return value can either be | ||
- False if HGPLAIN is not set, or feature is in HGPLAINEXCEPT | ||||
Yuya Nishihara
|
r35180 | - False if feature is disabled by default and not included in HGPLAIN | ||
"Yann E. MORIN"
|
r14372 | - True otherwise | ||
Augie Fackler
|
r46554 | """ | ||
Augie Fackler
|
r43346 | if ( | ||
Augie Fackler
|
r43347 | b'HGPLAIN' not in encoding.environ | ||
and b'HGPLAINEXCEPT' not in encoding.environ | ||||
Augie Fackler
|
r43346 | ): | ||
Brodie Rao
|
r13849 | return False | ||
Augie Fackler
|
r43346 | exceptions = ( | ||
Augie Fackler
|
r43347 | encoding.environ.get(b'HGPLAINEXCEPT', b'').strip().split(b',') | ||
Augie Fackler
|
r43346 | ) | ||
Yuya Nishihara
|
r35180 | # TODO: add support for HGPLAIN=+feature,-feature syntax | ||
Augie Fackler
|
r43347 | if b'+strictflags' not in encoding.environ.get(b'HGPLAIN', b'').split( | ||
b',' | ||||
): | ||||
exceptions.append(b'strictflags') | ||||
"Yann E. MORIN"
|
r14372 | if feature and exceptions: | ||
return feature not in exceptions | ||||
return True | ||||
Brodie Rao
|
r10455 | |||
Boris Feld
|
r34850 | def username(self, acceptempty=False): | ||
Thomas Arendsen Hein
|
r1985 | """Return default username to be used in commits. | ||
Searched in this order: $HGUSER, [ui] section of hgrcs, $EMAIL | ||||
and stop searching if one of these is set. | ||||
Boris Feld
|
r34850 | If not found and acceptempty is True, returns None. | ||
Benoit Boissinot
|
r6862 | If not found and ui.askusername is True, ask the user, else use | ||
($LOGNAME or $USER or $LNAME or $USERNAME) + "@full.hostname". | ||||
Boris Feld
|
r34850 | If no username could be found, raise an Abort error. | ||
Thomas Arendsen Hein
|
r1985 | """ | ||
Augie Fackler
|
r43347 | user = encoding.environ.get(b"HGUSER") | ||
Thomas Arendsen Hein
|
r1985 | if user is None: | ||
Augie Fackler
|
r43347 | user = self.config(b"ui", b"username") | ||
Chad Dombrova
|
r11225 | if user is not None: | ||
user = os.path.expandvars(user) | ||||
Thomas Arendsen Hein
|
r1985 | if user is None: | ||
Augie Fackler
|
r43347 | user = encoding.environ.get(b"EMAIL") | ||
Boris Feld
|
r34850 | if user is None and acceptempty: | ||
return user | ||||
Augie Fackler
|
r43347 | if user is None and self.configbool(b"ui", b"askusername"): | ||
user = self.prompt(_(b"enter a commit username:"), default=None) | ||||
Martin Geisler
|
r9613 | if user is None and not self.interactive(): | ||
Benoit Boissinot
|
r3721 | try: | ||
Augie Fackler
|
r43347 | user = b'%s@%s' % ( | ||
Augie Fackler
|
r43346 | procutil.getuser(), | ||
encoding.strtolocal(socket.getfqdn()), | ||||
) | ||||
Augie Fackler
|
r43347 | self.warn(_(b"no username found, using '%s' instead\n") % user) | ||
Benoit Boissinot
|
r3721 | except KeyError: | ||
Thomas Arendsen Hein
|
r4044 | pass | ||
if not user: | ||||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Augie Fackler
|
r43347 | _(b'no username supplied'), | ||
hint=_(b"use 'hg config --edit' " b'to set your username'), | ||||
Augie Fackler
|
r43346 | ) | ||
Augie Fackler
|
r43347 | if b"\n" in user: | ||
Augie Fackler
|
r43346 | raise error.Abort( | ||
Augie Fackler
|
r43347 | _(b"username %r contains a newline\n") % pycompat.bytestr(user) | ||
Augie Fackler
|
r43346 | ) | ||
Thomas Arendsen Hein
|
r1985 | return user | ||
Matt Mackall
|
r608 | |||
Matt Harbison
|
r50692 | def shortuser(self, user: bytes) -> bytes: | ||
Thomas Arendsen Hein
|
r1129 | """Return a short representation of a user name or email address.""" | ||
Matt Mackall
|
r10282 | if not self.verbose: | ||
Yuya Nishihara
|
r37102 | user = stringutil.shortuser(user) | ||
Thomas Arendsen Hein
|
r1129 | return user | ||
Gregory Szorc
|
r24250 | @util.propertycache | ||
def paths(self): | ||||
r47668 | return urlutil.paths(self) | |||
mpm@selenic.com
|
r506 | |||
Yuya Nishihara
|
r40579 | @property | ||
def fout(self): | ||||
return self._fout | ||||
r51339 | @util.propertycache | |||
def _fout_is_a_tty(self): | ||||
self._isatty(self._fout) | ||||
Yuya Nishihara
|
r40579 | @fout.setter | ||
def fout(self, f): | ||||
self._fout = f | ||||
Yuya Nishihara
|
r40580 | self._fmsgout, self._fmsgerr = _selectmsgdests(self) | ||
r51339 | if '_fout_is_a_tty' in vars(self): | |||
del self._fout_is_a_tty | ||||
Yuya Nishihara
|
r40579 | |||
@property | ||||
def ferr(self): | ||||
return self._ferr | ||||
@ferr.setter | ||||
def ferr(self, f): | ||||
self._ferr = f | ||||
Yuya Nishihara
|
r40580 | self._fmsgout, self._fmsgerr = _selectmsgdests(self) | ||
Yuya Nishihara
|
r40579 | |||
@property | ||||
def fin(self): | ||||
return self._fin | ||||
@fin.setter | ||||
def fin(self, f): | ||||
self._fin = f | ||||
Yuya Nishihara
|
r40623 | @property | ||
def fmsg(self): | ||||
"""Stream dedicated for status/error messages; may be None if | ||||
fout/ferr are used""" | ||||
return self._fmsg | ||||
@fmsg.setter | ||||
def fmsg(self, f): | ||||
self._fmsg = f | ||||
self._fmsgout, self._fmsgerr = _selectmsgdests(self) | ||||
Martin von Zweigbergk
|
r48230 | @contextlib.contextmanager | ||
Matt Harbison
|
r50692 | def silent( | ||
self, error: bool = False, subproc: bool = False, labeled: bool = False | ||||
): | ||||
Martin von Zweigbergk
|
r48230 | self.pushbuffer(error=error, subproc=subproc, labeled=labeled) | ||
try: | ||||
yield | ||||
finally: | ||||
self.popbuffer() | ||||
Matt Harbison
|
r50692 | def pushbuffer( | ||
self, error: bool = False, subproc: bool = False, labeled: bool = False | ||||
) -> None: | ||||
Mads Kiilerich
|
r23139 | """install a buffer to capture standard output of the ui object | ||
Pierre-Yves David
|
r21132 | |||
Pierre-Yves David
|
r24848 | If error is True, the error output will be captured too. | ||
If subproc is True, output from subprocesses (typically hooks) will be | ||||
Gregory Szorc
|
r27106 | captured too. | ||
Brodie Rao
|
r10815 | |||
If labeled is True, any labels associated with buffered | ||||
output will be handled. By default, this has no effect | ||||
on the output returned, but extensions and GUI tools may | ||||
handle this argument and returned styled output. If output | ||||
is being buffered so it can be captured and parsed or | ||||
processed, labeled should not be set to True. | ||||
Gregory Szorc
|
r27106 | """ | ||
Matt Mackall
|
r8202 | self._buffers.append([]) | ||
Gregory Szorc
|
r27106 | self._bufferstates.append((error, subproc, labeled)) | ||
self._bufferapplylabels = labeled | ||||
Matt Mackall
|
r3737 | |||
Matt Harbison
|
r50692 | def popbuffer(self) -> bytes: | ||
Gregory Szorc
|
r27109 | '''pop the last buffer and return the buffered output''' | ||
Pierre-Yves David
|
r21132 | self._bufferstates.pop() | ||
Gregory Szorc
|
r27106 | if self._bufferstates: | ||
self._bufferapplylabels = self._bufferstates[-1][2] | ||||
else: | ||||
self._bufferapplylabels = None | ||||
Augie Fackler
|
r43347 | return b"".join(self._buffers.pop()) | ||
Matt Mackall
|
r3737 | |||
Matt Harbison
|
r50692 | def _isbuffered(self, dest) -> bool: | ||
Yuya Nishihara
|
r40579 | if dest is self._fout: | ||
Yuya Nishihara
|
r40577 | return bool(self._buffers) | ||
Yuya Nishihara
|
r40579 | if dest is self._ferr: | ||
Yuya Nishihara
|
r40577 | return bool(self._bufferstates and self._bufferstates[-1][0]) | ||
return False | ||||
Matt Harbison
|
r50692 | def canwritewithoutlabels(self) -> bool: | ||
Joerg Sonnenberger
|
r35979 | '''check if write skips the label''' | ||
if self._buffers and not self._bufferapplylabels: | ||||
return True | ||||
return self._colormode is None | ||||
Matt Harbison
|
r50692 | def canbatchlabeledwrites(self) -> bool: | ||
Joerg Sonnenberger
|
r35979 | '''check if write calls with labels are batchable''' | ||
# Windows color printing is special, see ``write``. | ||||
Augie Fackler
|
r43347 | return self._colormode != b'win32' | ||
Joerg Sonnenberger
|
r35979 | |||
Matt Harbison
|
r50690 | def write(self, *args: bytes, **opts: _MsgOpts) -> None: | ||
Augie Fackler
|
r46554 | """write args to output | ||
Brodie Rao
|
r10815 | |||
Pierre-Yves David
|
r31091 | By default, this method simply writes to the buffer or stdout. | ||
Color mode can be set on the UI class to have the output decorated | ||||
with color modifier before being written to stdout. | ||||
Brodie Rao
|
r10815 | |||
Pierre-Yves David
|
r31091 | The color used is controlled by an optional keyword argument, "label". | ||
This should be a string containing label names separated by space. | ||||
Label names take the form of "topic.type". For example, ui.debug() | ||||
issues a label of "ui.debug". | ||||
Brodie Rao
|
r10815 | |||
Joerg Sonnenberger
|
r43559 | Progress reports via stderr are normally cleared before writing as | ||
stdout and stderr go to the same terminal. This can be skipped with | ||||
the optional keyword argument "keepprogressbar". The progress bar | ||||
will continue to occupy a partial line on stderr in that case. | ||||
This functionality is intended when Mercurial acts as data source | ||||
in a pipe. | ||||
Brodie Rao
|
r10815 | When labeling output for a specific command, a label of | ||
"cmdname.type" is recommended. For example, status issues | ||||
a label of "status.modified" for modified files. | ||||
Augie Fackler
|
r46554 | """ | ||
Yuya Nishihara
|
r41376 | dest = self._fout | ||
# inlined _write() for speed | ||||
Yuya Nishihara
|
r41378 | if self._buffers: | ||
Augie Fackler
|
r43906 | label = opts.get('label', b'') | ||
Yuya Nishihara
|
r41376 | if label and self._bufferapplylabels: | ||
self._buffers[-1].extend(self.label(a, label) for a in args) | ||||
else: | ||||
self._buffers[-1].extend(args) | ||||
Yuya Nishihara
|
r41377 | return | ||
Joerg Sonnenberger
|
r43559 | # inlined _writenobuf() for speed | ||
r51339 | if not opts.get('keepprogressbar', self._fout_is_a_tty): | |||
Joerg Sonnenberger
|
r43559 | self._progclear() | ||
Yuya Nishihara
|
r41377 | msg = b''.join(args) | ||
# opencode timeblockedsection because this is a critical path | ||||
starttime = util.timer() | ||||
try: | ||||
Augie Fackler
|
r43347 | if self._colormode == b'win32': | ||
Yuya Nishihara
|
r41377 | # windows color printing is its own can of crab, defer to | ||
# the color module and that is it. | ||||
color.win32print(self, dest.write, msg, **opts) | ||||
else: | ||||
if self._colormode is not None: | ||||
Augie Fackler
|
r43906 | label = opts.get('label', b'') | ||
Yuya Nishihara
|
r41377 | msg = self.label(msg, label) | ||
dest.write(msg) | ||||
except IOError as err: | ||||
raise error.StdioError(err) | ||||
finally: | ||||
Augie Fackler
|
r43347 | self._blockedtimes[b'stdio_blocked'] += ( | ||
Augie Fackler
|
r43346 | util.timer() - starttime | ||
) * 1000 | ||||
Yuya Nishihara
|
r40577 | |||
Matt Harbison
|
r50690 | def write_err(self, *args: bytes, **opts: _MsgOpts) -> None: | ||
Yuya Nishihara
|
r40579 | self._write(self._ferr, *args, **opts) | ||
Yuya Nishihara
|
r40577 | |||
Matt Harbison
|
r50690 | def _write(self, dest, *args: bytes, **opts: _MsgOpts) -> None: | ||
Yuya Nishihara
|
r41376 | # update write() as well if you touch this code | ||
Yuya Nishihara
|
r40577 | if self._isbuffered(dest): | ||
Augie Fackler
|
r43906 | label = opts.get('label', b'') | ||
Yuya Nishihara
|
r41375 | if label and self._bufferapplylabels: | ||
Pierre-Yves David
|
r31091 | self._buffers[-1].extend(self.label(a, label) for a in args) | ||
else: | ||||
self._buffers[-1].extend(args) | ||||
Yuya Nishihara
|
r35975 | else: | ||
Yuya Nishihara
|
r40577 | self._writenobuf(dest, *args, **opts) | ||
Yuya Nishihara
|
r35975 | |||
Matt Harbison
|
r50690 | def _writenobuf(self, dest, *args: bytes, **opts: _MsgOpts) -> None: | ||
Yuya Nishihara
|
r41377 | # update write() as well if you touch this code | ||
r51339 | if not opts.get('keepprogressbar', self._fout_is_a_tty): | |||
Joerg Sonnenberger
|
r43559 | self._progclear() | ||
Yuya Nishihara
|
r40557 | msg = b''.join(args) | ||
Pierre-Yves David
|
r31090 | |||
Yuya Nishihara
|
r31128 | # opencode timeblockedsection because this is a critical path | ||
starttime = util.timer() | ||||
try: | ||||
Yuya Nishihara
|
r40579 | if dest is self._ferr and not getattr(self._fout, 'closed', False): | ||
self._fout.flush() | ||||
Yuya Nishihara
|
r40625 | if getattr(dest, 'structured', False): | ||
# channel for machine-readable output with metadata, where | ||||
# no extra colorization is necessary. | ||||
dest.write(msg, **opts) | ||||
Augie Fackler
|
r43347 | elif self._colormode == b'win32': | ||
Yuya Nishihara
|
r40558 | # windows color printing is its own can of crab, defer to | ||
# the color module and that is it. | ||||
Yuya Nishihara
|
r40576 | color.win32print(self, dest.write, msg, **opts) | ||
Yuya Nishihara
|
r40558 | else: | ||
if self._colormode is not None: | ||||
Augie Fackler
|
r43906 | label = opts.get('label', b'') | ||
Yuya Nishihara
|
r40558 | msg = self.label(msg, label) | ||
Yuya Nishihara
|
r40576 | dest.write(msg) | ||
Yuya Nishihara
|
r40575 | # stderr may be buffered under win32 when redirected to files, | ||
# including stdout. | ||||
Manuel Jacob
|
r45525 | if dest is self._ferr and not getattr(dest, 'closed', False): | ||
Yuya Nishihara
|
r40575 | dest.flush() | ||
Bryan O'Sullivan
|
r31961 | except IOError as err: | ||
Augie Fackler
|
r43346 | if dest is self._ferr and err.errno in ( | ||
errno.EPIPE, | ||||
errno.EIO, | ||||
errno.EBADF, | ||||
): | ||||
Yuya Nishihara
|
r40575 | # no way to report the error, so ignore it | ||
return | ||||
Bryan O'Sullivan
|
r31961 | raise error.StdioError(err) | ||
Yuya Nishihara
|
r31128 | finally: | ||
Augie Fackler
|
r43347 | self._blockedtimes[b'stdio_blocked'] += ( | ||
Augie Fackler
|
r43346 | util.timer() - starttime | ||
) * 1000 | ||||
mpm@selenic.com
|
r565 | |||
Matt Harbison
|
r50690 | def _writemsg(self, dest, *args: bytes, **opts: _MsgOpts) -> None: | ||
Joerg Sonnenberger
|
r45564 | timestamp = self.showtimestamp and opts.get('type') in { | ||
b'debug', | ||||
b'error', | ||||
b'note', | ||||
b'status', | ||||
b'warning', | ||||
} | ||||
if timestamp: | ||||
args = ( | ||||
Joerg Sonnenberger
|
r45579 | b'[%s] ' | ||
% pycompat.bytestr(datetime.datetime.now().isoformat()), | ||||
Joerg Sonnenberger
|
r45564 | ) + args | ||
Yuya Nishihara
|
r40626 | _writemsgwith(self._write, dest, *args, **opts) | ||
Joerg Sonnenberger
|
r45564 | if timestamp: | ||
dest.flush() | ||||
Yuya Nishihara
|
r40626 | |||
Matt Harbison
|
r50690 | def _writemsgnobuf(self, dest, *args: bytes, **opts: _MsgOpts) -> None: | ||
Yuya Nishihara
|
r40626 | _writemsgwith(self._writenobuf, dest, *args, **opts) | ||
Matt Harbison
|
r50690 | def flush(self) -> None: | ||
Simon Farnsworth
|
r30978 | # opencode timeblockedsection because this is a critical path | ||
starttime = util.timer() | ||||
try: | ||||
Bryan O'Sullivan
|
r31963 | try: | ||
Yuya Nishihara
|
r40579 | self._fout.flush() | ||
Bryan O'Sullivan
|
r31963 | except IOError as err: | ||
Gregory Szorc
|
r33859 | if err.errno not in (errno.EPIPE, errno.EIO, errno.EBADF): | ||
raise error.StdioError(err) | ||||
Bryan O'Sullivan
|
r31963 | finally: | ||
try: | ||||
Yuya Nishihara
|
r40579 | self._ferr.flush() | ||
Bryan O'Sullivan
|
r31963 | except IOError as err: | ||
Gregory Szorc
|
r33859 | if err.errno not in (errno.EPIPE, errno.EIO, errno.EBADF): | ||
raise error.StdioError(err) | ||||
Simon Farnsworth
|
r30978 | finally: | ||
Augie Fackler
|
r43347 | self._blockedtimes[b'stdio_blocked'] += ( | ||
Augie Fackler
|
r43346 | util.timer() - starttime | ||
) * 1000 | ||||
Vadim Gelfer
|
r1837 | |||
Matt Harbison
|
r50692 | def _isatty(self, fh) -> bool: | ||
Augie Fackler
|
r43347 | if self.configbool(b'ui', b'nontty'): | ||
Matt Mackall
|
r16751 | return False | ||
Yuya Nishihara
|
r37138 | return procutil.isatty(fh) | ||
Vadim Gelfer
|
r1837 | |||
Yuya Nishihara
|
r41321 | def protectfinout(self): | ||
"""Duplicate ui streams and redirect original if they are stdio | ||||
Returns (fin, fout) which point to the original ui fds, but may be | ||||
copy of them. The returned streams can be considered "owned" in that | ||||
print(), exec(), etc. never reach to them. | ||||
""" | ||||
if self._finoutredirected: | ||||
# if already redirected, protectstdio() would just create another | ||||
# nullfd pair, which is equivalent to returning self._fin/_fout. | ||||
return self._fin, self._fout | ||||
fin, fout = procutil.protectstdio(self._fin, self._fout) | ||||
self._finoutredirected = (fin, fout) != (self._fin, self._fout) | ||||
return fin, fout | ||||
def restorefinout(self, fin, fout): | ||||
"""Restore ui streams from possibly duplicated (fin, fout)""" | ||||
if (fin, fout) == (self._fin, self._fout): | ||||
return | ||||
procutil.restorestdio(self._fin, self._fout, fin, fout) | ||||
# protectfinout() won't create more than one duplicated streams, | ||||
# so we can just turn the redirection flag off. | ||||
self._finoutredirected = False | ||||
Yuya Nishihara
|
r41320 | @contextlib.contextmanager | ||
def protectedfinout(self): | ||||
"""Run code block with protected standard streams""" | ||||
Yuya Nishihara
|
r41321 | fin, fout = self.protectfinout() | ||
Yuya Nishihara
|
r41320 | try: | ||
yield fin, fout | ||||
finally: | ||||
Yuya Nishihara
|
r41321 | self.restorefinout(fin, fout) | ||
Yuya Nishihara
|
r41320 | |||
Matt Harbison
|
r50692 | def disablepager(self) -> None: | ||
Augie Fackler
|
r31026 | self._disablepager = True | ||
Augie Fackler
|
r30994 | |||
Matt Harbison
|
r50692 | def pager(self, command: bytes) -> None: | ||
Augie Fackler
|
r30992 | """Start a pager for subsequent command output. | ||
Commands which produce a long stream of output should call | ||||
this function to activate the user's preferred pagination | ||||
mechanism (which may be no pager). Calling this function | ||||
precludes any future use of interactive functionality, such as | ||||
prompting the user or activating curses. | ||||
Args: | ||||
command: The full, non-aliased name of the command. That is, "log" | ||||
not "history, "summary" not "summ", etc. | ||||
""" | ||||
Augie Fackler
|
r43346 | if self._disablepager or self.pageractive: | ||
FUJIWARA Katsunori
|
r33622 | # how pager should do is already determined | ||
return | ||||
Augie Fackler
|
r43347 | if not command.startswith(b'internal-always-') and ( | ||
FUJIWARA Katsunori
|
r33622 | # explicit --pager=on (= 'internal-always-' prefix) should | ||
# take precedence over disabling factors below | ||||
Augie Fackler
|
r43347 | command in self.configlist(b'pager', b'ignore') | ||
or not self.configbool(b'ui', b'paginate') | ||||
or not self.configbool(b'pager', b'attend-' + command, True) | ||||
or encoding.environ.get(b'TERM') == b'dumb' | ||||
Augie Fackler
|
r30992 | # TODO: if we want to allow HGPLAINEXCEPT=pager, | ||
# formatted() will need some adjustment. | ||||
or not self.formatted() | ||||
or self.plain() | ||||
Jun Wu
|
r34023 | or self._buffers | ||
Augie Fackler
|
r30992 | # TODO: expose debugger-enabled on the UI object | ||
Augie Fackler
|
r43347 | or b'--debugger' in pycompat.sysargv | ||
Augie Fackler
|
r43346 | ): | ||
Augie Fackler
|
r30992 | # We only want to paginate if the ui appears to be | ||
# interactive, the user didn't say HGPLAIN or | ||||
# HGPLAINEXCEPT=pager, and the user didn't specify --debug. | ||||
return | ||||
Matt Harbison
|
r49966 | # py2exe doesn't appear to be able to use legacy I/O, and nothing is | ||
# output to the pager for paged commands. Piping to `more` in cmd.exe | ||||
# works, but is easy to forget. Just disable pager for py2exe, but | ||||
# leave it working for pyoxidizer and exewrapper builds. | ||||
if pycompat.iswindows and getattr(sys, "frozen", None) == "console_exe": | ||||
self.debug(b"pager is unavailable with py2exe packaging\n") | ||||
return | ||||
Augie Fackler
|
r43347 | pagercmd = self.config(b'pager', b'pager', rcutil.fallbackpager) | ||
Yuya Nishihara
|
r31079 | if not pagercmd: | ||
return | ||||
Jun Wu
|
r31954 | pagerenv = {} | ||
for name, value in rcutil.defaultpagerenv().items(): | ||||
if name not in encoding.environ: | ||||
pagerenv[name] = value | ||||
Augie Fackler
|
r43346 | self.debug( | ||
Augie Fackler
|
r43347 | b'starting pager for command %s\n' % stringutil.pprint(command) | ||
Augie Fackler
|
r43346 | ) | ||
Yuya Nishihara
|
r31490 | self.flush() | ||
Matt Harbison
|
r31690 | |||
wasformatted = self.formatted() | ||||
r51821 | if hasattr(signal, "SIGPIPE"): | |||
Augie Fackler
|
r30992 | signal.signal(signal.SIGPIPE, _catchterm) | ||
Jun Wu
|
r31954 | if self._runpager(pagercmd, pagerenv): | ||
Matt Harbison
|
r31690 | self.pageractive = True | ||
# Preserve the formatted-ness of the UI. This is important | ||||
# because we mess with stdout, which might confuse | ||||
# auto-detection of things being formatted. | ||||
Augie Fackler
|
r43347 | self.setconfig(b'ui', b'formatted', wasformatted, b'pager') | ||
self.setconfig(b'ui', b'interactive', False, b'pager') | ||||
Matt Harbison
|
r31691 | |||
# If pagermode differs from color.mode, reconfigure color now that | ||||
# pageractive is set. | ||||
cm = self._colormode | ||||
Augie Fackler
|
r43347 | if cm != self.config(b'color', b'pagermode', cm): | ||
Matt Harbison
|
r31691 | color.setup(self) | ||
Matt Harbison
|
r31690 | else: | ||
# If the pager can't be spawned in dispatch when --pager=on is | ||||
# given, don't try again when the command runs, to avoid a duplicate | ||||
# warning about a missing pager command. | ||||
self.disablepager() | ||||
Augie Fackler
|
r30992 | |||
Matt Harbison
|
r50692 | def _runpager(self, command: bytes, env=None) -> bool: | ||
Augie Fackler
|
r30992 | """Actually start the pager and set up file descriptors. | ||
This is separate in part so that extensions (like chg) can | ||||
override how a pager is invoked. | ||||
""" | ||||
Augie Fackler
|
r43347 | if command == b'cat': | ||
Augie Fackler
|
r31479 | # Save ourselves some work. | ||
Matt Harbison
|
r31690 | return False | ||
Augie Fackler
|
r31478 | # If the command doesn't contain any of these characters, we | ||
# assume it's a binary and exec it directly. This means for | ||||
# simple pager command configurations, we can degrade | ||||
# gracefully and tell the user about their broken pager. | ||||
Augie Fackler
|
r43347 | shell = any(c in command for c in b"|&;<>()$`\\\"' \t\n*?[#~=%") | ||
Matt Harbison
|
r31624 | |||
Jun Wu
|
r34646 | if pycompat.iswindows and not shell: | ||
Matt Harbison
|
r31624 | # Window's built-in `more` cannot be invoked with shell=False, but | ||
# its `more.com` can. Hide this implementation detail from the | ||||
# user so we can also get sane bad PAGER behavior. MSYS has | ||||
# `more.exe`, so do a cmd.exe style resolution of the executable to | ||||
# determine which one to use. | ||||
Yuya Nishihara
|
r37138 | fullcmd = procutil.findexe(command) | ||
Matt Harbison
|
r31624 | if not fullcmd: | ||
Augie Fackler
|
r43346 | self.warn( | ||
Augie Fackler
|
r43347 | _(b"missing pager command '%s', skipping pager\n") % command | ||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r31690 | return False | ||
Matt Harbison
|
r31624 | |||
command = fullcmd | ||||
Augie Fackler
|
r31478 | try: | ||
pager = subprocess.Popen( | ||||
Augie Fackler
|
r43346 | procutil.tonativestr(command), | ||
shell=shell, | ||||
bufsize=-1, | ||||
close_fds=procutil.closefds, | ||||
stdin=subprocess.PIPE, | ||||
stdout=procutil.stdout, | ||||
stderr=procutil.stderr, | ||||
env=procutil.tonativeenv(procutil.shellenviron(env)), | ||||
) | ||||
Manuel Jacob
|
r50201 | except FileNotFoundError: | ||
if not shell: | ||||
Augie Fackler
|
r43346 | self.warn( | ||
Augie Fackler
|
r43347 | _(b"missing pager command '%s', skipping pager\n") % command | ||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r31690 | return False | ||
Augie Fackler
|
r31478 | raise | ||
Augie Fackler
|
r30992 | |||
# back up original file descriptors | ||||
Jean-Francois Pieronne
|
r51891 | if pycompat.sysplatform != b'OpenVMS': | ||
stdoutfd = os.dup(procutil.stdout.fileno()) | ||||
stderrfd = os.dup(procutil.stderr.fileno()) | ||||
Augie Fackler
|
r30992 | |||
Yuya Nishihara
|
r37137 | os.dup2(pager.stdin.fileno(), procutil.stdout.fileno()) | ||
if self._isatty(procutil.stderr): | ||||
os.dup2(pager.stdin.fileno(), procutil.stderr.fileno()) | ||||
Augie Fackler
|
r30992 | |||
Bryan O'Sullivan
|
r31958 | @self.atexit | ||
Augie Fackler
|
r30992 | def killpager(): | ||
r51821 | if hasattr(signal, "SIGINT"): | |||
Augie Fackler
|
r30992 | signal.signal(signal.SIGINT, signal.SIG_IGN) | ||
# restore original fds, closing pager.stdin copies in the process | ||||
Jean-Francois Pieronne
|
r51891 | if pycompat.sysplatform == b'OpenVMS': | ||
pager.kill() | ||||
Yuya Nishihara
|
r37137 | os.dup2(stdoutfd, procutil.stdout.fileno()) | ||
os.dup2(stderrfd, procutil.stderr.fileno()) | ||||
Augie Fackler
|
r30992 | pager.stdin.close() | ||
pager.wait() | ||||
Matt Harbison
|
r31690 | return True | ||
Saurabh Singh
|
r34883 | @property | ||
def _exithandlers(self): | ||||
return _reqexithandlers | ||||
Bryan O'Sullivan
|
r31956 | def atexit(self, func, *args, **kwargs): | ||
Augie Fackler
|
r46554 | """register a function to run after dispatching a request | ||
Bryan O'Sullivan
|
r31956 | |||
Augie Fackler
|
r46554 | Handlers do not stay registered across request boundaries.""" | ||
Bryan O'Sullivan
|
r31956 | self._exithandlers.append((func, args, kwargs)) | ||
return func | ||||
Matt Harbison
|
r50692 | def interface(self, feature: bytes) -> bytes: | ||
Simon Farnsworth
|
r28542 | """what interface to use for interactive console features? | ||
The interface is controlled by the value of `ui.interface` but also by | ||||
the value of feature-specific configuration. For example: | ||||
ui.interface.histedit = text | ||||
ui.interface.chunkselector = curses | ||||
Here the features are "histedit" and "chunkselector". | ||||
The configuration above means that the default interfaces for commands | ||||
is curses, the interface for histedit is text and the interface for | ||||
selecting chunk is crecord (the best curses interface available). | ||||
Mads Kiilerich
|
r30332 | Consider the following example: | ||
Simon Farnsworth
|
r28542 | ui.interface = curses | ||
ui.interface.histedit = text | ||||
Then histedit will use the text interface and chunkselector will use | ||||
the default curses interface (crecord at the moment). | ||||
""" | ||||
Augie Fackler
|
r43347 | alldefaults = frozenset([b"text", b"curses"]) | ||
Simon Farnsworth
|
r28542 | |||
featureinterfaces = { | ||||
Augie Fackler
|
r46554 | b"chunkselector": [ | ||
b"text", | ||||
b"curses", | ||||
], | ||||
b"histedit": [ | ||||
b"text", | ||||
b"curses", | ||||
], | ||||
Simon Farnsworth
|
r28542 | } | ||
# Feature-specific interface | ||||
if feature not in featureinterfaces.keys(): | ||||
# Programming error, not user error | ||||
Augie Fackler
|
r43347 | raise ValueError(b"Unknown feature requested %s" % feature) | ||
Simon Farnsworth
|
r28542 | |||
availableinterfaces = frozenset(featureinterfaces[feature]) | ||||
if alldefaults > availableinterfaces: | ||||
# Programming error, not user error. We need a use case to | ||||
# define the right thing to do here. | ||||
raise ValueError( | ||||
Augie Fackler
|
r43347 | b"Feature %s does not handle all default interfaces" % feature | ||
Augie Fackler
|
r43346 | ) | ||
Simon Farnsworth
|
r28542 | |||
Augie Fackler
|
r43347 | if self.plain() or encoding.environ.get(b'TERM') == b'dumb': | ||
return b"text" | ||||
Simon Farnsworth
|
r28542 | |||
# Default interface for all the features | ||||
Augie Fackler
|
r43347 | defaultinterface = b"text" | ||
i = self.config(b"ui", b"interface") | ||||
Simon Farnsworth
|
r28542 | if i in alldefaults: | ||
Matt Harbison
|
r50692 | defaultinterface = cast(bytes, i) # cast to help pytype | ||
Simon Farnsworth
|
r28542 | |||
Matt Harbison
|
r50692 | choseninterface: bytes = defaultinterface | ||
Augie Fackler
|
r43347 | f = self.config(b"ui", b"interface.%s" % feature) | ||
Simon Farnsworth
|
r28542 | if f in availableinterfaces: | ||
Matt Harbison
|
r50692 | choseninterface = cast(bytes, f) # cast to help pytype | ||
Simon Farnsworth
|
r28542 | |||
if i is not None and defaultinterface != i: | ||||
if f is not None: | ||||
Augie Fackler
|
r43347 | self.warn(_(b"invalid value for ui.interface: %s\n") % (i,)) | ||
Simon Farnsworth
|
r28542 | else: | ||
Augie Fackler
|
r43346 | self.warn( | ||
Augie Fackler
|
r43347 | _(b"invalid value for ui.interface: %s (using %s)\n") | ||
Augie Fackler
|
r43346 | % (i, choseninterface) | ||
) | ||||
Simon Farnsworth
|
r28542 | if f is not None and choseninterface != f: | ||
Augie Fackler
|
r43346 | self.warn( | ||
Augie Fackler
|
r43347 | _(b"invalid value for ui.interface.%s: %s (using %s)\n") | ||
Augie Fackler
|
r43346 | % (feature, f, choseninterface) | ||
) | ||||
Simon Farnsworth
|
r28542 | |||
return choseninterface | ||||
Matt Mackall
|
r8208 | def interactive(self): | ||
Augie Fackler
|
r46554 | """is interactive input allowed? | ||
Dan Villiom Podlaski Christiansen
|
r11325 | |||
An interactive session is a session where input can be reasonably read | ||||
from `sys.stdin'. If this function returns false, any attempt to read | ||||
from stdin should fail with an error, unless a sensible default has been | ||||
specified. | ||||
Interactiveness is triggered by the value of the `ui.interactive' | ||||
configuration variable or - if it is unset - when `sys.stdin' points | ||||
to a terminal device. | ||||
This function refers to input only; for output, see `ui.formatted()'. | ||||
Augie Fackler
|
r46554 | """ | ||
Augie Fackler
|
r43347 | i = self.configbool(b"ui", b"interactive") | ||
Patrick Mezard
|
r8538 | if i is None: | ||
Idan Kamara
|
r14515 | # some environments replace stdin without implementing isatty | ||
# usually those are non-interactive | ||||
Yuya Nishihara
|
r40579 | return self._isatty(self._fin) | ||
Ronny Pfannschmidt
|
r10077 | |||
Patrick Mezard
|
r8538 | return i | ||
Matt Mackall
|
r8208 | |||
Matt Harbison
|
r50692 | def termwidth(self) -> int: | ||
Augie Fackler
|
r46554 | """how wide is the terminal in columns?""" | ||
Augie Fackler
|
r43347 | if b'COLUMNS' in encoding.environ: | ||
Augie Fackler
|
r12689 | try: | ||
Augie Fackler
|
r43347 | return int(encoding.environ[b'COLUMNS']) | ||
Augie Fackler
|
r12689 | except ValueError: | ||
pass | ||||
Yuya Nishihara
|
r30314 | return scmutil.termsize(self)[0] | ||
Augie Fackler
|
r12689 | |||
Dan Villiom Podlaski Christiansen
|
r11324 | def formatted(self): | ||
Augie Fackler
|
r46554 | """should formatted output be used? | ||
Dan Villiom Podlaski Christiansen
|
r11325 | |||
It is often desirable to format the output to suite the output medium. | ||||
Examples of this are truncating long lines or colorizing messages. | ||||
However, this is not often not desirable when piping output into other | ||||
utilities, e.g. `grep'. | ||||
Formatted output is triggered by the value of the `ui.formatted' | ||||
configuration variable or - if it is unset - when `sys.stdout' points | ||||
to a terminal device. Please note that `ui.formatted' should be | ||||
considered an implementation detail; it is not intended for use outside | ||||
Mercurial or its extensions. | ||||
This function refers to output only; for input, see `ui.interactive()'. | ||||
This function always returns false when in plain mode, see `ui.plain()'. | ||||
Augie Fackler
|
r46554 | """ | ||
Dan Villiom Podlaski Christiansen
|
r11324 | if self.plain(): | ||
return False | ||||
Augie Fackler
|
r43347 | i = self.configbool(b"ui", b"formatted") | ||
Dan Villiom Podlaski Christiansen
|
r11324 | if i is None: | ||
Idan Kamara
|
r14515 | # some environments replace stdout without implementing isatty | ||
# usually those are non-interactive | ||||
Yuya Nishihara
|
r40579 | return self._isatty(self._fout) | ||
Dan Villiom Podlaski Christiansen
|
r11324 | |||
return i | ||||
Matt Harbison
|
r50690 | def _readline( | ||
self, | ||||
prompt: bytes = b' ', | ||||
promptopts: Optional[Dict[str, _MsgOpts]] = None, | ||||
) -> bytes: | ||||
Yuya Nishihara
|
r36813 | # Replacing stdin/stdout temporarily is a hard problem on Python 3 | ||
# because they have to be text streams with *no buffering*. Instead, | ||||
# we use rawinput() only if call_readline() will be invoked by | ||||
# PyOS_Readline(), so no I/O will be made at Python layer. | ||||
Augie Fackler
|
r43346 | usereadline = ( | ||
self._isatty(self._fin) | ||||
and self._isatty(self._fout) | ||||
and procutil.isstdin(self._fin) | ||||
and procutil.isstdout(self._fout) | ||||
) | ||||
Yuya Nishihara
|
r36812 | if usereadline: | ||
Bryan O'Sullivan
|
r5036 | try: | ||
# magically add command line editing support, where | ||||
# available | ||||
import readline | ||||
Augie Fackler
|
r43346 | |||
Bryan O'Sullivan
|
r5036 | # force demandimport to really load the module | ||
readline.read_history_file | ||||
Brendan Cully
|
r7496 | # windows sometimes raises something other than ImportError | ||
except Exception: | ||||
Yuya Nishihara
|
r36812 | usereadline = False | ||
Idan Kamara
|
r14614 | |||
Augie Fackler
|
r43347 | if self._colormode == b'win32' or not usereadline: | ||
Kyle Lippincott
|
r42288 | if not promptopts: | ||
promptopts = {} | ||||
Augie Fackler
|
r43346 | self._writemsgnobuf( | ||
Augie Fackler
|
r43347 | self._fmsgout, prompt, type=b'prompt', **promptopts | ||
Augie Fackler
|
r43346 | ) | ||
Kyle Lippincott
|
r42288 | self.flush() | ||
Augie Fackler
|
r43347 | prompt = b' ' | ||
Kyle Lippincott
|
r42288 | else: | ||
Augie Fackler
|
r43347 | prompt = self.label(prompt, b'ui.prompt') + b' ' | ||
Kyle Lippincott
|
r42288 | |||
Yuya Nishihara
|
r22291 | # prompt ' ' must exist; otherwise readline may delete entire line | ||
# - http://bugs.python.org/issue12833 | ||||
Augie Fackler
|
r43347 | with self.timeblockedsection(b'stdio'): | ||
Yuya Nishihara
|
r36813 | if usereadline: | ||
Gregory Szorc
|
r43694 | self.flush() | ||
Denis Laxalde
|
r43388 | prompt = encoding.strfromlocal(prompt) | ||
Gregory Szorc
|
r49788 | line = encoding.strtolocal(input(prompt)) | ||
Yuya Nishihara
|
r36814 | # When stdin is in binary mode on Windows, it can cause | ||
Gregory Szorc
|
r49788 | # input() to emit an extra trailing carriage return | ||
Yuya Nishihara
|
r36814 | if pycompat.oslinesep == b'\r\n' and line.endswith(b'\r'): | ||
line = line[:-1] | ||||
Yuya Nishihara
|
r36813 | else: | ||
Kyle Lippincott
|
r42288 | self._fout.write(pycompat.bytestr(prompt)) | ||
Yuya Nishihara
|
r40579 | self._fout.flush() | ||
line = self._fin.readline() | ||||
Yuya Nishihara
|
r36813 | if not line: | ||
raise EOFError | ||||
Yuya Nishihara
|
r36852 | line = line.rstrip(pycompat.oslinesep) | ||
Idan Kamara
|
r14614 | |||
Steve Borho
|
r5613 | return line | ||
Bryan O'Sullivan
|
r5036 | |||
r52178 | if typing.TYPE_CHECKING: | |||
Matt Harbison
|
r50694 | |||
@overload | ||||
def prompt(self, msg: bytes, default: bytes) -> bytes: | ||||
pass | ||||
@overload | ||||
def prompt(self, msg: bytes, default: None) -> Optional[bytes]: | ||||
pass | ||||
Augie Fackler
|
r43347 | def prompt(self, msg, default=b"y"): | ||
Simon Heimberg
|
r9048 | """Prompt user with msg, read response. | ||
If ui is not interactive, the default is returned. | ||||
Kirill Smelkov
|
r5751 | """ | ||
Yuya Nishihara
|
r40628 | return self._prompt(msg, default=default) | ||
r52178 | if typing.TYPE_CHECKING: | |||
Matt Harbison
|
r50694 | |||
@overload | ||||
def _prompt( | ||||
self, msg: bytes, default: bytes, **opts: _MsgOpts | ||||
) -> bytes: | ||||
pass | ||||
@overload | ||||
def _prompt( | ||||
self, msg: bytes, default: None, **opts: _MsgOpts | ||||
) -> Optional[bytes]: | ||||
pass | ||||
Matt Harbison
|
r50693 | def _prompt(self, msg, default=b'y', **opts): | ||
opts = {**opts, 'default': default} | ||||
Matt Mackall
|
r8208 | if not self.interactive(): | ||
Augie Fackler
|
r43347 | self._writemsg(self._fmsgout, msg, b' ', type=b'prompt', **opts) | ||
Augie Fackler
|
r43346 | self._writemsg( | ||
Augie Fackler
|
r43347 | self._fmsgout, default or b'', b"\n", type=b'promptecho' | ||
Augie Fackler
|
r43346 | ) | ||
Peter Arrenbrecht
|
r7320 | return default | ||
Simon Heimberg
|
r9048 | try: | ||
Kyle Lippincott
|
r42288 | r = self._readline(prompt=msg, promptopts=opts) | ||
Simon Heimberg
|
r9048 | if not r: | ||
Mads Kiilerich
|
r22589 | r = default | ||
Augie Fackler
|
r43347 | if self.configbool(b'ui', b'promptecho'): | ||
Yuya Nishihara
|
r46380 | self._writemsg( | ||
self._fmsgout, r or b'', b"\n", type=b'promptecho' | ||||
) | ||||
Simon Heimberg
|
r9048 | return r | ||
except EOFError: | ||||
Siddharth Agarwal
|
r26896 | raise error.ResponseExpected() | ||
Simon Heimberg
|
r9048 | |||
FUJIWARA Katsunori
|
r20265 | @staticmethod | ||
Matt Harbison
|
r50690 | def extractchoices(prompt: bytes) -> Tuple[bytes, List[_PromptChoice]]: | ||
FUJIWARA Katsunori
|
r20265 | """Extract prompt message and list of choices from specified prompt. | ||
This returns tuple "(message, choices)", and "choices" is the | ||||
list of tuple "(response character, text without &)". | ||||
Matt Mackall
|
r27392 | |||
Yuya Nishihara
|
r34133 | >>> ui.extractchoices(b"awake? $$ &Yes $$ &No") | ||
Matt Mackall
|
r27392 | ('awake? ', [('y', 'Yes'), ('n', 'No')]) | ||
Yuya Nishihara
|
r34133 | >>> ui.extractchoices(b"line\\nbreak? $$ &Yes $$ &No") | ||
Matt Mackall
|
r27392 | ('line\\nbreak? ', [('y', 'Yes'), ('n', 'No')]) | ||
Yuya Nishihara
|
r34133 | >>> ui.extractchoices(b"want lots of $$money$$?$$Ye&s$$N&o") | ||
Matt Mackall
|
r27392 | ('want lots of $$money$$?', [('s', 'Yes'), ('o', 'No')]) | ||
FUJIWARA Katsunori
|
r20265 | """ | ||
Matt Mackall
|
r27392 | |||
# Sadly, the prompt string may have been built with a filename | ||||
# containing "$$" so let's try to find the first valid-looking | ||||
# prompt to start parsing. Sadly, we also can't rely on | ||||
# choices containing spaces, ASCII, or basically anything | ||||
# except an ampersand followed by a character. | ||||
Matt Harbison
|
r44473 | m = re.match(br'(?s)(.+?)\$\$([^$]*&[^ $].*)', prompt) | ||
Matt Harbison
|
r50688 | |||
assert m is not None # help pytype | ||||
Matt Mackall
|
r27392 | msg = m.group(1) | ||
Augie Fackler
|
r43347 | choices = [p.strip(b' ') for p in m.group(2).split(b'$$')] | ||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r33680 | def choicetuple(s): | ||
Augie Fackler
|
r43347 | ampidx = s.index(b'&') | ||
return s[ampidx + 1 : ampidx + 2].lower(), s.replace(b'&', b'', 1) | ||||
Augie Fackler
|
r43346 | |||
Augie Fackler
|
r33680 | return (msg, [choicetuple(s) for s in choices]) | ||
FUJIWARA Katsunori
|
r20265 | |||
Matt Harbison
|
r50690 | def promptchoice(self, prompt: bytes, default: int = 0) -> int: | ||
Matt Mackall
|
r19226 | """Prompt user with a message, read response, and ensure it matches | ||
one of the provided choices. The prompt is formatted as follows: | ||||
"would you like fries with that (Yn)? $$ &Yes $$ &No" | ||||
The index of the choice is returned. Responses are case | ||||
insensitive. If ui is not interactive, the default is | ||||
returned. | ||||
Simon Heimberg
|
r9048 | """ | ||
Matt Mackall
|
r19226 | |||
FUJIWARA Katsunori
|
r20265 | msg, choices = self.extractchoices(prompt) | ||
resps = [r for r, t in choices] | ||||
Thomas Arendsen Hein
|
r5671 | while True: | ||
Yuya Nishihara
|
r40628 | r = self._prompt(msg, default=resps[default], choices=choices) | ||
Simon Heimberg
|
r9048 | if r.lower() in resps: | ||
return resps.index(r.lower()) | ||||
Yuya Nishihara
|
r40580 | # TODO: shouldn't it be a warning? | ||
Augie Fackler
|
r43347 | self._writemsg(self._fmsgout, _(b"unrecognized response\n")) | ||
Simon Heimberg
|
r9048 | |||
Matt Harbison
|
r50690 | def getpass( | ||
self, prompt: Optional[bytes] = None, default: Optional[bytes] = None | ||||
) -> Optional[bytes]: | ||||
Matt Mackall
|
r10282 | if not self.interactive(): | ||
return default | ||||
Steve Borho
|
r7798 | try: | ||
Augie Fackler
|
r43346 | self._writemsg( | ||
self._fmsgerr, | ||||
Augie Fackler
|
r43347 | prompt or _(b'password: '), | ||
type=b'prompt', | ||||
Augie Fackler
|
r43346 | password=True, | ||
) | ||||
Yuya Nishihara
|
r21195 | # disable getpass() only if explicitly specified. it's still valid | ||
# to interact with tty even if fin is not a tty. | ||||
Augie Fackler
|
r43347 | with self.timeblockedsection(b'stdio'): | ||
if self.configbool(b'ui', b'nontty'): | ||||
Yuya Nishihara
|
r40579 | l = self._fin.readline() | ||
Simon Farnsworth
|
r30978 | if not l: | ||
raise EOFError | ||||
Augie Fackler
|
r43347 | return l.rstrip(b'\n') | ||
Simon Farnsworth
|
r30978 | else: | ||
Matt Harbison
|
r47949 | return util.get_password() | ||
Steve Borho
|
r7798 | except EOFError: | ||
Siddharth Agarwal
|
r26896 | raise error.ResponseExpected() | ||
Rodrigo Damazio Bovendorp
|
r38791 | |||
Matt Harbison
|
r50690 | def status(self, *msg: bytes, **opts: _MsgOpts) -> None: | ||
Augie Fackler
|
r46554 | """write status message to output (if ui.quiet is False) | ||
Brodie Rao
|
r10815 | |||
This adds an output label of "ui.status". | ||||
Augie Fackler
|
r46554 | """ | ||
Matt Mackall
|
r10282 | if not self.quiet: | ||
Augie Fackler
|
r43347 | self._writemsg(self._fmsgout, type=b'status', *msg, **opts) | ||
Rodrigo Damazio Bovendorp
|
r38791 | |||
Matt Harbison
|
r50690 | def warn(self, *msg: bytes, **opts: _MsgOpts) -> None: | ||
Augie Fackler
|
r46554 | """write warning message to output (stderr) | ||
Brodie Rao
|
r10815 | |||
This adds an output label of "ui.warning". | ||||
Augie Fackler
|
r46554 | """ | ||
Augie Fackler
|
r43347 | self._writemsg(self._fmsgerr, type=b'warning', *msg, **opts) | ||
Rodrigo Damazio Bovendorp
|
r38791 | |||
Matt Harbison
|
r50690 | def error(self, *msg: bytes, **opts: _MsgOpts) -> None: | ||
Augie Fackler
|
r46554 | """write error message to output (stderr) | ||
Rodrigo Damazio Bovendorp
|
r38791 | |||
This adds an output label of "ui.error". | ||||
Augie Fackler
|
r46554 | """ | ||
Augie Fackler
|
r43347 | self._writemsg(self._fmsgerr, type=b'error', *msg, **opts) | ||
Rodrigo Damazio Bovendorp
|
r38791 | |||
Matt Harbison
|
r50690 | def note(self, *msg: bytes, **opts: _MsgOpts) -> None: | ||
Augie Fackler
|
r46554 | """write note to output (if ui.verbose is True) | ||
Brodie Rao
|
r10815 | |||
This adds an output label of "ui.note". | ||||
Augie Fackler
|
r46554 | """ | ||
Matt Mackall
|
r10282 | if self.verbose: | ||
Augie Fackler
|
r43347 | self._writemsg(self._fmsgout, type=b'note', *msg, **opts) | ||
Rodrigo Damazio Bovendorp
|
r38791 | |||
Matt Harbison
|
r50690 | def debug(self, *msg: bytes, **opts: _MsgOpts) -> None: | ||
Augie Fackler
|
r46554 | """write debug message to output (if ui.debugflag is True) | ||
Brodie Rao
|
r10815 | |||
This adds an output label of "ui.debug". | ||||
Augie Fackler
|
r46554 | """ | ||
Matt Mackall
|
r10282 | if self.debugflag: | ||
Augie Fackler
|
r43347 | self._writemsg(self._fmsgout, type=b'debug', *msg, **opts) | ||
Yuya Nishihara
|
r40792 | self.log(b'debug', b'%s', b''.join(msg)) | ||
FUJIWARA Katsunori
|
r26750 | |||
Gregory Szorc
|
r43349 | # Aliases to defeat check-code. | ||
statusnoi18n = status | ||||
notenoi18n = note | ||||
warnnoi18n = warn | ||||
writenoi18n = write | ||||
Augie Fackler
|
r43346 | def edit( | ||
self, | ||||
Matt Harbison
|
r50692 | text: bytes, | ||
user: bytes, | ||||
extra: Optional[Dict[bytes, Any]] = None, # TODO: value type of bytes? | ||||
Augie Fackler
|
r43346 | editform=None, | ||
pending=None, | ||||
Matt Harbison
|
r50692 | repopath: Optional[bytes] = None, | ||
action: Optional[bytes] = None, | ||||
) -> bytes: | ||||
Michael Bolin
|
r34030 | if action is None: | ||
Augie Fackler
|
r43346 | self.develwarn( | ||
Augie Fackler
|
r43347 | b'action is None but will soon be a required ' | ||
b'parameter to ui.edit()' | ||||
Augie Fackler
|
r43346 | ) | ||
Jordi Gutiérrez Hermoso
|
r28635 | extra_defaults = { | ||
Augie Fackler
|
r43347 | b'prefix': b'editor', | ||
b'suffix': b'.txt', | ||||
Jordi Gutiérrez Hermoso
|
r28635 | } | ||
Mykola Nikishov
|
r27153 | if extra is not None: | ||
Augie Fackler
|
r43347 | if extra.get(b'suffix') is not None: | ||
Augie Fackler
|
r43346 | self.develwarn( | ||
Augie Fackler
|
r43347 | b'extra.suffix is not None but will soon be ' | ||
b'ignored by ui.edit()' | ||||
Augie Fackler
|
r43346 | ) | ||
Mykola Nikishov
|
r27153 | extra_defaults.update(extra) | ||
extra = extra_defaults | ||||
Sean Farley
|
r30835 | |||
Augie Fackler
|
r43347 | if action == b'diff': | ||
suffix = b'.diff' | ||||
Michael Bolin
|
r34056 | elif action: | ||
Augie Fackler
|
r43347 | suffix = b'.%s.hg.txt' % action | ||
Michael Bolin
|
r34030 | else: | ||
Augie Fackler
|
r43347 | suffix = extra[b'suffix'] | ||
Michael Bolin
|
r34030 | |||
Sean Farley
|
r30848 | rdir = None | ||
Augie Fackler
|
r43347 | if self.configbool(b'experimental', b'editortmpinhg'): | ||
Sean Farley
|
r30848 | rdir = repopath | ||
Augie Fackler
|
r43346 | (fd, name) = pycompat.mkstemp( | ||
Augie Fackler
|
r43347 | prefix=b'hg-' + extra[b'prefix'] + b'-', suffix=suffix, dir=rdir | ||
Augie Fackler
|
r43346 | ) | ||
Thomas Arendsen Hein
|
r1984 | try: | ||
Matt Harbison
|
r44328 | with os.fdopen(fd, 'wb') as f: | ||
f.write(util.tonativeeol(text)) | ||||
Thomas Arendsen Hein
|
r1984 | |||
Augie Fackler
|
r43347 | environ = {b'HGUSER': user} | ||
if b'transplant_source' in extra: | ||||
environ.update( | ||||
{b'HGREVISION': hex(extra[b'transplant_source'])} | ||||
) | ||||
for label in (b'intermediate-source', b'source', b'rebase_source'): | ||||
Alexander Drozdov
|
r20605 | if label in extra: | ||
Augie Fackler
|
r43347 | environ.update({b'HGREVISION': extra[label]}) | ||
Alexander Drozdov
|
r20605 | break | ||
FUJIWARA Katsunori
|
r22205 | if editform: | ||
Augie Fackler
|
r43347 | environ.update({b'HGEDITFORM': editform}) | ||
FUJIWARA Katsunori
|
r26750 | if pending: | ||
Augie Fackler
|
r43347 | environ.update({b'HG_PENDING': pending}) | ||
Alexander Drozdov
|
r20605 | |||
Osku Salerma
|
r5660 | editor = self.geteditor() | ||
mpm@selenic.com
|
r207 | |||
Augie Fackler
|
r43346 | self.system( | ||
Augie Fackler
|
r43347 | b"%s \"%s\"" % (editor, name), | ||
Augie Fackler
|
r43346 | environ=environ, | ||
Martin von Zweigbergk
|
r46489 | onerr=error.CanceledError, | ||
Augie Fackler
|
r43347 | errprefix=_(b"edit failed"), | ||
blockedtag=b'editor', | ||||
Augie Fackler
|
r43346 | ) | ||
Matt Mackall
|
r608 | |||
Matt Harbison
|
r44328 | with open(name, 'rb') as f: | ||
t = util.fromnativeeol(f.read()) | ||||
Thomas Arendsen Hein
|
r1984 | finally: | ||
os.unlink(name) | ||||
Radoslaw "AstralStorm" Szkodzinski
|
r662 | |||
mpm@selenic.com
|
r207 | return t | ||
Vadim Gelfer
|
r2200 | |||
Augie Fackler
|
r43346 | def system( | ||
self, | ||||
Matt Harbison
|
r50692 | cmd: bytes, | ||
Augie Fackler
|
r43346 | environ=None, | ||
Matt Harbison
|
r50692 | cwd: Optional[bytes] = None, | ||
onerr: Optional[Callable[[bytes], Exception]] = None, | ||||
errprefix: Optional[bytes] = None, | ||||
blockedtag: Optional[bytes] = None, | ||||
) -> int: | ||||
Augie Fackler
|
r46554 | """execute shell command with appropriate output stream. command | ||
Yuya Nishihara
|
r23269 | output will be redirected if fout is not stdout. | ||
Yuya Nishihara
|
r31108 | |||
if command fails and onerr is None, return status, else raise onerr | ||||
object as exception. | ||||
Augie Fackler
|
r46554 | """ | ||
Simon Farnsworth
|
r30979 | if blockedtag is None: | ||
Simon Farnsworth
|
r31535 | # Long cmds tend to be because of an absolute path on cmd. Keep | ||
# the tail end instead | ||||
cmdsuffix = cmd.translate(None, _keepalnum)[-85:] | ||||
Augie Fackler
|
r43347 | blockedtag = b'unknown_system_' + cmdsuffix | ||
Yuya Nishihara
|
r40579 | out = self._fout | ||
Augie Fackler
|
r25149 | if any(s[1] for s in self._bufferstates): | ||
Pierre-Yves David
|
r24848 | out = self | ||
Simon Farnsworth
|
r30979 | with self.timeblockedsection(blockedtag): | ||
Yuya Nishihara
|
r31108 | rc = self._runsystem(cmd, environ=environ, cwd=cwd, out=out) | ||
if rc and onerr: | ||||
Augie Fackler
|
r43347 | errmsg = b'%s %s' % ( | ||
Micha Wiedenmann
|
r45072 | procutil.shellsplit(cmd)[0], | ||
Augie Fackler
|
r43346 | procutil.explainexit(rc), | ||
) | ||||
Yuya Nishihara
|
r31108 | if errprefix: | ||
Augie Fackler
|
r43347 | errmsg = b'%s: %s' % (errprefix, errmsg) | ||
Yuya Nishihara
|
r31108 | raise onerr(errmsg) | ||
return rc | ||||
Yuya Nishihara
|
r31107 | |||
Matt Harbison
|
r50692 | def _runsystem(self, cmd: bytes, environ, cwd: Optional[bytes], out) -> int: | ||
Yuya Nishihara
|
r31107 | """actually execute the given shell command (can be overridden by | ||
extensions like chg)""" | ||||
Yuya Nishihara
|
r37138 | return procutil.system(cmd, environ=environ, cwd=cwd, out=out) | ||
Yuya Nishihara
|
r23269 | |||
Matt Harbison
|
r50692 | def traceback(self, exc=None, force: bool = False): | ||
Augie Fackler
|
r46554 | """print exception traceback if traceback printing enabled or forced. | ||
Vadim Gelfer
|
r2335 | only to call in exception handler. returns true if traceback | ||
Augie Fackler
|
r46554 | printed.""" | ||
Matt Harbison
|
r18966 | if self.tracebackflag or force: | ||
Matt Harbison
|
r18965 | if exc is None: | ||
exc = sys.exc_info() | ||||
cause = getattr(exc[1], 'cause', None) | ||||
if cause is not None: | ||||
causetb = traceback.format_tb(cause[2]) | ||||
exctb = traceback.format_tb(exc[2]) | ||||
exconly = traceback.format_exception_only(cause[0], cause[1]) | ||||
# exclude frame where 'exc' was chained and rethrown from exctb | ||||
Augie Fackler
|
r43346 | self.write_err( | ||
Augie Fackler
|
r43347 | b'Traceback (most recent call last):\n', | ||
Matt Harbison
|
r44329 | encoding.strtolocal(''.join(exctb[:-1])), | ||
encoding.strtolocal(''.join(causetb)), | ||||
encoding.strtolocal(''.join(exconly)), | ||||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r18965 | else: | ||
Matt Harbison
|
r25568 | output = traceback.format_exception(exc[0], exc[1], exc[2]) | ||
Augie Fackler
|
r43906 | self.write_err(encoding.strtolocal(''.join(output))) | ||
Matt Harbison
|
r18966 | return self.tracebackflag or force | ||
Osku Salerma
|
r5660 | |||
def geteditor(self): | ||||
'''return editor to use''' | ||||
Augie Fackler
|
r43347 | if pycompat.sysplatform == b'plan9': | ||
Steven Stallion
|
r16383 | # vi is the MIPS instruction simulator on Plan 9. We | ||
# instead default to E to plumb commit messages to | ||||
# avoid confusion. | ||||
Augie Fackler
|
r43347 | editor = b'E' | ||
Kyle Lippincott
|
r45085 | elif pycompat.isdarwin: | ||
# vi on darwin is POSIX compatible to a fault, and that includes | ||||
# exiting non-zero if you make any mistake when running an ex | ||||
# command. Proof: `vi -c ':unknown' -c ':qa'; echo $?` produces 1, | ||||
# while s/vi/vim/ doesn't. | ||||
editor = b'vim' | ||||
Steven Stallion
|
r16383 | else: | ||
Augie Fackler
|
r43347 | editor = b'vi' | ||
return encoding.environ.get(b"HGEDITOR") or self.config( | ||||
b"ui", b"editor", editor | ||||
Augie Fackler
|
r43346 | ) | ||
Matt Mackall
|
r9153 | |||
Pierre-Yves David
|
r25499 | @util.propertycache | ||
Matt Harbison
|
r50689 | def _progbar(self) -> Optional[progress.progbar]: | ||
Pierre-Yves David
|
r25499 | """setup the progbar singleton to the ui object""" | ||
Augie Fackler
|
r43346 | if ( | ||
self.quiet | ||||
or self.debugflag | ||||
Augie Fackler
|
r43347 | or self.configbool(b'progress', b'disable') | ||
Augie Fackler
|
r43346 | or not progress.shouldprint(self) | ||
): | ||||
Pierre-Yves David
|
r25499 | return None | ||
return getprogbar(self) | ||||
Matt Harbison
|
r50689 | def _progclear(self) -> None: | ||
Pierre-Yves David
|
r25499 | """clear progress bar output if any. use it before any output""" | ||
Augie Fackler
|
r43346 | if not haveprogbar(): # nothing loaded yet | ||
Pierre-Yves David
|
r25499 | return | ||
if self._progbar is not None and self._progbar.printed: | ||||
self._progbar.clear() | ||||
Matt Harbison
|
r50689 | def makeprogress( | ||
self, topic: bytes, unit: bytes = b"", total: Optional[int] = None | ||||
) -> scmutil.progress: | ||||
Yuya Nishihara
|
r41246 | """Create a progress helper for the specified topic""" | ||
if getattr(self._fmsgerr, 'structured', False): | ||||
# channel for machine-readable output with metadata, just send | ||||
# raw information | ||||
# TODO: consider porting some useful information (e.g. estimated | ||||
# time) from progbar. we might want to support update delay to | ||||
# reduce the cost of transferring progress messages. | ||||
def updatebar(topic, pos, item, unit, total): | ||||
Augie Fackler
|
r43346 | self._fmsgerr.write( | ||
None, | ||||
type=b'progress', | ||||
topic=topic, | ||||
pos=pos, | ||||
item=item, | ||||
unit=unit, | ||||
total=total, | ||||
) | ||||
Yuya Nishihara
|
r41246 | elif self._progbar is not None: | ||
updatebar = self._progbar.progress | ||||
else: | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r41246 | def updatebar(topic, pos, item, unit, total): | ||
pass | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r41246 | return scmutil.progress(self, updatebar, topic, unit, total) | ||
Martin von Zweigbergk
|
r38364 | |||
Yuya Nishihara
|
r40761 | def getlogger(self, name): | ||
"""Returns a logger of the given name; or None if not registered""" | ||||
return self._loggers.get(name) | ||||
Matt Harbison
|
r50692 | def setlogger(self, name, logger) -> None: | ||
Yuya Nishihara
|
r40761 | """Install logger which can be identified later by the given name | ||
More than one loggers can be registered. Use extension or module | ||||
name to uniquely identify the logger instance. | ||||
""" | ||||
self._loggers[name] = logger | ||||
Matt Harbison
|
r50692 | def log(self, event, msgfmt, *msgargs, **opts) -> None: | ||
Augie Fackler
|
r46554 | """hook for logging facility extensions | ||
Matt Mackall
|
r11984 | |||
Yuya Nishihara
|
r40714 | event should be a readily-identifiable subsystem, which will | ||
Matt Mackall
|
r11984 | allow filtering. | ||
Augie Fackler
|
r26235 | |||
Yuya Nishihara
|
r40793 | msgfmt should be a newline-terminated format string to log, and | ||
*msgargs are %-formatted into it. | ||||
Augie Fackler
|
r26235 | |||
**opts currently has no defined meanings. | ||||
Augie Fackler
|
r46554 | """ | ||
Yuya Nishihara
|
r40761 | if not self._loggers: | ||
return | ||||
Gregory Szorc
|
r49790 | activeloggers = [l for l in self._loggers.values() if l.tracked(event)] | ||
Yuya Nishihara
|
r40761 | if not activeloggers: | ||
return | ||||
Yuya Nishihara
|
r40793 | msg = msgfmt % msgargs | ||
Yuya Nishihara
|
r40794 | opts = pycompat.byteskwargs(opts) | ||
Yuya Nishihara
|
r40792 | # guard against recursion from e.g. ui.debug() | ||
registeredloggers = self._loggers | ||||
self._loggers = {} | ||||
try: | ||||
for logger in activeloggers: | ||||
logger.log(self, event, msg, opts) | ||||
finally: | ||||
self._loggers = registeredloggers | ||||
Matt Mackall
|
r11984 | |||
Matt Harbison
|
r50690 | def label(self, msg: bytes, label: bytes) -> bytes: | ||
Augie Fackler
|
r46554 | """style msg based on supplied label | ||
Brodie Rao
|
r10815 | |||
Pierre-Yves David
|
r31087 | If some color mode is enabled, this will add the necessary control | ||
characters to apply such color. In addition, 'debug' color mode adds | ||||
markup showing which label affects a piece of text. | ||||
Brodie Rao
|
r10815 | |||
ui.write(s, 'label') is equivalent to | ||||
ui.write(ui.label(s, 'label')). | ||||
Augie Fackler
|
r46554 | """ | ||
Pierre-Yves David
|
r31087 | if self._colormode is not None: | ||
return color.colorlabel(self, msg, label) | ||||
Brodie Rao
|
r10815 | return msg | ||
Gregory Szorc
|
r24250 | |||
Matt Harbison
|
r50690 | def develwarn( | ||
self, msg: bytes, stacklevel: int = 1, config: Optional[bytes] = None | ||||
) -> None: | ||||
Pierre-Yves David
|
r27274 | """issue a developer warning message | ||
Use 'stacklevel' to report the offender some layers further up in the | ||||
stack. | ||||
""" | ||||
Augie Fackler
|
r43347 | if not self.configbool(b'devel', b'all-warnings'): | ||
if config is None or not self.configbool(b'devel', config): | ||||
Pierre-Yves David
|
r29095 | return | ||
Augie Fackler
|
r43347 | msg = b'devel-warn: ' + msg | ||
Augie Fackler
|
r43346 | stacklevel += 1 # get in develwarn | ||
Pierre-Yves David
|
r25629 | if self.tracebackflag: | ||
Yuya Nishihara
|
r40579 | util.debugstacktrace(msg, stacklevel, self._ferr, self._fout) | ||
Augie Fackler
|
r43346 | self.log( | ||
Augie Fackler
|
r43347 | b'develwarn', | ||
b'%s at:\n%s' | ||||
% (msg, b''.join(util.getstackframes(stacklevel))), | ||||
Augie Fackler
|
r43346 | ) | ||
Pierre-Yves David
|
r25629 | else: | ||
curframe = inspect.currentframe() | ||||
calframe = inspect.getouterframes(curframe, 2) | ||||
Augie Fackler
|
r36144 | fname, lineno, fmsg = calframe[stacklevel][1:4] | ||
fname, fmsg = pycompat.sysbytes(fname), pycompat.sysbytes(fmsg) | ||||
Augie Fackler
|
r43347 | self.write_err(b'%s at: %s:%d (%s)\n' % (msg, fname, lineno, fmsg)) | ||
Augie Fackler
|
r43346 | self.log( | ||
Augie Fackler
|
r43347 | b'develwarn', b'%s at: %s:%d (%s)\n', msg, fname, lineno, fmsg | ||
Augie Fackler
|
r43346 | ) | ||
Matt Harbison
|
r44443 | |||
# avoid cycles | ||||
del curframe | ||||
del calframe | ||||
Pierre-Yves David
|
r25629 | |||
Matt Harbison
|
r50690 | def deprecwarn( | ||
r51148 | self, | |||
msg: bytes, | ||||
version: bytes, | ||||
stacklevel: int = 2, | ||||
Matt Harbison
|
r50690 | ) -> None: | ||
Pierre-Yves David
|
r27275 | """issue a deprecation warning | ||
- msg: message explaining what is deprecated and how to upgrade, | ||||
- version: last version where the API will be supported, | ||||
""" | ||||
Augie Fackler
|
r43346 | if not ( | ||
Augie Fackler
|
r43347 | self.configbool(b'devel', b'all-warnings') | ||
or self.configbool(b'devel', b'deprec-warn') | ||||
Augie Fackler
|
r43346 | ): | ||
Pierre-Yves David
|
r29082 | return | ||
Augie Fackler
|
r43346 | msg += ( | ||
Augie Fackler
|
r43347 | b"\n(compatibility will be dropped after Mercurial-%s," | ||
b" update your code.)" | ||||
Augie Fackler
|
r43346 | ) % version | ||
Augie Fackler
|
r43347 | self.develwarn(msg, stacklevel=stacklevel, config=b'deprec-warn') | ||
Pierre-Yves David
|
r25629 | |||
Matt Harbison
|
r30832 | def exportableenviron(self): | ||
"""The environment variables that are safe to export, e.g. through | ||||
hgweb. | ||||
""" | ||||
return self._exportableenviron | ||||
Kostia Balytskyi
|
r30480 | @contextlib.contextmanager | ||
Matt Harbison
|
r50692 | def configoverride(self, overrides: _ConfigItems, source: bytes = b""): | ||
Kostia Balytskyi
|
r30480 | """Context manager for temporary config overrides | ||
`overrides` must be a dict of the following structure: | ||||
{(section, name) : value}""" | ||||
backups = {} | ||||
Gregory Szorc
|
r30537 | try: | ||
for (section, name), value in overrides.items(): | ||||
backups[(section, name)] = self.backupconfig(section, name) | ||||
self.setconfig(section, name, value, source) | ||||
yield | ||||
finally: | ||||
for __, backup in backups.items(): | ||||
self.restoreconfig(backup) | ||||
# just restoring ui.quiet config to the previous value is not enough | ||||
# as it does not update ui.quiet class member | ||||
Augie Fackler
|
r43347 | if (b'ui', b'quiet') in overrides: | ||
self.fixconfig(section=b'ui') | ||||
Kostia Balytskyi
|
r30480 | |||
Matt Harbison
|
r50692 | def estimatememory(self) -> Optional[int]: | ||
Joerg Sonnenberger
|
r45621 | """Provide an estimate for the available system memory in Bytes. | ||
This can be overriden via ui.available-memory. It returns None, if | ||||
no estimate can be computed. | ||||
""" | ||||
value = self.config(b'ui', b'available-memory') | ||||
if value is not None: | ||||
try: | ||||
return util.sizetoint(value) | ||||
except error.ParseError: | ||||
raise error.ConfigError( | ||||
_(b"ui.available-memory value is invalid ('%s')") % value | ||||
) | ||||
return util._estimatememory() | ||||
Augie Fackler
|
r43346 | |||
Pierre-Yves David
|
r25498 | # we instantiate one globally shared progress bar to avoid | ||
# competing progress bars when multiple UI objects get created | ||||
Matt Harbison
|
r50689 | _progresssingleton: Optional[progress.progbar] = None | ||
Pierre-Yves David
|
r25498 | |||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r50689 | def getprogbar(ui: ui) -> progress.progbar: | ||
Pierre-Yves David
|
r25498 | global _progresssingleton | ||
if _progresssingleton is None: | ||||
# passing 'ui' object to the singleton is fishy, | ||||
# this is how the extension used to work but feel free to rework it. | ||||
_progresssingleton = progress.progbar(ui) | ||||
return _progresssingleton | ||||
Mark Thomas
|
r34346 | |||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r50689 | def haveprogbar() -> bool: | ||
Mark Thomas
|
r34346 | return _progresssingleton is not None | ||
Yuya Nishihara
|
r40580 | |||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r50692 | def _selectmsgdests(ui: ui): | ||
Yuya Nishihara
|
r40580 | name = ui.config(b'ui', b'message-output') | ||
Yuya Nishihara
|
r40625 | if name == b'channel': | ||
if ui.fmsg: | ||||
return ui.fmsg, ui.fmsg | ||||
else: | ||||
# fall back to ferr if channel isn't ready so that status/error | ||||
# messages can be printed | ||||
return ui.ferr, ui.ferr | ||||
Yuya Nishihara
|
r40580 | if name == b'stdio': | ||
return ui.fout, ui.ferr | ||||
if name == b'stderr': | ||||
return ui.ferr, ui.ferr | ||||
raise error.Abort(b'invalid ui.message-output destination: %s' % name) | ||||
Yuya Nishihara
|
r40626 | |||
Augie Fackler
|
r43346 | |||
Matt Harbison
|
r50690 | def _writemsgwith(write, dest, *args: bytes, **opts: _MsgOpts) -> None: | ||
Yuya Nishihara
|
r40626 | """Write ui message with the given ui._write*() function | ||
The specified message type is translated to 'ui.<type>' label if the dest | ||||
isn't a structured channel, so that the message will be colorized. | ||||
""" | ||||
# TODO: maybe change 'type' to a mandatory option | ||||
Augie Fackler
|
r43906 | if 'type' in opts and not getattr(dest, 'structured', False): | ||
opts['label'] = opts.get('label', b'') + b' ui.%s' % opts.pop('type') | ||||
Yuya Nishihara
|
r40626 | write(dest, *args, **opts) | ||