summaryrefslogtreecommitdiff
path: root/scripting/engine_spidermonkey.cpp
diff options
context:
space:
mode:
authorEliot Horowitz <eliot@10gen.com>2011-12-24 15:33:26 -0500
committerEliot Horowitz <eliot@10gen.com>2011-12-24 15:33:45 -0500
commitae1ecd9c786911f9f1f0242f0f7d702b3e5dfeba (patch)
tree92f8e1649e6f080b251ff5f1763679a72eb59b34 /scripting/engine_spidermonkey.cpp
parentdfa4cd7e2cf109b072440155fabc08a93c8045a0 (diff)
downloadmongo-ae1ecd9c786911f9f1f0242f0f7d702b3e5dfeba.tar.gz
bulk move of code to src/ SERVER-4551
Diffstat (limited to 'scripting/engine_spidermonkey.cpp')
-rw-r--r--scripting/engine_spidermonkey.cpp1766
1 files changed, 0 insertions, 1766 deletions
diff --git a/scripting/engine_spidermonkey.cpp b/scripting/engine_spidermonkey.cpp
deleted file mode 100644
index 70b89cddbb5..00000000000
--- a/scripting/engine_spidermonkey.cpp
+++ /dev/null
@@ -1,1766 +0,0 @@
-// engine_spidermonkey.cpp
-
-/* Copyright 2009 10gen Inc.
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#include "pch.h"
-#include "engine_spidermonkey.h"
-#include "../client/dbclient.h"
-
-#ifndef _WIN32
-#include <boost/date_time/posix_time/posix_time.hpp>
-#undef assert
-#define assert MONGO_assert
-#endif
-
-#define smuassert( cx , msg , val ) \
- if ( ! ( val ) ){ \
- JS_ReportError( cx , msg ); \
- return JS_FALSE; \
- }
-
-#define CHECKNEWOBJECT(xx,ctx,w) \
- if ( ! xx ){ \
- massert(13072,(string)"JS_NewObject failed: " + w ,xx); \
- }
-
-#define CHECKJSALLOC( newthing ) \
- massert( 13615 , "JS allocation failed, either memory leak or using too much memory" , newthing )
-
-namespace mongo {
-
- class InvalidUTF8Exception : public UserException {
- public:
- InvalidUTF8Exception() : UserException( 9006 , "invalid utf8" ) {
- }
- };
-
- string trim( string s ) {
- while ( s.size() && isspace( s[0] ) )
- s = s.substr( 1 );
-
- while ( s.size() && isspace( s[s.size()-1] ) )
- s = s.substr( 0 , s.size() - 1 );
-
- return s;
- }
-
- boost::thread_specific_ptr<SMScope> currentScope( dontDeleteScope );
- boost::recursive_mutex &smmutex = *( new boost::recursive_mutex );
-#define smlock recursive_scoped_lock ___lk( smmutex );
-
-#define GETHOLDER(x,o) ((BSONHolder*)JS_GetPrivate( x , o ))
-
- class BSONFieldIterator;
-
- class BSONHolder {
- public:
-
- BSONHolder( BSONObj obj ) {
- _obj = obj.getOwned();
- _inResolve = false;
- _modified = false;
- _magic = 17;
- }
-
- ~BSONHolder() {
- _magic = 18;
- }
-
- void check() {
- uassert( 10212 , "holder magic value is wrong" , _magic == 17 && _obj.isValid() );
- }
-
- BSONFieldIterator * it();
-
- BSONObj _obj;
- bool _inResolve;
- char _magic;
- list<string> _extra;
- set<string> _removed;
- bool _modified;
- };
-
- class BSONFieldIterator {
- public:
-
- BSONFieldIterator( BSONHolder * holder ) {
-
- set<string> added;
-
- BSONObjIterator it( holder->_obj );
- while ( it.more() ) {
- BSONElement e = it.next();
- if ( holder->_removed.count( e.fieldName() ) )
- continue;
- _names.push_back( e.fieldName() );
- added.insert( e.fieldName() );
- }
-
- for ( list<string>::iterator i = holder->_extra.begin(); i != holder->_extra.end(); i++ ) {
- if ( ! added.count( *i ) )
- _names.push_back( *i );
- }
-
- _it = _names.begin();
- }
-
- bool more() {
- return _it != _names.end();
- }
-
- string next() {
- string s = *_it;
- _it++;
- return s;
- }
-
- private:
- list<string> _names;
- list<string>::iterator _it;
- };
-
- BSONFieldIterator * BSONHolder::it() {
- return new BSONFieldIterator( this );
- }
-
- class TraverseStack {
- public:
- TraverseStack() {
- _o = 0;
- _parent = 0;
- }
-
- TraverseStack( JSObject * o , const TraverseStack * parent ) {
- _o = o;
- _parent = parent;
- }
-
- TraverseStack dive( JSObject * o ) const {
- if ( o ) {
- uassert( 13076 , (string)"recursive toObject" , ! has( o ) );
- }
- return TraverseStack( o , this );
- }
-
- int depth() const {
- int d = 0;
- const TraverseStack * s = _parent;
- while ( s ) {
- s = s->_parent;
- d++;
- }
- return d;
- }
-
- bool isTop() const {
- return _parent == 0;
- }
-
- bool has( JSObject * o ) const {
- if ( ! o )
- return false;
- const TraverseStack * s = this;
- while ( s ) {
- if ( s->_o == o )
- return true;
- s = s->_parent;
- }
- return false;
- }
-
- JSObject * _o;
- const TraverseStack * _parent;
- };
-
- class Convertor : boost::noncopyable {
- public:
- Convertor( JSContext * cx ) {
- _context = cx;
- }
-
- string toString( JSString * so ) {
- jschar * s = JS_GetStringChars( so );
- size_t srclen = JS_GetStringLength( so );
- if( srclen == 0 )
- return "";
-
- size_t len = srclen * 6; // we only need *3, but see note on len below
- char * dst = (char*)malloc( len );
-
- len /= 2;
- // doc re weird JS_EncodeCharacters api claims len expected in 16bit
- // units, but experiments suggest 8bit units expected. We allocate
- // enough memory that either will work.
-
- if ( !JS_EncodeCharacters( _context , s , srclen , dst , &len) ) {
- StringBuilder temp;
- temp << "Not proper UTF-16: ";
- for ( size_t i=0; i<srclen; i++ ) {
- if ( i > 0 )
- temp << ",";
- temp << s[i];
- }
- uasserted( 13498 , temp.str() );
- }
-
- string ss( dst , len );
- free( dst );
- if ( !JS_CStringsAreUTF8() )
- for( string::const_iterator i = ss.begin(); i != ss.end(); ++i )
- uassert( 10213 , "non ascii character detected", (unsigned char)(*i) <= 127 );
- return ss;
- }
-
- string toString( jsval v ) {
- return toString( JS_ValueToString( _context , v ) );
- }
-
- // NOTE No validation of passed in object
- long long toNumberLongUnsafe( JSObject *o ) {
- boost::uint64_t val;
- if ( hasProperty( o, "top" ) ) {
- val =
- ( (boost::uint64_t)(boost::uint32_t)getNumber( o , "top" ) << 32 ) +
- ( boost::uint32_t)( getNumber( o , "bottom" ) );
- }
- else {
- val = (boost::uint64_t)(boost::int64_t) getNumber( o, "floatApprox" );
- }
- return val;
- }
-
- int toNumberInt( JSObject *o ) {
- return (boost::uint32_t)(boost::int32_t) getNumber( o, "floatApprox" );
- }
-
- double toNumber( jsval v ) {
- double d;
- uassert( 10214 , "not a number" , JS_ValueToNumber( _context , v , &d ) );
- return d;
- }
-
- bool toBoolean( jsval v ) {
- JSBool b;
- assert( JS_ValueToBoolean( _context, v , &b ) );
- return b;
- }
-
- OID toOID( jsval v ) {
- JSContext * cx = _context;
- assert( JSVAL_IS_OID( v ) );
-
- JSObject * o = JSVAL_TO_OBJECT( v );
- OID oid;
- oid.init( getString( o , "str" ) );
- return oid;
- }
-
- BSONObj toObject( JSObject * o , const TraverseStack& stack=TraverseStack() ) {
- if ( ! o )
- return BSONObj();
-
- if ( JS_InstanceOf( _context , o , &bson_ro_class , 0 ) ) {
- BSONHolder * holder = GETHOLDER( _context , o );
- assert( holder );
- return holder->_obj.getOwned();
- }
-
- BSONObj orig;
- if ( JS_InstanceOf( _context , o , &bson_class , 0 ) ) {
- BSONHolder * holder = GETHOLDER(_context,o);
- assert( holder );
- if ( ! holder->_modified ) {
- return holder->_obj;
- }
- orig = holder->_obj;
- }
-
- BSONObjBuilder b;
-
- if ( ! appendSpecialDBObject( this , b , "value" , OBJECT_TO_JSVAL( o ) , o ) ) {
-
- if ( stack.isTop() ) {
- jsval theid = getProperty( o , "_id" );
- if ( ! JSVAL_IS_VOID( theid ) ) {
- append( b , "_id" , theid , EOO , stack.dive( o ) );
- }
- }
-
- JSIdArray * properties = JS_Enumerate( _context , o );
- assert( properties );
-
- for ( jsint i=0; i<properties->length; i++ ) {
- jsid id = properties->vector[i];
- jsval nameval;
- assert( JS_IdToValue( _context ,id , &nameval ) );
- string name = toString( nameval );
- if ( stack.isTop() && name == "_id" )
- continue;
-
- append( b , name , getProperty( o , name.c_str() ) , orig[name].type() , stack.dive( o ) );
- }
-
- JS_DestroyIdArray( _context , properties );
- }
-
- return b.obj();
- }
-
- BSONObj toObject( jsval v ) {
- if ( JSVAL_IS_NULL( v ) ||
- JSVAL_IS_VOID( v ) )
- return BSONObj();
-
- uassert( 10215 , "not an object" , JSVAL_IS_OBJECT( v ) );
- return toObject( JSVAL_TO_OBJECT( v ) );
- }
-
- string getFunctionCode( JSFunction * func ) {
- return toString( JS_DecompileFunction( _context , func , 0 ) );
- }
-
- string getFunctionCode( jsval v ) {
- uassert( 10216 , "not a function" , JS_TypeOfValue( _context , v ) == JSTYPE_FUNCTION );
- return getFunctionCode( JS_ValueToFunction( _context , v ) );
- }
-
- void appendRegex( BSONObjBuilder& b , const string& name , string s ) {
- assert( s[0] == '/' );
- s = s.substr(1);
- string::size_type end = s.rfind( '/' );
- b.appendRegex( name , s.substr( 0 , end ) , s.substr( end + 1 ) );
- }
-
- void append( BSONObjBuilder& b , string name , jsval val , BSONType oldType = EOO , const TraverseStack& stack=TraverseStack() ) {
- //cout << "name: " << name << "\t" << typeString( val ) << " oldType: " << oldType << endl;
- switch ( JS_TypeOfValue( _context , val ) ) {
-
- case JSTYPE_VOID: b.appendUndefined( name ); break;
- case JSTYPE_NULL: b.appendNull( name ); break;
-
- case JSTYPE_NUMBER: {
- double d = toNumber( val );
- if ( oldType == NumberInt && ((int)d) == d )
- b.append( name , (int)d );
- else
- b.append( name , d );
- break;
- }
- case JSTYPE_STRING: b.append( name , toString( val ) ); break;
- case JSTYPE_BOOLEAN: b.appendBool( name , toBoolean( val ) ); break;
-
- case JSTYPE_OBJECT: {
- JSObject * o = JSVAL_TO_OBJECT( val );
- if ( ! o || o == JSVAL_NULL ) {
- b.appendNull( name );
- }
- else if ( ! appendSpecialDBObject( this , b , name , val , o ) ) {
- BSONObj sub = toObject( o , stack );
- if ( JS_IsArrayObject( _context , o ) ) {
- b.appendArray( name , sub );
- }
- else {
- b.append( name , sub );
- }
- }
- break;
- }
-
- case JSTYPE_FUNCTION: {
- string s = toString(val);
- if ( s[0] == '/' ) {
- appendRegex( b , name , s );
- }
- else {
- b.appendCode( name , getFunctionCode( val ) );
- }
- break;
- }
-
- default: uassert( 10217 , (string)"can't append field. name:" + name + " type: " + typeString( val ) , 0 );
- }
- }
-
- // ---------- to spider monkey ---------
-
- bool hasFunctionIdentifier( const string& code ) {
- if ( code.size() < 9 || code.find( "function" ) != 0 )
- return false;
-
- return code[8] == ' ' || code[8] == '(';
- }
-
- bool isSimpleStatement( const string& code ) {
- if ( hasJSReturn( code ) )
- return false;
-
- if ( code.find( ';' ) != string::npos &&
- code.find( ';' ) != code.rfind( ';' ) )
- return false;
-
- if ( code.find( '\n') != string::npos )
- return false;
-
- if ( code.find( "for(" ) != string::npos ||
- code.find( "for (" ) != string::npos ||
- code.find( "while (" ) != string::npos ||
- code.find( "while(" ) != string::npos )
- return false;
-
- return true;
- }
-
- void addRoot( JSFunction * f , const char * name );
-
- JSFunction * compileFunction( const char * code, JSObject * assoc = 0 ) {
- const char * gcName = "unknown";
- JSFunction * f = _compileFunction( code , assoc , gcName );
- //addRoot( f , gcName );
- return f;
- }
-
- JSFunction * _compileFunction( const char * raw , JSObject * assoc , const char *& gcName ) {
- if ( ! assoc )
- assoc = JS_GetGlobalObject( _context );
-
- raw = jsSkipWhiteSpace( raw );
-
- //cout << "RAW\n---\n" << raw << "\n---" << endl;
-
- static int fnum = 1;
- stringstream fname;
- fname << "__cf__" << fnum++ << "__";
-
- if ( ! hasFunctionIdentifier( raw ) ) {
- string s = raw;
- if ( isSimpleStatement( s ) ) {
- s = "return " + s;
- }
- gcName = "cf anon";
- fname << "anon";
- return JS_CompileFunction( _context , assoc , fname.str().c_str() , 0 , 0 , s.c_str() , s.size() , "nofile_a" , 0 );
- }
-
- string code = raw;
-
- size_t start = code.find( '(' );
- assert( start != string::npos );
-
- string fbase;
- if ( start > 9 ) {
- fbase = trim( code.substr( 9 , start - 9 ) );
- }
- if ( fbase.length() == 0 ) {
- fbase = "anonymous_function";
- }
- fname << "f__" << fbase;
-
- code = code.substr( start + 1 );
- size_t end = code.find( ')' );
- assert( end != string::npos );
-
- string paramString = trim( code.substr( 0 , end ) );
- code = code.substr( end + 1 );
-
- vector<string> params;
- while ( paramString.size() ) {
- size_t c = paramString.find( ',' );
- if ( c == string::npos ) {
- params.push_back( paramString );
- break;
- }
- params.push_back( trim( paramString.substr( 0 , c ) ) );
- paramString = trim( paramString.substr( c + 1 ) );
- paramString = trim( paramString );
- }
-
- boost::scoped_array<const char *> paramArray (new const char*[params.size()]);
- for ( size_t i=0; i<params.size(); i++ )
- paramArray[i] = params[i].c_str();
-
- // avoid munging previously munged name (kludge; switching to v8 fixes underlying issue)
- if ( fbase.find("__cf__") != 0 && fbase.find("__f__") == string::npos ) {
- fbase = fname.str();
- }
-
- JSFunction * func = JS_CompileFunction( _context , assoc , fbase.c_str() , params.size() , paramArray.get() , code.c_str() , code.size() , "nofile_b" , 0 );
-
- if ( ! func ) {
- log() << "compile failed for: " << raw << endl;
- return 0;
- }
- gcName = "cf normal";
- return func;
- }
-
- jsval toval( double d ) {
- jsval val;
- assert( JS_NewNumberValue( _context, d , &val ) );
- return val;
- }
-
- jsval toval( const char * c ) {
- JSString * s = JS_NewStringCopyZ( _context , c );
- if ( s )
- return STRING_TO_JSVAL( s );
-
- // possibly unicode, try manual
-
- size_t len = strlen( c );
- size_t dstlen = len * 4;
- jschar * dst = (jschar*)malloc( dstlen );
-
- JSBool res = JS_DecodeBytes( _context , c , len , dst, &dstlen );
- if ( res ) {
- s = JS_NewUCStringCopyN( _context , dst , dstlen );
- }
-
- free( dst );
-
- if ( ! res ) {
- tlog() << "decode failed. probably invalid utf-8 string [" << c << "]" << endl;
- jsval v;
- if ( JS_GetPendingException( _context , &v ) )
- tlog() << "\t why: " << toString( v ) << endl;
- throw InvalidUTF8Exception();
- }
-
- CHECKJSALLOC( s );
- return STRING_TO_JSVAL( s );
- }
-
- JSObject * toJSObject( const BSONObj * obj , bool readOnly=false ) {
- static string ref = "$ref";
- if ( ref == obj->firstElementFieldName() ) {
- JSObject * o = JS_NewObject( _context , &dbref_class , NULL, NULL);
- CHECKNEWOBJECT(o,_context,"toJSObject1");
- assert( JS_SetPrivate( _context , o , (void*)(new BSONHolder( obj->getOwned() ) ) ) );
- return o;
- }
- JSObject * o = JS_NewObject( _context , readOnly ? &bson_ro_class : &bson_class , NULL, NULL);
- CHECKNEWOBJECT(o,_context,"toJSObject2");
- assert( JS_SetPrivate( _context , o , (void*)(new BSONHolder( obj->getOwned() ) ) ) );
- return o;
- }
-
- jsval toval( const BSONObj* obj , bool readOnly=false ) {
- JSObject * o = toJSObject( obj , readOnly );
- return OBJECT_TO_JSVAL( o );
- }
-
- void makeLongObj( long long n, JSObject * o ) {
- boost::uint64_t val = (boost::uint64_t)n;
- CHECKNEWOBJECT(o,_context,"NumberLong1");
- double floatApprox = (double)(boost::int64_t)val;
- setProperty( o , "floatApprox" , toval( floatApprox ) );
- if ( (boost::int64_t)val != (boost::int64_t)floatApprox ) {
- // using 2 doubles here instead of a single double because certain double
- // bit patterns represent undefined values and sm might trash them
- setProperty( o , "top" , toval( (double)(boost::uint32_t)( val >> 32 ) ) );
- setProperty( o , "bottom" , toval( (double)(boost::uint32_t)( val & 0x00000000ffffffff ) ) );
- }
- }
-
- jsval toval( long long n ) {
- JSObject * o = JS_NewObject( _context , &numberlong_class , 0 , 0 );
- makeLongObj( n, o );
- return OBJECT_TO_JSVAL( o );
- }
-
- void makeIntObj( int n, JSObject * o ) {
- boost::uint32_t val = (boost::uint32_t)n;
- CHECKNEWOBJECT(o,_context,"NumberInt1");
- double floatApprox = (double)(boost::int32_t)val;
- setProperty( o , "floatApprox" , toval( floatApprox ) );
- }
-
- jsval toval( int n ) {
- JSObject * o = JS_NewObject( _context , &numberint_class , 0 , 0 );
- makeIntObj( n, o );
- return OBJECT_TO_JSVAL( o );
- }
-
- jsval toval( const BSONElement& e ) {
-
- switch( e.type() ) {
- case EOO:
- case jstNULL:
- case Undefined:
- return JSVAL_NULL;
- case NumberDouble:
- case NumberInt:
- return toval( e.number() );
-// case NumberInt:
-// return toval( e.numberInt() );
- case Symbol: // TODO: should we make a special class for this
- case String:
- return toval( e.valuestr() );
- case Bool:
- return e.boolean() ? JSVAL_TRUE : JSVAL_FALSE;
- case Object: {
- BSONObj embed = e.embeddedObject().getOwned();
- return toval( &embed );
- }
- case Array: {
-
- BSONObj embed = e.embeddedObject().getOwned();
-
- if ( embed.isEmpty() ) {
- return OBJECT_TO_JSVAL( JS_NewArrayObject( _context , 0 , 0 ) );
- }
-
- JSObject * array = JS_NewArrayObject( _context , 1 , 0 );
- CHECKJSALLOC( array );
-
- jsval myarray = OBJECT_TO_JSVAL( array );
-
- BSONObjIterator i( embed );
- while ( i.more() ){
- const BSONElement& e = i.next();
- jsval v = toval( e );
- assert( JS_SetElement( _context , array , atoi(e.fieldName()) , &v ) );
- }
-
- return myarray;
- }
- case jstOID: {
- OID oid = e.__oid();
- JSObject * o = JS_NewObject( _context , &object_id_class , 0 , 0 );
- CHECKNEWOBJECT(o,_context,"jstOID");
- setProperty( o , "str" , toval( oid.str().c_str() ) );
- return OBJECT_TO_JSVAL( o );
- }
- case RegEx: {
- const char * flags = e.regexFlags();
- uintN flagNumber = 0;
- while ( *flags ) {
- switch ( *flags ) {
- case 'g': flagNumber |= JSREG_GLOB; break;
- case 'i': flagNumber |= JSREG_FOLD; break;
- case 'm': flagNumber |= JSREG_MULTILINE; break;
- //case 'y': flagNumber |= JSREG_STICKY; break;
-
- default:
- log() << "warning: unknown regex flag:" << *flags << endl;
- }
- flags++;
- }
-
- JSObject * r = JS_NewRegExpObject( _context , (char*)e.regex() , strlen( e.regex() ) , flagNumber );
- assert( r );
- return OBJECT_TO_JSVAL( r );
- }
- case Code: {
- JSFunction * func = compileFunction( e.valuestr() );
- if ( func )
- return OBJECT_TO_JSVAL( JS_GetFunctionObject( func ) );
- return JSVAL_NULL;
- }
- case CodeWScope: {
- JSFunction * func = compileFunction( e.codeWScopeCode() );
- if ( !func )
- return JSVAL_NULL;
-
- BSONObj extraScope = e.codeWScopeObject();
- if ( ! extraScope.isEmpty() ) {
- log() << "warning: CodeWScope doesn't transfer to db.eval" << endl;
- }
-
- return OBJECT_TO_JSVAL( JS_GetFunctionObject( func ) );
- }
- case Date:
- return OBJECT_TO_JSVAL( js_NewDateObjectMsec( _context , (jsdouble) ((long long)e.date().millis) ) );
-
- case MinKey:
- return OBJECT_TO_JSVAL( JS_NewObject( _context , &minkey_class , 0 , 0 ) );
-
- case MaxKey:
- return OBJECT_TO_JSVAL( JS_NewObject( _context , &maxkey_class , 0 , 0 ) );
-
- case Timestamp: {
- JSObject * o = JS_NewObject( _context , &timestamp_class , 0 , 0 );
- CHECKNEWOBJECT(o,_context,"Timestamp1");
- setProperty( o , "t" , toval( (double)(e.timestampTime()) ) );
- setProperty( o , "i" , toval( (double)(e.timestampInc()) ) );
- return OBJECT_TO_JSVAL( o );
- }
- case NumberLong: {
- return toval( e.numberLong() );
- }
- case DBRef: {
- JSObject * o = JS_NewObject( _context , &dbpointer_class , 0 , 0 );
- CHECKNEWOBJECT(o,_context,"DBRef1");
- setProperty( o , "ns" , toval( e.dbrefNS() ) );
-
- JSObject * oid = JS_NewObject( _context , &object_id_class , 0 , 0 );
- CHECKNEWOBJECT(oid,_context,"DBRef2");
- setProperty( oid , "str" , toval( e.dbrefOID().str().c_str() ) );
-
- setProperty( o , "id" , OBJECT_TO_JSVAL( oid ) );
- return OBJECT_TO_JSVAL( o );
- }
- case BinData: {
- JSObject * o = JS_NewObject( _context , &bindata_class , 0 , 0 );
- CHECKNEWOBJECT(o,_context,"Bindata_BinData1");
- int len;
- const char * data = e.binData( len );
- assert( data );
- assert( JS_SetPrivate( _context , o , new BinDataHolder( data , len ) ) );
-
- setProperty( o , "len" , toval( (double)len ) );
- setProperty( o , "type" , toval( (double)e.binDataType() ) );
- return OBJECT_TO_JSVAL( o );
- }
- }
-
- log() << "toval: unknown type: " << (int) e.type() << endl;
- uassert( 10218 , "not done: toval" , 0 );
- return 0;
- }
-
- // ------- object helpers ------
-
- JSObject * getJSObject( JSObject * o , const char * name ) {
- jsval v;
- assert( JS_GetProperty( _context , o , name , &v ) );
- return JSVAL_TO_OBJECT( v );
- }
-
- JSObject * getGlobalObject( const char * name ) {
- return getJSObject( JS_GetGlobalObject( _context ) , name );
- }
-
- JSObject * getGlobalPrototype( const char * name ) {
- return getJSObject( getGlobalObject( name ) , "prototype" );
- }
-
- bool hasProperty( JSObject * o , const char * name ) {
- JSBool res;
- assert( JS_HasProperty( _context , o , name , & res ) );
- return res;
- }
-
- jsval getProperty( JSObject * o , const char * field ) {
- uassert( 10219 , "object passed to getPropery is null" , o );
- jsval v;
- assert( JS_GetProperty( _context , o , field , &v ) );
- return v;
- }
-
- void setProperty( JSObject * o , const char * field , jsval v ) {
- assert( JS_SetProperty( _context , o , field , &v ) );
- }
-
- string typeString( jsval v ) {
- JSType t = JS_TypeOfValue( _context , v );
- return JS_GetTypeName( _context , t );
- }
-
- bool getBoolean( JSObject * o , const char * field ) {
- return toBoolean( getProperty( o , field ) );
- }
-
- double getNumber( JSObject * o , const char * field ) {
- return toNumber( getProperty( o , field ) );
- }
-
- string getString( JSObject * o , const char * field ) {
- return toString( getProperty( o , field ) );
- }
-
- JSClass * getClass( JSObject * o , const char * field ) {
- jsval v;
- assert( JS_GetProperty( _context , o , field , &v ) );
- if ( ! JSVAL_IS_OBJECT( v ) )
- return 0;
- return JS_GET_CLASS( _context , JSVAL_TO_OBJECT( v ) );
- }
-
- JSContext * _context;
-
-
- };
-
-
- void bson_finalize( JSContext * cx , JSObject * obj ) {
- BSONHolder * o = GETHOLDER( cx , obj );
- if ( o ) {
- delete o;
- assert( JS_SetPrivate( cx , obj , 0 ) );
- }
- }
-
- JSBool bson_enumerate( JSContext *cx, JSObject *obj, JSIterateOp enum_op, jsval *statep, jsid *idp ) {
-
- BSONHolder * o = GETHOLDER( cx , obj );
-
- if ( enum_op == JSENUMERATE_INIT ) {
- if ( o ) {
- BSONFieldIterator * it = o->it();
- *statep = PRIVATE_TO_JSVAL( it );
- }
- else {
- *statep = 0;
- }
- if ( idp )
- *idp = JSVAL_ZERO;
- return JS_TRUE;
- }
-
- BSONFieldIterator * it = (BSONFieldIterator*)JSVAL_TO_PRIVATE( *statep );
- if ( ! it ) {
- *statep = 0;
- return JS_TRUE;
- }
-
- if ( enum_op == JSENUMERATE_NEXT ) {
- if ( it->more() ) {
- string name = it->next();
- Convertor c(cx);
- assert( JS_ValueToId( cx , c.toval( name.c_str() ) , idp ) );
- }
- else {
- delete it;
- *statep = 0;
- }
- return JS_TRUE;
- }
-
- if ( enum_op == JSENUMERATE_DESTROY ) {
- if ( it )
- delete it;
- return JS_TRUE;
- }
-
- uassert( 10220 , "don't know what to do with this op" , 0 );
- return JS_FALSE;
- }
-
- JSBool noaccess( JSContext *cx, JSObject *obj, jsval idval, jsval *vp) {
- BSONHolder * holder = GETHOLDER( cx , obj );
- if ( ! holder ) {
- // in init code still
- return JS_TRUE;
- }
- if ( holder->_inResolve )
- return JS_TRUE;
- JS_ReportError( cx , "doing write op on read only operation" );
- return JS_FALSE;
- }
-
- JSClass bson_ro_class = {
- "bson_ro_object" , JSCLASS_HAS_PRIVATE | JSCLASS_NEW_RESOLVE | JSCLASS_NEW_ENUMERATE ,
- noaccess, noaccess, JS_PropertyStub, noaccess,
- (JSEnumerateOp)bson_enumerate, (JSResolveOp)(&resolveBSONField) , JS_ConvertStub, bson_finalize ,
- JSCLASS_NO_OPTIONAL_MEMBERS
- };
-
- JSBool bson_cons( JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval ) {
- cerr << "bson_cons : shouldn't be here!" << endl;
- JS_ReportError( cx , "can't construct bson object" );
- return JS_FALSE;
- }
-
- JSFunctionSpec bson_functions[] = {
- { 0 }
- };
-
- JSBool bson_add_prop( JSContext *cx, JSObject *obj, jsval idval, jsval *vp) {
- BSONHolder * holder = GETHOLDER( cx , obj );
- if ( ! holder ) {
- // static init
- return JS_TRUE;
- }
- if ( ! holder->_inResolve ) {
- Convertor c(cx);
- string name = c.toString( idval );
- if ( holder->_obj[name].eoo() ) {
- holder->_extra.push_back( name );
- }
- holder->_modified = true;
- }
- return JS_TRUE;
- }
-
-
- JSBool mark_modified( JSContext *cx, JSObject *obj, jsval idval, jsval *vp) {
- Convertor c(cx);
- BSONHolder * holder = GETHOLDER( cx , obj );
- if ( !holder ) // needed when we're messing with DBRef.prototype
- return JS_TRUE;
- if ( holder->_inResolve )
- return JS_TRUE;
- holder->_modified = true;
- holder->_removed.erase( c.toString( idval ) );
- return JS_TRUE;
- }
-
- JSBool mark_modified_remove( JSContext *cx, JSObject *obj, jsval idval, jsval *vp) {
- Convertor c(cx);
- BSONHolder * holder = GETHOLDER( cx , obj );
- if ( holder->_inResolve )
- return JS_TRUE;
- holder->_modified = true;
- holder->_removed.insert( c.toString( idval ) );
- return JS_TRUE;
- }
-
- JSClass bson_class = {
- "bson_object" , JSCLASS_HAS_PRIVATE | JSCLASS_NEW_RESOLVE | JSCLASS_NEW_ENUMERATE ,
- bson_add_prop, mark_modified_remove, JS_PropertyStub, mark_modified,
- (JSEnumerateOp)bson_enumerate, (JSResolveOp)(&resolveBSONField) , JS_ConvertStub, bson_finalize ,
- JSCLASS_NO_OPTIONAL_MEMBERS
- };
-
- 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
- };
-
- // --- global helpers ---
-
- JSBool hexToBinData(JSContext * cx, jsval *rval, int subtype, string s) {
- JSObject * o = JS_NewObject( cx , &bindata_class , 0 , 0 );
- CHECKNEWOBJECT(o,_context,"Bindata_BinData1");
- int len = s.size() / 2;
- char * data = new char[len];
- char *p = data;
- const char *src = s.c_str();
- for( size_t i = 0; i+1 < s.size(); i += 2 ) {
- *p++ = fromHex(src + i);
- }
- assert( JS_SetPrivate( cx , o , new BinDataHolder( data , len ) ) );
- Convertor c(cx);
- c.setProperty( o, "len", c.toval((double)len) );
- c.setProperty( o, "type", c.toval((double)subtype) );
- *rval = OBJECT_TO_JSVAL( o );
- delete data;
- return JS_TRUE;
- }
-
- JSBool _HexData( JSContext * cx , JSObject * obj , uintN argc, jsval *argv, jsval *rval ) {
- Convertor c( cx );
- if ( argc != 2 ) {
- JS_ReportError( cx , "HexData needs 2 arguments -- HexData(subtype,hexstring)" );
- return JS_FALSE;
- }
- int type = (int)c.toNumber( argv[ 0 ] );
- if ( type == 2 ) {
- JS_ReportError( cx , "BinData subtype 2 is deprecated" );
- return JS_FALSE;
- }
- string s = c.toString(argv[1]);
- return hexToBinData(cx, rval, type, s);
- }
-
- JSBool _UUID( JSContext * cx , JSObject * obj , uintN argc, jsval *argv, jsval *rval ) {
- Convertor c( cx );
- if ( argc != 1 ) {
- JS_ReportError( cx , "UUID needs argument -- UUID(hexstring)" );
- return JS_FALSE;
- }
- string s = c.toString(argv[0]);
- if( s.size() != 32 ) {
- JS_ReportError( cx , "bad UUID hex string len" );
- return JS_FALSE;
- }
- return hexToBinData(cx, rval, 3, s);
- }
-
- JSBool _MD5( JSContext * cx , JSObject * obj , uintN argc, jsval *argv, jsval *rval ) {
- Convertor c( cx );
- if ( argc != 1 ) {
- JS_ReportError( cx , "MD5 needs argument -- MD5(hexstring)" );
- return JS_FALSE;
- }
- string s = c.toString(argv[0]);
- if( s.size() != 32 ) {
- JS_ReportError( cx , "bad MD5 hex string len" );
- return JS_FALSE;
- }
- return hexToBinData(cx, rval, 5, s);
- }
-
- JSBool native_print( JSContext * cx , JSObject * obj , uintN argc, jsval *argv, jsval *rval ) {
- stringstream ss;
- Convertor c( cx );
- for ( uintN i=0; i<argc; i++ ) {
- if ( i > 0 )
- ss << " ";
- ss << c.toString( argv[i] );
- }
- ss << "\n";
- Logstream::logLockless( ss.str() );
- return JS_TRUE;
- }
-
- JSBool native_helper( JSContext *cx , JSObject *obj , uintN argc, jsval *argv , jsval *rval ) {
- Convertor c(cx);
-
- NativeFunction func = (NativeFunction)((long long)c.getNumber( obj , "x" ) );
- void* data = (void*)((long long)c.getNumber( obj , "y" ) );
- assert( func );
-
- BSONObj a;
- if ( argc > 0 ) {
- BSONObjBuilder args;
- for ( uintN i=0; i<argc; i++ ) {
- c.append( args , args.numStr( i ) , argv[i] );
- }
-
- a = args.obj();
- }
-
- BSONObj out;
- try {
- out = func( a, data );
- }
- catch ( std::exception& e ) {
- JS_ReportError( cx , e.what() );
- return JS_FALSE;
- }
-
- if ( out.isEmpty() ) {
- *rval = JSVAL_VOID;
- }
- else {
- *rval = c.toval( out.firstElement() );
- }
-
- return JS_TRUE;
- }
-
- JSBool native_load( JSContext *cx , JSObject *obj , uintN argc, jsval *argv , jsval *rval );
-
- JSBool native_gc( JSContext *cx , JSObject *obj , uintN argc, jsval *argv , jsval *rval ) {
- JS_GC( cx );
- return JS_TRUE;
- }
-
- JSFunctionSpec globalHelpers[] = {
- { "print" , &native_print , 0 , 0 , 0 } ,
- { "nativeHelper" , &native_helper , 1 , 0 , 0 } ,
- { "load" , &native_load , 1 , 0 , 0 } ,
- { "gc" , &native_gc , 1 , 0 , 0 } ,
- { "UUID", &_UUID, 0, 0, 0 } ,
- { "MD5", &_MD5, 0, 0, 0 } ,
- { "HexData", &_HexData, 0, 0, 0 } ,
- { 0 , 0 , 0 , 0 , 0 }
- };
-
- // ----END global helpers ----
-
- // Object helpers
-
- JSBool bson_get_size(JSContext *cx, JSObject *obj, uintN argc, jsval *argv, jsval *rval) {
- if ( argc != 1 || !JSVAL_IS_OBJECT( argv[ 0 ] ) ) {
- JS_ReportError( cx , "bsonsize requires one valid object" );
- return JS_FALSE;
- }
-
- Convertor c(cx);
-
- if ( argv[0] == JSVAL_VOID || argv[0] == JSVAL_NULL ) {
- *rval = c.toval( 0.0 );
- return JS_TRUE;
- }
-
- JSObject * o = JSVAL_TO_OBJECT( argv[0] );
-
- double size = 0;
-
- if ( JS_InstanceOf( cx , o , &bson_ro_class , 0 ) ||
- JS_InstanceOf( cx , o , &bson_class , 0 ) ) {
- BSONHolder * h = GETHOLDER( cx , o );
- if ( h ) {
- size = h->_obj.objsize();
- }
- }
- else {
- BSONObj temp = c.toObject( o );
- size = temp.objsize();
- }
-
- *rval = c.toval( size );
- return JS_TRUE;
- }
-
- JSFunctionSpec objectHelpers[] = {
- { "bsonsize" , &bson_get_size , 1 , 0 , 0 } ,
- { 0 , 0 , 0 , 0 , 0 }
- };
-
- // end Object helpers
-
- JSBool resolveBSONField( JSContext *cx, JSObject *obj, jsval id, uintN flags, JSObject **objp ) {
- assert( JS_EnterLocalRootScope( cx ) );
- Convertor c( cx );
-
- BSONHolder * holder = GETHOLDER( cx , obj );
- if ( ! holder ) {
- // static init
- *objp = 0;
- JS_LeaveLocalRootScope( cx );
- return JS_TRUE;
- }
- holder->check();
-
- string s = c.toString( id );
-
- BSONElement e = holder->_obj[ s.c_str() ];
-
- if ( e.type() == EOO || holder->_removed.count( s ) ) {
- *objp = 0;
- JS_LeaveLocalRootScope( cx );
- return JS_TRUE;
- }
-
- jsval val;
- try {
- val = c.toval( e );
- }
- catch ( InvalidUTF8Exception& ) {
- JS_LeaveLocalRootScope( cx );
- JS_ReportError( cx , "invalid utf8" );
- return JS_FALSE;
- }
-
- assert( ! holder->_inResolve );
- holder->_inResolve = true;
- assert( JS_SetProperty( cx , obj , s.c_str() , &val ) );
- holder->_inResolve = false;
-
- if ( val != JSVAL_NULL && val != JSVAL_VOID && JSVAL_IS_OBJECT( val ) ) {
- // TODO: this is a hack to get around sub objects being modified
- // basically right now whenever a sub object is read we mark whole obj as possibly modified
- JSObject * oo = JSVAL_TO_OBJECT( val );
- if ( JS_InstanceOf( cx , oo , &bson_class , 0 ) ||
- JS_IsArrayObject( cx , oo ) ) {
- holder->_modified = true;
- }
- }
-
- *objp = obj;
- JS_LeaveLocalRootScope( cx );
- return JS_TRUE;
- }
-
-
- class SMScope;
-
- class SMEngine : public ScriptEngine {
- public:
-
- SMEngine() {
-#ifdef SM18
- JS_SetCStringsAreUTF8();
-#endif
-
- _runtime = JS_NewRuntime(64L * 1024L * 1024L);
- uassert( 10221 , "JS_NewRuntime failed" , _runtime );
-
- if ( ! utf8Ok() ) {
- log() << "*** warning: spider monkey build without utf8 support. consider rebuilding with utf8 support" << endl;
- }
-
- int x = 0;
- assert( x = 1 );
- uassert( 10222 , "assert not being executed" , x == 1 );
- }
-
- ~SMEngine() {
- JS_DestroyRuntime( _runtime );
- JS_ShutDown();
- }
-
- Scope * createScope();
-
- void runTest();
-
- virtual bool utf8Ok() const { return JS_CStringsAreUTF8(); }
-
-#ifdef XULRUNNER
- JSClass * _dateClass;
- JSClass * _regexClass;
-#endif
-
-
- private:
- JSRuntime * _runtime;
- friend class SMScope;
- };
-
- SMEngine * globalSMEngine;
-
-
- void ScriptEngine::setup() {
- globalSMEngine = new SMEngine();
- globalScriptEngine = globalSMEngine;
- }
-
-
- // ------ scope ------
-
-
- JSBool no_gc(JSContext *cx, JSGCStatus status) {
- return JS_FALSE;
- }
-
- JSBool yes_gc(JSContext *cx, JSGCStatus status) {
- return JS_TRUE;
- }
-
- class SMScope : public Scope {
- public:
- SMScope() : _this( 0 ) , _externalSetup( false ) , _localConnect( false ) {
- smlock;
- _context = JS_NewContext( globalSMEngine->_runtime , 8192 );
- _convertor = new Convertor( _context );
- massert( 10431 , "JS_NewContext failed" , _context );
-
- JS_SetOptions( _context , JSOPTION_VAROBJFIX);
- //JS_SetVersion( _context , JSVERSION_LATEST); TODO
- JS_SetErrorReporter( _context , errorReporter );
-
- _global = JS_NewObject( _context , &global_class, NULL, NULL);
- massert( 10432 , "JS_NewObject failed for global" , _global );
- JS_SetGlobalObject( _context , _global );
- massert( 10433 , "js init failed" , JS_InitStandardClasses( _context , _global ) );
-
- JS_SetOptions( _context , JS_GetOptions( _context ) | JSOPTION_VAROBJFIX );
-
- JS_DefineFunctions( _context , _global , globalHelpers );
-
- JS_DefineFunctions( _context , _convertor->getGlobalObject( "Object" ), objectHelpers );
-
- //JS_SetGCCallback( _context , no_gc ); // this is useful for seeing if something is a gc problem
-
- _postCreateHacks();
- }
-
- ~SMScope() {
- smlock;
- uassert( 10223 , "deleted SMScope twice?" , _convertor );
-
- for ( list<void*>::iterator i=_roots.begin(); i != _roots.end(); i++ ) {
- JS_RemoveRoot( _context , *i );
- }
- _roots.clear();
-
- if ( _this ) {
- JS_RemoveRoot( _context , &_this );
- _this = 0;
- }
-
- if ( _convertor ) {
- delete _convertor;
- _convertor = 0;
- }
-
- if ( _context ) {
- // This is expected to reclaim _global as well.
- JS_DestroyContext( _context );
- _context = 0;
- }
-
- }
-
- void reset() {
- smlock;
- assert( _convertor );
- return;
- if ( _this ) {
- JS_RemoveRoot( _context , &_this );
- _this = 0;
- }
- currentScope.reset( this );
- _error = "";
- }
-
- void addRoot( void * root , const char * name ) {
- JS_AddNamedRoot( _context , root , name );
- _roots.push_back( root );
- }
-
- void init( const BSONObj * data ) {
- smlock;
- if ( ! data )
- return;
-
- BSONObjIterator i( *data );
- while ( i.more() ) {
- BSONElement e = i.next();
- _convertor->setProperty( _global , e.fieldName() , _convertor->toval( e ) );
- _initFieldNames.insert( e.fieldName() );
- }
-
- }
-
- bool hasOutOfMemoryException() {
- string err = getError();
- return err.find("out of memory") != string::npos;
- }
-
- void externalSetup() {
- smlock;
- uassert( 10224 , "already local connected" , ! _localConnect );
- if ( _externalSetup )
- return;
- initMongoJS( this , _context , _global , false );
- _externalSetup = true;
- }
-
- void localConnect( const char * dbName ) {
- {
- smlock;
- uassert( 10225 , "already setup for external db" , ! _externalSetup );
- if ( _localConnect ) {
- uassert( 10226 , "connected to different db" , _localDBName == dbName );
- return;
- }
-
- initMongoJS( this , _context , _global , true );
-
- exec( "_mongo = new Mongo();" );
- exec( ((string)"db = _mongo.getDB( \"" + dbName + "\" ); ").c_str() );
-
- _localConnect = true;
- _localDBName = dbName;
- }
- loadStored();
- }
-
- // ----- getters ------
- double getNumber( const char *field ) {
- smlock;
- jsval val;
- assert( JS_GetProperty( _context , _global , field , &val ) );
- return _convertor->toNumber( val );
- }
-
- string getString( const char *field ) {
- smlock;
- jsval val;
- assert( JS_GetProperty( _context , _global , field , &val ) );
- JSString * s = JS_ValueToString( _context , val );
- return _convertor->toString( s );
- }
-
- bool getBoolean( const char *field ) {
- smlock;
- return _convertor->getBoolean( _global , field );
- }
-
- BSONObj getObject( const char *field ) {
- smlock;
- return _convertor->toObject( _convertor->getProperty( _global , field ) );
- }
-
- JSObject * getJSObject( const char * field ) {
- smlock;
- return _convertor->getJSObject( _global , field );
- }
-
- int type( const char *field ) {
- smlock;
- jsval val;
- assert( JS_GetProperty( _context , _global , field , &val ) );
-
- switch ( JS_TypeOfValue( _context , val ) ) {
- case JSTYPE_VOID: return Undefined;
- case JSTYPE_NULL: return jstNULL;
- case JSTYPE_OBJECT: {
- if ( val == JSVAL_NULL )
- return jstNULL;
- JSObject * o = JSVAL_TO_OBJECT( val );
- if ( JS_IsArrayObject( _context , o ) )
- return Array;
- if ( isDate( _context , o ) )
- return Date;
- return Object;
- }
- case JSTYPE_FUNCTION: return Code;
- case JSTYPE_STRING: return String;
- case JSTYPE_NUMBER: return NumberDouble;
- case JSTYPE_BOOLEAN: return Bool;
- default:
- uassert( 10227 , "unknown type" , 0 );
- }
- return 0;
- }
-
- // ----- setters ------
-
- void setElement( const char *field , const BSONElement& val ) {
- smlock;
- jsval v = _convertor->toval( val );
- assert( JS_SetProperty( _context , _global , field , &v ) );
- }
-
- void setNumber( const char *field , double val ) {
- smlock;
- jsval v = _convertor->toval( val );
- assert( JS_SetProperty( _context , _global , field , &v ) );
- }
-
- void setString( const char *field , const char * val ) {
- smlock;
- jsval v = _convertor->toval( val );
- assert( JS_SetProperty( _context , _global , field , &v ) );
- }
-
- void setObject( const char *field , const BSONObj& obj , bool readOnly ) {
- smlock;
- jsval v = _convertor->toval( &obj , readOnly );
- JS_SetProperty( _context , _global , field , &v );
- }
-
- void setBoolean( const char *field , bool val ) {
- smlock;
- jsval v = BOOLEAN_TO_JSVAL( val );
- assert( JS_SetProperty( _context , _global , field , &v ) );
- }
-
- void setThis( const BSONObj * obj ) {
- smlock;
- if ( _this ) {
- JS_RemoveRoot( _context , &_this );
- _this = 0;
- }
-
- if ( obj ) {
- _this = _convertor->toJSObject( obj );
- JS_AddNamedRoot( _context , &_this , "scope this" );
- }
- }
-
- void setFunction( const char *field , const char * code ) {
- smlock;
- jsval v = OBJECT_TO_JSVAL(JS_GetFunctionObject(_convertor->compileFunction(code)));
- JS_SetProperty( _context , _global , field , &v );
- }
-
- void rename( const char * from , const char * to ) {
- smlock;
- jsval v;
- assert( JS_GetProperty( _context , _global , from , &v ) );
- assert( JS_SetProperty( _context , _global , to , &v ) );
- v = JSVAL_VOID;
- assert( JS_SetProperty( _context , _global , from , &v ) );
- }
-
- // ---- functions -----
-
- ScriptingFunction _createFunction( const char * code ) {
- smlock;
- precall();
- return (ScriptingFunction)_convertor->compileFunction( code );
- }
-
- struct TimeoutSpec {
- boost::posix_time::ptime start;
- boost::posix_time::time_duration timeout;
- int count;
- };
-
- // should not generate exceptions, as those can be caught in
- // javascript code; returning false without an exception exits
- // immediately
- static JSBool _interrupt( JSContext *cx ) {
- TimeoutSpec &spec = *(TimeoutSpec *)( JS_GetContextPrivate( cx ) );
- if ( ++spec.count % 1000 != 0 )
- return JS_TRUE;
- const char * interrupt = ScriptEngine::checkInterrupt();
- if ( interrupt && interrupt[ 0 ] ) {
- return JS_FALSE;
- }
- if ( spec.timeout.ticks() == 0 ) {
- return JS_TRUE;
- }
- boost::posix_time::time_duration elapsed = ( boost::posix_time::microsec_clock::local_time() - spec.start );
- if ( elapsed < spec.timeout ) {
- return JS_TRUE;
- }
- return JS_FALSE;
-
- }
-
- static JSBool interrupt( JSContext *cx, JSScript *script ) {
- return _interrupt( cx );
- }
-
- void installInterrupt( int timeoutMs ) {
- if ( timeoutMs != 0 || ScriptEngine::haveCheckInterruptCallback() ) {
- TimeoutSpec *spec = new TimeoutSpec;
- spec->timeout = boost::posix_time::millisec( timeoutMs );
- spec->start = boost::posix_time::microsec_clock::local_time();
- spec->count = 0;
- JS_SetContextPrivate( _context, (void*)spec );
-#if defined(SM181) && !defined(XULRUNNER190)
- JS_SetOperationCallback( _context, _interrupt );
-#else
- JS_SetBranchCallback( _context, interrupt );
-#endif
- }
- }
-
- void uninstallInterrupt( int timeoutMs ) {
- if ( timeoutMs != 0 || ScriptEngine::haveCheckInterruptCallback() ) {
-#if defined(SM181) && !defined(XULRUNNER190)
- JS_SetOperationCallback( _context , 0 );
-#else
- JS_SetBranchCallback( _context, 0 );
-#endif
- delete (TimeoutSpec *)JS_GetContextPrivate( _context );
- JS_SetContextPrivate( _context, 0 );
- }
- }
-
- void precall() {
- _error = "";
- currentScope.reset( this );
- }
-
- bool exec( const StringData& code , const string& name = "(anon)" , bool printResult = false , bool reportError = true , bool assertOnError = true, int timeoutMs = 0 ) {
- smlock;
- precall();
-
- jsval ret = JSVAL_VOID;
-
- installInterrupt( timeoutMs );
- JSBool worked = JS_EvaluateScript( _context , _global , code.data() , code.size() , name.c_str() , 1 , &ret );
- uninstallInterrupt( timeoutMs );
-
- if ( ! worked && _error.size() == 0 ) {
- jsval v;
- if ( JS_GetPendingException( _context , &v ) ) {
- _error = _convertor->toString( v );
- if ( reportError )
- cout << _error << endl;
- }
- }
-
- uassert( 10228 , str::stream() << name + " exec failed: " << _error , worked || ! assertOnError );
-
- if ( reportError && ! _error.empty() ) {
- // cout << "exec error: " << _error << endl;
- // already printed in reportError, so... TODO
- }
-
- if ( worked )
- _convertor->setProperty( _global , "__lastres__" , ret );
-
- if ( worked && printResult && ! JSVAL_IS_VOID( ret ) )
- cout << _convertor->toString( ret ) << endl;
-
- return worked;
- }
-
- int invoke( JSFunction * func , const BSONObj* args, const BSONObj* recv, int timeoutMs , bool ignoreReturn, bool readOnlyArgs, bool readOnlyRecv ) {
- smlock;
- precall();
-
- assert( JS_EnterLocalRootScope( _context ) );
-
- int nargs = args ? args->nFields() : 0;
- scoped_array<jsval> smargsPtr( new jsval[nargs] );
- if ( nargs ) {
- BSONObjIterator it( *args );
- for ( int i=0; i<nargs; i++ ) {
- smargsPtr[i] = _convertor->toval( it.next() );
- }
- }
-
- if ( !args ) {
- _convertor->setProperty( _global , "args" , JSVAL_NULL );
- }
- else {
- setObject( "args" , *args , true ); // this is for backwards compatability
- }
-
- JS_LeaveLocalRootScope( _context );
-
- installInterrupt( timeoutMs );
- jsval rval;
- setThis(recv);
- JSBool ret = JS_CallFunction( _context , _this ? _this : _global , func , nargs , smargsPtr.get() , &rval );
- setThis(0);
- uninstallInterrupt( timeoutMs );
-
- if ( !ret ) {
- return -3;
- }
-
- if ( ! ignoreReturn ) {
- assert( JS_SetProperty( _context , _global , "return" , &rval ) );
- }
-
- return 0;
- }
-
- int invoke( ScriptingFunction funcAddr , const BSONObj* args, const BSONObj* recv, int timeoutMs = 0 , bool ignoreReturn = 0, bool readOnlyArgs = false, bool readOnlyRecv = false ) {
- return invoke( (JSFunction*)funcAddr , args , recv, timeoutMs , ignoreReturn, readOnlyArgs, readOnlyRecv);
- }
-
- void gotError( string s ) {
- _error = s;
- }
-
- string getError() {
- return _error;
- }
-
- void injectNative( const char *field, NativeFunction func, void* data ) {
- smlock;
- string name = field;
- _convertor->setProperty( _global , (name + "_").c_str() , _convertor->toval( (double)(long long)func ) );
-
- stringstream code;
- if (data) {
- _convertor->setProperty( _global , (name + "_data_").c_str() , _convertor->toval( (double)(long long)data ) );
- code << field << "_" << " = { x : " << field << "_ , y: " << field << "_data_ }; ";
- } else {
- code << field << "_" << " = { x : " << field << "_ }; ";
- }
- code << field << " = function(){ return nativeHelper.apply( " << field << "_ , arguments ); }";
- exec( code.str() );
- }
-
- virtual void gc() {
- smlock;
- JS_GC( _context );
- }
-
- JSContext *SavedContext() const { return _context; }
-
- private:
-
- void _postCreateHacks() {
-#ifdef XULRUNNER
- exec( "__x__ = new Date(1);" );
- globalSMEngine->_dateClass = _convertor->getClass( _global , "__x__" );
- exec( "__x__ = /abc/i" );
- globalSMEngine->_regexClass = _convertor->getClass( _global , "__x__" );
-#endif
- }
-
- JSContext * _context;
- Convertor * _convertor;
-
- JSObject * _global;
- JSObject * _this;
-
- string _error;
- list<void*> _roots;
-
- bool _externalSetup;
- bool _localConnect;
-
- set<string> _initFieldNames;
-
- };
-
- /* used to make the logging not overly chatty in the mongo shell. */
- extern bool isShell;
-
- void errorReporter( JSContext *cx, const char *message, JSErrorReport *report ) {
- stringstream ss;
- if( !isShell )
- ss << "JS Error: ";
- ss << message;
-
- if ( report && report->filename ) {
- ss << " " << report->filename << ":" << report->lineno;
- }
-
- tlog() << ss.str() << endl;
-
- if ( currentScope.get() ) {
- currentScope->gotError( ss.str() );
- }
- }
-
- JSBool native_load( JSContext *cx , JSObject *obj , uintN argc, jsval *argv , jsval *rval ) {
- Convertor c(cx);
-
- Scope * s = currentScope.get();
-
- for ( uintN i=0; i<argc; i++ ) {
- string filename = c.toString( argv[i] );
- //cout << "load [" << filename << "]" << endl;
-
- if ( ! s->execFile( filename , false , true , false ) ) {
- JS_ReportError( cx , ((string)"error loading js file: " + filename ).c_str() );
- return JS_FALSE;
- }
- }
-
- return JS_TRUE;
- }
-
-
-
- void SMEngine::runTest() {
- SMScope s;
-
- s.localConnect( "foo" );
-
- s.exec( "assert( db.getMongo() )" );
- s.exec( "assert( db.bar , 'collection getting does not work' ); " );
- s.exec( "assert.eq( db._name , 'foo' );" );
- s.exec( "assert( _mongo == db.getMongo() ); " );
- s.exec( "assert( _mongo == db._mongo ); " );
- s.exec( "assert( typeof DB.bar == 'undefined' ); " );
- s.exec( "assert( typeof DB.prototype.bar == 'undefined' , 'resolution is happening on prototype, not object' ); " );
-
- s.exec( "assert( db.bar ); " );
- s.exec( "assert( typeof db.addUser == 'function' )" );
- s.exec( "assert( db.addUser == DB.prototype.addUser )" );
- s.exec( "assert.eq( 'foo.bar' , db.bar._fullName ); " );
- s.exec( "db.bar.verify();" );
-
- s.exec( "db.bar.silly.verify();" );
- s.exec( "assert.eq( 'foo.bar.silly' , db.bar.silly._fullName )" );
- s.exec( "assert.eq( 'function' , typeof _mongo.find , 'mongo.find is not a function' )" );
-
- assert( (string)"abc" == trim( "abc" ) );
- assert( (string)"abc" == trim( " abc" ) );
- assert( (string)"abc" == trim( "abc " ) );
- assert( (string)"abc" == trim( " abc " ) );
-
- }
-
- Scope * SMEngine::createScope() {
- return new SMScope();
- }
-
- void Convertor::addRoot( JSFunction * f , const char * name ) {
- if ( ! f )
- return;
-
- SMScope * scope = currentScope.get();
- uassert( 10229 , "need a scope" , scope );
-
- JSObject * o = JS_GetFunctionObject( f );
- assert( o );
- scope->addRoot( &o , name );
- }
-
-}
-
-#include "sm_db.cpp"