node.js - NodeJS Promise Firebase

标签 node.js firebase firebase-realtime-database promise google-cloud-functions

爱上 NodeJS 和异步特性!有了这个,我很惊讶如何继续,因为我不能保留嵌套的 promise ,当然这是不可能的,所以我举手,因为每个步骤都需要使用上一步的数据完成操作。

这就是我想要完成的任务,代码如下。

  1. 一所新学院进入/sessions/college
  2. 获得该 key 的值后,去找订阅该学院的顾问。
  3. 为订阅顾问获取 FCM 代币
  4. 显然还没有到达这一部分,但请向订阅者发送 FCM 通知。
  5. 多田!

exports.newSessionNotifer = functions.database.ref('/sessions/college').onCreate((snap, context) => {
    const college = snap.val(); 
    var promises = [];
    var getAdvisors = admin.database().ref('colleges').child(college).once('value').then((snapshot) => {
        const people = snapshot.val();
        var advisors = Object.keys(people);
        return advisors;
    }).then((advisors) => {
        return advisors.forEach((token) => {
            var advisorToken = admin.database().ref('users').child(token).child('fcmtoken').child('token').once('value');
            return console.log(advisorToken);
        });
    });

    return Promise.all(promises).then((values) => {
        console.log(promises);
        return console.log('Hi');
    });

最佳答案

你走在正确的道路上。 once() 返回一个 Promise,它是重复调用一次的 Promise 集合,必须使用 Promise.all() 收集和运行。

exports.newSessionNotifer = functions.database.ref('/sessions/college').onCreate((snap, context) => {
    const college = snap.val();
    return admin.database().ref('colleges').child(college).once('value');
}).then(snapshot => {
    const people = snapshot.val();
    let advisors = Object.keys(people);
    let promises = advisors.map(token => {
        return admin.database().ref('users').child(token).child('fcmtoken').child('token').once('value');
    });
    return Promise.all(promises);
});

编辑再次编辑,这次是OP的答案在手。关于样式,我不确定 lint 说什么,但我对不良嵌套样式的定义是当一个 then() block 包含另一个 then() block 时。另外关于风格,我使这些东西易于理解的方法是构建(和测试)小函数,每个异步任务一个。

在结构上,OP的新答案不必要地在返回顾问之后链接第二个 block 。由于顾问程序不是一个 promise ,因此我们可以使用同步代码继续进行。同样在结构上,OP 的解决方案创建了一系列 promise - 每个顾问有两个 promise (获取顾问 token 并推送) - 但除非应用并返回 Promise.all,否则这些 promise 不一定会完成。

总结所有这些,我的建议如下......

创建时,找到学院的顾问,给每个顾问发送一条消息。

exports.newSessionNotifer = functions.database.ref('/sessions/{sessionID}/college').onCreate((snap, context) => {
    const college = snap.val(); 
    return advisorsForCollege(college).then(advisors => {
        let promises = advisors.map(advisor => sendAdvisorMessage(advisor, college));
        return Promise.all(promises);
    });
});

大学的顾问显然是该大学对象的 key

function advisorsForCollege(college) {
    return admin.database().ref('colleges').child(college).once('value').then(snapshot => Object.keys(snapshot.val()));
}

发送顾问消息意味着获取顾问 token 并进行推送。返回一个执行此操作的两个 promise 链...

function sendAdvisorMessage(advisor, college) {
    return tokenForAdvisor(advisor).then(token => {
        let title = `There's a new session for ${college}!`;
        let body = 'Go to the middle tab and swipe right to accept the session before your peers do!'
        return sendToDevice(token, title, body);
    });
}

现在我们只需要一个来获取顾问的 token 和一个来进行推送......

function tokenForAdvisor(advisor) {
    return admin.database().ref('users').child(advisor).child('fcmtoken').child('token').once('value');
}

function sendToDevice(token, title, body) {
    const payload = { notification: { title: title, body: body } };
    return admin.messaging().sendToDevice(token, payload);
};

我认为 lint 应该将上述所有内容报告得很好,即使打开了 promise 嵌套警告。

关于node.js - NodeJS Promise Firebase,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52806164/

相关文章:

node.js - 如何在 knex 模式中指定浮点精度?

具有动态参数的 Angular Factory Provider

Firebase:如何按键查询?

java - 如何检查数据是否已成功插入 Firebase 数组中

node.js - ACCESS_TOKEN_SCOPE_INSUFFICIENT 403 请求的认证范围域不足

node.js - 如何在不使其不唯一/不可猜测的情况下缩短 UUID V4

node.js - 是否有一个模块相当于 Python 的 argparse for node.js?

Firebase UI 电子邮件链接登录意图额外为空

ios - Firebase iOS 在创建用户时返回

ios - 在 Firebase 上查询多个键