51518d4d96
The gnome-extensions tool code is really independent from the rest of the code base, and could be used either as part of the gnome-shell build or as stand-alone project (for example for the extension-ci docker image). We can actually support both cases by moving the code to a subproject. https://gitlab.gnome.org/GNOME/gnome-shell/merge_requests/877
516 lines
15 KiB
C
516 lines
15 KiB
C
/* command-pack.c
|
|
*
|
|
* Copyright 2018 Florian Müllner <fmuellner@gnome.org>
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
|
*/
|
|
|
|
#include <glib/gi18n.h>
|
|
#include <gio/gio.h>
|
|
|
|
#include <gnome-autoar/gnome-autoar.h>
|
|
#include <json-glib/json-glib.h>
|
|
|
|
#include "commands.h"
|
|
#include "common.h"
|
|
#include "config.h"
|
|
|
|
typedef struct _ExtensionPack {
|
|
GHashTable *files;
|
|
JsonObject *metadata;
|
|
GFile *tmpdir;
|
|
char *srcdir;
|
|
} ExtensionPack;
|
|
|
|
static void extension_pack_free (ExtensionPack *);
|
|
G_DEFINE_AUTOPTR_CLEANUP_FUNC (ExtensionPack, extension_pack_free);
|
|
|
|
static ExtensionPack *
|
|
extension_pack_new (const char *srcdir)
|
|
{
|
|
ExtensionPack *pack = g_new0 (ExtensionPack, 1);
|
|
pack->srcdir = g_strdup (srcdir);
|
|
pack->files = g_hash_table_new_full (g_str_hash, g_str_equal,
|
|
g_free, g_object_unref);
|
|
return pack;
|
|
}
|
|
|
|
static void
|
|
extension_pack_free (ExtensionPack *pack)
|
|
{
|
|
if (pack->tmpdir)
|
|
file_delete_recursively (pack->tmpdir, NULL);
|
|
|
|
g_clear_pointer (&pack->files, g_hash_table_destroy);
|
|
g_clear_pointer (&pack->metadata, json_object_unref);
|
|
g_clear_pointer (&pack->srcdir, g_free);
|
|
g_clear_object (&pack->tmpdir);
|
|
g_free (pack);
|
|
}
|
|
|
|
static void
|
|
extension_pack_add_source (ExtensionPack *pack,
|
|
const char *filename)
|
|
{
|
|
g_autoptr (GFile) file = NULL;
|
|
file = g_file_new_for_commandline_arg_and_cwd (filename, pack->srcdir);
|
|
if (g_file_query_exists (file, NULL))
|
|
g_hash_table_insert (pack->files,
|
|
g_path_get_basename (filename), g_steal_pointer (&file));
|
|
}
|
|
|
|
static gboolean
|
|
extension_pack_check_required_file (ExtensionPack *pack,
|
|
const char *filename,
|
|
GError **error)
|
|
{
|
|
if (!g_hash_table_contains (pack->files, filename))
|
|
{
|
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND,
|
|
"Missing %s in extension pack", filename);
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
ensure_tmpdir (ExtensionPack *pack,
|
|
GError **error)
|
|
{
|
|
g_autofree char *path = NULL;
|
|
|
|
if (pack->tmpdir != NULL)
|
|
return TRUE;
|
|
|
|
path = g_dir_make_tmp ("gnome-extensions.XXXXXX", error);
|
|
if (path != NULL)
|
|
pack->tmpdir = g_file_new_for_path (path);
|
|
|
|
return pack->tmpdir != NULL;
|
|
}
|
|
|
|
static gboolean
|
|
ensure_metadata (ExtensionPack *pack,
|
|
GError **error)
|
|
{
|
|
g_autoptr (JsonParser) parser = NULL;
|
|
g_autoptr (GInputStream) stream = NULL;
|
|
GFile *file = NULL;
|
|
|
|
if (pack->metadata != NULL)
|
|
return TRUE;
|
|
|
|
if (!extension_pack_check_required_file (pack, "metadata.json", error))
|
|
return FALSE;
|
|
|
|
file = g_hash_table_lookup (pack->files, "metadata.json");
|
|
stream = G_INPUT_STREAM (g_file_read (file, NULL, error));
|
|
|
|
if (stream == NULL)
|
|
return FALSE;
|
|
|
|
parser = json_parser_new_immutable ();
|
|
|
|
if (!json_parser_load_from_stream (parser, stream, NULL, error))
|
|
return FALSE;
|
|
|
|
pack->metadata = json_node_dup_object (json_parser_get_root (parser));
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
extension_pack_add_schemas (ExtensionPack *pack,
|
|
char **schemas,
|
|
GError **error)
|
|
{
|
|
g_autoptr (GSubprocess) proc = NULL;
|
|
g_autoptr (GFile) dstdir = NULL;
|
|
g_autofree char *dstpath = NULL;
|
|
char **s;
|
|
|
|
if (!ensure_tmpdir (pack, error))
|
|
return FALSE;
|
|
|
|
dstdir = g_file_get_child (pack->tmpdir, "schemas");
|
|
if (!g_file_make_directory (dstdir, NULL, error))
|
|
return FALSE;
|
|
|
|
for (s = schemas; s && *s; s++)
|
|
{
|
|
g_autoptr (GFile) src = NULL;
|
|
g_autoptr (GFile) dst = NULL;
|
|
g_autofree char *basename = NULL;
|
|
|
|
src = g_file_new_for_commandline_arg_and_cwd (*s, pack->srcdir);
|
|
|
|
basename = g_file_get_basename (src);
|
|
dst = g_file_get_child (dstdir, basename);
|
|
|
|
if (!g_file_copy (src, dst, G_FILE_COPY_NONE, NULL, NULL, NULL, error))
|
|
return FALSE;
|
|
}
|
|
|
|
dstpath = g_file_get_path (dstdir);
|
|
proc = g_subprocess_new (G_SUBPROCESS_FLAGS_STDERR_SILENCE, error,
|
|
"glib-compile-schemas", "--strict", dstpath, NULL);
|
|
|
|
if (!g_subprocess_wait_check (proc, NULL, error))
|
|
return FALSE;
|
|
|
|
g_hash_table_insert (pack->files,
|
|
g_strdup ("schemas"), g_steal_pointer (&dstdir));
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
extension_pack_add_locales (ExtensionPack *pack,
|
|
const char *podir,
|
|
const char *gettext_domain,
|
|
GError **error)
|
|
{
|
|
g_autoptr (GFile) dstdir = NULL;
|
|
g_autoptr (GFile) srcdir = NULL;
|
|
g_autoptr (GFileEnumerator) file_enum = NULL;
|
|
g_autofree char *dstpath = NULL;
|
|
g_autofree char *moname = NULL;
|
|
GFile *child;
|
|
GFileInfo *info;
|
|
|
|
if (!ensure_tmpdir (pack, error))
|
|
return FALSE;
|
|
|
|
dstdir = g_file_get_child (pack->tmpdir, "locale");
|
|
if (!g_file_make_directory (dstdir, NULL, error))
|
|
return FALSE;
|
|
|
|
srcdir = g_file_new_for_commandline_arg_and_cwd (podir, pack->srcdir);
|
|
file_enum = g_file_enumerate_children (srcdir,
|
|
G_FILE_ATTRIBUTE_STANDARD_NAME,
|
|
G_FILE_QUERY_INFO_NONE,
|
|
NULL,
|
|
error);
|
|
if (file_enum == NULL)
|
|
return FALSE;
|
|
|
|
if (gettext_domain == NULL)
|
|
{
|
|
if (!ensure_metadata (pack, error))
|
|
return FALSE;
|
|
|
|
if (json_object_has_member (pack->metadata, "gettext-domain"))
|
|
gettext_domain = json_object_get_string_member (pack->metadata,
|
|
"gettext-domain");
|
|
else
|
|
gettext_domain = json_object_get_string_member (pack->metadata,
|
|
"uuid");
|
|
}
|
|
|
|
dstpath = g_file_get_path (dstdir);
|
|
moname = g_strdup_printf ("%s.mo", gettext_domain);
|
|
|
|
while (TRUE)
|
|
{
|
|
g_autoptr (GSubprocess) proc = NULL;
|
|
g_autoptr (GFile) modir = NULL;
|
|
g_autofree char *popath = NULL;
|
|
g_autofree char *mopath = NULL;
|
|
g_autofree char *lang = NULL;
|
|
const char *name;
|
|
|
|
if (!g_file_enumerator_iterate (file_enum, &info, &child, NULL, error))
|
|
return FALSE;
|
|
|
|
if (info == NULL)
|
|
break;
|
|
|
|
name = g_file_info_get_name (info);
|
|
if (!g_str_has_suffix (name, ".po"))
|
|
continue;
|
|
|
|
lang = g_strndup (name, strlen (name) - 3 /* strlen (".po") */);
|
|
modir = g_file_new_build_filename (dstpath, lang, "LC_MESSAGES", NULL);
|
|
if (!g_file_make_directory_with_parents (modir, NULL, error))
|
|
return FALSE;
|
|
|
|
mopath = g_build_filename (dstpath, lang, "LC_MESSAGES", moname, NULL);
|
|
popath = g_file_get_path (child);
|
|
|
|
proc = g_subprocess_new (G_SUBPROCESS_FLAGS_STDERR_SILENCE, error,
|
|
"msgfmt", "-o", mopath, popath, NULL);
|
|
|
|
if (!g_subprocess_wait_check (proc, NULL, error))
|
|
return FALSE;
|
|
}
|
|
|
|
g_hash_table_insert (pack->files,
|
|
g_strdup ("locale"), g_steal_pointer (&dstdir));
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
on_error (AutoarCompressor *compressor,
|
|
GError *error,
|
|
gpointer data)
|
|
{
|
|
*((GError **)data) = g_error_copy (error);
|
|
}
|
|
|
|
static gboolean
|
|
extension_pack_compress (ExtensionPack *pack,
|
|
const char *outdir,
|
|
gboolean overwrite,
|
|
GError **error)
|
|
{
|
|
g_autoptr (AutoarCompressor) compressor = NULL;
|
|
g_autoptr (GError) err = NULL;
|
|
g_autoptr (GFile) outfile = NULL;
|
|
g_autofree char *name = NULL;
|
|
const char *uuid;
|
|
|
|
if (!ensure_metadata (pack, error))
|
|
return FALSE;
|
|
|
|
uuid = json_object_get_string_member (pack->metadata, "uuid");
|
|
name = g_strdup_printf ("%s.shell-extension.zip", uuid);
|
|
outfile = g_file_new_for_commandline_arg_and_cwd (name, outdir);
|
|
|
|
if (g_file_query_exists (outfile, NULL))
|
|
{
|
|
if (!overwrite)
|
|
{
|
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_EXISTS,
|
|
"%s exists and --force was not specified", name);
|
|
return FALSE;
|
|
}
|
|
else if (!g_file_delete (outfile, NULL, error))
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
compressor = autoar_compressor_new (g_hash_table_get_values (pack->files),
|
|
outfile,
|
|
AUTOAR_FORMAT_ZIP,
|
|
AUTOAR_FILTER_NONE,
|
|
FALSE);
|
|
autoar_compressor_set_output_is_dest (compressor, TRUE);
|
|
|
|
g_signal_connect (compressor, "error", G_CALLBACK (on_error), err);
|
|
|
|
autoar_compressor_start (compressor, NULL);
|
|
|
|
if (err != NULL)
|
|
{
|
|
g_propagate_error (error, err);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static char **
|
|
find_schemas (const char *basepath,
|
|
GError **error)
|
|
{
|
|
g_autoptr (GFile) basedir = NULL;
|
|
g_autoptr (GFile) schemadir = NULL;
|
|
g_autoptr (GFileEnumerator) file_enum = NULL;
|
|
g_autoptr (GPtrArray) schemas = NULL;
|
|
GFile *child;
|
|
GFileInfo *info;
|
|
|
|
basedir = g_file_new_for_path (basepath);
|
|
schemadir = g_file_get_child (basedir, "schemas");
|
|
file_enum = g_file_enumerate_children (schemadir,
|
|
G_FILE_ATTRIBUTE_STANDARD_NAME,
|
|
G_FILE_QUERY_INFO_NONE,
|
|
NULL, error);
|
|
|
|
if (error && *error)
|
|
{
|
|
if (g_error_matches (*error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND) ||
|
|
g_error_matches (*error, G_IO_ERROR, G_IO_ERROR_NOT_DIRECTORY))
|
|
g_clear_error (error);
|
|
return NULL;
|
|
}
|
|
|
|
schemas = g_ptr_array_new_with_free_func (g_free);
|
|
|
|
while (TRUE)
|
|
{
|
|
if (!g_file_enumerator_iterate (file_enum, &info, &child, NULL, error))
|
|
return NULL;
|
|
|
|
if (child == NULL)
|
|
break;
|
|
|
|
if (!g_str_has_suffix (g_file_info_get_name (info), ".gschema.xml"))
|
|
continue;
|
|
|
|
g_ptr_array_add (schemas, g_file_get_relative_path (basedir, child));
|
|
}
|
|
g_ptr_array_add (schemas, NULL);
|
|
|
|
return (char **)g_ptr_array_free (g_ptr_array_ref (schemas), FALSE);
|
|
}
|
|
|
|
static int
|
|
pack_extension (char *srcdir,
|
|
char *dstdir,
|
|
gboolean force,
|
|
char **extra_sources,
|
|
char **schemas,
|
|
char *podir,
|
|
char *gettext_domain)
|
|
{
|
|
g_autoptr (ExtensionPack) pack = NULL;
|
|
g_autoptr (GError) error = NULL;
|
|
char **s;
|
|
|
|
pack = extension_pack_new (srcdir);
|
|
extension_pack_add_source (pack, "extension.js");
|
|
extension_pack_add_source (pack, "metadata.json");
|
|
extension_pack_add_source (pack, "stylesheet.css");
|
|
extension_pack_add_source (pack, "prefs.js");
|
|
|
|
for (s = extra_sources; s && *s; s++)
|
|
extension_pack_add_source (pack, *s);
|
|
|
|
if (!extension_pack_check_required_file (pack, "extension.js", &error))
|
|
goto err;
|
|
|
|
if (!extension_pack_check_required_file (pack, "metadata.json", &error))
|
|
goto err;
|
|
|
|
if (schemas == NULL)
|
|
schemas = find_schemas (srcdir, &error);
|
|
|
|
if (schemas != NULL)
|
|
extension_pack_add_schemas (pack, schemas, &error);
|
|
|
|
if (error)
|
|
goto err;
|
|
|
|
if (podir == NULL)
|
|
{
|
|
g_autoptr (GFile) dir = NULL;
|
|
|
|
dir = g_file_new_for_commandline_arg_and_cwd ("po", srcdir);
|
|
if (g_file_query_exists (dir, NULL))
|
|
podir = (char *)"po";
|
|
}
|
|
|
|
if (podir != NULL)
|
|
extension_pack_add_locales (pack, podir, gettext_domain, &error);
|
|
|
|
if (error)
|
|
goto err;
|
|
|
|
extension_pack_compress (pack, dstdir, force, &error);
|
|
|
|
err:
|
|
if (error)
|
|
{
|
|
g_printerr ("%s\n", error->message);
|
|
return 2;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
handle_pack (int argc, char *argv[], gboolean do_help)
|
|
{
|
|
g_autoptr (GOptionContext) context = NULL;
|
|
g_autoptr (GError) error = NULL;
|
|
g_auto(GStrv) extra_sources = NULL;
|
|
g_auto(GStrv) schemas = NULL;
|
|
g_auto(GStrv) srcdirs = NULL;
|
|
g_autofree char *podir = NULL;
|
|
g_autofree char *srcdir = NULL;
|
|
g_autofree char *dstdir = NULL;
|
|
g_autofree char *gettext_domain = NULL;
|
|
gboolean force = FALSE;
|
|
GOptionEntry entries[] = {
|
|
{ .long_name = "extra-source",
|
|
.arg = G_OPTION_ARG_FILENAME_ARRAY, .arg_data = &extra_sources,
|
|
.arg_description = _("FILE"),
|
|
.description = _("Additional source to include in the bundle") },
|
|
{ .long_name = "schema",
|
|
.arg = G_OPTION_ARG_FILENAME_ARRAY, .arg_data = &schemas,
|
|
.arg_description = _("SCHEMA"),
|
|
.description = _("A GSettings schema that should be included") },
|
|
{ .long_name = "podir",
|
|
.arg_description = _("DIRECTORY"),
|
|
.arg = G_OPTION_ARG_FILENAME, .arg_data = &podir,
|
|
.description = _("The directory where translations are found") },
|
|
{ .long_name = "gettext-domain",
|
|
.arg_description = _("DOMAIN"),
|
|
.arg = G_OPTION_ARG_STRING, .arg_data = &gettext_domain,
|
|
.description = _("The gettext domain to use for translations") },
|
|
{ .long_name = "force", .short_name = 'f',
|
|
.arg = G_OPTION_ARG_NONE, .arg_data = &force,
|
|
.description = _("Overwrite an existing pack") },
|
|
{ .long_name = "out-dir", .short_name = 'o',
|
|
.arg_description = _("DIRECTORY"),
|
|
.arg = G_OPTION_ARG_FILENAME, .arg_data = &dstdir,
|
|
.description = _("The directory where the pack should be created") },
|
|
{ .long_name = G_OPTION_REMAINING,
|
|
.arg_description =_("SOURCE_DIRECTORY"),
|
|
.arg = G_OPTION_ARG_FILENAME_ARRAY, .arg_data = &srcdirs },
|
|
{ NULL }
|
|
};
|
|
|
|
g_set_prgname ("gnome-extensions pack");
|
|
|
|
context = g_option_context_new (NULL);
|
|
g_option_context_set_help_enabled (context, FALSE);
|
|
g_option_context_set_summary (context, _("Create an extension bundle"));
|
|
g_option_context_add_main_entries (context, entries, GETTEXT_PACKAGE);
|
|
|
|
if (do_help)
|
|
{
|
|
show_help (context, NULL);
|
|
return 0;
|
|
}
|
|
|
|
if (!g_option_context_parse (context, &argc, &argv, &error))
|
|
{
|
|
show_help (context, error->message);
|
|
return 1;
|
|
}
|
|
|
|
if (srcdirs)
|
|
{
|
|
if (g_strv_length (srcdirs) > 1)
|
|
{
|
|
show_help (context, _("More than one source directory specified"));
|
|
return 1;
|
|
}
|
|
srcdir = g_strdup (*srcdirs);
|
|
}
|
|
else
|
|
{
|
|
srcdir = g_get_current_dir ();
|
|
}
|
|
|
|
if (dstdir == NULL)
|
|
dstdir = g_get_current_dir ();
|
|
|
|
return pack_extension (srcdir, dstdir, force,
|
|
extra_sources, schemas, podir, gettext_domain);
|
|
}
|