I'm trying to call createTableIfNotExists
in this npm package, and do so synchronously in Meteor, server-side.
https://www.npmjs.com/package/azure-storage
However, the callback signature is of type function(error, result, response)
instead of the traditional function(error,result)
.
1) Because of that, I cannot use Meteor.wrapAsync
and instead have to use Meteor.bindEnvironment
2) I call 'bindEnvironment' as below. Note the callback with 3 arguments. This works, but now I would like to extract the return value
, back to the original method (i.e. the original fiber).
Note that simply defining 'addResult' outside the createTableService
does not work because the callback inside bindEnvironment
runs asynchrounously relative to the outside code...i.e. demoFunction()
returns before the callback sets addResult
.
var demoFunction = function(){
var addResult = null;
var tableService = azure.createTableService(acctName, acctKey);
tableService.createTableIfNotExists(asTableName, Meteor.bindEnvironment(function(error,result,response){
if (error) {
throw new Meteor.Error(500, "table create error - " + asTableName + ": "+ error);
}
console.log(result);
if(result){
addResult = /*call to meteor method*/
return addResult;
}
}));
return addResult; //this is what I would like to do, but will always be null, as written.
}
How can I call createTableIfNotExists
and still return addResult
back to the function that called demoFunction()
?
Thanks!
You can use the Future
from the fibers/future
(node-fibers) package, which is just a different abstraction layer of fibers as this async/await implementation or this promise implementation are,
too. (Note, that using a fiber directly is not recommended).
Generic Future pattern
As you already pointed out, you can solve this issue also using async/await or using promises.
However, if you want to know how to use a Future
instance, you can follow this popular pattern, which I make use of in many of my applications.
Future pattern with Meteor.bindEnvironment
Including the bindEnvironment
into this pattern, and restructuring the code a bit, it looks like the following:
import Future from 'fibers/future';
Meteor.methods({
myMeteorMethod: function() {
// load Future
const myFuture = new Future();
// create bound callback, that uses the future
const boundCallback = Meteor.bindEnvironment(function (error,results){
if(error){
myFuture.throw(error);
}else{
myFuture.return(results);
}
});
// call the function and pass bound callback
SomeAsynchronousFunction("foo", boundCallback);
return myFuture.wait();
}
});
Applied to code example
Applying this modification of the pattern to your code, it could result in something like the code below.
Note, that I separated the callback from being created inside the function simply because of readability. You can of course just create it inside the function head, as you typically do.
import Future from 'fibers/future';
var demoFunction = function(){
// load Future
const myFuture = new Future();
// create bound callback, that uses the future
const boundCallback = Meteor.bindEnvironment(function (error, result, response) {
if (error) {
myFuture.throw(new Meteor.Error(500, "table create error - " + asTableName + ": "+ error));
}
// do whatever you need to do here
// with result and response...
if (result) {
myFuture.return(result);
}
});
var tableService = azure.createTableService(acctName, acctKey);
// pass the bound callback here
tableService.createTableIfNotExists(asTableName, boundCallback);
return myFuture.wait();
}