summaryrefslogtreecommitdiff
path: root/scripting
diff options
context:
space:
mode:
authoragirbal <antoine@10gen.com>2011-10-07 00:33:51 -0700
committeragirbal <antoine@10gen.com>2011-10-07 00:33:51 -0700
commite03ce18ac0ddce659e808256bedcbb4acd8fb080 (patch)
treea70b2c978aa3fd4d079417c8b8565b8ab2905335 /scripting
parentd312c1abe458919c977cbb82ba3970f8af456478 (diff)
downloadmongo-e03ce18ac0ddce659e808256bedcbb4acd8fb080.tar.gz
- SERVER-2751: BinData Asserts and crashes shell
Diffstat (limited to 'scripting')
-rw-r--r--scripting/v8_db.cpp125
1 files changed, 52 insertions, 73 deletions
diff --git a/scripting/v8_db.cpp b/scripting/v8_db.cpp
index a8006987c1b..3e5b58b14ed 100644
--- a/scripting/v8_db.cpp
+++ b/scripting/v8_db.cpp
@@ -40,6 +40,16 @@ namespace mongo {
#define DDD(x)
+ static v8::Handle<v8::Value> newInstance( v8::Function* f, const v8::Arguments& args ) {
+ // need to translate arguments into an array
+ int argc = args.Length();
+ scoped_array< Handle<Value> > argv( new Handle<Value>[argc] );
+ for (int i = 0; i < argc; ++i) {
+ argv[i] = args[i];
+ }
+ return f->NewInstance(argc, argv.get());
+ }
+
v8::Handle<v8::FunctionTemplate> getMongoFunctionTemplate( V8Scope* scope, bool local ) {
v8::Handle<v8::FunctionTemplate> mongo;
if ( local ) {
@@ -99,31 +109,16 @@ namespace mongo {
v8::Handle<v8::FunctionTemplate> getUUIDFunctionTemplate(V8Scope* scope) {
v8::Handle<v8::FunctionTemplate> templ = scope->createV8Function(uuidInit);
- templ->InstanceTemplate()->SetInternalFieldCount(1);
- v8::Local<v8::Template> proto = templ->PrototypeTemplate();
- scope->injectV8Function("toString", binDataToString, proto);
- scope->injectV8Function("base64", binDataToBase64, proto);
- scope->injectV8Function("hex", binDataToHex, proto);
return templ;
}
v8::Handle<v8::FunctionTemplate> getMD5FunctionTemplate(V8Scope* scope) {
v8::Handle<v8::FunctionTemplate> templ = scope->createV8Function(md5Init);
- templ->InstanceTemplate()->SetInternalFieldCount(1);
- v8::Local<v8::Template> proto = templ->PrototypeTemplate();
- scope->injectV8Function("toString", binDataToString, proto);
- scope->injectV8Function("base64", binDataToBase64, proto);
- scope->injectV8Function("hex", binDataToHex, proto);
return templ;
}
v8::Handle<v8::FunctionTemplate> getHexDataFunctionTemplate(V8Scope* scope) {
v8::Handle<v8::FunctionTemplate> templ = scope->createV8Function(hexDataInit);
- templ->InstanceTemplate()->SetInternalFieldCount(1);
- v8::Local<v8::Template> proto = templ->PrototypeTemplate();
- scope->injectV8Function("toString", binDataToString, proto);
- scope->injectV8Function("base64", binDataToBase64, proto);
- scope->injectV8Function("hex", binDataToHex, proto);
return templ;
}
@@ -679,10 +674,9 @@ namespace mongo {
v8::Handle<v8::Value> objectIdInit( V8Scope* scope, const v8::Arguments& args ) {
v8::Handle<v8::Object> it = args.This();
-
if ( it->IsUndefined() || it == v8::Context::GetCurrent()->Global() ) {
v8::Function * f = scope->getObjectIdCons();
- it = f->NewInstance();
+ return newInstance(f, args);
}
OID oid;
@@ -708,18 +702,16 @@ namespace mongo {
}
v8::Handle<v8::Value> dbRefInit( V8Scope* scope, const v8::Arguments& args ) {
+ v8::Handle<v8::Object> it = args.This();
+ if ( it->IsUndefined() || it == v8::Context::GetCurrent()->Global() ) {
+ v8::Function * f = scope->getNamedCons( "DBRef" );
+ return newInstance(f, args);
+ }
if (args.Length() != 2 && args.Length() != 0) {
return v8::ThrowException( v8::String::New( "DBRef needs 2 arguments" ) );
}
- v8::Handle<v8::Object> it = args.This();
-
- if ( it->IsUndefined() || it == v8::Context::GetCurrent()->Global() ) {
- v8::Function* f = scope->getNamedCons( "DBRef" );
- it = f->NewInstance();
- }
-
if ( args.Length() == 2 ) {
it->Set( scope->getV8Str( "$ref" ) , args[0] );
it->Set( scope->getV8Str( "$id" ) , args[1] );
@@ -729,18 +721,16 @@ namespace mongo {
}
v8::Handle<v8::Value> dbPointerInit( V8Scope* scope, const v8::Arguments& args ) {
+ v8::Handle<v8::Object> it = args.This();
+ if ( it->IsUndefined() || it == v8::Context::GetCurrent()->Global() ) {
+ v8::Function * f = scope->getNamedCons( "DBPointer" );
+ return newInstance(f, args);
+ }
if (args.Length() != 2) {
return v8::ThrowException( v8::String::New( "DBPointer needs 2 arguments" ) );
}
- v8::Handle<v8::Object> it = args.This();
-
- if ( it->IsUndefined() || it == v8::Context::GetCurrent()->Global() ) {
- v8::Function* f = scope->getNamedCons( "DBPointer" );
- it = f->NewInstance();
- }
-
it->Set( scope->getV8Str( "ns" ) , args[0] );
it->Set( scope->getV8Str( "id" ) , args[1] );
it->SetHiddenValue( scope->getV8Str( "__DBPointer" ), v8::Number::New( 1 ) );
@@ -749,8 +739,11 @@ namespace mongo {
}
v8::Handle<v8::Value> dbTimestampInit( V8Scope* scope, const v8::Arguments& args ) {
-
v8::Handle<v8::Object> it = args.This();
+ if ( it->IsUndefined() || it == v8::Context::GetCurrent()->Global() ) {
+ v8::Function * f = scope->getNamedCons( "Timestamp" );
+ return newInstance(f, args);
+ }
if ( args.Length() == 0 ) {
it->Set( scope->getV8Str( "t" ) , v8::Number::New( 0 ) );
@@ -772,6 +765,10 @@ namespace mongo {
v8::Handle<v8::Value> binDataInit( V8Scope* scope, const v8::Arguments& args ) {
v8::Local<v8::Object> it = args.This();
+ if ( it->IsUndefined() || it == v8::Context::GetCurrent()->Global() ) {
+ v8::Function* f = scope->getNamedCons( "BinData" );
+ return newInstance(f, args);
+ }
Handle<Value> type;
Handle<Value> len;
@@ -779,12 +776,6 @@ namespace mongo {
char* data;
if (args.Length() == 3) {
// 3 args: len, type, data
-
- if ( it->IsUndefined() || it == v8::Context::GetCurrent()->Global() ) {
- v8::Function* f = scope->getNamedCons( "BinData" );
- it = f->NewInstance();
- }
-
len = args[0];
rlen = len->IntegerValue();
type = args[1];
@@ -795,12 +786,6 @@ namespace mongo {
}
else if ( args.Length() == 2 ) {
// 2 args: type, base64 string
-
- if ( it->IsUndefined() || it == v8::Context::GetCurrent()->Global() ) {
- v8::Function* f = scope->getNamedCons( "BinData" );
- it = f->NewInstance();
- }
-
type = args[0];
v8::String::Utf8Value utf( args[ 1 ] );
string decoded = base64::decode( *utf );
@@ -810,8 +795,10 @@ namespace mongo {
memcpy(data, tmp, rlen);
len = v8::Number::New(rlen);
// it->Set( scope->getV8Str( "data" ), v8::String::New( decoded.data(), decoded.length() ) );
- }
- else {
+ } else if (args.Length() == 0) {
+ // this is called by subclasses that will fill properties
+ return it;
+ } else {
return v8::ThrowException( v8::String::New( "BinData needs 2 or 3 arguments" ) );
}
@@ -887,19 +874,11 @@ namespace mongo {
return v8::ThrowException( v8::String::New( "UUIS string must have 32 characters" ) );
}
- return hexToBinData(scope, args.This(), bdtUUID, *utf);
+ v8::Function * f = scope->getNamedCons("BinData");
+ Local<v8::Object> it = f->NewInstance();
+ return hexToBinData(scope, it, bdtUUID, *utf);
}
-// v8::Handle<v8::Value> uuidToString( V8Scope* scope, const v8::Arguments& args ) {
-// v8::Handle<v8::Object> it = args.This();
-// Local<External> c = External::Cast( *(it->GetInternalField( 0 )) );
-// char* data = (char*)(c->Value());
-//
-// stringstream ss;
-// ss << "UUID(\"" << toHex(data, 16) << "\")";
-// return v8::String::New( ss.str().c_str() );
-// }
-
v8::Handle<v8::Value> md5Init( V8Scope* scope, const v8::Arguments& args ) {
if (args.Length() != 1) {
return v8::ThrowException( v8::String::New( "MD5 needs 1 argument" ) );
@@ -909,7 +888,9 @@ namespace mongo {
return v8::ThrowException( v8::String::New( "MD5 string must have 32 characters" ) );
}
- return hexToBinData(scope, args.This(), MD5Type, *utf);
+ v8::Function * f = scope->getNamedCons("BinData");
+ Local<v8::Object> it = f->NewInstance();
+ return hexToBinData(scope, it, MD5Type, *utf);
}
v8::Handle<v8::Value> hexDataInit( V8Scope* scope, const v8::Arguments& args ) {
@@ -917,22 +898,22 @@ namespace mongo {
return v8::ThrowException( v8::String::New( "HexData needs 2 arguments" ) );
}
v8::String::Utf8Value utf( args[ 1 ] );
- return hexToBinData(scope, args.This(), args[0]->IntegerValue(), *utf);
+ v8::Function * f = scope->getNamedCons("BinData");
+ Local<v8::Object> it = f->NewInstance();
+ return hexToBinData(scope, it, args[0]->IntegerValue(), *utf);
}
v8::Handle<v8::Value> numberLongInit( V8Scope* scope, const v8::Arguments& args ) {
+ v8::Handle<v8::Object> it = args.This();
+ if ( it->IsUndefined() || it == v8::Context::GetCurrent()->Global() ) {
+ v8::Function * f = scope->getNamedCons( "NumberLong" );
+ return newInstance(f, args);
+ }
if (args.Length() != 0 && args.Length() != 1 && args.Length() != 3) {
return v8::ThrowException( v8::String::New( "NumberLong needs 0, 1 or 3 arguments" ) );
}
- v8::Handle<v8::Object> it = args.This();
-
- if ( it->IsUndefined() || it == v8::Context::GetCurrent()->Global() ) {
- v8::Function* f = scope->getNamedCons( "NumberLong" );
- it = f->NewInstance();
- }
-
if ( args.Length() == 0 ) {
it->Set( scope->getV8Str( "floatApprox" ), v8::Number::New( 0 ) );
}
@@ -1009,18 +990,16 @@ namespace mongo {
}
v8::Handle<v8::Value> numberIntInit( V8Scope* scope, const v8::Arguments& args ) {
+ v8::Handle<v8::Object> it = args.This();
+ if ( it->IsUndefined() || it == v8::Context::GetCurrent()->Global() ) {
+ v8::Function * f = scope->getNamedCons( "NumberInt" );
+ return newInstance(f, args);
+ }
if (args.Length() != 0 && args.Length() != 1) {
return v8::ThrowException( v8::String::New( "NumberInt needs 0, 1 argument" ) );
}
- v8::Handle<v8::Object> it = args.This();
-
- if ( it->IsUndefined() || it == v8::Context::GetCurrent()->Global() ) {
- v8::Function* f = scope->getNamedCons( "NumberInt" );
- it = f->NewInstance();
- }
-
if ( args.Length() == 0 ) {
it->SetHiddenValue( scope->V8STR_NUMBERINT, v8::Number::New( 0 ) );
}