Skip to content

Commit

Permalink
Enhance asyncOptions to provide control over sync & async method vari…
Browse files Browse the repository at this point in the history
…ants.

Generalizes asyncOptions to allow changing the suffix used for sync and
async style methods, and also allow any of them to be omitted.
  • Loading branch information
jimlloyd committed Mar 13, 2015
1 parent a407d90 commit 7280eb9
Show file tree
Hide file tree
Showing 11 changed files with 481 additions and 26 deletions.
73 changes: 62 additions & 11 deletions src/java.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -87,6 +87,12 @@ NAN_METHOD(Java::New) {
Java::Java() {
this->m_jvm = NULL;
this->m_env = NULL;

m_SyncSuffix = "Sync";
m_AsyncSuffix = "";
doSync = true;
doAsync = true;
doPromise = false;
}

Java::~Java() {
Expand All @@ -95,28 +101,73 @@ Java::~Java() {

v8::Local<v8::Value> Java::ensureJvm() {
if(!m_jvm) {
return createJVM(&this->m_jvm, &this->m_env);
v8::Local<v8::Value> result = createJVM(&this->m_jvm, &this->m_env);
assert(result->IsNull());
return result;
}

return NanNull();
}

void Java::configureAsync(v8::Local<v8::Value>& asyncOptions) {
v8::Local<v8::Object> asyncOptionsObj = asyncOptions.As<v8::Object>();

m_SyncSuffix = "invalid";
m_AsyncSuffix = "invalid";
m_PromiseSuffix = "invalid";
doSync = false;
doAsync = false;
doPromise = false;

v8::Local<v8::Value> suffixValue = asyncOptionsObj->Get(NanNew<v8::String>("syncSuffix"));
if (suffixValue->IsString()) {
v8::Local<v8::String> suffix = suffixValue->ToString();
v8::String::Utf8Value utf8(suffix);
m_SyncSuffix.assign(*utf8);
doSync = true;
}

suffixValue = asyncOptionsObj->Get(NanNew<v8::String>("asyncSuffix"));
if (suffixValue->IsString()) {
v8::Local<v8::String> suffix = suffixValue->ToString();
v8::String::Utf8Value utf8(suffix);
m_AsyncSuffix.assign(*utf8);
doAsync = true;
}

suffixValue = asyncOptionsObj->Get(NanNew<v8::String>("promiseSuffix"));
if (suffixValue->IsString()) {
v8::Local<v8::String> suffix = suffixValue->ToString();
v8::String::Utf8Value utf8(suffix);
m_PromiseSuffix.assign(*utf8);
v8::Local<v8::Value> promisify = asyncOptionsObj->Get(NanNew<v8::String>("promisify"));
if (!promisify->IsFunction()) {
fprintf(stderr, "asyncOptions.promisify must be a function");
assert(promisify->IsFunction());
}
doPromise = true;
}

if (doSync && doAsync) {
assert(m_SyncSuffix != m_AsyncSuffix);
}
if (doSync && doPromise) {
assert(m_SyncSuffix != m_PromiseSuffix);
}
if (doAsync && doPromise) {
assert(m_AsyncSuffix != m_PromiseSuffix);
}

NanAssignPersistent(m_asyncOptions, asyncOptionsObj);
}

v8::Local<v8::Value> Java::createJVM(JavaVM** jvm, JNIEnv** env) {
JavaVM* jvmTemp;
JavaVMInitArgs args;

v8::Local<v8::Value> asyncOptions = NanObjectWrapHandle(this)->Get(NanNew<v8::String>("asyncOptions"));
if (asyncOptions->IsObject()) {
v8::Local<v8::Object> asyncOptionsObj = asyncOptions.As<v8::Object>();
v8::Local<v8::Value> promisify = asyncOptionsObj->Get(NanNew<v8::String>("promisify"));
if (!promisify->IsFunction()) {
return NanTypeError("asyncOptions.promisify must be a function");
}
v8::Local<v8::Value> suffix = asyncOptionsObj->Get(NanNew<v8::String>("promiseSuffix"));
if (!suffix->IsString()) {
return NanTypeError("asyncOptions.promiseSuffix must be a string");
}
NanAssignPersistent(m_asyncOptions, asyncOptionsObj);
configureAsync(asyncOptions);
}

// setup classpath
Expand Down
17 changes: 17 additions & 0 deletions src/java.h
Original file line number Diff line number Diff line change
Expand Up @@ -21,11 +21,20 @@ class Java : public node::ObjectWrap {
JNIEnv* getJavaEnv() { return m_env; } // can only be used safely by the main thread as this is the thread it belongs to
jobject getClassLoader() { return m_classLoader; }

public:
bool DoSync() const { return doSync; }
bool DoAsync() const { return doAsync; }
bool DoPromise() const { return doPromise; }
std::string SyncSuffix() const { return m_SyncSuffix; }
std::string AsyncSuffix() const { return m_AsyncSuffix; }
std::string PromiseSuffix() const { return m_PromiseSuffix; }

private:
Java();
~Java();
v8::Local<v8::Value> createJVM(JavaVM** jvm, JNIEnv** env);
void destroyJVM(JavaVM** jvm, JNIEnv** env);
void configureAsync(v8::Local<v8::Value>& asyncOptions);

static NAN_METHOD(New);
static NAN_METHOD(getClassLoader);
Expand Down Expand Up @@ -60,6 +69,14 @@ class Java : public node::ObjectWrap {
v8::Persistent<v8::Array> m_classPathArray;
v8::Persistent<v8::Array> m_optionsArray;
v8::Persistent<v8::Object> m_asyncOptions;

std::string m_SyncSuffix;
std::string m_AsyncSuffix;
std::string m_PromiseSuffix;

bool doSync;
bool doAsync;
bool doPromise;
};

#endif
26 changes: 11 additions & 15 deletions src/javaObject.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -27,17 +27,11 @@
std::replace(className.begin(), className.end(), '[', 'a');
className = "nodeJava_" + className;

// Set up promisification
v8::Local<v8::Object> asyncOptions = NanObjectWrapHandle(java)->Get(NanNew<v8::String>("asyncOptions")).As<v8::Object>();
v8::Local<v8::Function> promisify;
std::string promiseSuffix;
bool promisifying = asyncOptions->IsObject();
if(promisifying) {
if(java->DoPromise()) {
v8::Local<v8::Object> asyncOptions = NanObjectWrapHandle(java)->Get(NanNew<v8::String>("asyncOptions")).As<v8::Object>();
v8::Local<v8::Value> promisifyValue = asyncOptions->Get(NanNew<v8::String>("promisify"));
promisify = promisifyValue.As<v8::Function>();
v8::Local<v8::String> suffix = asyncOptions->Get(NanNew<v8::String>("promiseSuffix"))->ToString();
v8::String::Utf8Value utf8(suffix);
promiseSuffix.assign(*utf8);
}

v8::Local<v8::FunctionTemplate> funcTemplate;
Expand All @@ -60,15 +54,17 @@
assert(!env->ExceptionCheck());
std::string methodNameStr = javaToString(env, methodNameJava);

v8::Handle<v8::String> methodName = NanNew<v8::String>(methodNameStr.c_str());
v8::Local<v8::FunctionTemplate> methodCallTemplate = NanNew<v8::FunctionTemplate>(methodCall, methodName);
funcTemplate->PrototypeTemplate()->Set(methodName, methodCallTemplate->GetFunction());
v8::Handle<v8::String> baseMethodName = NanNew<v8::String>(methodNameStr.c_str());

v8::Handle<v8::String> methodNameSync = NanNew<v8::String>((methodNameStr + "Sync").c_str());
v8::Local<v8::FunctionTemplate> methodCallSyncTemplate = NanNew<v8::FunctionTemplate>(methodCallSync, methodName);
v8::Handle<v8::String> methodNameAsync = NanNew<v8::String>((methodNameStr + java->AsyncSuffix()).c_str());
v8::Local<v8::FunctionTemplate> methodCallTemplate = NanNew<v8::FunctionTemplate>(methodCall, baseMethodName);
funcTemplate->PrototypeTemplate()->Set(methodNameAsync, methodCallTemplate->GetFunction());

v8::Handle<v8::String> methodNameSync = NanNew<v8::String>((methodNameStr + java->SyncSuffix()).c_str());
v8::Local<v8::FunctionTemplate> methodCallSyncTemplate = NanNew<v8::FunctionTemplate>(methodCallSync, baseMethodName);
funcTemplate->PrototypeTemplate()->Set(methodNameSync, methodCallSyncTemplate->GetFunction());

if (promisifying) {
if (java->DoPromise()) {
v8::Local<v8::Object> recv = NanNew<v8::Object>();
v8::Local<v8::Value> argv[] = { methodCallTemplate->GetFunction() };
v8::Local<v8::Value> result = promisify->Call(recv, 1, argv);
Expand All @@ -77,7 +73,7 @@
assert(result->IsFunction());
}
v8::Local<v8::Function> promFunction = result.As<v8::Function>();
v8::Handle<v8::String> methodNamePromise = NanNew<v8::String>((methodNameStr + promiseSuffix).c_str());
v8::Handle<v8::String> methodNamePromise = NanNew<v8::String>((methodNameStr + java->PromiseSuffix()).c_str());
funcTemplate->PrototypeTemplate()->Set(methodNamePromise, promFunction);
}
}
Expand Down
67 changes: 67 additions & 0 deletions testAsyncOptions/testAllThreeSuffix.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
// testAllThreeSuffix.js

// All three variants have non-empty suffix, i.e a suffix is required for any variant.

var java = require("../");
var assert = require("assert");
var _ = require('lodash');

java.asyncOptions = {
syncSuffix: "Sync",
asyncSuffix: "Async",
promiseSuffix: 'Promise',
promisify: require('when/node').lift // https://github.com/cujojs/when
};

module.exports = {
testAPI: function(test) {
test.expect(6);
var arrayList = java.newInstanceSync("java.util.ArrayList");
test.ok(arrayList);
test.ok(java.instanceOf(arrayList, "java.util.ArrayList"));

var api = _.functions(arrayList);
test.ok(_.includes(api, 'addSync'), 'Expected `addSync` to be present, but it is NOT.');
test.ok(_.includes(api, 'addAsync'), 'Expected `addAsync` to be present, but it is NOT.');
test.ok(_.includes(api, 'addPromise'), 'Expected addPromise to be present, but it is NOT.');
test.ok(!_.includes(api, 'add'), 'Expected add to NOT be present, but it is.');
test.done();
},

testSyncCalls: function(test) {
test.expect(1);
var arrayList = java.newInstanceSync("java.util.ArrayList");
arrayList.addSync("hello");
arrayList.addSync("world");
test.strictEqual(arrayList.sizeSync(), 2);
test.done();
},

testAsyncCalls: function(test) {
test.expect(4);
var arrayList = java.newInstanceSync("java.util.ArrayList");
arrayList.addAsync("hello", function(err, result) {
test.ifError(err);
arrayList.addAsync("world", function(err, result) {
test.ifError(err);
arrayList.sizeAsync(function(err, size) {
test.ifError(err);
test.strictEqual(size, 2);
test.done();
});
});
});
},

testPromiseCalls: function(test) {
test.expect(1);
var arrayList = java.newInstanceSync("java.util.ArrayList");
arrayList.addPromise("hello")
.then(function () { return arrayList.addPromise("world"); })
.then(function () { return arrayList.sizePromise(); })
.then(function (size) {
test.strictEqual(size, 2);
test.done();
});
}
}
52 changes: 52 additions & 0 deletions testAsyncOptions/testAsyncSuffixSyncDefault.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
// testAsyncSuffixSyncDefault.js

// Use "Async" for the asyncSuffix, and "" for the syncSuffix.

var java = require("../");
var assert = require("assert");
var _ = require('lodash');

java.asyncOptions = {
syncSuffix: "",
asyncSuffix: "Async"
};

module.exports = {
testAPI: function(test) {
test.expect(5);
var arrayList = java.newInstanceSync("java.util.ArrayList");
test.ok(arrayList);
test.ok(java.instanceOf(arrayList, "java.util.ArrayList"));

var api = _.functions(arrayList);
test.ok(_.includes(api, 'addAsync'), 'Expected `addAsync` to be present, but it is NOT.');
test.ok(_.includes(api, 'add'), 'Expected `add` to be present, but it is NOT.');
test.ok(!_.includes(api, 'addPromise'), 'Expected addPromise to NOT be present, but it is.');
test.done();
},

testSyncCalls: function(test) {
test.expect(1);
var arrayList = java.newInstanceSync("java.util.ArrayList");
arrayList.add("hello");
arrayList.add("world");
test.strictEqual(arrayList.size(), 2);
test.done();
},

testAsyncCalls: function(test) {
test.expect(4);
var arrayList = java.newInstanceSync("java.util.ArrayList");
arrayList.addAsync("hello", function(err, result) {
test.ifError(err);
arrayList.addAsync("world", function(err, result) {
test.ifError(err);
arrayList.sizeAsync(function(err, size) {
test.ifError(err);
test.strictEqual(size, 2);
test.done();
});
});
});
}
}
52 changes: 52 additions & 0 deletions testAsyncOptions/testDefacto.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
// testDefacto.js

// In the defacto case, the developer sets asyncOptions, but specifies the defacto standard behavior.

var _ = require('lodash');
var java = require("../");
var nodeunit = require("nodeunit");

java.asyncOptions = {
syncSuffix: "Sync",
asyncSuffix: ""
};

module.exports = {
testAPI: function(test) {
test.expect(5);
var arrayList = java.newInstanceSync("java.util.ArrayList");
test.ok(arrayList);
test.ok(java.instanceOf(arrayList, "java.util.ArrayList"));

var api = _.functions(arrayList);
test.ok(_.includes(api, 'addSync'), 'Expected `addSync` to be present, but it is NOT.');
test.ok(_.includes(api, 'add'), 'Expected `add` to be present, but it is NOT.');
test.ok(!_.includes(api, 'addPromise'), 'Expected addPromise to NOT be present, but it is.');
test.done();
},

testSyncCalls: function(test) {
test.expect(1);
var arrayList = java.newInstanceSync("java.util.ArrayList");
arrayList.addSync("hello");
arrayList.addSync("world");
test.strictEqual(arrayList.sizeSync(), 2);
test.done();
},

testAsyncCalls: function(test) {
test.expect(4);
var arrayList = java.newInstanceSync("java.util.ArrayList");
arrayList.add("hello", function(err, result) {
test.ifError(err);
arrayList.add("world", function(err, result) {
test.ifError(err);
arrayList.size(function(err, size) {
test.ifError(err);
test.strictEqual(size, 2);
test.done();
});
});
});
}
}
Loading

0 comments on commit 7280eb9

Please sign in to comment.