All of lore.kernel.org
 help / color / mirror / Atom feed
From: Michael J Gruber <git@drmicha.warpmail.net>
To: Giuseppe Scrivano <gscrivano@gnu.org>
Cc: git@vger.kernel.org
Subject: Re: [RFC][PATCH] Print the usage string on stdout instead of stderr.
Date: Tue, 18 May 2010 11:43:42 +0200	[thread overview]
Message-ID: <4BF2614E.50003@drmicha.warpmail.net> (raw)
In-Reply-To: <87d3wutt34.fsf@thor.thematica.it>

Giuseppe Scrivano venit, vidit, dixit 17.05.2010 18:02:
> Michael J Gruber <git@drmicha.warpmail.net> writes:
> 
>> Your patch puts both on stdout, and that is a problem, not only for
>> several test which could be adjusted, but also for scripters.
>>
>> A patch which *really* only changes '-h' to use stdout would certainly
>> not be objected. Actually, most calling sites are probably the "error
>> case" so that you only need to make sure that the "-h" path get's to use
>> a different output file descriptor.
> 
> I have changed my patches following your suggestions.

Thanks!

> I have also fixed two tests.
> 
> Cheers,
> Giuseppe
> 
> 
> 
> From 00100c61db30725011edf62e7e0e7bc6ac685cb0 Mon Sep 17 00:00:00 2001
> From: Giuseppe Scrivano <gscrivano@gnu.org>
> Date: Mon, 17 May 2010 17:34:41 +0200
> Subject: [PATCH] print the usage string on stdout instead of stderr
> 
> When -h is used, print usage messages on stdout.  If a command is invoked with
> wrong arguments then print the usage messages on stderr.
> 
> Signed-off-by: Giuseppe Scrivano <gscrivano@gnu.org>
> ---
>  parse-options.c               |   64 +++++++++++++++++++++--------------------
>  t/t0040-parse-options.sh      |    8 +++--
>  t/t1502-rev-parse-parseopt.sh |    6 ++--
>  3 files changed, 41 insertions(+), 37 deletions(-)
> 
> diff --git a/parse-options.c b/parse-options.c
> index 8546d85..c8aaf95 100644
> --- a/parse-options.c
> +++ b/parse-options.c
> @@ -5,7 +5,7 @@
>  #include "color.h"
>  
>  static int parse_options_usage(const char * const *usagestr,
> -			       const struct option *opts);
> +			       const struct option *opts, int err);
>  
>  #define OPT_SHORT 1
>  #define OPT_UNSET 2
> @@ -352,7 +352,7 @@ void parse_options_start(struct parse_opt_ctx_t *ctx,
>  }
>  
>  static int usage_with_options_internal(const char * const *,
> -				       const struct option *, int);
> +				       const struct option *, int, int);
>  
>  int parse_options_step(struct parse_opt_ctx_t *ctx,
>  		       const struct option *options,
> @@ -380,10 +380,10 @@ int parse_options_step(struct parse_opt_ctx_t *ctx,
>  		if (arg[1] != '-') {
>  			ctx->opt = arg + 1;
>  			if (internal_help && *ctx->opt == 'h')
> -				return parse_options_usage(usagestr, options);
> +				return parse_options_usage(usagestr, options, 0);
>  			switch (parse_short_opt(ctx, options)) {
>  			case -1:
> -				return parse_options_usage(usagestr, options);
> +				return parse_options_usage(usagestr, options, 1);
>  			case -2:
>  				goto unknown;
>  			}
> @@ -391,10 +391,10 @@ int parse_options_step(struct parse_opt_ctx_t *ctx,
>  				check_typos(arg + 1, options);
>  			while (ctx->opt) {
>  				if (internal_help && *ctx->opt == 'h')
> -					return parse_options_usage(usagestr, options);
> +					return parse_options_usage(usagestr, options, 0);
>  				switch (parse_short_opt(ctx, options)) {
>  				case -1:
> -					return parse_options_usage(usagestr, options);
> +					return parse_options_usage(usagestr, options, 1);
>  				case -2:
>  					/* fake a short option thing to hide the fact that we may have
>  					 * started to parse aggregated stuff
> @@ -418,12 +418,12 @@ int parse_options_step(struct parse_opt_ctx_t *ctx,
>  		}
>  
>  		if (internal_help && !strcmp(arg + 2, "help-all"))
> -			return usage_with_options_internal(usagestr, options, 1);
> +			return usage_with_options_internal(usagestr, options, 1, 0);
>  		if (internal_help && !strcmp(arg + 2, "help"))
> -			return parse_options_usage(usagestr, options);
> +			return parse_options_usage(usagestr, options, 0);
>  		switch (parse_long_opt(ctx, arg + 2, options)) {
>  		case -1:
> -			return parse_options_usage(usagestr, options);
> +			return parse_options_usage(usagestr, options, 1);
>  		case -2:
>  			goto unknown;
>  		}
> @@ -468,7 +468,7 @@ int parse_options(int argc, const char **argv, const char *prefix,
>  	return parse_options_end(&ctx);
>  }
>  
> -static int usage_argh(const struct option *opts)
> +static int usage_argh(const struct option *opts, FILE *outfile)
>  {
>  	const char *s;
>  	int literal = (opts->flags & PARSE_OPT_LITERAL_ARGHELP) || !opts->argh;
> @@ -479,72 +479,74 @@ static int usage_argh(const struct option *opts)
>  			s = literal ? "[%s]" : "[<%s>]";
>  	else
>  		s = literal ? " %s" : " <%s>";
> -	return fprintf(stderr, s, opts->argh ? opts->argh : "...");
> +	return fprintf(outfile, s, opts->argh ? opts->argh : "...");
>  }
>  
>  #define USAGE_OPTS_WIDTH 24
>  #define USAGE_GAP         2
>  
>  static int usage_with_options_internal(const char * const *usagestr,
> -				const struct option *opts, int full)
> +				const struct option *opts, int full, int err)
>  {
> +	FILE *outfile = err ? stderr : stdout;
> +
>  	if (!usagestr)
>  		return PARSE_OPT_HELP;
>  
> -	fprintf(stderr, "usage: %s\n", *usagestr++);
> +	fprintf(outfile, "usage: %s\n", *usagestr++);
>  	while (*usagestr && **usagestr)
> -		fprintf(stderr, "   or: %s\n", *usagestr++);
> +		fprintf(outfile, "   or: %s\n", *usagestr++);
>  	while (*usagestr) {
> -		fprintf(stderr, "%s%s\n",
> +		fprintf(outfile, "%s%s\n",
>  				**usagestr ? "    " : "",
>  				*usagestr);
>  		usagestr++;
>  	}
>  
>  	if (opts->type != OPTION_GROUP)
> -		fputc('\n', stderr);
> +		fputc('\n', outfile);
>  
>  	for (; opts->type != OPTION_END; opts++) {
>  		size_t pos;
>  		int pad;
>  
>  		if (opts->type == OPTION_GROUP) {
> -			fputc('\n', stderr);
> +			fputc('\n', outfile);
>  			if (*opts->help)
> -				fprintf(stderr, "%s\n", opts->help);
> +				fprintf(outfile, "%s\n", opts->help);
>  			continue;
>  		}
>  		if (!full && (opts->flags & PARSE_OPT_HIDDEN))
>  			continue;
>  
> -		pos = fprintf(stderr, "    ");
> +		pos = fprintf(outfile, "    ");
>  		if (opts->short_name && !(opts->flags & PARSE_OPT_NEGHELP)) {
>  			if (opts->flags & PARSE_OPT_NODASH)
> -				pos += fprintf(stderr, "%c", opts->short_name);
> +				pos += fprintf(outfile, "%c", opts->short_name);
>  			else
> -				pos += fprintf(stderr, "-%c", opts->short_name);
> +				pos += fprintf(outfile, "-%c", opts->short_name);
>  		}
>  		if (opts->long_name && opts->short_name)
> -			pos += fprintf(stderr, ", ");
> +			pos += fprintf(outfile, ", ");
>  		if (opts->long_name)
> -			pos += fprintf(stderr, "--%s%s",
> +			pos += fprintf(outfile, "--%s%s",
>  				(opts->flags & PARSE_OPT_NEGHELP) ?  "no-" : "",
>  				opts->long_name);
>  		if (opts->type == OPTION_NUMBER)
> -			pos += fprintf(stderr, "-NUM");
> +			pos += fprintf(outfile, "-NUM");
>  
>  		if (!(opts->flags & PARSE_OPT_NOARG))
> -			pos += usage_argh(opts);
> +			pos += usage_argh(opts, outfile);
>  
>  		if (pos <= USAGE_OPTS_WIDTH)
>  			pad = USAGE_OPTS_WIDTH - pos;
>  		else {
> -			fputc('\n', stderr);
> +			fputc('\n', outfile);
>  			pad = USAGE_OPTS_WIDTH;
>  		}
> -		fprintf(stderr, "%*s%s\n", pad + USAGE_GAP, "", opts->help);
> +		fprintf(outfile, "%*s%s\n", pad + USAGE_GAP, "", opts->help);
>  	}
> -	fputc('\n', stderr);
> +	fputc('\n', outfile);
>  
>  	return PARSE_OPT_HELP;
>  }
> @@ -552,7 +554,7 @@ static int usage_with_options_internal(const char * const *usagestr,
>  void usage_with_options(const char * const *usagestr,
>  			const struct option *opts)
>  {
> -	usage_with_options_internal(usagestr, opts, 0);
> +	usage_with_options_internal(usagestr, opts, 0, 1);
>  	exit(129);
>  }
>  
> @@ -565,9 +567,9 @@ void usage_msg_opt(const char *msg,
>  }
>  
>  static int parse_options_usage(const char * const *usagestr,
> -			       const struct option *opts)
> +			       const struct option *opts, int err)
>  {
> -	return usage_with_options_internal(usagestr, opts, 0);
> +	return usage_with_options_internal(usagestr, opts, 0, err);
>  }
>  
>  
> diff --git a/t/t0040-parse-options.sh b/t/t0040-parse-options.sh
> index 3d450ed..2092450 100755
> --- a/t/t0040-parse-options.sh
> +++ b/t/t0040-parse-options.sh
> @@ -7,7 +7,7 @@ test_description='our own option parser'
>  
>  . ./test-lib.sh
>  
> -cat > expect.err << EOF
> +cat > expect << EOF
>  usage: test-parse-options <options>
>  
>      -b, --boolean         get a boolean
> @@ -46,10 +46,12 @@ EOF
>  
>  test_expect_success 'test help' '
>  	test_must_fail test-parse-options -h > output 2> output.err &&
> -	test ! -s output &&
> -	test_cmp expect.err output.err
> +	test ! -s output.err &&
> +	test_cmp expect output
>  '
>  
> +mv expect expect.err
> +
>  cat > expect << EOF
>  boolean: 2
>  integer: 1729
> diff --git a/t/t1502-rev-parse-parseopt.sh b/t/t1502-rev-parse-parseopt.sh
> index e504058..660487d 100755
> --- a/t/t1502-rev-parse-parseopt.sh
> +++ b/t/t1502-rev-parse-parseopt.sh
> @@ -3,7 +3,7 @@
>  test_description='test git rev-parse --parseopt'
>  . ./test-lib.sh
>  
> -cat > expect.err <<EOF
> +cat > expect <<EOF
>  usage: some-command [options] <args>...
>  
>      some-command does foo and bar!
> @@ -38,8 +38,8 @@ extra1    line above used to cause a segfault but no longer does
>  EOF
>  
>  test_expect_success 'test --parseopt help output' '
> -	git rev-parse --parseopt -- -h 2> output.err < optionspec
> -	test_cmp expect.err output.err
> +	git rev-parse --parseopt -- -h > output < optionspec
> +	test_cmp expect output
>  '
>  
>  cat > expect <<EOF

I haven't checked whether this covers all code paths but other than that
it looks OK to me, and the tests pass.

Cheers,
Michael

  reply	other threads:[~2010-05-18  9:43 UTC|newest]

Thread overview: 14+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2010-05-17  9:48 [RFC][PATCH] Print the usage string on stdout instead of stderr Giuseppe Scrivano
2010-05-17 11:46 ` Michael J Gruber
2010-05-17 12:07   ` Miles Bader
2010-05-17 13:30     ` Michael J Gruber
2010-05-17 13:54       ` Miles Bader
2010-05-17 14:07       ` Giuseppe Scrivano
2010-05-17 14:11         ` Tay Ray Chuan
2010-05-17 12:40   ` Giuseppe Scrivano
2010-05-17 13:30     ` Michael J Gruber
2010-05-17 16:02       ` Giuseppe Scrivano
2010-05-18  9:43         ` Michael J Gruber [this message]
2010-05-24 20:51           ` Giuseppe Scrivano
2010-05-25  6:46             ` Michael J Gruber
2010-05-25  8:40               ` [PATCH v2] " Giuseppe Scrivano

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=4BF2614E.50003@drmicha.warpmail.net \
    --to=git@drmicha.warpmail.net \
    --cc=git@vger.kernel.org \
    --cc=gscrivano@gnu.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.