我已经阅读了文档,并在GitHub示例页面上阅读了this示例,但是仍然有一个小问题,我没有看到它。
为了说明我要做的事情,我将在下面解释。
我有这个数据库结构(很抱歉,图像和带有西班牙语的代码)
我的目标是每当有一个新的turno
被写入数据库时就触发一个事件,所以可以说它将是76,77,78..
,依此类推。
现在,我的客户将clientes - userID
号发送到notificar
下,并生成了device_token
。
现在,我正在尝试获取与turno
中的值匹配的每个客户端的值,因此,如果值匹配,我会向与之匹配的clientes
发送推式通知相同的数字。
但是我的Firebase功能控制台出现问题
TypeError:无法读取null的属性“ notificar” 在databaseRoot.child.once.then(/user_code/index.js:23:51) 在process._tickDomainCallback(internal / process / next_tick.js:135:7)
这是我为触发此行为而编写的代码
const functions = require('firebase-functions');
const admin = require('firebase-admin');
admin.initializeApp();
//Uso el trigger onWrite, para ver cuando hay un nuevo turno, se ejecute toda esta funcion.
exports.sendClientesNotification = functions.database.ref('/farmacia/turno')
.onWrite((snapshot, context) => {
//Obtenemos el turno que va cambindo y lo guardamos en nuevoTurno
const nuevoTurno = snapshot.after.val();
console.log('Turno nuevo: '+nuevoTurno);
const databaseRoot = snapshot.before.ref.root;
return databaseRoot.child('/farmacia/clientes/{userID}').once('value').then((snapshot) =>{
//Obtengo los valores de la referencia con el wildcard
const getNumeroNotificar = snapshot.child('notificar').val();
const getDeviceToken = snapshot.child('device_token').val();
console.log('Token: '+deviceToken+" Notificar: "+numeroNotificar);
//Envio los valores a la promesa para luego mandar la notificacion
return Promise.all([getNumeroNotificar, getDeviceToken]).then(results => {
//Guardamos
const devicetoken = results[0];
const numeroturnocliente = results[1];
console.log('Nueva notificacion para '+devicetoken+' para el turno '+numeroturnocliente);
if(numeroturnocliente === nuevoTurno){
const payload = {
"data": {
"title": "Su turno ah llegado",
"body": "el turno "+numeroturnocliente+" es el siguiente a ser atendido.",
"icon": "ic_launcher",
"sound": "default",
"click_action": "delete_token"
}
};
return admin.messaging().sendToDevice(devicetoken,payload);
}else{
return 0;
}
});
});
});
我知道错误在这里
const getNumeroNotificar = snapshot.child('notificar').val();
,它也将在其下一行,但我无法弄清楚为什么不获取该位置的值。我已经更改了参考,并查找了几次代码。由于我正在使用javascript进行学习,因此通常我有时会遇到此类错误。
简而言之,我正在尝试
答案 0 :(得分:1)
您将获取特定节点的所有子节点并检索其值。因此,对于您的示例,您可以将其更改为以下内容-
select * from foo order by order
此外, databaseRoot.child('/farmacia/clientes').once('value').then(snapshot => {
const clientes = {}
snapshot.forEach(data => {
const key = data.key
const value = data.val()
//Do what you need, example-
clientes[key] = value
})
})
getNumeroNotificar
不是函数,因此您不需要在getDeviceToken
中运行它们。 Promise.all()
是同步操作。
答案 1 :(得分:1)
您的问题来自以下事实:userId
在您执行操作时未定义
return databaseRoot.child('/farmacia/clientes/{userID}')...
看看您引用的示例:
exports.sendFollowerNotification = functions.database.ref('/followers/{followedUid}/{followerUid}')
.onWrite(async (change, context) => {
.....
const followedUid = context.params.followedUid;
.....
.....
const getDeviceTokensPromise = admin.database()
.ref(`/users/${followedUid}/notificationTokens`).once('value');
最后一行使用模板文字,这是ES6的“新”功能,请参见https://developers.google.com/web/updates/2015/01/ES6-Template-Strings或https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Template_literals,并注意使用反向标记(``)而不是您使用的单引号。
然后,请注意如何从followedUid
占位符计算{followedUid}
常数,该占位符位于函数触发路径(使用context.params.followedUid;
)中。
在您的情况下,您的路径中没有userID
占位符,也没有在函数中使用context.params
来定义userId,因此最后'/farmacia/clientes/{userID}'
不会返回您的路径期待。