From: "Johannes Schindelin via GitGitGadget" <gitgitgadget@gmail.com>
To: git@vger.kernel.org
Cc: "brian m. carlson" <sandals@crustytoothpaste.net>,
Phillip Wood <phillip.wood123@gmail.com>,
Andreas Schwab <schwab@linux-m68k.org>,
Ondrej Pohorelsky <opohorel@redhat.com>,
Patrick Steinhardt <ps@pks.im>, Jeff King <peff@peff.net>,
Johannes Schindelin <johannes.schindelin@gmx.de>
Subject: [PATCH v3 0/5] Sanitize sideband channel messages
Date: Fri, 16 Jan 2026 22:26:08 +0000 [thread overview]
Message-ID: <pull.1853.v3.git.1768602373.gitgitgadget@gmail.com> (raw)
In-Reply-To: <pull.1853.v2.git.1765981422.gitgitgadget@gmail.com>
Git's sideband channel passes server output directly to the client terminal
without sanitization. This includes progress messages, error output, and
diagnostics from remote hooks during clone, fetch, and push operations.
This creates an ANSI escape sequence injection vulnerability (CWE-150
[https://cwe.mitre.org/data/definitions/150.html]). A malicious or
compromised server can corrupt terminal state, obscure information, or
inject characters into the terminal's input buffer. The client has no
mechanism to distinguish between legitimate output and attack sequences.
This series fixes the vulnerability by sanitizing control characters in the
sideband output. ANSI color sequences (SGR codes) pass through by default,
since server-side hooks exist that use these for visibility (e.g.
https://github.com/kikeonline/githook-explode). By default, all other
control characters are rendered in caret notation (e.g., ESC becomes ^[).
Users who need different behavior get configuration options:
sideband.allowControlCharacters provides an escape hatch for environments
that require raw passthrough. The defaults are secure.
Note: This series applies cleanly on v2.47.3. Integrating this into newer
versions is a bit cumbersome; I pushed a version of the branch as rebased to
v2.53.0-rc0 here:
https://github.com/dscho/git/tree/refs/heads/sanitize-sideband-2.53.0-rc0
Changes since v2:
* Added curly brackets around a single-line if clause.
* Enclosed the values in the documentation within backticks.
* Aligned the enum values for better readability.
* Added support for sideband.<url>.allowControlCharacters (à la
http.<url>.*) on top of sideband.allowControlCharacters.
Changes since v1:
* Applied the suggestions by Phillip and brian.
* Rebased onto v2.47.3.
* Added more categories of ANSI Escape sequences that can be enabled (but
that are off by default because they could be used to hide information).
Johannes Schindelin (5):
sideband: mask control characters
sideband: introduce an "escape hatch" to allow control characters
sideband: do allow ANSI color sequences by default
sideband: add options to allow more control sequences to be passed
through
sideband: offer to configure sanitizing on a per-URL basis
Documentation/config.txt | 2 +
Documentation/config/sideband.txt | 28 +++++
sideband.c | 181 +++++++++++++++++++++++++++-
sideband.h | 14 +++
t/t5409-colorize-remote-messages.sh | 92 ++++++++++++++
transport.c | 3 +
6 files changed, 318 insertions(+), 2 deletions(-)
create mode 100644 Documentation/config/sideband.txt
base-commit: a52a24e03c8c711f1d5e252fba78f9276908129b
Published-As: https://github.com/gitgitgadget/git/releases/tag/pr-1853%2Fdscho%2Fsanitize-sideband-v3
Fetch-It-Via: git fetch https://github.com/gitgitgadget/git pr-1853/dscho/sanitize-sideband-v3
Pull-Request: https://github.com/gitgitgadget/git/pull/1853
Range-diff vs v2:
1: 8d70476559 ! 1: e6b71af0ca sideband: mask control characters
@@ sideband.c: void list_config_color_sideband_slots(struct string_list *list, cons
+{
+ strbuf_grow(dest, n);
+ for (; n && *src; src++, n--) {
-+ if (!iscntrl(*src) || *src == '\t' || *src == '\n')
++ if (!iscntrl(*src) || *src == '\t' || *src == '\n') {
+ strbuf_addch(dest, *src);
-+ else {
++ } else {
+ strbuf_addch(dest, '^');
+ strbuf_addch(dest, *src == 0x7f ? '?' : 0x40 + *src);
+ }
2: 2615abd8c5 ! 2: 8f64d65844 sideband: introduce an "escape hatch" to allow control characters
@@ sideband.c: void list_config_color_sideband_slots(struct string_list *list, cons
+
strbuf_grow(dest, n);
for (; n && *src; src++, n--) {
- if (!iscntrl(*src) || *src == '\t' || *src == '\n')
+ if (!iscntrl(*src) || *src == '\t' || *src == '\n') {
## t/t5409-colorize-remote-messages.sh ##
@@ t/t5409-colorize-remote-messages.sh: test_expect_success 'disallow (color) control sequences in sideband' '
3: 44585ba1f4 ! 3: 44838acacc sideband: do allow ANSI color sequences by default
@@ Documentation/config/sideband.txt
+ this config setting to override this behavior:
++
+--
-+ default::
-+ color::
++ `default`::
++ `color`::
+ Allow ANSI color sequences, line feeds and horizontal tabs,
+ but mask all other control characters. This is the default.
-+ false::
++ `false`::
+ Mask all control characters other than line feeds and
+ horizontal tabs.
-+ true::
++ `true`::
+ Allow all control characters to be sent to the terminal.
+--
@@ sideband.c: static struct keyword_entry keywords[] = {
-static int allow_control_characters;
+static enum {
-+ ALLOW_NO_CONTROL_CHARACTERS = 0,
-+ ALLOW_ANSI_COLOR_SEQUENCES = 1<<0,
++ ALLOW_NO_CONTROL_CHARACTERS = 0,
++ ALLOW_ANSI_COLOR_SEQUENCES = 1<<0,
+ ALLOW_DEFAULT_ANSI_SEQUENCES = ALLOW_ANSI_COLOR_SEQUENCES,
+ ALLOW_ALL_CONTROL_CHARACTERS = 1<<1,
+} allow_control_characters = ALLOW_ANSI_COLOR_SEQUENCES;
@@ sideband.c: void list_config_color_sideband_slots(struct string_list *list, cons
}
@@ sideband.c: static void strbuf_add_sanitized(struct strbuf *dest, const char *src, int n)
for (; n && *src; src++, n--) {
- if (!iscntrl(*src) || *src == '\t' || *src == '\n')
+ if (!iscntrl(*src) || *src == '\t' || *src == '\n') {
strbuf_addch(dest, *src);
-- else {
-+ else if ((i = handle_ansi_color_sequence(dest, src, n))) {
++ } else if ((i = handle_ansi_color_sequence(dest, src, n))) {
+ src += i;
+ n -= i;
-+ } else {
+ } else {
strbuf_addch(dest, '^');
strbuf_addch(dest, *src == 0x7f ? '?' : 0x40 + *src);
- }
## t/t5409-colorize-remote-messages.sh ##
@@ t/t5409-colorize-remote-messages.sh: test_expect_success 'fallback to color.ui' '
4: fe109cd331 ! 4: cc578465b9 sideband: add options to allow more control sequences to be passed through
@@ Documentation/config/sideband.txt: sideband.allowControlCharacters::
+ a comma-separated list of the following keywords):
+
--
- default::
- color::
+ `default`::
+ `color`::
Allow ANSI color sequences, line feeds and horizontal tabs,
but mask all other control characters. This is the default.
-+ cursor::
++ `cursor:`:
+ Allow control sequences that move the cursor. This is
+ disabled by default.
-+ erase::
++ `erase`::
+ Allow control sequences that erase charactrs. This is
+ disabled by default.
- false::
+ `false`::
Mask all control characters other than line feeds and
horizontal tabs.
## sideband.c ##
@@ sideband.c: static struct keyword_entry keywords[] = {
static enum {
- ALLOW_NO_CONTROL_CHARACTERS = 0,
- ALLOW_ANSI_COLOR_SEQUENCES = 1<<0,
-+ ALLOW_ANSI_CURSOR_MOVEMENTS = 1<<1,
-+ ALLOW_ANSI_ERASE = 1<<2,
+ ALLOW_NO_CONTROL_CHARACTERS = 0,
+ ALLOW_ANSI_COLOR_SEQUENCES = 1<<0,
++ ALLOW_ANSI_CURSOR_MOVEMENTS = 1<<1,
++ ALLOW_ANSI_ERASE = 1<<2,
ALLOW_DEFAULT_ANSI_SEQUENCES = ALLOW_ANSI_COLOR_SEQUENCES,
- ALLOW_ALL_CONTROL_CHARACTERS = 1<<1,
-} allow_control_characters = ALLOW_ANSI_COLOR_SEQUENCES;
@@ sideband.c: static void strbuf_add_sanitized(struct strbuf *dest, const char *sr
}
@@ sideband.c: static void strbuf_add_sanitized(struct strbuf *dest, const char *src, int n)
for (; n && *src; src++, n--) {
- if (!iscntrl(*src) || *src == '\t' || *src == '\n')
+ if (!iscntrl(*src) || *src == '\t' || *src == '\n') {
strbuf_addch(dest, *src);
-- else if ((i = handle_ansi_color_sequence(dest, src, n))) {
-+ else if (allow_control_characters != ALLOW_NO_CONTROL_CHARACTERS &&
-+ (i = handle_ansi_sequence(dest, src, n))) {
+- } else if ((i = handle_ansi_color_sequence(dest, src, n))) {
++ } else if (allow_control_characters != ALLOW_NO_CONTROL_CHARACTERS &&
++ (i = handle_ansi_sequence(dest, src, n))) {
src += i;
n -= i;
} else {
-: ---------- > 5: f2eb0a758c sideband: offer to configure sanitizing on a per-URL basis
--
gitgitgadget
next prev parent reply other threads:[~2026-01-16 22:26 UTC|newest]
Thread overview: 85+ messages / expand[flat|nested] mbox.gz Atom feed top
2025-01-14 18:19 [PATCH 0/3] Sanitize sideband channel messages Johannes Schindelin via GitGitGadget
2025-01-14 18:19 ` [PATCH 1/3] sideband: mask control characters Johannes Schindelin via GitGitGadget
2025-01-15 14:49 ` Phillip Wood
2025-12-02 15:43 ` Johannes Schindelin
2025-01-15 15:17 ` Andreas Schwab
2025-01-15 16:24 ` Junio C Hamano
2025-01-14 18:19 ` [PATCH 2/3] sideband: introduce an "escape hatch" to allow " Johannes Schindelin via GitGitGadget
2025-01-14 18:19 ` [PATCH 3/3] sideband: do allow ANSI color sequences by default Johannes Schindelin via GitGitGadget
2025-01-14 22:50 ` [PATCH 0/3] Sanitize sideband channel messages brian m. carlson
2025-01-16 6:45 ` Junio C Hamano
2025-01-28 16:03 ` Ondrej Pohorelsky
2025-01-31 17:55 ` Junio C Hamano
2025-12-02 14:11 ` Johannes Schindelin
2025-12-03 0:47 ` brian m. carlson
2025-12-03 8:04 ` Johannes Schindelin
2025-01-15 14:49 ` Phillip Wood
2025-12-02 14:56 ` Johannes Schindelin
2025-12-17 14:23 ` [PATCH v2 0/4] " Johannes Schindelin via GitGitGadget
2025-12-17 14:23 ` [PATCH v2 1/4] sideband: mask control characters Johannes Schindelin via GitGitGadget
2026-01-09 12:38 ` Patrick Steinhardt
2026-01-16 19:29 ` Johannes Schindelin
2025-12-17 14:23 ` [PATCH v2 2/4] sideband: introduce an "escape hatch" to allow " Johannes Schindelin via GitGitGadget
2025-12-18 2:22 ` Junio C Hamano
2025-12-18 17:59 ` Johannes Schindelin
2025-12-19 13:33 ` Junio C Hamano
2026-01-16 19:25 ` Johannes Schindelin
2026-01-09 12:38 ` Patrick Steinhardt
2025-12-17 14:23 ` [PATCH v2 3/4] sideband: do allow ANSI color sequences by default Johannes Schindelin via GitGitGadget
2026-01-09 12:38 ` Patrick Steinhardt
2026-01-16 19:38 ` Johannes Schindelin
2025-12-17 14:23 ` [PATCH v2 4/4] sideband: add options to allow more control sequences to be passed through Johannes Schindelin via GitGitGadget
2026-01-09 12:38 ` Patrick Steinhardt
2026-01-10 17:26 ` brian m. carlson
2026-01-15 21:14 ` Jeff King
2026-01-15 21:36 ` Junio C Hamano
2026-01-15 23:12 ` Johannes Schindelin
2026-01-16 6:45 ` Patrick Steinhardt
2026-01-16 12:12 ` Ondrej Pohorelsky
2026-01-16 15:21 ` Junio C Hamano
2026-01-16 18:46 ` Johannes Schindelin
2026-01-16 19:24 ` Junio C Hamano
2026-01-19 7:20 ` Patrick Steinhardt
2026-01-19 22:16 ` brian m. carlson
2026-01-20 2:41 ` D. Ben Knoble
2026-01-20 17:05 ` Junio C Hamano
2026-01-20 19:31 ` Jeff King
2026-01-20 20:11 ` Junio C Hamano
2026-01-21 7:39 ` Patrick Steinhardt
2026-01-22 12:29 ` Johannes Schindelin
2026-01-22 17:58 ` Junio C Hamano
2026-01-15 23:10 ` brian m. carlson
2026-02-03 1:11 ` Junio C Hamano
2026-02-03 7:12 ` Johannes Schindelin
2026-02-03 19:00 ` Junio C Hamano
2026-02-04 19:35 ` Junio C Hamano
2026-01-16 19:47 ` Johannes Schindelin
2026-01-16 22:26 ` Johannes Schindelin via GitGitGadget [this message]
2026-01-16 22:26 ` [PATCH v3 1/5] sideband: mask control characters Johannes Schindelin via GitGitGadget
2026-01-16 22:26 ` [PATCH v3 2/5] sideband: introduce an "escape hatch" to allow " Johannes Schindelin via GitGitGadget
2026-01-16 22:26 ` [PATCH v3 3/5] sideband: do allow ANSI color sequences by default Johannes Schindelin via GitGitGadget
2026-01-16 22:26 ` [PATCH v3 4/5] sideband: add options to allow more control sequences to be passed through Johannes Schindelin via GitGitGadget
2026-01-16 22:26 ` [PATCH v3 5/5] sideband: offer to configure sanitizing on a per-URL basis Johannes Schindelin via GitGitGadget
2026-01-16 22:32 ` [PATCH v3 0/5] Sanitize sideband channel messages Johannes Schindelin
2026-02-03 10:17 ` [PATCH v4 0/6] " Johannes Schindelin via GitGitGadget
2026-02-03 10:17 ` [PATCH v4 1/6] sideband: mask control characters Johannes Schindelin via GitGitGadget
2026-02-03 10:17 ` [PATCH v4 2/6] sideband: introduce an "escape hatch" to allow " Johannes Schindelin via GitGitGadget
2026-02-03 10:17 ` [PATCH v4 3/6] sideband: do allow ANSI color sequences by default Johannes Schindelin via GitGitGadget
2026-02-03 10:18 ` [PATCH v4 4/6] sideband: add options to allow more control sequences to be passed through Johannes Schindelin via GitGitGadget
2026-02-03 10:18 ` [PATCH v4 5/6] sideband: offer to configure sanitizing on a per-URL basis Johannes Schindelin via GitGitGadget
2026-02-03 10:18 ` [PATCH v4 6/6] sideband: delay sanitizing by default to Git v3.0 Johannes Schindelin via GitGitGadget
2026-02-04 19:26 ` [PATCH v4 0/6] Sanitize sideband channel messages Junio C Hamano
2026-02-05 14:48 ` Junio C Hamano
2026-02-13 23:50 ` Junio C Hamano
2026-03-02 18:11 ` [PATCH 0/3] Sanitizing sideband output Junio C Hamano
2026-03-02 18:11 ` [PATCH 1/3] sideband: drop 'default' configuration Junio C Hamano
2026-03-02 18:11 ` [PATCH 2/3] sideband: delay sanitizing by default to Git v3.0 Junio C Hamano
2026-03-02 18:11 ` [PATCH 3/3] sideband: conditional documentation fix Junio C Hamano
2026-03-05 23:34 ` [PATCH v5 0/7] Sanitizing sideband output Junio C Hamano
2026-03-05 23:34 ` [PATCH v5 1/7] sideband: mask control characters Junio C Hamano
2026-03-05 23:34 ` [PATCH v5 2/7] sideband: introduce an "escape hatch" to allow " Junio C Hamano
2026-03-05 23:34 ` [PATCH v5 3/7] sideband: do allow ANSI color sequences by default Junio C Hamano
2026-03-05 23:34 ` [PATCH v5 4/7] sideband: add options to allow more control sequences to be passed through Junio C Hamano
2026-03-05 23:34 ` [PATCH v5 5/7] sideband: offer to configure sanitizing on a per-URL basis Junio C Hamano
2026-03-05 23:34 ` [PATCH v5 6/7] sideband: drop 'default' configuration Junio C Hamano
2026-03-05 23:34 ` [PATCH v5 7/7] sideband: delay sanitizing by default to Git v3.0 Junio C Hamano
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=pull.1853.v3.git.1768602373.gitgitgadget@gmail.com \
--to=gitgitgadget@gmail.com \
--cc=git@vger.kernel.org \
--cc=johannes.schindelin@gmx.de \
--cc=opohorel@redhat.com \
--cc=peff@peff.net \
--cc=phillip.wood123@gmail.com \
--cc=ps@pks.im \
--cc=sandals@crustytoothpaste.net \
--cc=schwab@linux-m68k.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox