Mercurial > pymonkey
view context.c @ 43:5727675b1bcb
JS-wrapped python functions now take a context object as their first parameter.
author | Atul Varma <varmaa@toolness.com> |
---|---|
date | Sun, 05 Jul 2009 23:52:14 -0700 |
parents | e62b1801f9af |
children | 0b9a316ce4ef |
line wrap: on
line source
#include "context.h" #include "object.h" #include "function.h" #include "utils.h" static void PYM_JSContextDealloc(PYM_JSContextObject *self) { if (self->cx) { JS_DestroyContext(self->cx); self->cx = NULL; } Py_DECREF(self->runtime); self->runtime = NULL; self->ob_type->tp_free((PyObject *) self); } static PyObject * PYM_getRuntime(PYM_JSContextObject *self, PyObject *args) { Py_INCREF(self->runtime); return (PyObject *) self->runtime; } static PyObject * PYM_newObject(PYM_JSContextObject *self, PyObject *args) { JSObject *obj = JS_NewObject(self->cx, &PYM_JS_ObjectClass, NULL, NULL); if (obj == NULL) { PyErr_SetString(PYM_error, "JS_NewObject() failed"); return NULL; } // If this fails, we don't need to worry about cleaning up // obj because it'll get cleaned up at the next GC. return (PyObject *) PYM_newJSObject(self, obj, NULL); } static PyObject * PYM_getProperty(PYM_JSContextObject *self, PyObject *args) { #ifndef Py_UNICODE_WIDE PYM_JSObject *object; Py_UNICODE *string; if (!PyArg_ParseTuple(args, "O!u", &PYM_JSObjectType, &object, &string)) return NULL; JSString *jsString = JS_NewUCStringCopyZ(self->cx, (const jschar *) string); if (jsString == NULL) { PyErr_SetString(PYM_error, "JS_NewStringCopyZ() failed"); return NULL; } jsval val; if (!JS_GetUCProperty(self->cx, object->obj, JS_GetStringChars(jsString), JS_GetStringLength(jsString), &val)) { // TODO: Get the actual JS exception. Any exception that exists // here will probably still be pending on the JS context. PyErr_SetString(PYM_error, "Getting property failed."); return NULL; } return PYM_jsvalToPyObject(self, val); #else PyErr_SetString(PyExc_NotImplementedError, "This function is not yet implemented for wide " "unicode builds of Python."); return NULL; #endif } static PyObject * PYM_initStandardClasses(PYM_JSContextObject *self, PyObject *args) { PYM_JSObject *object; if (!PyArg_ParseTuple(args, "O!", &PYM_JSObjectType, &object)) return NULL; if (!JS_InitStandardClasses(self->cx, object->obj)) { PyErr_SetString(PYM_error, "JS_InitStandardClasses() failed"); return NULL; } Py_RETURN_NONE; } static PyObject * PYM_evaluateScript(PYM_JSContextObject *self, PyObject *args) { PYM_JSObject *object; const char *source; int sourceLen; const char *filename; int lineNo; if (!PyArg_ParseTuple(args, "O!s#si", &PYM_JSObjectType, &object, &source, &sourceLen, &filename, &lineNo)) return NULL; JS_BeginRequest(self->cx); jsval rval; if (!JS_EvaluateScript(self->cx, object->obj, source, sourceLen, filename, lineNo, &rval)) { // TODO: Actually get the error that was raised. PyErr_SetString(PYM_error, "Script failed"); JS_EndRequest(self->cx); return NULL; } PyObject *pyRval = PYM_jsvalToPyObject(self, rval); JS_EndRequest(self->cx); return pyRval; } static PyObject * PYM_defineProperty(PYM_JSContextObject *self, PyObject *args) { PYM_JSObject *object; PyObject *value; const char *name; if (!PyArg_ParseTuple(args, "O!sO", &PYM_JSObjectType, &object, &name, &value)) return NULL; jsval jsValue; if (PYM_pyObjectToJsval(self->cx, value, &jsValue) == -1) return NULL; // TODO: Support unicode naming. if (!JS_DefineProperty(self->cx, object->obj, name, jsValue, NULL, NULL, JSPROP_ENUMERATE)) { // TODO: There's probably an exception pending on self->cx, // what should we do about it? PyErr_SetString(PYM_error, "JS_DefineProperty() failed"); return NULL; } Py_RETURN_NONE; } static PyObject * PYM_callFunction(PYM_JSContextObject *self, PyObject *args) { PYM_JSObject *obj; PYM_JSFunction *fun; PyObject *funcArgs; if (!PyArg_ParseTuple(args, "O!O!O!", &PYM_JSObjectType, &obj, &PYM_JSFunctionType, &fun, &PyTuple_Type, &funcArgs)) return NULL; uintN argc = PyTuple_Size(funcArgs); jsval argv[argc]; jsval *currArg = argv; for (unsigned int i = 0; i < argc; i++) { PyObject *item = PyTuple_GET_ITEM(funcArgs, i); if (item == NULL || PYM_pyObjectToJsval(self->cx, item, currArg) == -1) return NULL; currArg++; } jsval rval; // TODO: This assumes that a JSFunction * is actually a subclass of // a JSObject *, which may or may not be regarded as an implementation // detail. if (!JS_CallFunction(self->cx, obj->obj, (JSFunction *) fun->base.obj, argc, argv, &rval)) { // TODO: There's a pending exception on the JS context, should we // do something about it? // TODO: Convert the JS exception to a Python one. PyErr_SetString(PYM_error, "Function failed"); return NULL; } return PYM_jsvalToPyObject(self, rval); } static PyObject * PYM_newFunction(PYM_JSContextObject *self, PyObject *args) { PyObject *callable; const char *name; if (!PyArg_ParseTuple(args, "Os", &callable, &name)) return NULL; return (PyObject *) PYM_newJSFunctionFromCallable(self, callable, name); } static PyMethodDef PYM_JSContextMethods[] = { {"get_runtime", (PyCFunction) PYM_getRuntime, METH_VARARGS, "Get the JavaScript runtime associated with this context."}, {"new_object", (PyCFunction) PYM_newObject, METH_VARARGS, "Create a new JavaScript object."}, {"init_standard_classes", (PyCFunction) PYM_initStandardClasses, METH_VARARGS, "Add standard classes and functions to the given object."}, {"evaluate_script", (PyCFunction) PYM_evaluateScript, METH_VARARGS, "Evaluate the given JavaScript code in the context of the given " "global object, using the given filename" "and line number information."}, {"call_function", (PyCFunction) PYM_callFunction, METH_VARARGS, "Calls a JS function."}, {"new_function", (PyCFunction) PYM_newFunction, METH_VARARGS, "Creates a new function callable from JS."}, {"define_property", (PyCFunction) PYM_defineProperty, METH_VARARGS, "Defines a property on an object."}, {"get_property", (PyCFunction) PYM_getProperty, METH_VARARGS, "Gets the given property for the given JavaScript object."}, {NULL, NULL, 0, NULL} }; PyTypeObject PYM_JSContextType = { PyObject_HEAD_INIT(NULL) 0, /*ob_size*/ "pymonkey.Context", /*tp_name*/ sizeof(PYM_JSContextObject), /*tp_basicsize*/ 0, /*tp_itemsize*/ /*tp_dealloc*/ (destructor) PYM_JSContextDealloc, 0, /*tp_print*/ 0, /*tp_getattr*/ 0, /*tp_setattr*/ 0, /*tp_compare*/ 0, /*tp_repr*/ 0, /*tp_as_number*/ 0, /*tp_as_sequence*/ 0, /*tp_as_mapping*/ 0, /*tp_hash */ 0, /*tp_call*/ 0, /*tp_str*/ 0, /*tp_getattro*/ 0, /*tp_setattro*/ 0, /*tp_as_buffer*/ Py_TPFLAGS_DEFAULT, /*tp_flags*/ /* tp_doc */ "JavaScript Context.", 0, /* tp_traverse */ 0, /* tp_clear */ 0, /* tp_richcompare */ 0, /* tp_weaklistoffset */ 0, /* tp_iter */ 0, /* tp_iternext */ PYM_JSContextMethods, /* tp_methods */ 0, /* tp_members */ 0, /* tp_getset */ 0, /* tp_base */ 0, /* tp_dict */ 0, /* tp_descr_get */ 0, /* tp_descr_set */ 0, /* tp_dictoffset */ 0, /* tp_init */ 0, /* tp_alloc */ 0, /* tp_new */ }; extern PYM_JSContextObject * PYM_newJSContextObject(PYM_JSRuntimeObject *runtime, JSContext *cx) { PYM_JSContextObject *context = PyObject_New(PYM_JSContextObject, &PYM_JSContextType); if (context == NULL) return NULL; context->runtime = runtime; Py_INCREF(runtime); context->cx = cx; JS_SetContextPrivate(cx, context); #ifdef JS_GC_ZEAL // TODO: Consider exposing JS_SetGCZeal() to Python instead of // hard-coding it here. JS_SetGCZeal(cx, 2); #endif return context; }