On Mon, Jun 06 2022, Taylor Blau wrote: > On Mon, Jun 06, 2022 at 03:52:19PM -0700, Junio C Hamano wrote: >> Taylor Blau <me@xxxxxxxxxxxx> writes: >> >> > This short patch series adds support for a new `--count` argument for limiting >> > the output of `show-ref` (à-la the `for-each-ref` option by the same name). >> >> It makes me wonder why we limit this to show-ref. >> >> $ git --pipe-to-head-N=3 any-command args... >> >> IOW, having to add an option like this feels absurd. > > I don't disagree. But `--pipe-to-head-N=3` feels like too broad a > stroke. This series at least imitates `for-each-ref`'s `--count` > option, which makes it feel acceptable to me (if not a little silly). Yeah, although I do think it's worthwhile to think about where certain UX decisions are leading us, i.e. the logical conclusion here is to have every command that emits >1 lines support --count, which as your patch here shows needs special support, and even in your case you haven't implemented it in a way that's compatible with all existing options. B.t.w. why would a --count for --verify not just by supported have these be equivalent: # same git tag --count=3 --verify <name> git tag --verify <name> | head -n 3 >> > This is useful in contexts where a caller wants to avoid enumerating more >> > references than necessary (e.g., they only care whether a tag exists, but not >> > how many or what they are called) but doesn't have control of the output stream >> > (e.g., they are in Ruby and can't pipe the output to `head -n 1`). >> >> Are you saying that Ruby is incapable of run a command line like >> >> av[0] = "sh" >> av[1] = "-c" >> av[2] = "git show-ref blah | head -n 1" >> av[3] = NULL > > No, Ruby is perfectly capable of doing that. But it involves an extra > process (two, if `head` isn't a shell builtin) [...] Maybe this really is a limitation of ruby, or maybe I'm missing something, but doesn't it support just opening a process without "sh -c" and piping the output to your current process, as this perl command which makes use of execve() will do: $ perl -Mautodie=:all -wE ' my $i = 0; my $lim = shift; open my $fh, "-|", qw(git show-ref master); while (<$fh>) { last if $i++ >= $lim; print "$.: $_"; };' 10 Some quick searching for docs online suggests that Ruby's Open3 and/or Process.spawn might be the equivalent. Note that if you replace that qw(git show-ref master) with e.g.: "git show-ref master | tail -n 20" That you'll get 3x execve(), one sh -c, another for "git" and another for "tail", but in the first case you'll only get the execve(). Isn't that something that would make this workaround unnecessary? Well, maybe not because... > [...]and the additional > overhead of creating a pipe and passing data through it instead of > writing directly to stdout. It wouldn't take care of this part, but I'm struggling to think of cases where you'd be running this in the context of github.com and not already need to capture the output of the command. I.e. surely you're already piping stdout/stderr into your program, no?