Hi, Han-Wen Nienhuys wrote:
> The colorization is controlled with the config setting "color.remote". > > Supported keywords are "error", "warning", "hint" and "success". They > are highlighted if they appear at the start of the line, which is > common in error messages, eg. > > ERROR: commit is missing Change-Id A few questions: - should this be documented somewhere in Documentation/technical/*protocol*? That way, server implementors can know to take advantage of it. - how does this interact with (future) internationalization of server messages? If my server knows that the client speaks French, should they send "ERROR:" anyway and rely on the client to translate it? Or are we deferring that question to a later day? [...] > The Git push process itself prints lots of non-actionable messages > (eg. bandwidth statistics, object counters for different phases of the > process), and my hypothesis is that these obscure the actionable error > messages that our server sends back. Highlighting keywords in the > draws more attention to actionable messages. I'd be interested in ways to minimize Git's contribution to this as well. E.g. can we make more use of \r to make client-produced progress messages take less screen real estate? Should some of the servers involved (e.g., Gerrit) do so as well? > The highlighting is done on the client rather than server side, so > servers don't have to grow capabilities to understand terminal escape > codes and terminal state. It also consistent with the current state > where Git is control of the local display (eg. prefixing messages with > "remote: "). As a strawman idea, what would you think of a way to allow the server to do some coloration by using color tags like <red>Erreur</red>: ... ? As an advantage, this would give the server more control. As a disadvantage, it is not so useful as "semantic markup", meaning it is harder to figure out how to present to accessibility tools in a meaningful way. Plus, there's the issue of usefulness with existing servers you mentioned: > Finally, this solution is backwards compatible: many servers already > prefix their messages with "error", and they will benefit from this > change without requiring a server update. Yes, this seems like a compelling reason to follow this approach. [...] > --- a/Documentation/config.txt > +++ b/Documentation/config.txt > @@ -1229,6 +1229,15 @@ color.push:: > color.push.error:: > Use customized color for push errors. > > +color.remote:: > + A boolean to enable/disable colored remote output. If unset, > + then the value of `color.ui` is used (`auto` by default). > + > +color.remote.<slot>:: > + Use customized color for each remote keywords. `<slot>` may be > + `hint`, `warning`, `success` or `error` which match the > + corresponding keyword. What positions in a remote message are matched? If a server prints a message about something being discouraged because it is error-prone, would the "error" in error-prone turn red? [...] > --- a/sideband.c > +++ b/sideband.c > @@ -1,6 +1,103 @@ > #include "cache.h" > +#include "color.h" > +#include "config.h" > #include "pkt-line.h" > #include "sideband.h" > +#include "help.h" > + > +struct kwtable { nit: perhaps kwtable_entry? > + /* > + * We use keyword as config key so it can't contain funny chars like > + * spaces. When we do that, we need a separate field for slot name in > + * load_sideband_colors(). > + */ > + const char *keyword; > + char color[COLOR_MAXLEN]; > +}; > + > +static struct kwtable keywords[] = { > + { "hint", GIT_COLOR_YELLOW }, [...] > +// Returns a color setting (GIT_COLOR_NEVER, etc). nit: Git uses C89-style /* */ comments. > +static int use_sideband_colors(void) Makes sense. [...] > +void list_config_color_sideband_slots(struct string_list *list, const char > *prefix) > +{ > + int i; > + > + for (i = 0; i < ARRAY_SIZE(keywords); i++) > + list_config_item(list, prefix, keywords[i].keyword); > +} Not about this patch: I wonder if we can eliminate this boilerplate some time in the future. [...] > +/* > + * Optionally highlight some keywords in remote output if they appear at the > + * start of the line. This should be called for a single line only. > + */ > +void maybe_colorize_sideband(struct strbuf *dest, const char *src, int n) Can this be static? What does 'n' represent? Can the comment say? (Or if it's the length of src, can it be renamed to srclen?) Super-pedantic nit: even if there are multiple special words at the start, this will only highlight one. :) So it could say something like "Optionally check if the first word on the line is a keyword and highlight it if so." > +{ > + int i; > + if (!want_color_stderr(use_sideband_colors())) { nit: whitespace damage (you can check for this with "git show --check"). There's a bit more elsewhere. > + strbuf_add(dest, src, n); > + return; > + } > + > + while (isspace(*src)) { > + strbuf_addch(dest, *src); > + src++; > + n--; > + } > + > + for (i = 0; i < ARRAY_SIZE(keywords); i++) { > + struct kwtable* p = keywords + i; nit: * should attach to the variable, C-style. You can use "make style" to do some automatic formatting (though this is a bit experimental, so do double-check the results). > + int len = strlen(p->keyword); > + /* > + * Match case insensitively, so we colorize output from > existing > + * servers regardless of the case that they use for their > + * messages. We only highlight the word precisely, so > + * "successful" stays uncolored. > + */ > + if (!strncasecmp(p->keyword, src, len) && !isalnum(src[len])) { > + strbuf_addstr(dest, p->color); > + strbuf_add(dest, src, len); > + strbuf_addstr(dest, GIT_COLOR_RESET); > + n -= len; > + src += len; > + break; > + } Sensible. [...] > @@ -48,8 +145,10 @@ int recv_sideband(const char *me, int in_stream, int out) > len--; > switch (band) { > case 3: > - strbuf_addf(&outbuf, "%s%s%s", outbuf.len ? "\n" : "", > - DISPLAY_PREFIX, buf + 1); > + strbuf_addf(&outbuf, "%s%s", outbuf.len ? "\n" : "", > + DISPLAY_PREFIX); > + maybe_colorize_sideband(&outbuf, buf + 1, len); > + > retval = SIDEBAND_REMOTE_ERROR; > break; > case 2: > @@ -69,20 +168,22 @@ int recv_sideband(const char *me, int in_stream, int out) > if (!outbuf.len) > strbuf_addstr(&outbuf, DISPLAY_PREFIX); > if (linelen > 0) { > - strbuf_addf(&outbuf, "%.*s%s%c", > - linelen, b, suffix, *brk); > - } else { > - strbuf_addch(&outbuf, *brk); > + maybe_colorize_sideband(&outbuf, b, > linelen); > + strbuf_addstr(&outbuf, suffix); > } > + > + strbuf_addch(&outbuf, *brk); > xwrite(2, outbuf.buf, outbuf.len); Nice. This looks cleaner than what was there before, which is always a good sign. [...] > --- /dev/null > +++ b/t/t5409-colorize-remote-messages.sh Thanks for these. It may make sense to have a test with some leading whitespace as well. [...] > +test_expect_success 'setup' ' > + mkdir .git/hooks && > + cat << EOF > .git/hooks/update && > +#!/bin/sh > +echo error: error > +echo hint: hint > +echo success: success > +echo warning: warning > +echo prefixerror: error > +exit 0 > +EOF > + chmod +x .git/hooks/update && Please use write_script for this, since on esoteric platforms it picks an appropriate shell. If you use <<-\EOF instead of <<EOF, that has two advantages: - it lets you indent the script - it saves the reviewer from having to look for $ escapes inside the script body > + echo 1 >file && > + git add file && > + git commit -m 1 && This can use test_commit. See t/README for details. [...] > +test_expect_success 'push' ' > + git -c color.remote=always push -f origin HEAD:refs/heads/newbranch > 2>output && > + test_decode_color <output >decoded && > + grep "<BOLD;RED>error<RESET>:" decoded && > + grep "<YELLOW>hint<RESET>:" decoded && > + grep "<BOLD;GREEN>success<RESET>:" decoded && > + grep "<BOLD;YELLOW>warning<RESET>:" decoded && > + grep "prefixerror: error" decoded > +' > + > +test_expect_success 'push with customized color' ' > + git -c color.remote=always -c color.remote.error=white push -f origin > HEAD:refs/heads/newbranch2 2>output && > + test_decode_color <output >decoded && > + grep "<WHITE>error<RESET>:" decoded && > + grep "<YELLOW>hint<RESET>:" decoded && > + grep "<BOLD;GREEN>success<RESET>:" decoded && > + grep "<BOLD;YELLOW>warning<RESET>:" decoded > +' Nice. Thanks and hope that helps, Jonathan