mirror of
				https://gitlab.gnome.org/GNOME/glib.git
				synced 2025-11-04 10:08:56 +01: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;
 | 
						|
}
 |