Files
mongo/scripting/engine_spidermonkey.cpp

282 lines
8.2 KiB
C++
Raw Normal View History

2009-04-23 17:40:43 -04:00
// engine_spidermonkey.cpp
#include "engine.h"
2009-04-23 22:28:11 -04:00
#ifdef _WIN32
#define XP_WIN
#else
#define XP_UNIX
#endif
2009-04-28 08:47:56 -04:00
#ifdef MOZJS
#include "mozjs/jsapi.h"
#else
2009-04-23 17:40:43 -04:00
#include "js/jsapi.h"
2009-04-28 08:47:56 -04:00
#endif
2009-04-23 17:40:43 -04:00
namespace mongo {
2009-04-29 10:16:39 -04:00
class Convertor {
public:
Convertor( JSContext * cx ){
_context = cx;
}
string toString( JSString * so ){
jschar * s = JS_GetStringChars( so );
size_t srclen = JS_GetStringLength( so );
size_t len = srclen * 2;
char * dst = (char*)malloc( len );
assert( JS_EncodeCharacters( _context , s , srclen , dst , &len) );
string ss( dst , len );
free( dst );
return ss;
}
string toString( jsval v ){
return toString( JS_ValueToString( _context , v ) );
}
2009-04-29 10:29:56 -04:00
// ---------- to spider monke ---------
2009-04-29 10:16:39 -04:00
jsval toval( double d ){
jsval val;
assert( JS_NewNumberValue( _context, d , &val ) );
return val;
}
2009-04-29 10:29:56 -04:00
jsval toval( const char * c ){
JSString * s = JS_NewStringCopyZ( _context , c );
return STRING_TO_JSVAL( s );
}
2009-04-29 10:16:39 -04:00
private:
JSContext * _context;
};
2009-04-23 17:40:43 -04:00
static JSClass global_class = {
"global", JSCLASS_GLOBAL_FLAGS,
JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub,
JS_EnumerateStub, JS_ResolveStub, JS_ConvertStub, JS_FinalizeStub,
JSCLASS_NO_OPTIONAL_MEMBERS
};
2009-04-29 10:16:39 -04:00
JSBool resolveBSONField( JSContext *cx, JSObject *obj, jsval id, uintN flags, JSObject **objp ){
Convertor c( cx );
BSONObj * o = (BSONObj*)(JS_GetPrivate( cx , obj ));
string s = c.toString( id );
jsval val;
BSONElement e = (*o)[ s.c_str() ];
switch( e.type() ){
case EOO:
*objp = 0;
return JS_TRUE;
case NumberDouble:
case NumberInt:
val = c.toval( e.number() );
2009-04-29 10:29:56 -04:00
break;
case String:
val = c.toval( e.valuestr() );
2009-04-29 10:16:39 -04:00
break;
default:
log() << "resolveBSONField can't handle type: " << (int)(e.type()) << endl;
2009-04-29 10:29:56 -04:00
uassert( "not done" , 0 );
2009-04-29 10:16:39 -04:00
return JS_FALSE;
}
2009-04-29 10:29:56 -04:00
assert( JS_SetProperty( cx , obj , s.c_str() , &val ) );
2009-04-29 10:16:39 -04:00
*objp = obj;
return JS_TRUE;
}
static JSClass bson_ro_class = {
"bson_object" , JSCLASS_HAS_PRIVATE | JSCLASS_NEW_RESOLVE ,
JS_PropertyStub, JS_PropertyStub, JS_PropertyStub, JS_PropertyStub,
JS_EnumerateStub, (JSResolveOp)(&resolveBSONField) , JS_ConvertStub, JS_FinalizeStub,
JSCLASS_NO_OPTIONAL_MEMBERS
};
2009-04-23 17:40:43 -04:00
class SMScope;
class SMEngine : public ScriptEngine {
public:
SMEngine(){
_runtime = JS_NewRuntime(8L * 1024L * 1024L);
uassert( "JS_NewRuntime failed" , _runtime );
}
~SMEngine(){
JS_DestroyRuntime( _runtime );
JS_ShutDown();
}
2009-04-23 23:52:47 -04:00
Scope * createScope();
2009-04-23 17:40:43 -04:00
void runTest();
2009-04-29 10:16:39 -04:00
2009-04-23 17:40:43 -04:00
private:
JSRuntime * _runtime;
friend class SMScope;
};
SMEngine * globalSMEngine;
void ScriptEngine::setup(){
globalSMEngine = new SMEngine();
globalScriptEngine = globalSMEngine;
}
2009-04-23 23:52:47 -04:00
class SMScope : public Scope {
2009-04-23 17:40:43 -04:00
public:
SMScope(){
_context = JS_NewContext( globalSMEngine->_runtime , 8192 );
2009-04-29 10:16:39 -04:00
_convertor = new Convertor( _context );
2009-04-23 17:40:43 -04:00
massert( "JS_NewContext failed" , _context );
JS_SetOptions( _context , JSOPTION_VAROBJFIX);
//JS_SetVersion( _context , JSVERSION_LATEST); TODO
//JS_SetErrorReporter( _context , reportError); TODO
_global = JS_NewObject( _context , &global_class, NULL, NULL);
massert( "JS_NewObject failed for global" , _global );
2009-04-23 23:52:47 -04:00
massert( "js init failed" , JS_InitStandardClasses( _context , _global ) );
2009-04-23 17:40:43 -04:00
}
~SMScope(){
2009-04-29 10:16:39 -04:00
delete _convertor;
2009-04-23 17:40:43 -04:00
JS_DestroyContext( _context );
}
2009-04-23 23:52:47 -04:00
void reset(){
massert( "not implemented yet" , 0 );
}
void init( BSONObj * data ){
massert( "not implemented yet" , 0 );
}
2009-04-23 17:40:43 -04:00
// ----- getters ------
double getNumber( const char *field ){
jsval val;
assert( JS_GetProperty( _context , _global , field , &val ) );
return toNumber( val );
}
double toNumber( jsval v ){
double d;
uassert( "not a number" , JS_ValueToNumber( _context , v , &d ) );
return d;
}
2009-04-23 23:52:47 -04:00
string getString( const char *field ){
jsval val;
assert( JS_GetProperty( _context , _global , field , &val ) );
JSString * s = JS_ValueToString( _context , val );
2009-04-29 10:16:39 -04:00
return _convertor->toString( s );
2009-04-23 23:52:47 -04:00
}
bool getBoolean( const char *field ){
jsval val;
assert( JS_GetProperty( _context , _global , field , &val ) );
JSBool b;
assert( JS_ValueToBoolean( _context, val , &b ) );
return b;
}
BSONObj getObject( const char *field ){
massert( "not implemented yet: getObject()" , 0 ); throw -1;
}
int type( const char *field ){
massert( "not implemented yet: type()" , 0 ); throw -1;
}
// ----- setters ------
void setNumber( const char *field , double val ){
2009-04-29 10:16:39 -04:00
jsval v = _convertor->toval( val );
2009-04-23 23:52:47 -04:00
assert( JS_SetProperty( _context , _global , field , &v ) );
}
void setString( const char *field , const char * val ){
2009-04-29 10:29:56 -04:00
jsval v = _convertor->toval( val );
2009-04-23 23:52:47 -04:00
assert( JS_SetProperty( _context , _global , field , &v ) );
}
void setObject( const char *field , const BSONObj& obj ){
2009-04-29 10:16:39 -04:00
JSObject * o = JS_NewObject( _context , &bson_ro_class , NULL, NULL);
JS_SetPrivate( _context , o , (void*)(&obj) );
// TODO: make a copy and then delete it
jsval v = OBJECT_TO_JSVAL( o );
JS_SetProperty( _context , _global , field , &v );
2009-04-23 23:52:47 -04:00
}
void setBoolean( const char *field , bool val ){
jsval v = BOOLEAN_TO_JSVAL( val );
assert( JS_SetProperty( _context , _global , field , &v ) );
}
void setThis( const BSONObj * obj ){
massert( "not implemented yet: setThis()" , 0 );
}
2009-04-23 17:40:43 -04:00
// ---- functions -----
JSFunction * compileFunction( const char * code ){
2009-04-24 16:41:40 -04:00
if ( strstr( code , "function(" ) != code )
return JS_CompileFunction( _context , 0 , "anonymous" , 0 , 0 , code , strlen( code ) , "nofile" , 0 );
// TODO: there must be a way in spider monkey to do this - this is a total hack
string s = "return ";
s += code;
s += ";";
JSFunction * func = JS_CompileFunction( _context , 0 , "anonymous" , 0 , 0 , s.c_str() , strlen( s.c_str() ) , "nofile" , 0 );
jsval ret;
JS_CallFunction( _context , 0 , func , 0 , 0 , &ret );
return JS_ValueToFunction( _context , ret );
2009-04-23 17:40:43 -04:00
}
ScriptingFunction createFunction( const char * code ){
return (ScriptingFunction)compileFunction( code );
}
int invoke( JSFunction * func , const BSONObj& args ){
jsval rval;
JS_CallFunction( _context , 0 , func , 0 , 0 , &rval );
2009-04-24 16:41:40 -04:00
assert( JS_SetProperty( _context , _global , "return" , &rval ) );
2009-04-23 17:40:43 -04:00
return 0;
}
int invoke( ScriptingFunction funcAddr , const BSONObj& args ){
return invoke( (JSFunction*)funcAddr , args );
}
private:
JSContext * _context;
2009-04-29 10:16:39 -04:00
Convertor * _convertor;
2009-04-23 17:40:43 -04:00
JSObject * _global;
};
void SMEngine::runTest(){
2009-04-23 22:44:19 -04:00
// this is deprecated
2009-04-23 17:40:43 -04:00
}
2009-04-23 23:52:47 -04:00
Scope * SMEngine::createScope(){
return new SMScope();
}
2009-04-23 17:40:43 -04:00
}