Han-Wen Nienhuys <hanwen@xxxxxxxxxx> writes: > Supported keywords are "error", "warning", "hint" and "success". > > TODO: > * make the coloring optional? What variable to use? > * doc for the coloring option. > * how to test? At the sideband layer, the sender only has the same level of information as the receiver has regarding what the random bytes that happens to be sent on the sideband, and that shows from the need to guess with "keywords" in this code. So if you are operating on the sideband layer, I think the right way to do so would be to add the coloring on the receiving end. That way, talking to the same sender, each receiver can decide if s/he prefers or capable of color. Hold your objection a bit. I'll come back to it soon ;-) It theoretically may make more sense to color on the sender side, but that is true only if done at a higher layer that prepares a string and calls into the sideband code to send. That code must know what the bytes _mean_ a lot better than the code at the sideband layer, so we do not have to guess. Having written all the above, I think you are doing this at the receiving end, so this actually makes quite a lot of sense. I was fooled greatly by "EMIT_sideband", which in reality does NOT emit at all. That function is badly misnamed. The function is more like "color sideband payload"; actual "emitting" is still done at the places the code originally "emitted" them to the receiving user. > Signed-off-by: Han-Wen Nienhuys <hanwen@xxxxxxxxxx> > Change-Id: I090412a1288bc2caef0916447e28c2d0199da47d That's an unusual trailer we do not use in this project. > --- > sideband.c | 69 +++++++++++++++++++++++++++++++++++++++++++++++------- > 1 file changed, 60 insertions(+), 9 deletions(-) > > diff --git a/sideband.c b/sideband.c > index 325bf0e97..c8b7cb6dd 100644 > --- a/sideband.c > +++ b/sideband.c > @@ -1,6 +1,53 @@ > #include "cache.h" > #include "pkt-line.h" > #include "sideband.h" > +#include "color.h" > + > +/* > + * Optionally highlight some keywords in remote output if they appear at the > + * start of the line. > + */ > +void emit_sideband(struct strbuf *dest, const char *src, int n) { Open brace on its own line. > + // NOSUBMIT - maybe use transport.color property? Avoid // comment. > + int want_color = want_color_stderr(GIT_COLOR_AUTO); > + > + if (!want_color) { > + strbuf_add(dest, src, n); > + return; > + } > + > + struct kwtable { > + const char* keyword; > + const char* color; In our codebase in C, asterisk sticks to the variable not the type. > + } keywords[] = { > + {"hint", GIT_COLOR_YELLOW}, > + {"warning", GIT_COLOR_BOLD_YELLOW}, > + {"success", GIT_COLOR_BOLD_GREEN}, > + {"error", GIT_COLOR_BOLD_RED}, > + {}, Drop the last sentinel element, and instead stop iteration over the array using (i < ARRAY_SIZE(keywords)). > + }; > + > + while (isspace(*src)) { > + strbuf_addch(dest, *src); > + src++; > + n--; > + } > + > + for (struct kwtable* p = keywords; p->keyword; p++) { Does anybody know if we already use the variable decl inside the "for (...)" construct like this? I know we discussed the idea of using it somewhere as a weather-balloon to see if people with exotic environment would mind, and I certainly do not mind making this patch serve as such a weather-baloon, but if that is what we are doing, I want the commit log message clearly marked as such, so that we can later "git log --grep=C99" to see how long ago such an experiment started. Note that we did attempt to start such an experiment once cf. https://public-inbox.org/git/20170719181956.15845-1-sbeller@xxxxxxxxxx/ but failed. > + int len = strlen(p->keyword); > + 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; > + } > + } > + > + strbuf_add(dest, src, n); > +} > + I wonder how this interacts with exotics like the progress output, which is essentially a single long line with many CRs sprinkled in, sent on many packets? I offhand do not think of a possible bad interaction, but others may be able to think of some. > /* > * Receive multiplexed output stream over git native protocol. > @@ -48,8 +95,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); > + emit_sideband(&outbuf, buf+1, len); > + Let's not lose SP around "+" on both sides. Also you seem to be indenting some lines with all SP and some with mixture of HT and SP. We prefer to use as many 8-column HT and then fill the remainder with SP if needed to align with the opening parenthesis on line above it (imitate the way strbuf_addf() is split into two lines in the original in this hunk). > @@ -69,20 +118,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); > + emit_sideband(&outbuf, b, linelen); > + strbuf_addstr(&outbuf, suffix); Here, suffix is the "clear to the end of the line" sequence (used mostly for progress display), so it is correct to exclude that from the call to "color this sideband payload" helper. Good. Thanks. While there are need for mostly minor fix-ups, the logic seems quite sane. I think we can start without configuration and then "fix" it later. While I am OK with calling that variable "transport.<something>", we should not define/explain it as "color output coming from the other end over the wire transport". Those who want to see messages emitted remotely during "git fetch" in color would want to see the messages generated by "git fetch" locally painted in the same color scheme, so it makes sense to let "git fetch" pay attention and honor that variable even for its own locally generated messages. The variable instead means "color any message, either generated locally or remotely, during an operation that has something to do with object transport", or something like that. Thanks.