2018-08-27 01:24:10 +00:00
|
|
|
/* command-create.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 <gio/gunixinputstream.h>
|
|
|
|
|
|
|
|
#include "commands.h"
|
|
|
|
#include "common.h"
|
2018-08-27 02:30:53 +00:00
|
|
|
#include "config.h"
|
2018-08-27 01:24:10 +00:00
|
|
|
|
|
|
|
static char *
|
|
|
|
get_shell_version (GError **error)
|
|
|
|
{
|
|
|
|
g_autoptr (GDBusProxy) proxy = NULL;
|
|
|
|
g_autoptr (GVariant) variant = NULL;
|
|
|
|
g_auto (GStrv) split_version = NULL;
|
|
|
|
|
|
|
|
proxy = get_shell_proxy (error);
|
|
|
|
if (proxy == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
variant = g_dbus_proxy_get_cached_property (proxy, "ShellVersion");
|
|
|
|
if (variant == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
split_version = g_strsplit (g_variant_get_string (variant, NULL), ".", 3);
|
|
|
|
if (g_ascii_strtoll (split_version[1], NULL, 10) % 2 == 0)
|
|
|
|
g_clear_pointer (&split_version[2], g_free);
|
|
|
|
|
|
|
|
return g_strjoinv (".", split_version);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
create_metadata (GFile *target_dir,
|
|
|
|
const char *uuid,
|
|
|
|
const char *name,
|
|
|
|
const char *description,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
g_autoptr (GFile) target = NULL;
|
|
|
|
g_autoptr (GString) json = NULL;
|
|
|
|
g_autofree char *version = NULL;
|
|
|
|
|
|
|
|
version = get_shell_version (error);
|
|
|
|
if (version == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
json = g_string_new ("{\n");
|
|
|
|
|
|
|
|
g_string_append_printf (json, " \"name\": \"%s\",\n", name);
|
|
|
|
g_string_append_printf (json, " \"description\": \"%s\",\n", description);
|
|
|
|
g_string_append_printf (json, " \"uuid\": \"%s\",\n", uuid);
|
|
|
|
g_string_append_printf (json, " \"shell-version\": [\n");
|
|
|
|
g_string_append_printf (json, " \"%s\"\n", version);
|
|
|
|
g_string_append_printf (json, " ]\n}\n");
|
|
|
|
|
|
|
|
target = g_file_get_child (target_dir, "metadata.json");
|
|
|
|
return g_file_replace_contents (target,
|
|
|
|
json->str,
|
|
|
|
json->len,
|
|
|
|
NULL,
|
|
|
|
FALSE,
|
|
|
|
0,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
error);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define TEMPLATE_PATH "/org/gnome/extensions-tool/template"
|
|
|
|
static gboolean
|
|
|
|
copy_extension_template (GFile *target_dir, GError **error)
|
|
|
|
{
|
2020-03-02 17:16:25 +00:00
|
|
|
g_auto (GStrv) templates = NULL;
|
2018-08-27 01:24:10 +00:00
|
|
|
char **s;
|
|
|
|
|
|
|
|
templates = g_resources_enumerate_children (TEMPLATE_PATH, 0, NULL);
|
|
|
|
for (s = templates; *s; s++)
|
|
|
|
{
|
|
|
|
g_autoptr (GFile) target = NULL;
|
|
|
|
g_autoptr (GFile) source = NULL;
|
|
|
|
g_autofree char *uri = NULL;
|
|
|
|
|
|
|
|
uri = g_strdup_printf ("resource://%s/%s", TEMPLATE_PATH, *s);
|
|
|
|
source = g_file_new_for_uri (uri);
|
|
|
|
target = g_file_get_child (target_dir, *s);
|
|
|
|
|
|
|
|
if (!g_file_copy (source, target, G_FILE_COPY_TARGET_DEFAULT_PERMS, NULL, NULL, NULL, error))
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
launch_extension_source (GFile *dir, GError **error)
|
|
|
|
{
|
|
|
|
g_autoptr (GFile) main_source = NULL;
|
|
|
|
g_autoptr (GAppInfo) handler = NULL;
|
|
|
|
GList l;
|
|
|
|
|
|
|
|
main_source = g_file_get_child (dir, "extension.js");
|
2019-12-19 19:00:51 +00:00
|
|
|
handler = g_file_query_default_handler (main_source, NULL, NULL);
|
2018-08-27 01:24:10 +00:00
|
|
|
if (handler == NULL)
|
2019-12-19 19:00:51 +00:00
|
|
|
{
|
|
|
|
/* Translators: a file path to an extension directory */
|
|
|
|
g_print (_("The new extension was successfully created in %s.\n"),
|
|
|
|
g_file_peek_path (dir));
|
|
|
|
return TRUE;
|
|
|
|
}
|
2018-08-27 01:24:10 +00:00
|
|
|
|
|
|
|
l.data = main_source;
|
|
|
|
l.next = l.prev = NULL;
|
|
|
|
|
|
|
|
return g_app_info_launch (handler, &l, NULL, error);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
create_extension (const char *uuid, const char *name, const char *description)
|
|
|
|
{
|
|
|
|
g_autoptr (GFile) dir = NULL;
|
|
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
|
|
|
|
dir = g_file_new_build_filename (g_get_user_data_dir (),
|
|
|
|
"gnome-shell",
|
|
|
|
"extensions",
|
|
|
|
uuid,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (!g_file_make_directory_with_parents (dir, NULL, &error))
|
|
|
|
{
|
|
|
|
g_printerr ("%s\n", error->message);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!create_metadata (dir, uuid, name, description, &error))
|
|
|
|
{
|
|
|
|
g_printerr ("%s\n", error->message);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!copy_extension_template (dir, &error))
|
|
|
|
{
|
|
|
|
g_printerr ("%s\n", error->message);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!launch_extension_source (dir, &error))
|
|
|
|
{
|
|
|
|
g_printerr ("%s\n", error->message);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
prompt_metadata (char **uuid, char **name, char **description)
|
|
|
|
{
|
|
|
|
g_autoptr (GInputStream) stdin = NULL;
|
|
|
|
g_autoptr (GDataInputStream) istream = NULL;
|
|
|
|
|
2018-08-27 02:30:53 +00:00
|
|
|
if ((uuid == NULL || *uuid != NULL) &&
|
|
|
|
(name == NULL || *name != NULL) &&
|
|
|
|
(description == NULL || *description != NULL))
|
|
|
|
return;
|
|
|
|
|
2018-08-27 01:24:10 +00:00
|
|
|
stdin = g_unix_input_stream_new (0, FALSE);
|
|
|
|
istream = g_data_input_stream_new (stdin);
|
|
|
|
|
2018-08-27 02:30:53 +00:00
|
|
|
if (name != NULL && *name == NULL)
|
2018-08-27 01:24:10 +00:00
|
|
|
{
|
2020-04-06 23:49:30 +00:00
|
|
|
char *line = NULL;
|
2018-08-27 01:24:10 +00:00
|
|
|
|
|
|
|
g_print (
|
|
|
|
_("Name should be a very short (ideally descriptive) string.\n"
|
|
|
|
"Examples are: %s"),
|
|
|
|
"“Click To Focus”, “Adblock”, “Shell Window Shrinker”\n");
|
|
|
|
|
2020-04-06 23:49:30 +00:00
|
|
|
while (line == NULL)
|
|
|
|
{
|
|
|
|
g_print ("%s: ", _("Name"));
|
|
|
|
|
|
|
|
line = g_data_input_stream_read_line_utf8 (istream, NULL, NULL, NULL);
|
|
|
|
}
|
2018-08-27 01:24:10 +00:00
|
|
|
*name = g_strdelimit (line, "\n", '\0');
|
2019-11-10 02:00:37 +00:00
|
|
|
|
|
|
|
g_print ("\n");
|
2018-08-27 01:24:10 +00:00
|
|
|
}
|
|
|
|
|
2018-08-27 02:30:53 +00:00
|
|
|
if (description != NULL && *description == NULL)
|
2018-08-27 01:24:10 +00:00
|
|
|
{
|
2020-04-06 23:49:30 +00:00
|
|
|
char *line = NULL;
|
2018-08-27 01:24:10 +00:00
|
|
|
|
|
|
|
g_print (
|
|
|
|
_("Description is a single-sentence explanation of what your extension does.\n"
|
|
|
|
"Examples are: %s"),
|
|
|
|
"“Make windows visible on click”, “Block advertisement popups”, “Animate windows shrinking on minimize”\n");
|
|
|
|
|
2020-04-06 23:49:30 +00:00
|
|
|
while (line == NULL)
|
|
|
|
{
|
|
|
|
g_print ("%s: ", _("Description"));
|
|
|
|
|
|
|
|
line = g_data_input_stream_read_line_utf8 (istream, NULL, NULL, NULL);
|
|
|
|
}
|
2018-08-27 01:24:10 +00:00
|
|
|
*description = g_strdelimit (line, "\n", '\0');
|
2019-11-10 02:00:37 +00:00
|
|
|
|
|
|
|
g_print ("\n");
|
2018-08-27 01:24:10 +00:00
|
|
|
}
|
|
|
|
|
2018-08-27 02:30:53 +00:00
|
|
|
if (uuid != NULL && *uuid == NULL)
|
2018-08-27 01:24:10 +00:00
|
|
|
{
|
2020-04-06 23:49:30 +00:00
|
|
|
char *line = NULL;
|
2018-08-27 01:24:10 +00:00
|
|
|
|
|
|
|
g_print (
|
|
|
|
_("UUID is a globally-unique identifier for your extension.\n"
|
|
|
|
"This should be in the format of an email address (clicktofocus@janedoe.example.com)\n"));
|
|
|
|
|
2020-04-06 23:49:30 +00:00
|
|
|
while (line == NULL)
|
|
|
|
{
|
|
|
|
g_print ("UUID: ");
|
|
|
|
|
|
|
|
line = g_data_input_stream_read_line_utf8 (istream, NULL, NULL, NULL);
|
|
|
|
}
|
2018-08-27 01:24:10 +00:00
|
|
|
*uuid = g_strdelimit (line, "\n", '\0');
|
2019-11-10 02:00:37 +00:00
|
|
|
|
|
|
|
g_print ("\n");
|
2018-08-27 01:24:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
handle_create (int argc, char *argv[], gboolean do_help)
|
|
|
|
{
|
|
|
|
g_autoptr (GOptionContext) context = NULL;
|
|
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
g_autofree char *name = NULL;
|
|
|
|
g_autofree char *description = NULL;
|
|
|
|
g_autofree char *uuid = NULL;
|
2018-08-27 02:30:53 +00:00
|
|
|
gboolean interactive = FALSE;
|
|
|
|
GOptionEntry entries[] = {
|
|
|
|
{ .long_name = "uuid",
|
|
|
|
.arg = G_OPTION_ARG_STRING, .arg_data = &uuid,
|
|
|
|
.arg_description = "UUID",
|
|
|
|
.description = _("The unique identifier of the new extension") },
|
|
|
|
{ .long_name = "name",
|
|
|
|
.arg = G_OPTION_ARG_STRING, .arg_data = &name,
|
|
|
|
.arg_description = _("NAME"),
|
|
|
|
.description = _("The user-visible name of the new extension") },
|
|
|
|
{ .long_name = "description",
|
|
|
|
.arg_description = _("DESCRIPTION"),
|
|
|
|
.arg = G_OPTION_ARG_STRING, .arg_data = &description,
|
|
|
|
.description = _("A short description of what the extension does") },
|
|
|
|
{ .long_name = "interactive", .short_name = 'i',
|
|
|
|
.arg = G_OPTION_ARG_NONE, .arg_data = &interactive,
|
|
|
|
.description = _("Enter extension information interactively") },
|
|
|
|
{ NULL }
|
|
|
|
};
|
2018-08-27 01:24:10 +00:00
|
|
|
|
|
|
|
g_set_prgname ("gnome-extensions create");
|
|
|
|
|
|
|
|
context = g_option_context_new (NULL);
|
|
|
|
g_option_context_set_help_enabled (context, FALSE);
|
|
|
|
g_option_context_set_summary (context, _("Create a new extension"));
|
2018-08-27 02:30:53 +00:00
|
|
|
g_option_context_add_main_entries (context, entries, GETTEXT_PACKAGE);
|
2020-03-14 11:58:04 +00:00
|
|
|
g_option_context_add_group (context, get_option_group ());
|
2018-08-27 01:24:10 +00:00
|
|
|
|
|
|
|
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 (argc > 1)
|
|
|
|
{
|
|
|
|
show_help (context, _("Unknown arguments"));
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2018-08-27 02:30:53 +00:00
|
|
|
if (interactive)
|
|
|
|
prompt_metadata (&uuid, &name, &description);
|
|
|
|
|
|
|
|
if (uuid == NULL || name == NULL || description == NULL)
|
|
|
|
{
|
|
|
|
show_help (context, _("UUID, name and description are required"));
|
|
|
|
return 1;
|
|
|
|
}
|
2018-08-27 01:24:10 +00:00
|
|
|
|
|
|
|
return create_extension (uuid, name, description) ? 0 : 2;
|
|
|
|
}
|