mirror of
				https://gitlab.gnome.org/GNOME/glib.git
				synced 2025-10-24 22:12:16 +02:00 
			
		
		
		
	If the help output is explicitly requested by the user, it’s conventional for it to be printed to stdout rather than stderr. Signed-off-by: Philip Withnall <philip@tecnocode.co.uk>
		
			
				
	
	
		
			357 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			357 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright 2015 Red Hat, Inc.
 | |
|  *
 | |
|  * SPDX-License-Identifier: LGPL-2.1-or-later
 | |
|  *
 | |
|  * This library is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU Lesser General Public
 | |
|  * License as published by the Free Software Foundation; either
 | |
|  * version 2.1 of the License, or (at your option) any later version.
 | |
|  *
 | |
|  * This library is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 | |
|  * Lesser General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Lesser General Public
 | |
|  * License along with this library; if not, see <http://www.gnu.org/licenses/>.
 | |
|  *
 | |
|  * Author: Matthias Clasen <mclasen@redhat.com>
 | |
|  */
 | |
| 
 | |
| #include "config.h"
 | |
| 
 | |
| #include <gio/gio.h>
 | |
| #include <gi18n.h>
 | |
| #include <locale.h>
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include <stdlib.h>
 | |
| #include <errno.h>
 | |
| 
 | |
| #include "gio-tool.h"
 | |
| #include "glib/glib-private.h"
 | |
| 
 | |
| void
 | |
| print_error (const gchar *format, ...)
 | |
| {
 | |
|   gchar *message;
 | |
|   va_list args;
 | |
| 
 | |
|   va_start (args, format);
 | |
|   message = g_strdup_vprintf (format, args);
 | |
|   va_end (args);
 | |
| 
 | |
|   g_printerr ("gio: %s\n", message);
 | |
|   g_free (message);
 | |
| }
 | |
| 
 | |
| void
 | |
| print_file_error (GFile *file, const gchar *message)
 | |
| {
 | |
|   gchar *uri;
 | |
| 
 | |
|   uri = g_file_get_uri (file);
 | |
|   print_error ("%s: %s", uri, message);
 | |
|   g_free (uri);
 | |
| }
 | |
| 
 | |
| void
 | |
| show_help (GOptionContext *context, const char *message)
 | |
| {
 | |
|   char *help;
 | |
| 
 | |
|   if (message)
 | |
|     g_printerr ("gio: %s\n\n", message);
 | |
| 
 | |
|   help = g_option_context_get_help (context, TRUE, NULL);
 | |
|   g_printerr ("%s", help);
 | |
|   g_free (help);
 | |
| }
 | |
| 
 | |
| const char *
 | |
| file_type_to_string (GFileType type)
 | |
| {
 | |
|   switch (type)
 | |
|     {
 | |
|     case G_FILE_TYPE_UNKNOWN:
 | |
|       return "unknown";
 | |
|     case G_FILE_TYPE_REGULAR:
 | |
|       return "regular";
 | |
|     case G_FILE_TYPE_DIRECTORY:
 | |
|       return "directory";
 | |
|     case G_FILE_TYPE_SYMBOLIC_LINK:
 | |
|       return "symlink";
 | |
|     case G_FILE_TYPE_SPECIAL:
 | |
|       return "special";
 | |
|     case G_FILE_TYPE_SHORTCUT:
 | |
|       return "shortcut";
 | |
|     case G_FILE_TYPE_MOUNTABLE:
 | |
|       return "mountable";
 | |
|     default:
 | |
|       return "invalid type";
 | |
|     }
 | |
| }
 | |
| 
 | |
| const char *
 | |
| attribute_type_to_string (GFileAttributeType type)
 | |
| {
 | |
|   switch (type)
 | |
|     {
 | |
|     case G_FILE_ATTRIBUTE_TYPE_INVALID:
 | |
|       return "invalid";
 | |
|     case G_FILE_ATTRIBUTE_TYPE_STRING:
 | |
|       return "string";
 | |
|     case G_FILE_ATTRIBUTE_TYPE_BYTE_STRING:
 | |
|       return "bytestring";
 | |
|     case G_FILE_ATTRIBUTE_TYPE_BOOLEAN:
 | |
|       return "boolean";
 | |
|     case G_FILE_ATTRIBUTE_TYPE_UINT32:
 | |
|       return "uint32";
 | |
|     case G_FILE_ATTRIBUTE_TYPE_INT32:
 | |
|       return "int32";
 | |
|     case G_FILE_ATTRIBUTE_TYPE_UINT64:
 | |
|       return "uint64";
 | |
|     case G_FILE_ATTRIBUTE_TYPE_INT64:
 | |
|       return "int64";
 | |
|     case G_FILE_ATTRIBUTE_TYPE_OBJECT:
 | |
|       return "object";
 | |
|     default:
 | |
|       return "unknown type";
 | |
|     }
 | |
| }
 | |
| 
 | |
| GFileAttributeType
 | |
| attribute_type_from_string (const char *str)
 | |
