On Fri, Dec 05, 2014 at 10:00:05AM -0800, Junio C Hamano wrote: > Jeff King <peff@xxxxxxxx> writes: > > > The only downside I can think of is that we may truncate the message in > > exceptional circumstances. But is it really any less helpful to say: > > > > error: unable to open file: some-incredibly-long-filename-aaaaaa... > > > > than printing out an extra 100 lines of "a"? And I mean the "..." > > literally. I think mkerror() should indicate the truncation with a > > "...", just so that it is clear to the user. It should almost never > > happen, but when it does, it can be helpful to show the user that yes, > > we know we are truncating the message, and it is not that git truncated > > your filename during the operation. > > > > Is this truncation really a concern, and/or is there some other downside > > I'm not thinking of? > > I am more worried about variable length part pushing the information > that is given later out to the right, e.g. "error: missing file '%s' > prevents us from doing X". Chomping to [1024] is not a good > strategy for that kind of message; abbreviating %s to /path/name/... > (again, with literally "...") would be. True. Unfortunately I do not think there is an easy way to truncate the expanded strings used by placeholders. The two approaches I could think of are: 1. Loop over the va_list and tweak any pointers we find. Except that loop means we actually need to loop over the _format string_, since that is the only thing which tells us which placeholders are strings. But I am not even sure there is a portable way to munge a va_list, as opposed to just reading it. 2. Transparently rewrite '%s' in the format string to '%.128s' or similar. This also requires iterating over the format string, but it's fairly straightforward. Code is below, but it's not _quite_ right. We would want to conditionally add "..." based on whether or not the particular item was truncated. But we cannot know when munging the format string if that will happen or not (we know _something_ will be truncated, but not which string). I don't think you can do it right without reimplementing vsnprintf yourself. Which is obviously a terrible idea. I still think truncation is going to be a non-issue in practice, and I'd rather deal with that potential fallout than have to deal with memory management in every error handler that uses this technique. But I don't have much more to say on the matter, so if you disagree, I guess that's that. Unless we can do something clever with a set of global error strbufs or something (i.e., that expand as needed, but the caller does not have to free themselves, as they will get recycled eventually). That has its own corner cases, though. Anyway, the truncating-fmt code is below, for fun. -Peff static int abbrev_fmt(char *buf, int len, int abbrev, const char *fmt, va_list ap) { va_list cp; int got; va_copy(cp, ap); got = vsnprintf(buf, len, fmt, ap); if (got >= len) { struct strbuf munged = STRBUF_INIT; while (*fmt) { char ch = *fmt++; strbuf_addch(&munged, ch); if (ch == '%') { if (*fmt == 's') { strbuf_addf(&munged, ".%ds...", abbrev); fmt++; } else /* skips past %%-quoting */ strbuf_addch(&munged, *fmt); } } got = vsnprintf(buf, len, munged.buf, cp); strbuf_release(&munged); } return got; } -- To unsubscribe from this list: send the line "unsubscribe git" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html