Merge branch 'ow/ref-filter-omit-empty'
"git branch --format=..." and "git format-patch --format=..." learns "--omit-empty" to hide refs that whose formatting result becomes an empty string from the output. * ow/ref-filter-omit-empty: branch, for-each-ref, tag: add option to omit empty linesmaint
						commit
						b64894c206
					
				|  | @ -156,6 +156,10 @@ in another worktree linked to the same repository. | ||||||
| --ignore-case:: | --ignore-case:: | ||||||
| 	Sorting and filtering branches are case insensitive. | 	Sorting and filtering branches are case insensitive. | ||||||
|  |  | ||||||
|  | --omit-empty:: | ||||||
|  | 	Do not print a newline after formatted refs where the format expands | ||||||
|  | 	to the empty string. | ||||||
|  |  | ||||||
| --column[=<options>]:: | --column[=<options>]:: | ||||||
| --no-column:: | --no-column:: | ||||||
| 	Display branch listing in columns. See configuration variable | 	Display branch listing in columns. See configuration variable | ||||||
|  |  | ||||||
|  | @ -98,6 +98,10 @@ OPTIONS | ||||||
| --ignore-case:: | --ignore-case:: | ||||||
| 	Sorting and filtering refs are case insensitive. | 	Sorting and filtering refs are case insensitive. | ||||||
|  |  | ||||||
|  | --omit-empty:: | ||||||
|  | 	Do not print a newline after formatted refs where the format expands | ||||||
|  | 	to the empty string. | ||||||
|  |  | ||||||
| FIELD NAMES | FIELD NAMES | ||||||
| ----------- | ----------- | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @ -131,6 +131,10 @@ options for details. | ||||||
| --ignore-case:: | --ignore-case:: | ||||||
| 	Sorting and filtering tags are case insensitive. | 	Sorting and filtering tags are case insensitive. | ||||||
|  |  | ||||||
|  | --omit-empty:: | ||||||
|  | 	Do not print a newline after formatted refs where the format expands | ||||||
|  | 	to the empty string. | ||||||
|  |  | ||||||
| --column[=<options>]:: | --column[=<options>]:: | ||||||
| --no-column:: | --no-column:: | ||||||
| 	Display tag listing in columns. See configuration variable | 	Display tag listing in columns. See configuration variable | ||||||
|  |  | ||||||
|  | @ -44,6 +44,7 @@ static const char *head; | ||||||
| static struct object_id head_oid; | static struct object_id head_oid; | ||||||
| static int recurse_submodules = 0; | static int recurse_submodules = 0; | ||||||
| static int submodule_propagate_branches = 0; | static int submodule_propagate_branches = 0; | ||||||
|  | static int omit_empty = 0; | ||||||
|  |  | ||||||
| static int branch_use_color = -1; | static int branch_use_color = -1; | ||||||
| static char branch_colors[][COLOR_MAXLEN] = { | static char branch_colors[][COLOR_MAXLEN] = { | ||||||
|  | @ -483,7 +484,8 @@ static void print_ref_list(struct ref_filter *filter, struct ref_sorting *sortin | ||||||
| 			string_list_append(output, out.buf); | 			string_list_append(output, out.buf); | ||||||
| 		} else { | 		} else { | ||||||
| 			fwrite(out.buf, 1, out.len, stdout); | 			fwrite(out.buf, 1, out.len, stdout); | ||||||
| 			putchar('\n'); | 			if (out.len || !omit_empty) | ||||||
|  | 				putchar('\n'); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | @ -692,6 +694,8 @@ int cmd_branch(int argc, const char **argv, const char *prefix) | ||||||
| 		OPT_BIT('D', NULL, &delete, N_("delete branch (even if not merged)"), 2), | 		OPT_BIT('D', NULL, &delete, N_("delete branch (even if not merged)"), 2), | ||||||
| 		OPT_BIT('m', "move", &rename, N_("move/rename a branch and its reflog"), 1), | 		OPT_BIT('m', "move", &rename, N_("move/rename a branch and its reflog"), 1), | ||||||
| 		OPT_BIT('M', NULL, &rename, N_("move/rename a branch, even if target exists"), 2), | 		OPT_BIT('M', NULL, &rename, N_("move/rename a branch, even if target exists"), 2), | ||||||
|  | 		OPT_BOOL(0, "omit-empty",  &omit_empty, | ||||||
|  | 			N_("do not output a newline after empty formatted refs")), | ||||||
| 		OPT_BIT('c', "copy", ©, N_("copy a branch and its reflog"), 1), | 		OPT_BIT('c', "copy", ©, N_("copy a branch and its reflog"), 1), | ||||||
| 		OPT_BIT('C', NULL, ©, N_("copy a branch, even if target exists"), 2), | 		OPT_BIT('C', NULL, ©, N_("copy a branch, even if target exists"), 2), | ||||||
| 		OPT_BOOL('l', "list", &list, N_("list branch names")), | 		OPT_BOOL('l', "list", &list, N_("list branch names")), | ||||||
|  |  | ||||||
|  | @ -22,7 +22,7 @@ int cmd_for_each_ref(int argc, const char **argv, const char *prefix) | ||||||
| 	int i; | 	int i; | ||||||
| 	struct ref_sorting *sorting; | 	struct ref_sorting *sorting; | ||||||
| 	struct string_list sorting_options = STRING_LIST_INIT_DUP; | 	struct string_list sorting_options = STRING_LIST_INIT_DUP; | ||||||
| 	int maxcount = 0, icase = 0; | 	int maxcount = 0, icase = 0, omit_empty = 0; | ||||||
| 	struct ref_array array; | 	struct ref_array array; | ||||||
| 	struct ref_filter filter; | 	struct ref_filter filter; | ||||||
| 	struct ref_format format = REF_FORMAT_INIT; | 	struct ref_format format = REF_FORMAT_INIT; | ||||||
|  | @ -40,6 +40,8 @@ int cmd_for_each_ref(int argc, const char **argv, const char *prefix) | ||||||
| 			N_("quote placeholders suitably for python"), QUOTE_PYTHON), | 			N_("quote placeholders suitably for python"), QUOTE_PYTHON), | ||||||
| 		OPT_BIT(0 , "tcl",  &format.quote_style, | 		OPT_BIT(0 , "tcl",  &format.quote_style, | ||||||
| 			N_("quote placeholders suitably for Tcl"), QUOTE_TCL), | 			N_("quote placeholders suitably for Tcl"), QUOTE_TCL), | ||||||
|  | 		OPT_BOOL(0, "omit-empty",  &omit_empty, | ||||||
|  | 			N_("do not output a newline after empty formatted refs")), | ||||||
|  |  | ||||||
| 		OPT_GROUP(""), | 		OPT_GROUP(""), | ||||||
| 		OPT_INTEGER( 0 , "count", &maxcount, N_("show only <n> matched refs")), | 		OPT_INTEGER( 0 , "count", &maxcount, N_("show only <n> matched refs")), | ||||||
|  | @ -112,7 +114,8 @@ int cmd_for_each_ref(int argc, const char **argv, const char *prefix) | ||||||
| 		if (format_ref_array_item(array.items[i], &format, &output, &err)) | 		if (format_ref_array_item(array.items[i], &format, &output, &err)) | ||||||
| 			die("%s", err.buf); | 			die("%s", err.buf); | ||||||
| 		fwrite(output.buf, 1, output.len, stdout); | 		fwrite(output.buf, 1, output.len, stdout); | ||||||
| 		putchar('\n'); | 		if (output.len || !omit_empty) | ||||||
|  | 			putchar('\n'); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	strbuf_release(&err); | 	strbuf_release(&err); | ||||||
|  |  | ||||||
|  | @ -41,6 +41,7 @@ static const char * const git_tag_usage[] = { | ||||||
| static unsigned int colopts; | static unsigned int colopts; | ||||||
| static int force_sign_annotate; | static int force_sign_annotate; | ||||||
| static int config_sign_tag = -1; /* unspecified */ | static int config_sign_tag = -1; /* unspecified */ | ||||||
|  | static int omit_empty = 0; | ||||||
|  |  | ||||||
| static int list_tags(struct ref_filter *filter, struct ref_sorting *sorting, | static int list_tags(struct ref_filter *filter, struct ref_sorting *sorting, | ||||||
| 		     struct ref_format *format) | 		     struct ref_format *format) | ||||||
|  | @ -79,7 +80,8 @@ static int list_tags(struct ref_filter *filter, struct ref_sorting *sorting, | ||||||
| 		if (format_ref_array_item(array.items[i], format, &output, &err)) | 		if (format_ref_array_item(array.items[i], format, &output, &err)) | ||||||
| 			die("%s", err.buf); | 			die("%s", err.buf); | ||||||
| 		fwrite(output.buf, 1, output.len, stdout); | 		fwrite(output.buf, 1, output.len, stdout); | ||||||
| 		putchar('\n'); | 		if (output.len || !omit_empty) | ||||||
|  | 			putchar('\n'); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	strbuf_release(&err); | 	strbuf_release(&err); | ||||||
|  | @ -474,6 +476,8 @@ int cmd_tag(int argc, const char **argv, const char *prefix) | ||||||
| 		OPT_WITHOUT(&filter.no_commit, N_("print only tags that don't contain the commit")), | 		OPT_WITHOUT(&filter.no_commit, N_("print only tags that don't contain the commit")), | ||||||
| 		OPT_MERGED(&filter, N_("print only tags that are merged")), | 		OPT_MERGED(&filter, N_("print only tags that are merged")), | ||||||
| 		OPT_NO_MERGED(&filter, N_("print only tags that are not merged")), | 		OPT_NO_MERGED(&filter, N_("print only tags that are not merged")), | ||||||
|  | 		OPT_BOOL(0, "omit-empty",  &omit_empty, | ||||||
|  | 			N_("do not output a newline after empty formatted refs")), | ||||||
| 		OPT_REF_SORT(&sorting_options), | 		OPT_REF_SORT(&sorting_options), | ||||||
| 		{ | 		{ | ||||||
| 			OPTION_CALLBACK, 0, "points-at", &filter.points_at, N_("object"), | 			OPTION_CALLBACK, 0, "points-at", &filter.points_at, N_("object"), | ||||||
|  |  | ||||||
|  | @ -355,6 +355,30 @@ test_expect_success 'git branch with --format=%(rest) must fail' ' | ||||||
| 	test_must_fail git branch --format="%(rest)" >actual | 	test_must_fail git branch --format="%(rest)" >actual | ||||||
| ' | ' | ||||||
|  |  | ||||||
|  | test_expect_success 'git branch --format --omit-empty' ' | ||||||
|  | 	cat >expect <<-\EOF && | ||||||
|  | 	Refname is (HEAD detached from fromtag) | ||||||
|  | 	Refname is refs/heads/ambiguous | ||||||
|  | 	Refname is refs/heads/branch-one | ||||||
|  | 	Refname is refs/heads/branch-two | ||||||
|  |  | ||||||
|  | 	Refname is refs/heads/ref-to-branch | ||||||
|  | 	Refname is refs/heads/ref-to-remote | ||||||
|  | 	EOF | ||||||
|  | 	git branch --format="%(if:notequals=refs/heads/main)%(refname)%(then)Refname is %(refname)%(end)" >actual && | ||||||
|  | 	test_cmp expect actual && | ||||||
|  | 	cat >expect <<-\EOF && | ||||||
|  | 	Refname is (HEAD detached from fromtag) | ||||||
|  | 	Refname is refs/heads/ambiguous | ||||||
|  | 	Refname is refs/heads/branch-one | ||||||
|  | 	Refname is refs/heads/branch-two | ||||||
|  | 	Refname is refs/heads/ref-to-branch | ||||||
|  | 	Refname is refs/heads/ref-to-remote | ||||||
|  | 	EOF | ||||||
|  | 	git branch --omit-empty --format="%(if:notequals=refs/heads/main)%(refname)%(then)Refname is %(refname)%(end)" >actual && | ||||||
|  | 	test_cmp expect actual | ||||||
|  | ' | ||||||
|  |  | ||||||
| test_expect_success 'worktree colors correct' ' | test_expect_success 'worktree colors correct' ' | ||||||
| 	cat >expect <<-EOF && | 	cat >expect <<-EOF && | ||||||
| 	* <GREEN>(HEAD detached from fromtag)<RESET> | 	* <GREEN>(HEAD detached from fromtag)<RESET> | ||||||
|  |  | ||||||
|  | @ -1374,6 +1374,14 @@ test_expect_success 'for-each-ref --ignore-case ignores case' ' | ||||||
| 	test_cmp expect actual | 	test_cmp expect actual | ||||||
| ' | ' | ||||||
|  |  | ||||||
|  | test_expect_success 'for-each-ref --omit-empty works' ' | ||||||
|  | 	git for-each-ref --format="%(refname)" >actual && | ||||||
|  | 	test_line_count -gt 1 actual && | ||||||
|  | 	git for-each-ref --format="%(if:equals=refs/heads/main)%(refname)%(then)%(refname)%(end)" --omit-empty >actual && | ||||||
|  | 	echo refs/heads/main >expect && | ||||||
|  | 	test_cmp expect actual | ||||||
|  | ' | ||||||
|  |  | ||||||
| test_expect_success 'for-each-ref --ignore-case works on multiple sort keys' ' | test_expect_success 'for-each-ref --ignore-case works on multiple sort keys' ' | ||||||
| 	# name refs numerically to avoid case-insensitive filesystem conflicts | 	# name refs numerically to avoid case-insensitive filesystem conflicts | ||||||
| 	nr=0 && | 	nr=0 && | ||||||
|  |  | ||||||
|  | @ -2046,6 +2046,22 @@ test_expect_success '--format should list tags as per format given' ' | ||||||
| 	test_cmp expect actual | 	test_cmp expect actual | ||||||
| ' | ' | ||||||
|  |  | ||||||
|  | test_expect_success '--format --omit-empty works' ' | ||||||
|  | 	cat >expect <<-\EOF && | ||||||
|  | 	refname : refs/tags/v1.0 | ||||||
|  |  | ||||||
|  | 	refname : refs/tags/v1.1.3 | ||||||
|  | 	EOF | ||||||
|  | 	git tag -l --format="%(if:notequals=refs/tags/v1.0.1)%(refname)%(then)refname : %(refname)%(end)" "v1*" >actual && | ||||||
|  | 	test_cmp expect actual && | ||||||
|  | 	cat >expect <<-\EOF && | ||||||
|  | 	refname : refs/tags/v1.0 | ||||||
|  | 	refname : refs/tags/v1.1.3 | ||||||
|  | 	EOF | ||||||
|  | 	git tag -l --omit-empty --format="%(if:notequals=refs/tags/v1.0.1)%(refname)%(then)refname : %(refname)%(end)" "v1*" >actual && | ||||||
|  | 	test_cmp expect actual | ||||||
|  | ' | ||||||
|  |  | ||||||
| test_expect_success 'git tag -l with --format="%(rest)" must fail' ' | test_expect_success 'git tag -l with --format="%(rest)" must fail' ' | ||||||
| 	test_must_fail git tag -l --format="%(rest)" "v1*" | 	test_must_fail git tag -l --format="%(rest)" "v1*" | ||||||
| ' | ' | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue
	
	 Junio C Hamano
						Junio C Hamano