[gjs/esm/static-imports] console: Add --module evaluation option.
- From: Evan Welsh <ewlsh src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gjs/esm/static-imports] console: Add --module evaluation option.
- Date: Sun, 28 Jun 2020 17:10:16 +0000 (UTC)
commit 4a16ac18822b40c5e1cf2d1e60516902e31613af
Author: Evan Welsh <noreply evanwelsh com>
Date: Sat Apr 11 11:04:57 2020 -0500
console: Add --module evaluation option.
gjs/console.cpp | 42 ++++++++++++++++++++-----
gjs/context-private.h | 9 ++++++
gjs/context.cpp | 85 +++++++++++++++++++++++++++++++++++++++++++++++++--
gjs/context.h | 12 ++++++++
4 files changed, 139 insertions(+), 9 deletions(-)
---
diff --git a/gjs/console.cpp b/gjs/console.cpp
index 9e335594..3b375775 100644
--- a/gjs/console.cpp
+++ b/gjs/console.cpp
@@ -24,6 +24,7 @@
#include <config.h> // for PACKAGE_STRING
#include <locale.h> // for setlocale, LC_ALL
+#include <stdint.h> // for uint8_t
#include <stdlib.h> // for exit
#include <string.h> // for strcmp, strlen
@@ -38,6 +39,7 @@
#include <glib.h>
#include <gjs/gjs.h>
+#include <gjs/jsapi-util.h>
static char **include_path = NULL;
static char **coverage_prefixes = NULL;
@@ -47,6 +49,7 @@ static char *command = NULL;
static gboolean print_version = false;
static gboolean print_js_version = false;
static gboolean debugging = false;
+static gboolean exec_as_module = false;
static bool enable_profiler = false;
static gboolean parse_profile_arg(const char *, const char *, void *, GError **);
@@ -60,6 +63,8 @@ static GOptionEntry entries[] = {
{ "coverage-prefix", 'C', 0, G_OPTION_ARG_STRING_ARRAY, &coverage_prefixes, "Add the prefix PREFIX to
the list of files to generate coverage info for", "PREFIX" },
{ "coverage-output", 0, 0, G_OPTION_ARG_STRING, &coverage_output_path, "Write coverage output to a
directory DIR. This option is mandatory when using --coverage-path", "DIR", },
{ "include-path", 'I', 0, G_OPTION_ARG_STRING_ARRAY, &include_path, "Add the directory DIR to the list
of directories to search for js files.", "DIR" },
+ { "module", 'm', 0, G_OPTION_ARG_NONE, &exec_as_module,
+ "Execute the input as a js module (implies strict mode)" },
{ "profile", 0, G_OPTION_FLAG_OPTIONAL_ARG | G_OPTION_FLAG_FILENAME,
G_OPTION_ARG_CALLBACK, reinterpret_cast<void *>(&parse_profile_arg),
"Enable the profiler and write output to FILE (default: gjs-$PID.syscap)",
@@ -201,7 +206,23 @@ int define_argv_and_eval_script(GjsContext* js_context, int argc,
/* evaluate the script */
int code;
- if (!gjs_context_eval(js_context, script, len, filename, &code, &error)) {
+ if (exec_as_module) {
+ GjsAutoUnref<GFile> output = g_file_new_for_commandline_arg(filename);
+ char* full_path = g_file_get_path(output);
+ if (!gjs_context_register_module(js_context, full_path, full_path,
+ script, len, &error)) {
+ g_printerr("%s\n", error->message);
+ code = 1;
+ }
+
+ uint8_t code_8 = 0;
+ if (!gjs_context_eval_module(js_context, full_path, &code_8, &error)) {
+ code = code_8;
+ if (!g_error_matches(error, GJS_ERROR, GJS_ERROR_SYSTEM_EXIT))
+ g_critical("%s", error->message);
+ }
+ } else if (!gjs_context_eval(js_context, script, len, filename, &code,
+ &error)) {
if (!g_error_matches(error, GJS_ERROR, GJS_ERROR_SYSTEM_EXIT))
g_critical("%s", error->message);
g_clear_error(&error);
@@ -269,6 +290,7 @@ main(int argc, char **argv)
print_version = false;
print_js_version = false;
debugging = false;
+ exec_as_module = false;
g_option_context_set_ignore_unknown_options(context, false);
g_option_context_set_help_enabled(context, true);
if (!g_option_context_parse_strv(context, &gjs_argv, &error)) {
@@ -298,7 +320,14 @@ main(int argc, char **argv)
filename = "<command line>";
program_name = gjs_argv[0];
} else if (gjs_argc == 1) {
- script = g_strdup("const Console = imports.console; Console.interact();");
+ if (exec_as_module) {
+ script = g_strdup(
+ "throw new Error('Console interaction is not implemented with "
+ "the --module option. Exiting with error.')");
+ } else {
+ script = g_strdup(
+ "const Console = imports.console; Console.interact();");
+ }
len = strlen(script);
filename = "<stdin>";
program_name = gjs_argv[0];
@@ -335,11 +364,10 @@ main(int argc, char **argv)
g_unsetenv("GJS_TRACE_FD"); /* ignore env var in eval() */
}
- js_context = (GjsContext*) g_object_new(GJS_TYPE_CONTEXT,
- "search-path", include_path,
- "program-name", program_name,
- "profiler-enabled", enable_profiler,
- NULL);
+ js_context = (GjsContext*)g_object_new(
+ GJS_TYPE_CONTEXT, "search-path", include_path, "program-name",
+ program_name, "profiler-enabled", enable_profiler, "modules",
+ exec_as_module, NULL);
env_coverage_output_path = g_getenv("GJS_COVERAGE_OUTPUT");
if (env_coverage_output_path != NULL) {
diff --git a/gjs/context-private.h b/gjs/context-private.h
index 1b1e5562..7721a915 100644
--- a/gjs/context-private.h
+++ b/gjs/context-private.h
@@ -26,6 +26,7 @@
#include <config.h>
+#include <stddef.h> // for size_t
#include <stdint.h>
#include <sys/types.h> // for ssize_t
@@ -141,6 +142,7 @@ class GjsContextPrivate : public JS::JobQueue {
bool m_force_gc : 1;
bool m_draining_job_queue : 1;
bool m_should_profile : 1;
+ bool m_exec_as_module : 1;
bool m_should_listen_sigusr2 : 1;
int64_t m_sweep_begin_time;
@@ -190,6 +192,7 @@ class GjsContextPrivate : public JS::JobQueue {
void set_program_name(char* value) { m_program_name = value; }
void set_search_path(char** value) { m_search_path = value; }
void set_should_profile(bool value) { m_should_profile = value; }
+ void set_execute_as_module(bool value) { m_exec_as_module = value; }
void set_should_listen_sigusr2(bool value) {
m_should_listen_sigusr2 = value;
}
@@ -218,6 +221,9 @@ class GjsContextPrivate : public JS::JobQueue {
ssize_t script_len, const char* filename,
JS::MutableHandleValue retval);
GJS_JSAPI_RETURN_CONVENTION
+ bool eval_module(const char* identifier, uint8_t* exit_code_p,
+ GError** error);
+ GJS_JSAPI_RETURN_CONVENTION
bool call_function(JS::HandleObject this_obj, JS::HandleValue func_val,
const JS::HandleValueArray& args,
JS::MutableHandleValue rval);
@@ -245,6 +251,9 @@ class GjsContextPrivate : public JS::JobQueue {
void register_unhandled_promise_rejection(uint64_t id, GjsAutoChar&& stack);
void unregister_unhandled_promise_rejection(uint64_t id);
+ bool register_module(const char* identifier, const char* filename,
+ const char* mod_text, size_t mod_len, GError** error);
+
void set_sweeping(bool value);
static void trace(JSTracer* trc, void* data);
diff --git a/gjs/context.cpp b/gjs/context.cpp
index 97f40cc2..3fcf13a1 100644
--- a/gjs/context.cpp
+++ b/gjs/context.cpp
@@ -24,6 +24,7 @@
#include <config.h>
#include <signal.h> // for sigaction, SIGUSR1, sa_handler
+#include <stdint.h> // for uint64_t
#include <stdio.h> // for FILE, fclose, size_t
#include <string.h> // for memset
@@ -34,7 +35,8 @@
#endif
#include <new>
-#include <string> // for u16string
+#include <string> // for u16string
+#include <type_traits> // for remove_reference<>::type
#include <unordered_map>
#include <utility> // for move
#include <vector>
@@ -150,6 +152,7 @@ enum {
PROP_PROGRAM_NAME,
PROP_PROFILER_ENABLED,
PROP_PROFILER_SIGUSR2,
+ PROP_MODULES_ENABLED,
};
static GMutex contexts_lock;
@@ -299,6 +302,13 @@ gjs_context_class_init(GjsContextClass *klass)
g_object_class_install_property(object_class, PROP_PROFILER_SIGUSR2, pspec);
g_param_spec_unref(pspec);
+ pspec = g_param_spec_boolean(
+ "modules", "Modules enabled",
+ "Whether to profile JS code run by this context", FALSE,
+ GParamFlags(G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY));
+ g_object_class_install_property(object_class, PROP_MODULES_ENABLED, pspec);
+ g_param_spec_unref(pspec);
+
/* For GjsPrivate */
{
#ifdef G_OS_WIN32
@@ -573,8 +583,11 @@ gjs_context_set_property (GObject *object,
case PROP_PROFILER_SIGUSR2:
gjs->set_should_listen_sigusr2(g_value_get_boolean(value));
break;
+ case PROP_MODULES_ENABLED:
+ gjs->set_execute_as_module(g_value_get_boolean(value));
+ break;
default:
- G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
break;
}
}
@@ -938,6 +951,28 @@ gjs_context_eval(GjsContext *js_context,
return gjs->eval(script, script_len, filename, exit_status_p, error);
}
+bool gjs_context_eval_module(GjsContext* js_context, const char* name,
+ uint8_t* exit_code, GError** error) {
+ g_return_val_if_fail(GJS_IS_CONTEXT(js_context), false);
+
+ GjsAutoUnref<GjsContext> js_context_ref(js_context, GjsAutoTakeOwnership());
+
+ GjsContextPrivate* gjs = GjsContextPrivate::from_object(js_context);
+ return gjs->eval_module(name, exit_code, error);
+}
+
+bool gjs_context_register_module(GjsContext* js_context, const char* identifier,
+ const char* filename, const char* module,
+ size_t module_len, GError** error) {
+ g_return_val_if_fail(GJS_IS_CONTEXT(js_context), false);
+
+ GjsAutoUnref<GjsContext> js_context_ref(js_context, GjsAutoTakeOwnership());
+
+ GjsContextPrivate* gjs = GjsContextPrivate::from_object(js_context);
+ return gjs->register_module(identifier, filename, module, module_len,
+ error);
+}
+
bool GjsContextPrivate::eval(const char* script, ssize_t script_len,
const char* filename, int* exit_status_p,
GError** error) {
@@ -1010,6 +1045,36 @@ bool GjsContextPrivate::eval(const char* script, ssize_t script_len,
return true;
}
+bool GjsContextPrivate::eval_module(const char* identifier,
+ uint8_t* exit_status_p,
+ GError** error G_GNUC_UNUSED) {
+ // TODO Implement eval_module
+ g_print("GjsContextPrivate::eval_module called on identifier %s.",
+ identifier);
+
+ *exit_status_p = 1;
+
+ g_error(
+ "GjsContextPrivate::eval_module is not implemented. Exiting with "
+ "error.");
+
+ return false;
+}
+
+bool GjsContextPrivate::register_module(const char* identifier,
+ const char* filename,
+ const char* module, size_t module_len,
+ GError** error G_GNUC_UNUSED) {
+ // TODO Implement register_module
+ g_warning(
+ "GjsContextPrivate::register_module is not yet implemented. Printing "
+ "module...");
+ g_warning("Identifier: %s\nFilename: %s\nModule Length: %lu\n", identifier,
+ filename, module_len);
+ g_warning("%s", module);
+ return true;
+}
+
bool
gjs_context_eval_file(GjsContext *js_context,
const char *filename,
@@ -1029,6 +1094,22 @@ gjs_context_eval_file(GjsContext *js_context,
exit_status_p, error);
}
+bool gjs_context_eval_module_file(GjsContext* js_context, const char* filename,
+ uint8_t* exit_status_p, GError** error) {
+ char* module;
+ size_t module_len;
+ GjsAutoUnref<GFile> file = g_file_new_for_commandline_arg(filename);
+
+ if (!g_file_load_contents(file, nullptr, &module, &module_len, nullptr,
+ error))
+ return false;
+ GjsAutoChar module_ref = module;
+
+ return gjs_context_register_module(js_context, filename, filename, module,
+ module_len, error) &&
+ gjs_context_eval_module(js_context, filename, exit_status_p, error);
+}
+
/*
* GjsContextPrivate::eval_with_scope:
* @scope_object: an object to use as the global scope, or nullptr
diff --git a/gjs/context.h b/gjs/context.h
index 13be25b9..b015ca92 100644
--- a/gjs/context.h
+++ b/gjs/context.h
@@ -29,6 +29,7 @@
#endif
#include <stdbool.h> /* IWYU pragma: keep */
+#include <stdint.h>
#ifndef _WIN32
# include <signal.h> /* for siginfo_t */
@@ -61,10 +62,21 @@ GJS_EXPORT GJS_USE bool gjs_context_eval_file(GjsContext* js_context,
const char* filename,
int* exit_status_p,
GError** error);
+GJS_EXPORT GJS_USE bool gjs_context_eval_module_file(GjsContext* js_context,
+ const char* filename,
+ uint8_t* exit_status_p,
+ GError** error);
GJS_EXPORT GJS_USE bool gjs_context_eval(GjsContext* js_context,
const char* script, gssize script_len,
const char* filename,
int* exit_status_p, GError** error);
+GJS_EXPORT GJS_USE bool gjs_context_register_module(
+ GjsContext* context, const char* name, const char* filename,
+ const char* module, gsize module_len, GError** error);
+GJS_EXPORT GJS_USE bool gjs_context_eval_module(GjsContext* context,
+ const char* name,
+ uint8_t* exit_code,
+ GError** error);
GJS_EXPORT GJS_USE bool gjs_context_define_string_array(
GjsContext* js_context, const char* array_name, gssize array_length,
const char** array_values, GError** error);
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]