| {
 | |
|   if (strcmp (str, "string") == 0)
 | |
|     return G_FILE_ATTRIBUTE_TYPE_STRING;
 | |
|   if (strcmp (str, "stringv") == 0)
 | |
|     return G_FILE_ATTRIBUTE_TYPE_STRINGV;
 | |
|   if (strcmp (str, "bytestring") == 0)
 | |
|     return G_FILE_ATTRIBUTE_TYPE_BYTE_STRING;
 | |
|   if (strcmp (str, "boolean") == 0)
 | |
|     return G_FILE_ATTRIBUTE_TYPE_BOOLEAN;
 | |
|   if (strcmp (str, "uint32") == 0)
 | |
|     return G_FILE_ATTRIBUTE_TYPE_UINT32;
 | |
|   if (strcmp (str, "int32") == 0)
 | |
|     return G_FILE_ATTRIBUTE_TYPE_INT32;
 | |
|   if (strcmp (str, "uint64") == 0)
 | |
|     return G_FILE_ATTRIBUTE_TYPE_UINT64;
 | |
|   if (strcmp (str, "int64") == 0)
 | |
|     return G_FILE_ATTRIBUTE_TYPE_INT64;
 | |
|   if (strcmp (str, "object") == 0)
 | |
|     return G_FILE_ATTRIBUTE_TYPE_OBJECT;
 | |
|   if (strcmp (str, "unset") == 0)
 | |
|     return G_FILE_ATTRIBUTE_TYPE_INVALID;
 | |
|   return -1;
 | |
| }
 | |
| 
 | |
| char *
 | |
| attribute_flags_to_string (GFileAttributeInfoFlags flags)
 | |
