On Sat, Feb 13, 2016 at 03:24:16PM +0100, larsxschnei...@gmail.com wrote:

> From: Lars Schneider <larsxschnei...@gmail.com>
> 
> If config values are queried using 'git config' (e.g. via --get,
> --get-all, --get-regexp, or --list flag) then it is sometimes hard to
> find the configuration file where the values were defined.
> 
> Teach 'git config' the '--show-origin' option to print the source
> configuration file for every printed value.

Thanks, I think this version fixes the correctness issues I mentioned
earlier. I do still have nits to pick (of course :) ), that we may or
may not want to deal with.

> +static void show_config_origin(struct strbuf *buf)
> +{
> +     const char term = end_null ? '\0' : '\t';
> +     const char *type;
> +     const char *name;
> +
> +     current_config_type_name(&type, &name);

This double out-parameter feels like a clunky interface.

I was tempted to suggest that we simply make the "struct config_source"
available to builtin/config.c (which is already pretty intimate with the
rest of the config code), and then it can pick out what it wants. But
there _is_ some logic in the function to convert the NULL "cf" into
"cmdline".

Perhaps it would be simpler to just have two accessor functions, and do:

  strbuf_addstr(buf, current_config_type());
  ...
  strbuf_addstr(buf, current_config_name());

I admit it is a pretty minor point, though.

>  static int show_all_config(const char *key_, const char *value_, void *cb)
>  {
> +     if (show_origin) {
> +       struct strbuf buf = STRBUF_INIT;
> +       show_config_origin(&buf);
> +       fwrite(buf.buf, 1, buf.len, stdout);
> +       strbuf_release(&buf);
> +     }

The indentation is funky here.

The use of fwrite() to catch the embedded NULs is subtle enough that it
might be worth a comment.

It also made me wonder how format_config() handles this. It looks like
we send the result eventually to fwrite() there, so it all works (and it
does _not_ have the comment I mentioned :) ).

> +test_expect_success '--show-origin' '
> +     >.git/config &&
> +     >"$HOME"/.gitconfig &&
> +     INCLUDE_DIR="$HOME/include" &&
> +     mkdir -p "$INCLUDE_DIR" &&
> +     cat >"$INCLUDE_DIR"/absolute.include <<-EOF &&
> +             [user]
> +                     absolute = include
> +     EOF
> +     cat >"$INCLUDE_DIR"/relative.include <<-EOF &&
> +             [user]
> +                     relative = include
> +     EOF
> +     test_config_global user.global "true" &&
> +     test_config_global user.override "global" &&
> +     test_config_global include.path "$INCLUDE_DIR"/absolute.include &&
> +     test_config include.path ../include/relative.include &&
> +     test_config user.local "true" &&
> +     test_config user.override "local" &&
> +
> +     cat >expect <<-EOF &&
> +             file:$HOME/.gitconfig   user.global=true
> +             file:$HOME/.gitconfig   user.override=global
> +             file:$HOME/.gitconfig   
> include.path=$INCLUDE_DIR/absolute.include
> +             file:$INCLUDE_DIR/absolute.include      user.absolute=include
> +             file:.git/config        include.path=../include/relative.include
> +             file:.git/../include/relative.include   user.relative=include
> +             file:.git/config        user.local=true
> +             file:.git/config        user.override=local
> +             cmdline:        user.cmdline=true
> +     EOF
> +     git -c user.cmdline=true config --list --show-origin >output &&
> +     test_cmp expect output &&
> +
> +     cat >expect <<-EOF &&
> +             file:$HOME/.gitconfigQuser.global
> +             trueQfile:$HOME/.gitconfigQuser.override
> +             globalQfile:$HOME/.gitconfigQinclude.path
> +             
> $INCLUDE_DIR/absolute.includeQfile:$INCLUDE_DIR/absolute.includeQuser.absolute
> +             includeQfile:.git/configQinclude.path
> +             
> ../include/relative.includeQfile:.git/../include/relative.includeQuser.relative
> +             includeQfile:.git/configQuser.local
> +             trueQfile:.git/configQuser.override
> +             localQcmdline:Quser.cmdline
> +             trueQ
> +     EOF

I see you split this up more, but there's still quite a bit going on in
this one block. IMHO, it would be more customary in our tests to put the
setup into one test_expect_success block, then each of these
expect-run-cmp blocks into their own test_expect_success.

It does mean that the setup mutates the global test state for further
tests (and you should stop using test_config_*, which clean up at the
end of the block), but I think that's the right thing here. The point of
test_config is "flip on this switch just for a moment, so we can test
its effect without hurting further tests". But these are config tests in
the first place, and it is OK for them to show a progression of
mutations of the config (you'll note that like the other tests in this
script, you are clearing out .git/config in the first place).

-Peff
--
To unsubscribe from this list: send the line "unsubscribe git" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to