diff --git a/gio/Makefile.am b/gio/Makefile.am index b20c87417..f350a0cbe 100644 --- a/gio/Makefile.am +++ b/gio/Makefile.am @@ -590,7 +590,9 @@ gdbus_LDADD = libgio-2.0.la \ $(top_builddir)/gobject/libgobject-2.0.la completiondir = $(sysconfdir)/bash_completion.d -completion_SCRIPTS = gdbus-bash-completion.sh +completion_SCRIPTS = \ + gdbus-bash-completion.sh \ + gsettings-bash-completion.sh EXTRA_DIST += $(completion_SCRIPTS) # ------------------------------------------------------------------------ diff --git a/gio/gsettings-bash-completion.sh b/gio/gsettings-bash-completion.sh new file mode 100644 index 000000000..47ffcbeb5 --- /dev/null +++ b/gio/gsettings-bash-completion.sh @@ -0,0 +1,33 @@ + +# Check for bash +[ -z "$BASH_VERSION" ] && return + +#################################################################################################### + + +__gsettings() { + local IFS=$'\n' + local cur=`_get_cword :` + + local suggestions=$(gsettings complete "${COMP_LINE}" ${COMP_POINT}) + COMPREPLY=($(compgen -W "$suggestions" -- "$cur")) + + # Remove colon-word prefix from COMPREPLY items + case "$cur" in + *:*) + case "$COMP_WORDBREAKS" in + *:*) + local colon_word=${cur%${cur##*:}} + local i=${#COMPREPLY[*]} + while [ $((--i)) -ge 0 ]; do + COMPREPLY[$i]=${COMPREPLY[$i]#"$colon_word"} + done + ;; + esac + ;; + esac +} + +#################################################################################################### + +complete -o nospace -F __gsettings gsettings diff --git a/gio/gsettings-tool.c b/gio/gsettings-tool.c index 04d95e01e..0e7b2ab21 100644 --- a/gio/gsettings-tool.c +++ b/gio/gsettings-tool.c @@ -21,10 +21,48 @@ #include "config.h" +#include +#include #include #include #include +static gchar * +pick_word_at (const gchar *s, + gint cursor, + gint *out_word_begins_at) +{ + gint begin; + gint end; + + if (s[0] == '\0') + { + if (out_word_begins_at != NULL) + *out_word_begins_at = -1; + return NULL; + } + + if (g_ascii_isspace (s[cursor]) && + ((cursor > 0 && g_ascii_isspace (s[cursor-1])) || cursor == 0)) + { + if (out_word_begins_at != NULL) + *out_word_begins_at = cursor; + return g_strdup (""); + } + while (!g_ascii_isspace (s[cursor - 1]) && cursor > 0) + cursor--; + begin = cursor; + + end = begin; + while (!g_ascii_isspace (s[end]) && s[end] != '\0') + end++; + + if (out_word_begins_at != NULL) + *out_word_begins_at = begin; + + return g_strndup (s + begin, end - begin); +} + static gint usage (gint *argc, gchar **argv[], @@ -73,6 +111,7 @@ remove_arg (gint num, gint *argc, gchar **argv[]) (*argc) = (*argc) - 1; } + static void modify_argv0_for_command (gint *argc, gchar **argv[], @@ -87,10 +126,86 @@ modify_argv0_for_command (gint *argc, (*argv)[0] = s; } +static gboolean +schema_exists (const gchar *name) +{ + const gchar * const *schemas; + gint i; + + schemas = g_settings_list_schemas (); + for (i = 0; schemas[i]; i++) + if (g_strcmp0 (name, schemas[i]) == 0) + return TRUE; + + return FALSE; +} + +static void +list_schemas (const gchar *prefix) +{ + const gchar * const *schemas; + gint i; + + schemas = g_settings_list_schemas (); + for (i = 0; schemas[i]; i++) + if (prefix == NULL || g_str_has_prefix (schemas[i], prefix)) + g_print ("%s \n", schemas[i]); +} + +static gboolean +key_exists (GSettings *settings, + const gchar *name) +{ + const gchar **keys; + gint i; + gboolean ret; + + ret = FALSE; + + keys = g_settings_list_keys (settings); + for (i = 0; keys[i]; i++) + if (g_strcmp0 (keys[i], name) == 0) + { + ret = TRUE; + break; + } + g_free (keys); + + return ret; +} + +static void +list_keys (GSettings *settings, + const gchar *prefix) +{ + const gchar **keys; + gint i; + + keys = g_settings_list_keys (settings); + for (i = 0; keys[i]; i++) + if (prefix == NULL || g_str_has_prefix (keys[i], prefix)) + g_print ("%s \n", keys[i]); + g_free (keys); +} + +static void +list_options (GOptionContext *context, + const gchar *prefix) +{ + /* FIXME extract options from context */ + const gchar *options[] = { "--help", "--path", NULL }; + gint i; + for (i = 0; options[i]; i++) + if (g_str_has_prefix (options[i], prefix)) + g_print ("%s \n", options[i]); +} static gint -handle_get (gint *argc, - gchar **argv[]) +handle_get (gint *argc, + gchar **argv[], + gboolean request_completion, + gchar *completion_cur, + gchar *completion_prev) { gchar *schema; gchar *path; @@ -116,41 +231,79 @@ handle_get (gint *argc, " KEY The name of the key\n")); g_option_context_add_main_entries (context, entries, GETTEXT_PACKAGE); + settings = NULL; path = NULL; + schema = NULL; + key = NULL; error = NULL; - if (!g_option_context_parse (context, argc, argv, NULL) || *argc != 3) + if (!g_option_context_parse (context, argc, argv, NULL)) { - gchar *s; - s = g_option_context_get_help (context, FALSE, NULL); - g_printerr ("%s", s); - g_free (s); + if (!request_completion) + { + gchar *s; + s = g_option_context_get_help (context, FALSE, NULL); + g_printerr ("%s", s); + g_free (s); + goto out; + } + } + + if (*argc > 1) + schema = (*argv)[1]; + if (*argc > 2) + key = (*argv)[2]; + + if (request_completion && completion_cur[0] == '-') + { + list_options (context, completion_cur); + ret = 0; goto out; } - schema = (*argv)[1]; - key = (*argv)[2]; + if (request_completion && !schema_exists (schema)) + { + list_schemas (schema); + ret = 0; + goto out; + } if (path) settings = g_settings_new_with_path (schema, path); else settings = g_settings_new (schema); - v = g_settings_get_value (settings, key); - g_print ("%s\n", g_variant_print (v, FALSE)); - g_variant_unref (v); - ret = 0; + if (request_completion && !key_exists (settings, key)) + { + list_keys (settings, key); + ret = 0; + goto out; + } + + if (!request_completion) + { + v = g_settings_get_value (settings, key); + g_print ("%s\n", g_variant_print (v, FALSE)); + g_variant_unref (v); + ret = 0; + } out: + if (settings) + g_object_unref (settings); + g_option_context_free (context); return ret; } static gint -handle_set (gint *argc, - gchar **argv[]) +handle_set (gint *argc, + gchar **argv[], + gboolean request_completion, + gchar *completion_cur, + gchar *completion_prev) { gchar *schema; gchar *path; @@ -179,50 +332,85 @@ handle_set (gint *argc, " VALUE The value to set key to, as a serialized GVariant\n")); g_option_context_add_main_entries (context, entries, GETTEXT_PACKAGE); + settings = NULL; path = NULL; + schema = NULL; + key = NULL; error = NULL; - if (!g_option_context_parse (context, argc, argv, NULL) || *argc != 4) + if (!g_option_context_parse (context, argc, argv, NULL)) { - gchar *s; - s = g_option_context_get_help (context, FALSE, NULL); - g_printerr ("%s", s); - g_free (s); + if (!request_completion) + { + gchar *s; + s = g_option_context_get_help (context, FALSE, NULL); + g_printerr ("%s", s); + g_free (s); + goto out; + } + } + if (*argc > 1) + schema = (*argv)[1]; + if (*argc > 2) + key = (*argv)[2]; + if (*argc > 3) + value = (*argv)[3]; + + if (request_completion && completion_cur[0] == '-') + { + list_options (context, completion_cur); + ret = 0; goto out; } - schema = (*argv)[1]; - key = (*argv)[2]; - value = (*argv)[3]; + if (request_completion && !schema_exists (schema)) + { + list_schemas (schema); + ret = 0; + goto out; + } if (path) settings = g_settings_new_with_path (schema, path); else settings = g_settings_new (schema); - default_v = g_settings_get_value (settings, key); - type = g_variant_get_type (default_v); - - error = NULL; - v = g_variant_parse (type, value, NULL, NULL, &error); - g_variant_unref (default_v); - if (v == NULL) + if (request_completion && !key_exists (settings, key)) { - g_printerr ("%s\n", error->message); + list_keys (settings, key); + ret = 0; goto out; } - if (!g_settings_set_value (settings, key, v)) + if (!request_completion) { - g_printerr (_("Key %s is not writable\n"), key); - goto out; - } + default_v = g_settings_get_value (settings, key); + type = g_variant_get_type (default_v); - g_settings_sync (); - ret = 0; + error = NULL; + v = g_variant_parse (type, value, NULL, NULL, &error); + g_variant_unref (default_v); + if (v == NULL) + { + g_printerr ("%s\n", error->message); + goto out; + } + + if (!g_settings_set_value (settings, key, v)) + { + g_printerr (_("Key %s is not writable\n"), key); + goto out; + } + + g_settings_sync (); + ret = 0; + } out: + if (settings) + g_object_unref (settings); + g_option_context_free (context); return ret; @@ -230,7 +418,10 @@ handle_set (gint *argc, static gint handle_writable (gint *argc, - gchar **argv[]) + gchar **argv[], + gboolean request_completion, + gchar *completion_cur, + gchar *completion_prev) { gchar *schema; gchar *path; @@ -255,34 +446,68 @@ handle_writable (gint *argc, " KEY The name of the key\n")); g_option_context_add_main_entries (context, entries, GETTEXT_PACKAGE); + settings = NULL; path = NULL; + schema = NULL; + key = NULL; error = NULL; - if (!g_option_context_parse (context, argc, argv, NULL) || *argc != 3) + if (!g_option_context_parse (context, argc, argv, NULL)) { - gchar *s; - s = g_option_context_get_help (context, FALSE, NULL); - g_printerr ("%s", s); - g_free (s); + if (!request_completion) + { + gchar *s; + s = g_option_context_get_help (context, FALSE, NULL); + g_printerr ("%s", s); + g_free (s); + goto out; + } + } + if (*argc > 1) + schema = (*argv)[1]; + if (*argc > 2) + key = (*argv)[2]; + + if (request_completion && completion_cur[0] == '-') + { + list_options (context, completion_cur); + ret = 0; goto out; } - schema = (*argv)[1]; - key = (*argv)[2]; + if (request_completion && !schema_exists (schema)) + { + list_schemas (schema); + ret = 0; + goto out; + } if (path) settings = g_settings_new_with_path (schema, path); else settings = g_settings_new (schema); - if (g_settings_is_writable (settings, key)) - g_print ("true\n"); - else - g_print ("false\n"); - ret = 0; + if (request_completion && !key_exists (settings, key)) + { + list_keys (settings, key); + ret = 0; + goto out; + } + + if (!request_completion) + { + if (g_settings_is_writable (settings, key)) + g_print ("true\n"); + else + g_print ("false\n"); + ret = 0; + } out: + if (settings) + g_object_unref (settings); + g_option_context_free (context); return ret; @@ -303,8 +528,11 @@ key_changed (GSettings *settings, } static gint -handle_monitor (gint *argc, - gchar **argv[]) +handle_monitor (gint *argc, + gchar **argv[], + gboolean request_completion, + gchar *completion_cur, + gchar *completion_prev) { gchar *schema; gchar *path; @@ -334,36 +562,71 @@ handle_monitor (gint *argc, " KEY The name of the key\n")); g_option_context_add_main_entries (context, entries, GETTEXT_PACKAGE); + settings = NULL; path = NULL; + schema = NULL; + key = NULL; error = NULL; - if (!g_option_context_parse (context, argc, argv, NULL) || *argc != 3) + if (!g_option_context_parse (context, argc, argv, NULL)) { - gchar *s; - s = g_option_context_get_help (context, FALSE, NULL); - g_printerr ("%s", s); - g_free (s); + if (!request_completion) + { + gchar *s; + s = g_option_context_get_help (context, FALSE, NULL); + g_printerr ("%s", s); + g_free (s); + goto out; + } + } + if (*argc > 1) + schema = (*argv)[1]; + if (*argc > 2) + key = (*argv)[2]; + + if (request_completion && completion_cur[0] == '-') + { + list_options (context, completion_cur); + ret = 0; goto out; } - schema = (*argv)[1]; - key = (*argv)[2]; + if (request_completion && !schema_exists (schema)) + { + list_schemas (schema); + ret = 0; + goto out; + } if (path) settings = g_settings_new_with_path (schema, path); else settings = g_settings_new (schema); - detailed_signal = g_strdup_printf ("changed::%s", key); - g_signal_connect (settings, detailed_signal, - G_CALLBACK (key_changed), NULL); + if (request_completion && !key_exists (settings, key)) + { + list_keys (settings, key); + ret = 0; + goto out; + } - loop = g_main_loop_new (NULL, FALSE); - g_main_loop_run (loop); - g_main_loop_unref (loop); + if (!request_completion) + { + detailed_signal = g_strdup_printf ("changed::%s", key); + g_signal_connect (settings, detailed_signal, + G_CALLBACK (key_changed), NULL); + + loop = g_main_loop_new (NULL, FALSE); + g_main_loop_run (loop); + g_main_loop_unref (loop); + ret = 0; + } out: + if (settings) + g_object_unref (settings); + g_option_context_free (context); return ret; @@ -371,29 +634,106 @@ handle_monitor (gint *argc, int main (int argc, char *argv[]) { - gboolean ret = 1; + gboolean ret; + gchar *command; + gboolean request_completion; + gchar *completion_cur; + gchar *completion_prev; setlocale (LC_ALL, ""); g_type_init (); + ret = 1; + completion_cur = NULL; + completion_prev = NULL; + request_completion = FALSE; + if (argc < 2) - ret = usage (&argc, &argv, FALSE); - else if (g_strcmp0 (argv[1], "help") == 0) - ret = usage (&argc, &argv, TRUE); - else if (g_strcmp0 (argv[1], "get") == 0) - ret = handle_get (&argc, &argv); - else if (g_strcmp0 (argv[1], "set") == 0) - ret = handle_set (&argc, &argv); - else if (g_strcmp0 (argv[1], "monitor") == 0) - ret = handle_monitor (&argc, &argv); - else if (g_strcmp0 (argv[1], "writable") == 0) - ret = handle_writable (&argc, &argv); + { + ret = usage (&argc, &argv, FALSE); + goto out; + } + + again: + command = argv[1]; + + if (g_strcmp0 (command, "help") == 0) + { + if (!request_completion) + ret = usage (&argc, &argv, TRUE); + } + else if (g_strcmp0 (command, "get") == 0) + ret = handle_get (&argc, &argv, request_completion, completion_cur, completion_prev); + else if (g_strcmp0 (command, "set") == 0) + ret = handle_set (&argc, &argv, request_completion, completion_cur, completion_prev); + else if (g_strcmp0 (command, "monitor") == 0) + ret = handle_monitor (&argc, &argv, request_completion, completion_cur, completion_prev); + else if (g_strcmp0 (command, "writable") == 0) + ret = handle_writable (&argc, &argv, request_completion, completion_cur, completion_prev); + else if (g_strcmp0 (command, "complete") == 0 && argc == 4 && !request_completion) + { + gchar *completion_line; + gint completion_point; + gchar *endp; + gchar **completion_argv; + gint completion_argc; + gint cur_begin; + + request_completion = TRUE; + + completion_line = argv[2]; + completion_point = strtol (argv[3], &endp, 10); + if (endp == argv[3] || *endp != '\0') + goto out; + + if (!g_shell_parse_argv (completion_line, + &completion_argc, + &completion_argv, + NULL)) + { + /* can't parse partical cmdline, don't attempt completion */ + goto out; + } + + completion_prev = NULL; + completion_cur = pick_word_at (completion_line, completion_point, &cur_begin); + if (cur_begin > 0) + { + gint prev_end; + for (prev_end = cur_begin - 1; prev_end >= 0; prev_end--) + { + if (!g_ascii_isspace (completion_line[prev_end])) + { + completion_prev = pick_word_at (completion_line, prev_end, NULL); + break; + } + } + } + + argc = completion_argc; + argv = completion_argv; + + ret = 0; + goto again; + } else { - g_printerr (_("Unknown command '%s'\n"), argv[1]); - ret = usage (&argc, &argv, FALSE); + if (request_completion) + { + g_print ("help \nget \nmonitor \nwritable \nset \n"); + ret = 0; + } + else + { + g_printerr (_("Unknown command '%s'\n"), argv[1]); + ret = usage (&argc, &argv, FALSE); + } } + out: + g_free (completion_cur); + g_free (completion_prev); + return ret; }