[glib] Implement bash completion for gsettings



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]