39

我想在 Meteor 方法中调用一个异步函数,然后将该函数的结果返回给 Meteor.call。

(如何)这可能吗?

Meteor.methods({
  my_function: function(arg1, arg2) {
    //Call other asynchronous function and return result or throw error
  }
});
4

4 回答 4

36

使用 Future 来做到这一点。像这样:

Meteor.methods({
  my_function: function(arg1, arg2) {

    // Set up a future
    var fut = new Future();

    // This should work for any async method
    setTimeout(function() {

      // Return the results
      fut.ret(message + " (delayed for 3 seconds)");

    }, 3 * 1000);

    // Wait for async to finish before returning
    // the result
    return fut.wait();
  }
});

更新

要从 Meteor 0.5.1 开始使用 Future,您必须在 Meteor.startup 方法中运行以下代码:

Meteor.startup(function () {
  var require = __meteor_bootstrap__.require
  Future = require('fibers/future');

  // use Future here
});

  更新 2

要从 Meteor 0.6 开始使用 Future,您必须在 Meteor.startup 方法中运行以下代码:

Meteor.startup(function () {
  Future = Npm.require('fibers/future');

  // use Future here
});

然后使用return方法而不是ret方法:

Meteor.methods({
  my_function: function(arg1, arg2) {

    // Set up a future
    var fut = new Future();

    // This should work for any async method
    setTimeout(function() {

      // Return the results
      fut['return'](message + " (delayed for 3 seconds)");

    }, 3 * 1000);

    // Wait for async to finish before returning
    // the result
    return fut.wait();
  }
});

请参阅此要点

于 2012-09-25T17:14:50.810 回答
26

最近版本的 Meteor 提供了未记录Meteor._wrapAsync的函数,它将带有标准(err, res)回调的函数转换为同步函数,这意味着当前的 Fiber 产生直到回调返回,然后使用 Meteor.bindEnvironment 来确保您保留当前的 ​​Meteor 环境变量(比如Meteor.userId())

一个简单的用法如下:

asyncFunc = function(arg1, arg2, callback) {
  // callback has the form function (err, res) {}

};

Meteor.methods({
  "callFunc": function() {
     syncFunc = Meteor._wrapAsync(asyncFunc);

     res = syncFunc("foo", "bar"); // Errors will be thrown     
  }
});

您可能还需要使用function#bind确保asyncFunc在包装之前使用正确的上下文调用它。

有关更多信息,请参阅:https ://www.eventedmind.com/tracks/feed-archive/meteor-meteor-wrapasync

于 2014-02-04T03:21:37.333 回答
19

安德鲁·毛是对的。Meteor 现在有Meteor.wrapAsync()用于这种情况。

这是通过条带进行收费并传递回调函数的最简单方法:

var stripe = StripeAPI("key");    
Meteor.methods({

    yourMethod: function(callArg) {

        var charge = Meteor.wrapAsync(stripe.charges.create, stripe.charges);
        charge({
            amount: amount,
            currency: "usd",
            //I passed the stripe token in callArg
            card: callArg.stripeToken,
        }, function(err, charge) {
            if (err && err.type === 'StripeCardError') {
              // The card has been declined
              throw new Meteor.Error("stripe-charge-error", err.message);
            }

            //Insert your 'on success' code here

        });
    }
});

我发现这篇文章真的很有帮助: Meteor: Proper use of Meteor.wrapAsync on server

于 2015-01-07T06:56:53.583 回答
5

另一种选择是实现类似目标的这个包。

meteor add meteorhacks:async

从包自述文件中:

异步包装(函数)

包装一个异步函数并允许它在 Meteor 中运行而无需回调。

//declare a simple async function
function delayedMessge(delay, message, callback) {
  setTimeout(function() {
    callback(null, message);
  }, delay);
}

//wrapping
var wrappedDelayedMessage = Async.wrap(delayedMessge);

//usage
Meteor.methods({
  'delayedEcho': function(message) {
    var response = wrappedDelayedMessage(500, message);
    return response;
  }
});
于 2015-03-05T04:38:39.693 回答