| {
 | |
|   GString *s;
 | |
|   gsize i;
 | |
|   gboolean first;
 | |
|   struct {
 | |
|     guint32 mask;
 | |
|     char *descr;
 | |
|   } flag_descr[] = {
 | |
|     {
 | |
|       G_FILE_ATTRIBUTE_INFO_COPY_WITH_FILE,
 | |
|       N_("Copy with file")
 | |
|     },
 | |
|     {
 | |
|       G_FILE_ATTRIBUTE_INFO_COPY_WHEN_MOVED,
 | |
|       N_("Keep with file when moved")
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   first = TRUE;
 | |
| 
 | |
|   s = g_string_new ("");
 | |
|   for (i = 0; i < G_N_ELEMENTS (flag_descr); i++)
 | |
|     {
 | |
|       if (flags & flag_descr[i].mask)
 | |
|         {
 | |
|           if (!first)
 | |
|             g_string_append (s, ", ");
 | |
|           g_string_append (s, gettext (flag_descr[i].descr));
 | |
|           first = FALSE;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|   return g_string_free (s, FALSE);
 | |
| }
 | |
| 
 | |
| gboolean
 | |
| file_is_dir (GFile *file)
 | |
| {
 | |
|   GFileInfo *info;
 | |
|   gboolean res;
 | |
| 
 | |
|   info = g_file_query_info (file, G_FILE_ATTRIBUTE_STANDARD_TYPE, 0, NULL, NULL);
 | |
|   res = info && g_file_info_get_file_type (info) == G_FILE_TYPE_DIRECTORY;
 | |
|   if (info)
 | |
|     g_object_unref (info);
 | |
|   return res;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int
 | |
| handle_version (int argc, char *argv[], gboolean do_help)
 | |
| {
 | |
|   if (do_help || argc > 1)
 | |
|     {
 | |
|       if (!do_help)
 | |
|         g_printerr ("gio: %s\n\n", _("“version” takes no arguments"));
 | |
| 
 | |
|       g_printerr ("%s\n", _("Usage:"));
 | |
|       g_printerr ("  gio version\n");
 | |
|       g_printerr ("\n");
 | |
|       g_printerr ("%s\n", _("Print version information and exit."));
 | |
| 
 | |
|       return do_help ? 0 : 2;
 | |
|     }
 | |
| 
 | |
|   g_print ("%d.%d.%d\n", glib_major_version, glib_minor_version, glib_micro_version);
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| typedef int (*HandleSubcommand) (int argc, char *argv[], gboolean do_help);
 | |
| 
 | |
| static const struct
 | |
| {
 | |
|   const char *name;  /* as used on the command line */
 | |
|   HandleSubcommand handle_func;  /* (nullable) for "help" only */
 | |
|   const char *description;  /* translatable */
 | |
| } gio_subcommands[] = {
 | |
|   { "help", NULL, N_("Print help") },
 | |
|   { "version", handle_version, N_("Print version") },
 | |
|   { "cat", handle_cat, N_("Concatenate files to standard output") },
 | |
|   { "copy", handle_copy, N_("Copy one or more files") },
 | |
|   { "info", handle_info, N_("Show information about locations") },
 | |
|   { "launch", handle_launch, N_("Launch an application from a desktop file") },
 | |
|   { "list", handle_list, N_("List the contents of locations") },
 | |
|   { "mime", handle_mime, N_("Get or set the handler for a mimetype") },
 | |
|   { "mkdir", handle_mkdir, N_("Create directories") },
 | |
|   { "monitor", handle_monitor, N_("Monitor files and directories for changes") },
 | |
|   { "mount", handle_mount, N_("Mount or unmount the locations") },
 | |
|   { "move", handle_move, N_("Move one or more files") },
 | |
|   { "open", handle_open, N_("Open files with the default application") },
 | |
|   { "rename", handle_rename, N_("Rename a file") },
 | |
|   { "remove", handle_remove, N_("Delete one or more files") },
 | |
|   { "save", handle_save, N_("Read from standard input and save") },
 | |
|   { "set", handle_set, N_("Set a file attribute") },
 | |
|   { "trash", handle_trash, N_("Move files or directories to the trash") },
 | |
|   { "tree", handle_tree, N_("Lists the contents of locations in a tree") },
 | |
| };
 | |
| 
 | |
| static void
 | |
| usage (gboolean is_error)
 | |
| {
 | |
|   GString *out = NULL;
 | |
|   size_t name_width = 0;
 | |
| 
 | |
|   out = g_string_new ("");
 | |
|   g_string_append_printf (out, "%s\n", _("Usage:"));
 | |
|   g_string_append_printf (out, "  gio %s %s\n", _("COMMAND"), _("[ARGS…]"));
 | |
|   g_string_append_c (out, '\n');
 | |
|   g_string_append_printf (out, "%s\n", _("Commands:"));
 | |
| 
 | |
|   /* Work out the maximum name length for column alignment. */
 | |
|   for (size_t i = 0; i < G_N_ELEMENTS (gio_subcommands); i++)
 | |
|     name_width = MAX (name_width, strlen (gio_subcommands[i].name));
 | |
| 
 | |
|   for (size_t i = 0; i < G_N_ELEMENTS (gio_subcommands); i++)
 | |
|     {
 | |
|       g_string_append_printf (out, "  %-*s  %s\n",
 | |
|                               (int) name_width, gio_subcommands[i].name,
 | |
|                               _(gio_subcommands[i].description));
 | |
|     }
 | |
| 
 | |
|   g_string_append_c (out, '\n');
 | |
|   g_string_append_printf (out, _("Use %s to get detailed help.\n"), "“gio help COMMAND”");
 | |
| 
 | |
|   if (is_error)
 | |
|     g_printerr ("%s", out->str);
 | |
|   else
 | |
|     g_print ("%s", out->str);
 | |
| 
 | |
|   g_string_free (out, TRUE);
 | |
| }
 | |
| 
 | |
| int
 | |
| main (int argc, char **argv)
 | |
| {
 | |
|   const char *command;
 | |
|   gboolean do_help;
 | |
| 
 | |
| #ifdef G_OS_WIN32
 | |
|   gchar *localedir;
 | |
| #endif
 | |
| 
 | |
|   setlocale (LC_ALL, GLIB_DEFAULT_LOCALE);
 | |
|   textdomain (GETTEXT_PACKAGE);
 | |
| 
 | |
| #ifdef G_OS_WIN32
 | |
|   localedir = _glib_get_locale_dir ();
 | |
|   bindtextdomain (GETTEXT_PACKAGE, localedir);
 | |
|   g_free (localedir);
 | |
| #else
 | |
|   bindtextdomain (GETTEXT_PACKAGE, GLIB_LOCALE_DIR);
 | |
| #endif
 | |
| 
 | |
| #ifdef HAVE_BIND_TEXTDOMAIN_CODESET
 | |
|   bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
 | |
| #endif
 | |
| 
 | |
|   if (argc < 2)
 | |
|     {
 | |
|       usage (TRUE);
 | |
|       return 1;
 | |
|     }
 | |
| 
 | |
|   command = argv[1];
 | |
|   argc -= 1;
 | |
|   argv += 1;
 | |
| 
 | |
|   do_help = FALSE;
 | |
|   if (g_str_equal (command, "help"))
 | |
|     {
 | |
|       if (argc == 1)
 | |
|         {
 | |
|           usage (FALSE);
 | |
|           return 0;
 | |
|         }
 | |
|       else
 | |
|         {
 | |
|           command = argv[1];
 | |
|           do_help = TRUE;
 | |
|         }
 | |
|     }
 | |
|   else if (g_str_equal (command, "--help"))
 | |
|     {
 | |
|       usage (FALSE);
 | |
|       return 0;
 | |
|     }
 | |
|   else if (g_str_equal (command, "--version"))
 | |
|     command = "version";
 | |
| 
 | |
|   /* Work out which subcommand it is. */
 | |
|   for (size_t i = 0; i < G_N_ELEMENTS (gio_subcommands); i++)
 | |
|     {
 | |
|       if (g_str_equal (command, gio_subcommands[i].name))
 | |
|         {
 | |
|           g_assert (gio_subcommands[i].handle_func != NULL);
 | |
|           return gio_subcommands[i].handle_func (argc, argv, do_help);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|   /* Unknown subcommand. */
 | |
|   usage (TRUE);
 | |
| 
 | |
|   return 1;
 | |
| }
 |