On Fri, Mar 30, 2018 at 01:27:19AM -0400, Taylor Blau wrote:

> > If you really want to go all-out, I think the ACTION flags could use the
> > same cleanup. We treat them as bitflags, and then issue an error when
> > you set more than one, which is just silly.
> 
> Agreed, and I think that this is a good candidate for a future patch.
> Thoughts? :-).

I actually worked this up for fun, though I had second thoughts while
writing the commit message.

Besides the cleanup, my primary motivation was following last-one-wins
rules as we often do elsewhere. But actually, last-one-wins applies only
to a _single_ option, not necessarily unrelated ones. Many other
multi-action commands actually have a series of separate boolean flags,
and then complain when more than one of the flags is set.

So maybe it's not such a good idea for the actions (I do still think
it's the right path for the types).

For reference, here's the patch I wrote:

 builtin/config.c | 137 +++++++++++++++++++++++++----------------------
 1 file changed, 72 insertions(+), 65 deletions(-)

diff --git a/builtin/config.c b/builtin/config.c
index 01169dd628..5581f48ac8 100644
--- a/builtin/config.c
+++ b/builtin/config.c
@@ -25,35 +25,36 @@ static char term = '\n';
 
 static int use_global_config, use_system_config, use_local_config;
 static struct git_config_source given_config_source;
-static int actions, types;
+static int types;
 static int end_null;
 static int respect_includes_opt = -1;
 static struct config_options config_options;
 static int show_origin;
 
-#define ACTION_GET (1<<0)
-#define ACTION_GET_ALL (1<<1)
-#define ACTION_GET_REGEXP (1<<2)
-#define ACTION_REPLACE_ALL (1<<3)
-#define ACTION_ADD (1<<4)
-#define ACTION_UNSET (1<<5)
-#define ACTION_UNSET_ALL (1<<6)
-#define ACTION_RENAME_SECTION (1<<7)
-#define ACTION_REMOVE_SECTION (1<<8)
-#define ACTION_LIST (1<<9)
-#define ACTION_EDIT (1<<10)
-#define ACTION_SET (1<<11)
-#define ACTION_SET_ALL (1<<12)
-#define ACTION_GET_COLOR (1<<13)
-#define ACTION_GET_COLORBOOL (1<<14)
-#define ACTION_GET_URLMATCH (1<<15)
-
+enum config_action {
+       ACTION_NONE = 0,
+       ACTION_GET,
+       ACTION_GET_ALL,
+       ACTION_GET_REGEXP,
+       ACTION_REPLACE_ALL,
+       ACTION_ADD,
+       ACTION_UNSET,
+       ACTION_UNSET_ALL,
+       ACTION_RENAME_SECTION,
+       ACTION_REMOVE_SECTION,
+       ACTION_LIST,
+       ACTION_EDIT,
+       ACTION_SET,
+       ACTION_SET_ALL,
+       ACTION_GET_COLOR,
+       ACTION_GET_COLORBOOL,
+       ACTION_GET_URLMATCH,
+};
 /*
- * The actions "ACTION_LIST | ACTION_GET_*" which may produce more than
- * one line of output and which should therefore be paged.
+ * This must be an int and not an enum because we pass it by address
+ * to OPT_SETINT.
  */
-#define PAGING_ACTIONS (ACTION_LIST | ACTION_GET_ALL | \
-                       ACTION_GET_REGEXP | ACTION_GET_URLMATCH)
+static int action;
 
 #define TYPE_BOOL (1<<0)
 #define TYPE_INT (1<<1)
