爱上 NodeJS 和异步特性!有了这个,我很惊讶如何继续,因为我不能保留嵌套的 promise ,当然这是不可能的,所以我举手,因为每个步骤都需要使用上一步的数据完成操作。
这就是我想要完成的任务,代码如下。
- 一所新学院进入/sessions/college
- 获得该 key 的值后,去找订阅该学院的顾问。
- 为订阅顾问获取 FCM 代币
- 显然还没有到达这一部分,但请向订阅者发送 FCM 通知。
- 多田!
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/