New gapplication(1) tool
[platform/upstream/glib.git] / gio / gapplication-tool.c
1 /*
2  * Copyright © 2013 Canonical Limited
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Lesser General Public
6  * License as published by the Free Software Foundation; either
7  * version 2 of the licence, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Lesser General Public License for more details.
13  *
14  * You should have received a copy of the GNU Lesser General Public
15  * License along with this library; if not, write to the
16  * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17  * Boston, MA 02111-1307, USA.
18  *
19  * Author: Ryan Lortie <desrt@desrt.ca>
20  */
21
22 #include "config.h"
23
24 #include <gio/gdesktopappinfo.h>
25
26 #include <glib/gi18n.h>
27 #include <gio/gio.h>
28
29 #include <string.h>
30 #include <locale.h>
31
32 struct help_topic
33 {
34   const gchar *command;
35   const gchar *summary;
36   const gchar *description;
37   const gchar *synopsis;
38 };
39
40 struct help_substvar
41 {
42   const gchar *var;
43   const gchar *description;
44 };
45
46 static const struct help_topic topics[] = {
47   { "help",         N_("Print help"),
48                     N_("Print help"),
49                     N_("[COMMAND]")
50   },
51   { "version",      N_("Print version"),
52                     N_("Print version information and exit")
53   },
54   { "list-apps",    N_("List applications"),
55                     N_("List the installed D-Bus activatable applications (by .desktop files)")
56   },
57   { "launch",       N_("Launch an application"),
58                     N_("Launch the application (with optional files to open)"),
59                     N_("APPID [FILE...]")
60   },
61   { "action",       N_("Activate an action"),
62                     N_("Invoke an action on the application"),
63                     N_("APPID ACTION [PARAMETER]")
64   },
65   { "list-actions", N_("List available actions"),
66                     N_("List static actions for an application (from .desktop file)"),
67                     N_("APPID")
68   }
69 };
70
71 static const struct help_substvar substvars[] = {
72   { N_("COMMAND"),   N_("The command to print detailed help for")                             },
73   { N_("APPID"),     N_("Application identifier in D-Bus format (eg: org.example.viewer)")    },
74   { N_("FILE"),      N_("Optional relative or relative filenames, or URIs to open")           },
75   { N_("ACTION"),    N_("The action name to invoke")                                          },
76   { N_("PARAMETER"), N_("Optional parameter to the action invocation, in GVariant format")    }
77 };
78
79 static int
80 app_help (gboolean     requested,
81           const gchar *command)
82 {
83   const struct help_topic *topic = NULL;
84   GString *string;
85
86   string = g_string_new (NULL);
87
88   if (command)
89     {
90       gint i;
91
92       for (i = 0; i < G_N_ELEMENTS (topics); i++)
93         if (g_str_equal (topics[i].command, command))
94           topic = &topics[i];
95
96       if (!topic)
97         {
98           g_string_printf (string, _("Unknown command %s\n\n"), command);
99           requested = FALSE;
100         }
101     }
102
103   g_string_append (string, _("Usage:\n"));
104
105   if (topic)
106     {
107       gint maxwidth;
108       gint i;
109
110       g_string_append_printf (string, "\n  %s %s %s\n\n", "gapplication",
111                               topic->command, topic->synopsis ? _(topic->synopsis) : "");
112       g_string_append_printf (string, "%s\n\n", _(topic->description));
113
114       if (topic->synopsis)
115         {
116           g_string_append (string, _("Arguments:\n"));
117
118           maxwidth = 0;
119           for (i = 0; i < G_N_ELEMENTS (substvars); i++)
120             if (strstr (topic->synopsis, substvars[i].var))
121               maxwidth = MAX(maxwidth, strlen (_(substvars[i].var)));
122
123           for (i = 0; i < G_N_ELEMENTS (substvars); i++)
124             if (strstr (topic->synopsis, substvars[i].var))
125               g_string_append_printf (string, "  %-*.*s   %s\n", maxwidth, maxwidth,
126                                       _(substvars[i].var), _(substvars[i].description));
127           g_string_append (string, "\n");
128         }
129     }
130   else
131     {
132       gint maxwidth;
133       gint i;
134
135       g_string_append_printf (string, "\n  %s %s %s\n\n", "gapplication", _("COMMAND"), _("[ARGS...]"));
136       g_string_append_printf (string, _("Commands:\n"));
137
138       maxwidth = 0;
139       for (i = 0; i < G_N_ELEMENTS (topics); i++)
140         maxwidth = MAX(maxwidth, strlen (topics[i].command));
141
142       for (i = 0; i < G_N_ELEMENTS (topics); i++)
143         g_string_append_printf (string, "  %-*.*s   %s\n", maxwidth, maxwidth,
144                                 topics[i].command, _(topics[i].summary));
145
146       g_string_append (string, "\n");
147       /* Translators: do not translate 'help', but please translate 'COMMAND'. */
148       g_string_append_printf (string, _("Use '%s help COMMAND' to get detailed help.\n\n"), "gapplication");
149     }
150
151   if (requested)
152     g_print ("%s", string->str);
153   else
154     g_printerr ("%s\n", string->str);
155
156   g_string_free (string, TRUE);
157
158   return requested ? 0 : 1;
159 }
160
161 static gboolean
162 app_check_name (gchar       **args,
163                 const gchar  *command)
164 {
165   if (args[0] == NULL)
166     {
167       g_printerr (_("%s command requires an application id to directly follow\n\n"), command);
168       return FALSE;
169     }
170
171   if (!g_dbus_is_name (args[0]))
172     {
173       g_printerr (_("invalid application id: '%s'\n"), args[0]);
174       return FALSE;
175     }
176
177   return TRUE;
178 }
179
180 static int
181 app_no_args (const gchar *command)
182 {
183   /* Translators: %s is replaced with a command name like 'list-actions' */
184   g_printerr (_("'%s' takes no arguments\n\n"), command);
185   return app_help (FALSE, command);
186 }
187
188 static int
189 app_version (gchar **args)
190 {
191   if (g_strv_length (args))
192     return app_no_args ("version");
193
194   g_print (PACKAGE_VERSION "\n");
195   return 0;
196 }
197
198 static int
199 app_list (gchar **args)
200 {
201   GList *apps;
202
203   if (g_strv_length (args))
204     return app_no_args ("list");
205
206   apps = g_app_info_get_all ();
207
208   while (apps)
209     {
210       GDesktopAppInfo *info = apps->data;
211
212       if (G_IS_DESKTOP_APP_INFO (info))
213         if (g_desktop_app_info_get_boolean (info, "DBusActivatable"))
214           {
215             const gchar *filename;
216
217             filename = g_app_info_get_id (G_APP_INFO (info));
218             if (g_str_has_suffix (filename, ".desktop"))
219               {
220                 gchar *id;
221
222                 id = g_strndup (filename, strlen (filename) - 8);
223                 g_print ("%s\n", id);
224                 g_free (id);
225               }
226           }
227
228       apps = g_list_delete_link (apps, apps);
229       g_object_unref (info);
230     }
231
232   return 0;
233 }
234
235 static gchar *
236 app_path_for_id (const gchar *app_id)
237 {
238   gchar *path;
239   gint i;
240
241   path = g_strconcat ("/", app_id, NULL);
242   for (i = 0; path[i]; i++)
243     if (path[i] == '.')
244       path[i] = '/';
245
246   return path;
247 }
248
249 static int
250 app_call (const gchar *app_id,
251           const gchar *method_name,
252           GVariant    *parameters)
253 {
254   GDBusConnection *session;
255   GError *error = NULL;
256   gchar *object_path;
257   GVariant *result;
258
259
260   session = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error);
261   if (!session)
262     {
263       g_variant_unref (g_variant_ref_sink (parameters));
264       g_printerr (_("unable to connect to D-Bus: %s\n"), error->message);
265       g_error_free (error);
266       return 1;
267     }
268
269   object_path = app_path_for_id (app_id);
270
271   result = g_dbus_connection_call_sync (session, app_id, object_path, "org.freedesktop.Application",
272                                         method_name, parameters, G_VARIANT_TYPE_UNIT,
273                                         G_DBUS_CALL_FLAGS_NONE, -1, NULL, &error);
274
275   g_free (object_path);
276
277   if (result)
278     {
279       g_variant_unref (result);
280       return 0;
281     }
282   else
283     {
284       g_printerr (_("error sending %s message to application: %s\n"), method_name, error->message);
285       g_error_free (error);
286       return 1;
287     }
288 }
289
290 static GVariant *
291 app_get_platform_data (void)
292 {
293   GVariantBuilder builder;
294   const gchar *startup_id;
295
296   g_variant_builder_init (&builder, G_VARIANT_TYPE_VARDICT);
297
298   if ((startup_id = g_getenv ("DESKTOP_STARTUP_iD")))
299     g_variant_builder_add (&builder, "{sv}", "desktop-startup-id", g_variant_new_string (startup_id));
300
301   return g_variant_builder_end (&builder);
302 }
303
304 static int
305 app_action (gchar **args)
306 {
307   GVariantBuilder params;
308   const gchar *name;
309
310   if (!app_check_name (args, "action"))
311     return 1;
312
313   if (args[1] == NULL)
314     {
315       g_printerr (_("action name must be given after application id\n"));
316       return 1;
317     }
318
319   name = args[1];
320
321   if (!g_action_name_is_valid (name))
322     {
323       g_printerr (_("invalid action name: '%s'\n"
324                     "action names must consist of only alphanumerics, '-' and '.'\n"), name);
325       return 1;
326     }
327
328   g_variant_builder_init (&params, G_VARIANT_TYPE ("av"));
329
330   if (args[2])
331     {
332       GError *error = NULL;
333       GVariant *parameter;
334
335       parameter = g_variant_parse (NULL, args[2], NULL, NULL, &error);
336
337       if (!parameter)
338         {
339           g_printerr (_("error parsing action parameter: %s\n"), error->message);
340           g_variant_builder_clear (&params);
341           g_error_free (error);
342           return 1;
343         }
344
345       g_variant_builder_add (&params, "v", parameter);
346       g_variant_unref (parameter);
347
348       if (args[3])
349         {
350           g_printerr (_("actions accept a maximum of one parameter\n"));
351           g_variant_builder_clear (&params);
352           return 1;
353         }
354     }
355
356   return app_call (args[0], "ActivateAction", g_variant_new ("(sav@a{sv})", name, &params, app_get_platform_data ()));
357 }
358
359 static int
360 app_activate (const gchar *app_id)
361 {
362   return app_call (app_id, "Activate", g_variant_new ("(@a{sv})", app_get_platform_data ()));
363 }
364
365 static int
366 app_launch (gchar **args)
367 {
368   GVariantBuilder files;
369   gint i;
370
371   if (!app_check_name (args, "launch"))
372     return 1;
373
374   if (args[1] == NULL)
375     return app_activate (args[0]);
376
377   g_variant_builder_init (&files, G_VARIANT_TYPE_STRING_ARRAY);
378
379   for (i = 1; args[i]; i++)
380     {
381       GFile *file;
382
383       /* "This operation never fails" */
384       file = g_file_new_for_commandline_arg (args[i]);
385       g_variant_builder_add_value (&files, g_variant_new_take_string (g_file_get_uri (file)));
386       g_object_unref (file);
387     }
388
389   return app_call (args[0], "Open", g_variant_new ("(as@a{sv})", &files, app_get_platform_data ()));
390 }
391
392 static int
393 app_list_actions (gchar **args)
394 {
395   const gchar * const *actions;
396   GDesktopAppInfo *app_info;
397   gchar *filename;
398   gint i;
399
400   if (!app_check_name (args, "list-actions"))
401     return 1;
402
403   if (args[1])
404     {
405       g_printerr (_("list-actions command takes only the application id"));
406       app_help (FALSE, "list-actions");
407     }
408
409   filename = g_strconcat (args[0], ".desktop", NULL);
410   app_info = g_desktop_app_info_new (filename);
411   g_free (filename);
412
413   if (app_info == NULL)
414     {
415       g_printerr (_("unable to find desktop file for application %s\n"), args[0]);
416       return 1;
417     }
418
419   actions = g_desktop_app_info_list_actions (app_info);
420
421   for (i = 0; actions[i]; i++)
422     g_print ("%s\n", actions[i]);
423
424   g_object_unref (app_info);
425
426   return 0;
427 }
428
429 int
430 main (int argc, char **argv)
431 {
432   setlocale (LC_ALL, "");
433   textdomain (GETTEXT_PACKAGE);
434   bindtextdomain (GETTEXT_PACKAGE, GLIB_LOCALE_DIR);
435 #ifdef HAVE_BIND_TEXTDOMAIN_CODESET
436   bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
437 #endif
438
439   if (argc < 2)
440     return app_help (TRUE, NULL);
441
442   if (g_str_equal (argv[1], "help"))
443     return app_help (TRUE, argv[2]);
444
445   if (g_str_equal (argv[1], "version"))
446     return app_version (argv + 2);
447
448   if (g_str_equal (argv[1], "list-apps"))
449     return app_list (argv + 2);
450
451   if (g_str_equal (argv[1], "launch"))
452     return app_launch (argv + 2);
453
454   if (g_str_equal (argv[1], "action"))
455     return app_action (argv + 2);
456
457   if (g_str_equal (argv[1], "list-actions"))
458     return app_list_actions (argv + 2);
459
460   g_printerr (_("unrecognised command: %s\n\n"), argv[1]);
461
462   return app_help (FALSE, NULL);
463 }