glib/gio/gio-tool.c
Philip Withnall 96ee1ea086 gio-tool: Print help output to stdout when --help is passed
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>
2023-11-07 13:51:56 +00:00

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;
}