2023-07-12 21:50:31 +08:00

830 lines
28 KiB
C

#include <Python.h>
#include <time.h>
#include "third-party/quickjs.h"
// Node of Python callable that the context needs to keep available.
typedef struct PythonCallableNode PythonCallableNode;
struct PythonCallableNode {
PyObject *obj;
// Internal ID of the callable function. "magic" is QuickJS terminology.
int magic;
PythonCallableNode *next;
};
// Keeps track of the time if we are using a time limit.
typedef struct {
clock_t start;
clock_t limit;
} InterruptData;
// The data of the type _quickjs.Context.
typedef struct {
PyObject_HEAD JSRuntime *runtime;
JSContext *context;
int has_time_limit;
clock_t time_limit;
// Used when releasing the GIL.
PyThreadState *thread_state;
InterruptData interrupt_data;
// NULL-terminated singly linked list of callable Python objects that we need to keep alive.
PythonCallableNode *python_callables;
} ContextData;
// The data of the type _quickjs.Object.
typedef struct {
PyObject_HEAD;
ContextData *context;
JSValue object;
} ObjectData;
// The exception raised by this module.
static PyObject *JSException = NULL;
static PyObject *StackOverflow = NULL;
// Converts a JSValue to a Python object.
//
// Takes ownership of the JSValue and will deallocate it (refcount reduced by 1).
static PyObject *quickjs_to_python(ContextData *context_obj, JSValue value);
// Whether converting item to QuickJS would be possible.
static int python_to_quickjs_possible(ContextData *context, PyObject *item);
// Converts item to QuickJS.
//
// If the Python object is not possible to convert to JS, undefined will be returned. This fallback
// will not be used if python_to_quickjs_possible returns 1.
static JSValueConst python_to_quickjs(ContextData *context, PyObject *item);
static PyTypeObject Object;
// Returns nonzero if we should stop due to a time limit.
static int js_interrupt_handler(JSRuntime *rt, void *opaque) {
InterruptData *data = opaque;
if (clock() - data->start >= data->limit) {
return 1;
} else {
return 0;
}
}
// Sets up a context and an InterruptData struct if the context has a time limit.
static void setup_time_limit(ContextData *context, InterruptData *interrupt_data) {
if (context->has_time_limit) {
JS_SetInterruptHandler(context->runtime, js_interrupt_handler, interrupt_data);
interrupt_data->limit = context->time_limit;
interrupt_data->start = clock();
}
}
// Restores the context if the context has a time limit.
static void teardown_time_limit(ContextData *context) {
if (context->has_time_limit) {
JS_SetInterruptHandler(context->runtime, NULL, NULL);
}
}
// This method is always called in a context before running JS code in QuickJS. It sets up time
// limites, releases the GIL etc.
static void prepare_call_js(ContextData *context) {
// We release the GIL in order to speed things up for certain use cases.
assert(!context->thread_state);
context->thread_state = PyEval_SaveThread();
setup_time_limit(context, &context->interrupt_data);
}
// This method is called right after returning from running JS code. Aquires the GIL etc.
static void end_call_js(ContextData *context) {
teardown_time_limit(context);
assert(context->thread_state);
PyEval_RestoreThread(context->thread_state);
context->thread_state = NULL;
}
// Called when Python is called again from inside QuickJS.
static void prepare_call_python(ContextData *context) {
assert(context->thread_state);
PyEval_RestoreThread(context->thread_state);
context->thread_state = NULL;
}
// Called when the operation started by prepare_call_python is done.
static void end_call_python(ContextData *context) {
assert(!context->thread_state);
context->thread_state = PyEval_SaveThread();
}
// GC traversal.
static int object_traverse(ObjectData *self, visitproc visit, void *arg) {
Py_VISIT(self->context);
return 0;
}
// Creates an instance of the Object class.
static PyObject *object_new(PyTypeObject *type, PyObject *args, PyObject *kwds) {
ObjectData *self = PyObject_GC_New(ObjectData, type);
if (self != NULL) {
self->context = NULL;
}
return (PyObject *)self;
}
// Deallocates an instance of the Object class.
static void object_dealloc(ObjectData *self) {
if (self->context) {
PyObject_GC_UnTrack(self);
JS_FreeValue(self->context->context, self->object);
// We incremented the refcount of the context when we created this object, so we should
// decrease it now so we don't leak memory.
Py_DECREF(self->context);
}
PyObject_GC_Del(self);
}
// _quickjs.Object.get
//
// Gets a Javascript property of the object.
static PyObject *object_get(ObjectData *self, PyObject *args) {
const char *name;
if (!PyArg_ParseTuple(args, "s", &name)) {
return NULL;
}
JSValue value = JS_GetPropertyStr(self->context->context, self->object, name);
return quickjs_to_python(self->context, value);
}
static JSValue js_c_function(
JSContext *ctx, JSValueConst this_val, int argc, JSValueConst *argv, int magic) {
ContextData *context = (ContextData *)JS_GetContextOpaque(ctx);
if (context->has_time_limit) {
return JS_ThrowInternalError(ctx, "Can not call into Python with a time limit set.");
}
PythonCallableNode *node = context->python_callables;
while (node && node->magic != magic) {
node = node->next;
}
if (!node) {
return JS_ThrowInternalError(ctx, "Internal error.");
}
prepare_call_python(context);
PyObject *args = PyTuple_New(argc);
if (!args) {
end_call_python(context);
return JS_ThrowOutOfMemory(ctx);
}
int tuple_success = 1;
for (int i = 0; i < argc; ++i) {
PyObject *arg = quickjs_to_python(context, JS_DupValue(ctx, argv[i]));
if (!arg) {
tuple_success = 0;
break;
}
PyTuple_SET_ITEM(args, i, arg);
}
if (!tuple_success) {
Py_DECREF(args);
end_call_python(context);
return JS_ThrowInternalError(ctx, "Internal error: could not convert args.");
}
PyObject *result = PyObject_CallObject(node->obj, args);
Py_DECREF(args);
if (!result) {
end_call_python(context);
return JS_ThrowInternalError(ctx, "Python call failed.");
}
JSValue js_result = JS_NULL;
if (python_to_quickjs_possible(context, result)) {
js_result = python_to_quickjs(context, result);
} else {
PyErr_Clear();
js_result = JS_ThrowInternalError(ctx, "Can not convert Python result to JS.");
}
Py_DECREF(result);
end_call_python(context);
return js_result;
}
// _quickjs.Object.set
//
// Sets a Javascript property to the object. Callables are supported.
static PyObject *object_set(ObjectData *self, PyObject *args) {
const char *name;
PyObject *item;
if (!PyArg_ParseTuple(args, "sO", &name, &item)) {
return NULL;
}
int ret = 0;
if (PyCallable_Check(item) && (!PyObject_IsInstance(item, (PyObject *)&Object) || JS_IsFunction(
self->context->context, ((ObjectData *)item)->object))) {
PythonCallableNode *node = PyMem_Malloc(sizeof(PythonCallableNode));
if (!node) {
return NULL;
}
Py_INCREF(item);
node->magic = 0;
if (self->context->python_callables) {
node->magic = self->context->python_callables->magic + 1;
}
node->obj = item;
node->next = self->context->python_callables;
self->context->python_callables = node;
JSValue function = JS_NewCFunctionMagic(
self->context->context,
js_c_function,
name,
0, // TODO: Should we allow setting the .length of the function to something other than 0?
JS_CFUNC_generic_magic,
node->magic);
// If this fails we don't notify the caller of this function.
ret = JS_SetPropertyStr(self->context->context, self->object, name, function);
if (ret != 1) {
PyErr_SetString(PyExc_TypeError, "Failed setting the variable as a callable.");
return NULL;
} else {
Py_RETURN_NONE;
}
} else {
if (python_to_quickjs_possible(self->context, item)) {
ret = JS_SetPropertyStr(self->context->context, self->object, name,
python_to_quickjs(self->context, item));
if (ret != 1) {
PyErr_SetString(PyExc_TypeError, "Failed setting the variable.");
}
}
if (ret == 1) {
Py_RETURN_NONE;
} else {
return NULL;
}
}
}
// _quickjs.Object.__call__
static PyObject *object_call(ObjectData *self, PyObject *args, PyObject *kwds);
// _quickjs.Object.json
//
// Returns the JSON representation of the object as a Python string.
static PyObject *object_json(ObjectData *self) {
JSContext *context = self->context->context;
JSValue json_string = JS_JSONStringify(context, self->object, JS_UNDEFINED, JS_UNDEFINED);
return quickjs_to_python(self->context, json_string);
}
// All methods of the _quickjs.Object class.
static PyMethodDef object_methods[] = {
{"get", (PyCFunction)object_get, METH_VARARGS, "Gets a Javascript property of the object."},
{"set", (PyCFunction)object_set, METH_VARARGS, "Sets a Javascript property to the object."},
{"json", (PyCFunction)object_json, METH_NOARGS, "Converts to a JSON string."},
{NULL} /* Sentinel */
};
// Define the quickjs.Object type.
static PyTypeObject Object = {PyVarObject_HEAD_INIT(NULL, 0).tp_name = "_quickjs.Object",
.tp_doc = "Quickjs object",
.tp_basicsize = sizeof(ObjectData),
.tp_itemsize = 0,
.tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC,
.tp_traverse = (traverseproc)object_traverse,
.tp_new = object_new,
.tp_dealloc = (destructor)object_dealloc,
.tp_call = (ternaryfunc)object_call,
.tp_methods = object_methods};
// Whether converting item to QuickJS would be possible.
static int python_to_quickjs_possible(ContextData *context, PyObject *item) {
if (PyBool_Check(item)) {
return 1;
} else if (PyLong_Check(item)) {
return 1;
} else if (PyFloat_Check(item)) {
return 1;
} else if (item == Py_None) {
return 1;
} else if (PyUnicode_Check(item)) {
return 1;
} else if (PyObject_IsInstance(item, (PyObject *)&Object)) {
ObjectData *object = (ObjectData *)item;
if (object->context != context) {
PyErr_Format(PyExc_ValueError, "Can not mix JS objects from different contexts.");
return 0;
}
return 1;
} else {
PyErr_Format(PyExc_TypeError,
"Unsupported type when converting a Python object to quickjs: %s.",
Py_TYPE(item)->tp_name);
return 0;
}
}
// Converts item to QuickJS.
//
// If the Python object is not possible to convert to JS, undefined will be returned. This fallback
// will not be used if python_to_quickjs_possible returns 1.
static JSValueConst python_to_quickjs(ContextData *context, PyObject *item) {
if (PyBool_Check(item)) {
return JS_MKVAL(JS_TAG_BOOL, item == Py_True ? 1 : 0);
} else if (PyLong_Check(item)) {
int overflow;
long value = PyLong_AsLongAndOverflow(item, &overflow);
if (overflow) {
PyObject *float_value = PyNumber_Float(item);
double double_value = PyFloat_AsDouble(float_value);
Py_DECREF(float_value);
return JS_NewFloat64(context->context, double_value);
} else {
return JS_MKVAL(JS_TAG_INT, value);
}
} else if (PyFloat_Check(item)) {
return JS_NewFloat64(context->context, PyFloat_AsDouble(item));
} else if (item == Py_None) {
return JS_NULL;
} else if (PyUnicode_Check(item)) {
return JS_NewString(context->context, PyUnicode_AsUTF8(item));
} else if (PyObject_IsInstance(item, (PyObject *)&Object)) {
return JS_DupValue(context->context, ((ObjectData *)item)->object);
} else {
// Can not happen if python_to_quickjs_possible passes.
return JS_UNDEFINED;
}
}
// _quickjs.Object.__call__
static PyObject *object_call(ObjectData *self, PyObject *args, PyObject *kwds) {
if (self->context == NULL) {
// This object does not have a context and has not been created by this module.
Py_RETURN_NONE;
}
// We first loop through all arguments and check that they are supported without doing anything.
// This makes the cleanup code simpler for the case where we have to raise an error.
const int nargs = PyTuple_Size(args);
for (int i = 0; i < nargs; ++i) {
PyObject *item = PyTuple_GetItem(args, i);
if (!python_to_quickjs_possible(self->context, item)) {
return NULL;
}
}
// Now we know that all arguments are supported and we can convert them.
JSValueConst *jsargs = malloc(nargs * sizeof(JSValueConst));
for (int i = 0; i < nargs; ++i) {
PyObject *item = PyTuple_GetItem(args, i);
jsargs[i] = python_to_quickjs(self->context, item);
}
prepare_call_js(self->context);
JSValue value;
value = JS_Call(self->context->context, self->object, JS_NULL, nargs, jsargs);
for (int i = 0; i < nargs; ++i) {
JS_FreeValue(self->context->context, jsargs[i]);
}
free(jsargs);
end_call_js(self->context);
return quickjs_to_python(self->context, value);
}
// Converts the current Javascript exception to a Python exception via a C string.
static void quickjs_exception_to_python(JSContext *context) {
JSValue exception = JS_GetException(context);
const char *cstring = JS_ToCString(context, exception);
const char *stack_cstring = NULL;
if (!JS_IsNull(exception) && !JS_IsUndefined(exception)) {
JSValue stack = JS_GetPropertyStr(context, exception, "stack");
if (!JS_IsException(stack)) {
stack_cstring = JS_ToCString(context, stack);
JS_FreeValue(context, stack);
}
}
if (cstring != NULL) {
const char *safe_stack_cstring = stack_cstring ? stack_cstring : "";
if (strstr(cstring, "stack overflow") != NULL) {
PyErr_Format(StackOverflow, "%s\n%s", cstring, safe_stack_cstring);
} else {
PyErr_Format(JSException, "%s\n%s", cstring, safe_stack_cstring);
}
} else {
// This has been observed to happen when different threads have used the same QuickJS
// runtime, but not at the same time.
// Could potentially be another problem though, since JS_ToCString may return NULL.
PyErr_Format(JSException,
"(Failed obtaining QuickJS error string. Concurrency issue?)");
}
JS_FreeCString(context, cstring);
JS_FreeCString(context, stack_cstring);
JS_FreeValue(context, exception);
}
// Converts a JSValue to a Python object.
//
// Takes ownership of the JSValue and will deallocate it (refcount reduced by 1).
static PyObject *quickjs_to_python(ContextData *context_obj, JSValue value) {
JSContext *context = context_obj->context;
int tag = JS_VALUE_GET_TAG(value);
// A return value of NULL means an exception.
PyObject *return_value = NULL;
if (tag == JS_TAG_INT) {
return_value = Py_BuildValue("i", JS_VALUE_GET_INT(value));
} else if (tag == JS_TAG_BIG_INT) {
const char *cstring = JS_ToCString(context, value);
return_value = PyLong_FromString(cstring, NULL, 10);
JS_FreeCString(context, cstring);
} else if (tag == JS_TAG_BOOL) {
return_value = Py_BuildValue("O", JS_VALUE_GET_BOOL(value) ? Py_True : Py_False);
} else if (tag == JS_TAG_NULL) {
return_value = Py_None;
} else if (tag == JS_TAG_UNDEFINED) {
return_value = Py_None;
} else if (tag == JS_TAG_EXCEPTION) {
quickjs_exception_to_python(context);
} else if (tag == JS_TAG_FLOAT64) {
return_value = Py_BuildValue("d", JS_VALUE_GET_FLOAT64(value));
} else if (tag == JS_TAG_STRING) {
const char *cstring = JS_ToCString(context, value);
return_value = Py_BuildValue("s", cstring);
JS_FreeCString(context, cstring);
} else if (tag == JS_TAG_OBJECT || tag == JS_TAG_MODULE || tag == JS_TAG_SYMBOL) {
// This is a Javascript object or function. We wrap it in a _quickjs.Object.
return_value = PyObject_CallObject((PyObject *)&Object, NULL);
ObjectData *object = (ObjectData *)return_value;
// This is important. Otherwise, the context may be deallocated before the object, which
// will result in a segfault with high probability.
Py_INCREF(context_obj);
object->context = context_obj;
PyObject_GC_Track(object);
object->object = JS_DupValue(context, value);
} else {
PyErr_Format(PyExc_TypeError, "Unknown quickjs tag: %d", tag);
}
JS_FreeValue(context, value);
if (return_value == Py_None) {
// Can not simply return PyNone for refcounting reasons.
Py_RETURN_NONE;
}
return return_value;
}
static PyObject *test(PyObject *self, PyObject *args) {
return Py_BuildValue("i", 42);
}
// Global state of the module. Currently none.
struct module_state {};
// GC traversal.
static int context_traverse(ContextData *self, visitproc visit, void *arg) {
PythonCallableNode *node = self->python_callables;
while (node) {
Py_VISIT(node->obj);
node = node->next;
}
return 0;
}
// GC clearing. Object does not have a clearing method, therefore dependency cycles
// between Context and Object will always be cleared starting here.
static int context_clear(ContextData *self) {
PythonCallableNode *node = self->python_callables;
while (node) {
Py_CLEAR(node->obj);
node = node->next;
}
return 0;
}
// Creates an instance of the _quickjs.Context class.
static PyObject *context_new(PyTypeObject *type, PyObject *args, PyObject *kwds) {
ContextData *self = PyObject_GC_New(ContextData, type);
if (self != NULL) {
// We never have different contexts for the same runtime. This way, different
// _quickjs.Context can be used concurrently.
self->runtime = JS_NewRuntime();
self->context = JS_NewContext(self->runtime);
self->has_time_limit = 0;
self->time_limit = 0;
self->thread_state = NULL;
self->python_callables = NULL;
JS_SetContextOpaque(self->context, self);
PyObject_GC_Track(self);
}
return (PyObject *)self;
}
// Deallocates an instance of the _quickjs.Context class.
static void context_dealloc(ContextData *self) {
JS_FreeContext(self->context);
JS_FreeRuntime(self->runtime);
PyObject_GC_UnTrack(self);
PythonCallableNode *node = self->python_callables;
self->python_callables = NULL;
while (node) {
PythonCallableNode *this = node;
node = node->next;
// this->obj may already be NULL if GC'ed right before through context_clear.
Py_XDECREF(this->obj);
PyMem_Free(this);
}
PyObject_GC_Del(self);
}
// Evaluates a Python string as JS and returns the result as a Python object. Will return
// _quickjs.Object for complex types (other than e.g. str, int).
static PyObject *context_eval_internal(ContextData *self, PyObject *args, int eval_type) {
const char *code;
if (!PyArg_ParseTuple(args, "s", &code)) {
return NULL;
}
prepare_call_js(self);
JSValue value;
value = JS_Eval(self->context, code, strlen(code), "<input>", eval_type);
end_call_js(self);
return quickjs_to_python(self, value);
}
// _quickjs.Context.eval
//
// Evaluates a Python string as JS and returns the result as a Python object. Will return
// _quickjs.Object for complex types (other than e.g. str, int).
static PyObject *context_eval(ContextData *self, PyObject *args) {
return context_eval_internal(self, args, JS_EVAL_TYPE_GLOBAL);
}
// _quickjs.Context.module
//
// Evaluates a Python string as JS module. Otherwise identical to eval.
static PyObject *context_module(ContextData *self, PyObject *args) {
return context_eval_internal(self, args, JS_EVAL_TYPE_MODULE);
}
// _quickjs.Context.execute_pending_job
//
// If there are pending jobs, executes one and returns True. Else returns False.
static PyObject *context_execute_pending_job(ContextData *self) {
prepare_call_js(self);
JSContext *ctx;
int ret = JS_ExecutePendingJob(self->runtime, &ctx);
end_call_js(self);
if (ret > 0) {
Py_RETURN_TRUE;
} else if (ret == 0) {
Py_RETURN_FALSE;
} else {
quickjs_exception_to_python(ctx);
return NULL;
}
}
// _quickjs.Context.parse_json
//
// Evaluates a Python string as JSON and returns the result as a Python object. Will
// return _quickjs.Object for complex types (other than e.g. str, int).
static PyObject *context_parse_json(ContextData *self, PyObject *args) {
const char *data;
if (!PyArg_ParseTuple(args, "s", &data)) {
return NULL;
}
JSValue value;
Py_BEGIN_ALLOW_THREADS;
value = JS_ParseJSON(self->context, data, strlen(data), "context_parse_json.json");
Py_END_ALLOW_THREADS;
return quickjs_to_python(self, value);
}
// _quickjs.Context.get_global
//
// Retrieves the global object of the JS context.
static PyObject *context_get_global(ContextData *self) {
return quickjs_to_python(self, JS_GetGlobalObject(self->context));
}
// _quickjs.Context.get
//
// Retrieves a global variable from the JS context.
static PyObject *context_get(ContextData *self, PyObject *args) {
PyErr_WarnEx(PyExc_DeprecationWarning,
"Context.get is deprecated, use Context.get_global().get instead.", 1);
PyObject *global = context_get_global(self);
if (global == NULL) {
return NULL;
}
PyObject *ret = object_get((ObjectData *)global, args);
Py_DECREF(global);
return ret;
}
// _quickjs.Context.set
//
// Sets a global variable to the JS context.
static PyObject *context_set(ContextData *self, PyObject *args) {
PyErr_WarnEx(PyExc_DeprecationWarning,
"Context.set is deprecated, use Context.get_global().set instead.", 1);
PyObject *global = context_get_global(self);
if (global == NULL) {
return NULL;
}
PyObject *ret = object_set((ObjectData *)global, args);
Py_DECREF(global);
return ret;
}
// _quickjs.Context.set_memory_limit
//
// Sets the memory limit of the context.
static PyObject *context_set_memory_limit(ContextData *self, PyObject *args) {
Py_ssize_t limit;
if (!PyArg_ParseTuple(args, "n", &limit)) {
return NULL;
}
JS_SetMemoryLimit(self->runtime, limit);
Py_RETURN_NONE;
}
// _quickjs.Context.set_time_limit
//
// Sets the CPU time limit of the context. This will be used in an interrupt handler.
static PyObject *context_set_time_limit(ContextData *self, PyObject *args) {
double limit;
if (!PyArg_ParseTuple(args, "d", &limit)) {
return NULL;
}
if (limit < 0) {
self->has_time_limit = 0;
} else {
self->has_time_limit = 1;
self->time_limit = (clock_t)(limit * CLOCKS_PER_SEC);
}
Py_RETURN_NONE;
}
// _quickjs.Context.set_max_stack_size
//
// Sets the max stack size in bytes.
static PyObject *context_set_max_stack_size(ContextData *self, PyObject *args) {
Py_ssize_t limit;
if (!PyArg_ParseTuple(args, "n", &limit)) {
return NULL;
}
JS_SetMaxStackSize(self->runtime, limit);
Py_RETURN_NONE;
}
// _quickjs.Context.memory
//
// Sets the CPU time limit of the context. This will be used in an interrupt handler.
static PyObject *context_memory(ContextData *self) {
PyObject *dict = PyDict_New();
if (dict == NULL) {
return NULL;
}
JSMemoryUsage usage;
JS_ComputeMemoryUsage(self->runtime, &usage);
#define MEM_USAGE_ADD_TO_DICT(key) \
{ \
PyObject *value = PyLong_FromLongLong(usage.key); \
if (PyDict_SetItemString(dict, #key, value) != 0) { \
return NULL; \
} \
Py_DECREF(value); \
}
MEM_USAGE_ADD_TO_DICT(malloc_size);
MEM_USAGE_ADD_TO_DICT(malloc_limit);
MEM_USAGE_ADD_TO_DICT(memory_used_size);
MEM_USAGE_ADD_TO_DICT(malloc_count);
MEM_USAGE_ADD_TO_DICT(memory_used_count);
MEM_USAGE_ADD_TO_DICT(atom_count);
MEM_USAGE_ADD_TO_DICT(atom_size);
MEM_USAGE_ADD_TO_DICT(str_count);
MEM_USAGE_ADD_TO_DICT(str_size);
MEM_USAGE_ADD_TO_DICT(obj_count);
MEM_USAGE_ADD_TO_DICT(obj_size);
MEM_USAGE_ADD_TO_DICT(prop_count);
MEM_USAGE_ADD_TO_DICT(prop_size);
MEM_USAGE_ADD_TO_DICT(shape_count);
MEM_USAGE_ADD_TO_DICT(shape_size);
MEM_USAGE_ADD_TO_DICT(js_func_count);
MEM_USAGE_ADD_TO_DICT(js_func_size);
MEM_USAGE_ADD_TO_DICT(js_func_code_size);
MEM_USAGE_ADD_TO_DICT(js_func_pc2line_count);
MEM_USAGE_ADD_TO_DICT(js_func_pc2line_size);
MEM_USAGE_ADD_TO_DICT(c_func_count);
MEM_USAGE_ADD_TO_DICT(array_count);
MEM_USAGE_ADD_TO_DICT(fast_array_count);
MEM_USAGE_ADD_TO_DICT(fast_array_elements);
MEM_USAGE_ADD_TO_DICT(binary_object_count);
MEM_USAGE_ADD_TO_DICT(binary_object_size);
return dict;
}
// _quickjs.Context.gc
//
// Runs garbage collection.
static PyObject *context_gc(ContextData *self) {
JS_RunGC(self->runtime);
Py_RETURN_NONE;
}
static PyObject *context_add_callable(ContextData *self, PyObject *args) {
PyErr_WarnEx(PyExc_DeprecationWarning,
"Context.add_callable is deprecated, use Context.get_global().set instead.", 1);
PyObject *global = context_get_global(self);
if (global == NULL) {
return NULL;
}
PyObject *ret = object_set((ObjectData *)global, args);
Py_DECREF(global);
return ret;
}
// All methods of the _quickjs.Context class.
static PyMethodDef context_methods[] = {
{"eval", (PyCFunction)context_eval, METH_VARARGS, "Evaluates a Javascript string."},
{"module",
(PyCFunction)context_module,
METH_VARARGS,
"Evaluates a Javascript string as a module."},
{"execute_pending_job", (PyCFunction)context_execute_pending_job, METH_NOARGS, "Executes a pending job."},
{"parse_json", (PyCFunction)context_parse_json, METH_VARARGS, "Parses a JSON string."},
{"get_global", (PyCFunction)context_get_global, METH_NOARGS, "Gets the Javascript global object."},
{"get", (PyCFunction)context_get, METH_VARARGS, "Gets a Javascript global variable."},
{"set", (PyCFunction)context_set, METH_VARARGS, "Sets a Javascript global variable."},
{"set_memory_limit",
(PyCFunction)context_set_memory_limit,
METH_VARARGS,
"Sets the memory limit in bytes."},
{"set_time_limit",
(PyCFunction)context_set_time_limit,
METH_VARARGS,
"Sets the CPU time limit in seconds (C function clock() is used)."},
{"set_max_stack_size",
(PyCFunction)context_set_max_stack_size,
METH_VARARGS,
"Sets the maximum stack size in bytes. Default is 256kB."},
{"memory", (PyCFunction)context_memory, METH_NOARGS, "Returns the memory usage as a dict."},
{"gc", (PyCFunction)context_gc, METH_NOARGS, "Runs garbage collection."},
{"add_callable", (PyCFunction)context_add_callable, METH_VARARGS, "Wraps a Python callable."},
{NULL} /* Sentinel */
};
// Define the _quickjs.Context type.
static PyTypeObject Context = {PyVarObject_HEAD_INIT(NULL, 0).tp_name = "_quickjs.Context",
.tp_doc = "Quickjs context",
.tp_basicsize = sizeof(ContextData),
.tp_itemsize = 0,
.tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC,
.tp_traverse = (traverseproc)context_traverse,
.tp_clear = (inquiry)context_clear,
.tp_new = context_new,
.tp_dealloc = (destructor)context_dealloc,
.tp_methods = context_methods};
// All global methods in _quickjs.
static PyMethodDef myextension_methods[] = {{"test", (PyCFunction)test, METH_NOARGS, NULL},
{NULL, NULL}};
// Define the _quickjs module.
static struct PyModuleDef moduledef = {PyModuleDef_HEAD_INIT,
"quickjs",
NULL,
sizeof(struct module_state),
myextension_methods,
NULL,
NULL,
NULL,
NULL};
// This function runs when the module is first imported.
PyMODINIT_FUNC PyInit__quickjs(void) {
if (PyType_Ready(&Context) < 0) {
return NULL;
}
if (PyType_Ready(&Object) < 0) {
return NULL;
}
PyObject *module = PyModule_Create(&moduledef);
if (module == NULL) {
return NULL;
}
JSException = PyErr_NewException("_quickjs.JSException", NULL, NULL);
if (JSException == NULL) {
return NULL;
}
StackOverflow = PyErr_NewException("_quickjs.StackOverflow", JSException, NULL);
if (StackOverflow == NULL) {
return NULL;
}
Py_INCREF(&Context);
PyModule_AddObject(module, "Context", (PyObject *)&Context);
Py_INCREF(&Object);
PyModule_AddObject(module, "Object", (PyObject *)&Object);
PyModule_AddObject(module, "JSException", JSException);
PyModule_AddObject(module, "StackOverflow", StackOverflow);
return module;
}