已经调用了回调。但在此之前我从未打过电话回叫。 async.parallel中的一个错误?

时间:2017-07-12 08:51:57

标签: javascript node.js async.js

我在互联网上对此问题进行了大量研究,但仍无法知道问题的原因。

问题是,当我从异步库使用并行函数时,我收到一个错误:

Callback was already called.

这是我的代码:(我已经评论了我收到此错误的行)

var sendNotificationAddToMyRememberNew = function(request, response, restaurant) {

    async.parallel({
        notification_list: function (callback) {
            models.NotificationList.create({
                alert_msg: "Restaurant " + restaurant.name + " remembered.",
                payload: JSON.stringify({
                    restaurant_id: restaurant.id,
                    restaurant_name: restaurant.name,
                    restaurant_image: restaurant.image
                }),
                type: constants.NOTIFICATION_TYPE.RESTAURANT_REMEMBERED,
                platform_type: constants.MOBILE_PLATFORM_TYPE.ALL,
                status: constants.NOTIFICATION_STATUS.SENT,
                device_token: null,
                device_arn: null,
                schedule_time: moment.utc(),
                customer_id: request.token.customer_id,
                customer_email: request.token.email,
                created_date: moment.utc(),
                updated_date: moment.utc()
            }).then(function(notificationList){
                callback(null, notificationList);
            }).catch(function(error){
                callback(error);
            });
        },
        badge_count: function(callback) {

            models.CustomerBadgeCount.findOrCreate({
                where: {
                    customer_id: request.token.customer_id
                },
                defaults: {
                    badge_count: 1,
                    customer_id: request.token.customer_id,
                    created_date: moment.utc(),
                    updated_date: moment.utc()
                }
            }).spread(function (badgeCount, created) {
                if(!created){
                    badgeCount.update(
                        {
                            badge_count: badgeCount.badge_count + 1,
                            updated_date: moment.utc()
                        },
                        {
                            fields:[
                                "badge_count",
                                "updated_date"
                            ]
                        }
                    ).then(function(badgeCount) {
                        callback(null, badgeCount);
                    }).catch(function (error) {
                        callback(error); // Getting error of callback here
                    });
                }else{
                    callback(null, badgeCount)
                }
            }).catch(function (error) {
                callback(error);
            });

        },
        devices: function(callback) {
            models.CustomerDeviceTokens.findAll({
                where: {
                    customer_email: request.token.email
                }
            }).then(function(devices) {
                callback(null, devices);
            }).catch(function(error) {
                callback(error);
            });
        }
    }, function(error, results){

        if(error) {
            Logger.logDbError(error,request);
            return console.log(error);
        }
        //callback function
        console.log("-------------------------------------------------------");
        console.log("Notification List: " + JSON.stringify(results.notification_list, null, 4));
        console.log("badge_count: " + JSON.stringify(results.badge_count, null, 4));

        if(results.devices && result.devices.count > 0) {
            results.devices.forEach(function(device) {
                console.log("device_arn: " + device.device_arn);
            }, this);
        }
    });

}

2 个答案:

答案 0 :(得分:1)

如果callback(null, badgeCount)抛出错误(如果最终回调因某种原因引发错误,可能会发生这种情况),就会发生这种情况。

在这种情况下会发生的情况是调用.catch处理程序,再次调用callback,导致“已经调用”消息。

为了防止这种情况(或者更确切地说,要捕获这些错误),您不应该使用.catch,而是使用.then的双参数版本:

.then(function(badgeCount) {
  callback(null, badgeCount);
}, function (error) {
  callback(error);
});

(对于您在.then().catch()操作中使用async的代码中的所有其他位置,类似

答案 1 :(得分:0)

我在代码中使用了findOrCreate续集方法。 findOrCreate的原始语法是:

model.SomeModelName.findOrCreate({
    ....
    ....
}).spread(function(data, created) {

}).fail(function(err) {

});

正如您在上面的代码中看到.spread()应该跟.fail()一样,但在我的代码中,我使用.spread()后跟.catch()。 因此,两个块都被执行,因此回调被调用了两次