[glib] Implement bash completion for gsettings
- From: Matthias Clasen <matthiasc src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [glib] Implement bash completion for gsettings
- Date: Sun, 27 Jun 2010 20:02:39 +0000 (UTC)
commit 025435329a7159a906a66f6b15facc2d611be4fb
Author: Matthias Clasen <mclasen redhat com>
Date: Sun Jun 27 16:00:20 2010 -0400
Implement bash completion for gsettings
gio/Makefile.am | 4 +-
gio/gsettings-bash-completion.sh | 33 +++
gio/gsettings-tool.c | 498 ++++++++++++++++++++++++++++++++------
3 files changed, 455 insertions(+), 80 deletions(-)
---
diff --git a/gio/Makefile.am b/gio/Makefile.am
index b20c874..f350a0c 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 0000000..47ffcbe
--- /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 04d95e0..0e7b2ab 100644
--- a/gio/gsettings-tool.c
+++ b/gio/gsettings-tool.c
@@ -21,10 +21,48 @@
#include "config.h"
+#include <stdlib.h>
+#include <stdio.h>
#include <locale.h>
#include <gi18n.h>
#include <gio.h>
+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);
+
+ 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;
}
- 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);
- else
{
- g_printerr (_("Unknown command '%s'\n"), argv[1]);
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
+ {
+ 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;
}
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]