logtoprocess.py
83 lines
| 2.8 KiB
| text/x-python
|
PythonLexer
/ hgext / logtoprocess.py
Martijn Pieters
|
r28901 | # logtoprocess.py - send ui.log() data to a subprocess | ||
# | ||||
# Copyright 2016 Facebook, Inc. | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
Jun Wu
|
r31601 | """send ui.log() data to a subprocess (EXPERIMENTAL) | ||
Martijn Pieters
|
r28901 | |||
This extension lets you specify a shell command per ui.log() event, | ||||
sending all remaining arguments to as environment variables to that command. | ||||
Yuya Nishihara
|
r40656 | Positional arguments construct a log message, which is passed in the `MSG1` | ||
environment variables. Each keyword argument is set as a `OPT_UPPERCASE_KEY` | ||||
variable (so the key is uppercased, and prefixed with `OPT_`). The original | ||||
event name is passed in the `EVENT` environment variable, and the process ID | ||||
of mercurial is given in `HGPID`. | ||||
Martijn Pieters
|
r28901 | |||
Yuya Nishihara
|
r40656 | So given a call `ui.log('foo', 'bar %s\n', 'baz', spam='eggs'), a script | ||
configured for the `foo` event can expect an environment with `MSG1=bar baz`, | ||||
and `OPT_SPAM=eggs`. | ||||
Martijn Pieters
|
r28901 | |||
Scripts are configured in the `[logtoprocess]` section, each key an event name. | ||||
For example:: | ||||
[logtoprocess] | ||||
Yuya Nishihara
|
r40656 | commandexception = echo "$MSG1" > /var/log/mercurial_exceptions.log | ||
Martijn Pieters
|
r28901 | |||
would log the warning message and traceback of any failed command dispatch. | ||||
Mads Kiilerich
|
r30332 | Scripts are run asynchronously as detached daemon processes; mercurial will | ||
Martijn Pieters
|
r28901 | not ensure that they exit cleanly. | ||
""" | ||||
Matt Harbison
|
r52756 | from __future__ import annotations | ||
Martijn Pieters
|
r28901 | |||
import os | ||||
Pulkit Goyal
|
r30638 | |||
Augie Fackler
|
r43346 | from mercurial.utils import procutil | ||
Matt Harbison
|
r39851 | |||
Augie Fackler
|
r29841 | # Note for extension authors: ONLY specify testedwith = 'ships-with-hg-core' for | ||
Martijn Pieters
|
r28901 | # extensions which SHIP WITH MERCURIAL. Non-mainline extensions should | ||
# be specifying the version(s) of Mercurial they are tested with, or | ||||
# leave the attribute unspecified. | ||||
Augie Fackler
|
r43347 | testedwith = b'ships-with-hg-core' | ||
Martijn Pieters
|
r28901 | |||
Augie Fackler
|
r43346 | |||
Gregory Szorc
|
r49801 | class processlogger: | ||
Yuya Nishihara
|
r40713 | """Map log events to external commands | ||
Arguments are passed on as environment variables. | ||||
""" | ||||
def __init__(self, ui): | ||||
self._scripts = dict(ui.configitems(b'logtoprocess')) | ||||
def tracked(self, event): | ||||
return bool(self._scripts.get(event)) | ||||
def log(self, ui, event, msg, opts): | ||||
Yuya Nishihara
|
r40761 | script = self._scripts[event] | ||
Valentin Gatien-Baron
|
r44904 | maxmsg = 100000 | ||
if len(msg) > maxmsg: | ||||
# Each env var has a 128KiB limit on linux. msg can be long, in | ||||
# particular for command event, where it's the full command line. | ||||
# Prefer truncating the message than raising "Argument list too | ||||
# long" error. | ||||
msg = msg[:maxmsg] + b' (truncated)' | ||||
Yuya Nishihara
|
r40713 | env = { | ||
b'EVENT': event, | ||||
b'HGPID': os.getpid(), | ||||
Yuya Nishihara
|
r40793 | b'MSG1': msg, | ||
Yuya Nishihara
|
r40713 | } | ||
# keyword arguments get prefixed with OPT_ and uppercased | ||||
Augie Fackler
|
r43346 | env.update( | ||
(b'OPT_%s' % key.upper(), value) for key, value in opts.items() | ||||
) | ||||
Yuya Nishihara
|
r40713 | fullenv = procutil.shellenviron(env) | ||
procutil.runbgcommand(script, fullenv, shell=True) | ||||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r40761 | def uipopulate(ui): | ||
ui.setlogger(b'logtoprocess', processlogger(ui)) | ||||