@@ -69,20 +70,20 @@ static struct option builtin_config_options[] = {
        OPT_STRING('f', "file", &given_config_source.file, N_("file"), N_("use 
given config file")),
        OPT_STRING(0, "blob", &given_config_source.blob, N_("blob-id"), 
N_("read config from given blob object")),
        OPT_GROUP(N_("Action")),
-       OPT_BIT(0, "get", &actions, N_("get value: name [value-regex]"), 
ACTION_GET),
-       OPT_BIT(0, "get-all", &actions, N_("get all values: key 
[value-regex]"), ACTION_GET_ALL),
-       OPT_BIT(0, "get-regexp", &actions, N_("get values for regexp: 
name-regex [value-regex]"), ACTION_GET_REGEXP),
-       OPT_BIT(0, "get-urlmatch", &actions, N_("get value specific for the 
URL: section[.var] URL"), ACTION_GET_URLMATCH),
-       OPT_BIT(0, "replace-all", &actions, N_("replace all matching variables: 
name value [value_regex]"), ACTION_REPLACE_ALL),
-       OPT_BIT(0, "add", &actions, N_("add a new variable: name value"), 
ACTION_ADD),
-       OPT_BIT(0, "unset", &actions, N_("remove a variable: name 
[value-regex]"), ACTION_UNSET),
-       OPT_BIT(0, "unset-all", &actions, N_("remove all matches: name 
[value-regex]"), ACTION_UNSET_ALL),
-       OPT_BIT(0, "rename-section", &actions, N_("rename section: old-name 
new-name"), ACTION_RENAME_SECTION),
-       OPT_BIT(0, "remove-section", &actions, N_("remove a section: name"), 
ACTION_REMOVE_SECTION),
-       OPT_BIT('l', "list", &actions, N_("list all"), ACTION_LIST),
-       OPT_BIT('e', "edit", &actions, N_("open an editor"), ACTION_EDIT),
-       OPT_BIT(0, "get-color", &actions, N_("find the color configured: slot 
[default]"), ACTION_GET_COLOR),
-       OPT_BIT(0, "get-colorbool", &actions, N_("find the color setting: slot 
[stdout-is-tty]"), ACTION_GET_COLORBOOL),
+       OPT_SET_INT(0, "get", &action, N_("get value: name [value-regex]"), 
ACTION_GET),
+       OPT_SET_INT(0, "get-all", &action, N_("get all values: key 
[value-regex]"), ACTION_GET_ALL),
+       OPT_SET_INT(0, "get-regexp", &action, N_("get values for regexp: 
name-regex [value-regex]"), ACTION_GET_REGEXP),
+       OPT_SET_INT(0, "get-urlmatch", &action, N_("get value specific for the 
URL: section[.var] URL"), ACTION_GET_URLMATCH),
+       OPT_SET_INT(0, "replace-all", &action, N_("replace all matching 
variables: name value [value_regex]"), ACTION_REPLACE_ALL),
+       OPT_SET_INT(0, "add", &action, N_("add a new variable: name value"), 
ACTION_ADD),
+       OPT_SET_INT(0, "unset", &action, N_("remove a variable: name 
[value-regex]"), ACTION_UNSET),
+       OPT_SET_INT(0, "unset-all", &action, N_("remove all matches: name 
[value-regex]"), ACTION_UNSET_ALL),
+       OPT_SET_INT(0, "rename-section", &action, N_("rename section: old-name 
new-name"), ACTION_RENAME_SECTION),
+       OPT_SET_INT(0, "remove-section", &action, N_("remove a section: name"), 
ACTION_REMOVE_SECTION),
+       OPT_SET_INT('l', "list", &action, N_("list all"), ACTION_LIST),
+       OPT_SET_INT('e', "edit", &action, N_("open an editor"), ACTION_EDIT),
+       OPT_SET_INT(0, "get-color", &action, N_("find the color configured: 
slot [default]"), ACTION_GET_COLOR),
+       OPT_SET_INT(0, "get-colorbool", &action, N_("find the color setting: 
slot [stdout-is-tty]"), ACTION_GET_COLORBOOL),
        OPT_GROUP(N_("Type")),
        OPT_BIT(0, "bool", &types, N_("value is \"true\" or \"false\""), 
TYPE_BOOL),
        OPT_BIT(0, "int", &types, N_("value is decimal number"), TYPE_INT),
@@ -571,40 +572,46 @@ int cmd_config(int argc, const char **argv, const char 
*prefix)
                usage_with_options(builtin_config_usage, 
builtin_config_options);
        }
 
-       if ((actions & (ACTION_GET_COLOR|ACTION_GET_COLORBOOL)) && types) {
+       if (types &&
+           (action == ACTION_GET_COLOR ||
+            action == ACTION_GET_COLORBOOL)) {
                error("--get-color and variable type are incoherent");
                usage_with_options(builtin_config_usage, 
builtin_config_options);
        }
 
-       if (HAS_MULTI_BITS(actions)) {
-               error("only one action at a time.");
-               usage_with_options(builtin_config_usage, 
builtin_config_options);
-       }
-       if (actions == 0)
+       if (action == ACTION_NONE) {
                switch (argc) {
-               case 1: actions = ACTION_GET; break;
-               case 2: actions = ACTION_SET; break;
-               case 3: actions = ACTION_SET_ALL; break;
+               case 1: action = ACTION_GET; break;
+               case 2: action = ACTION_SET; break;
+               case 3: action = ACTION_SET_ALL; break;
                default:
                        usage_with_options(builtin_config_usage, 
builtin_config_options);
                }
+       }
        if (omit_values &&
-           !(actions == ACTION_LIST || actions == ACTION_GET_REGEXP)) {
+           !(action == ACTION_LIST ||
+             action == ACTION_GET_REGEXP)) {
                error("--name-only is only applicable to --list or 
--get-regexp");
                usage_with_options(builtin_config_usage, 
builtin_config_options);
        }
 
-       if (show_origin && !(actions &
-               (ACTION_GET|ACTION_GET_ALL|ACTION_GET_REGEXP|ACTION_LIST))) {
+       if (show_origin &&
+           !(action == ACTION_GET ||
+             action == ACTION_GET_ALL ||
+             action == ACTION_GET_REGEXP ||
+             action == ACTION_LIST)) {
                error("--show-origin is only applicable to --get, --get-all, "
                          "--get-regexp, and --list.");
                usage_with_options(builtin_config_usage, 
builtin_config_options);
        }
 
-       if (actions & PAGING_ACTIONS)
+       if (action == ACTION_LIST ||
+           action == ACTION_GET_ALL ||
+           action == ACTION_GET_REGEXP ||
+           action == ACTION_GET_URLMATCH)
                setup_auto_pager("config", 1);
 
-       if (actions == ACTION_LIST) {
+       if (action == ACTION_LIST) {
                check_argc(argc, 0, 0);
                if (config_with_options(show_all_config, NULL,
                                        &given_config_source,
@@ -616,7 +623,7 @@ int cmd_config(int argc, const char **argv, const char 
*prefix)
                                die("error processing config file(s)");
                }
        }
-       else if (actions == ACTION_EDIT) {
+       else if (action == ACTION_EDIT) {
                char *config_file;
 
                check_argc(argc, 0, 0);
@@ -644,7 +651,7 @@ int cmd_config(int argc, const char **argv, const char 
*prefix)
                launch_editor(config_file, NULL, NULL);
                free(config_file);
        }
-       else if (actions == ACTION_SET) {
+       else if (action == ACTION_SET) {
                int ret;
                check_write();
                check_argc(argc, 2, 2);
@@ -656,7 +663,7 @@ int cmd_config(int argc, const char **argv, const char 
*prefix)
                        "       Use a regexp, --add or --replace-all to change 
%s."), argv[0]);
                return ret;
        }
-       else if (actions == ACTION_SET_ALL) {
+       else if (action == ACTION_SET_ALL) {
                check_write();
                check_argc(argc, 2, 3);
                value = normalize_value(argv[0], argv[1]);
@@ -664,7 +671,7 @@ int cmd_config(int argc, const char **argv, const char 
*prefix)
                return 
git_config_set_multivar_in_file_gently(given_config_source.file,
                                                              argv[0], value, 
argv[2], 0);
        }
-       else if (actions == ACTION_ADD) {
+       else if (action == ACTION_ADD) {
                check_write();
                check_argc(argc, 2, 2);
                value = normalize_value(argv[0], argv[1]);
@@ -673,7 +680,7 @@ int cmd_config(int argc, const char **argv, const char 
*prefix)
                                                              argv[0], value,
                                                              
CONFIG_REGEX_NONE, 0);
        }
-       else if (actions == ACTION_REPLACE_ALL) {
+       else if (action == ACTION_REPLACE_ALL) {
                check_write();
                check_argc(argc, 2, 3);
                value = normalize_value(argv[0], argv[1]);
@@ -681,27 +688,27 @@ int cmd_config(int argc, const char **argv, const char 
*prefix)
                return 
git_config_set_multivar_in_file_gently(given_config_source.file,
                                                              argv[0], value, 
argv[2], 1);
        }
-       else if (actions == ACTION_GET) {
+       else if (action == ACTION_GET) {
                check_argc(argc, 1, 2);
                return get_value(argv[0], argv[1]);
        }
-       else if (actions == ACTION_GET_ALL) {
+       else if (action == ACTION_GET_ALL) {
                do_all = 1;
                check_argc(argc, 1, 2);
                return get_value(argv[0], argv[1]);
        }
-       else if (actions == ACTION_GET_REGEXP) {
+       else if (action == ACTION_GET_REGEXP) {
                show_keys = 1;
                use_key_regexp = 1;
                do_all = 1;
                check_argc(argc, 1, 2);
                return get_value(argv[0], argv[1]);
        }
-       else if (actions == ACTION_GET_URLMATCH) {
+       else if (action == ACTION_GET_URLMATCH) {
                check_argc(argc, 2, 2);
                return get_urlmatch(argv[0], argv[1]);
        }
-       else if (actions == ACTION_UNSET) {
+       else if (action == ACTION_UNSET) {
                check_write();
                check_argc(argc, 1, 2);
                if (argc == 2)
@@ -711,13 +718,13 @@ int cmd_config(int argc, const char **argv, const char 
*prefix)
                        return 
git_config_set_in_file_gently(given_config_source.file,
                                                             argv[0], NULL);
        }
-       else if (actions == ACTION_UNSET_ALL) {
+       else if (action == ACTION_UNSET_ALL) {
                check_write();
                check_argc(argc, 1, 2);
                return 
git_config_set_multivar_in_file_gently(given_config_source.file,
                                                              argv[0], NULL, 
argv[1], 1);
        }
-       else if (actions == ACTION_RENAME_SECTION) {
+       else if (action == ACTION_RENAME_SECTION) {
                int ret;
                check_write();
                check_argc(argc, 2, 2);
@@ -728,7 +735,7 @@ int cmd_config(int argc, const char **argv, const char 
*prefix)
                if (ret == 0)
                        die("No such section!");
        }
-       else if (actions == ACTION_REMOVE_SECTION) {
+       else if (action == ACTION_REMOVE_SECTION) {
                int ret;
                check_write();
                check_argc(argc, 1, 1);
@@ -739,11 +746,11 @@ int cmd_config(int argc, const char **argv, const char 
*prefix)
                if (ret == 0)
                        die("No such section!");
        }
-       else if (actions == ACTION_GET_COLOR) {
+       else if (action == ACTION_GET_COLOR) {
                check_argc(argc, 1, 2);
                get_color(argv[0], argv[1]);
        }
-       else if (actions == ACTION_GET_COLORBOOL) {
+       else if (action == ACTION_GET_COLORBOOL) {
                check_argc(argc, 1, 2);
                if (argc == 2)
                        color_stdout_is_tty = git_config_bool("command line", 
argv[1]);
-- 
2.17.0.rc2.594.gdb94a0ce02

Reply via email to