【问题标题】:NodeJS sending APN to app through firebaseNodeJS通过firebase向应用发送APN
【发布时间】:2018-02-23 08:31:08
【问题描述】:

您好,我想将 APN 发送到我的应用程序。我能够成功生成通知并将它们发送到我的应用程序。

我的问题是,服务器非常频繁地以块的形式发送通知。我猜我的脚本算法有问题。

我想做什么:

我希望每次帖子中有评论时都将通知发送到设备。我想从 Firebase 数据库中显式获取用户名和评论。

我正在附加服务器脚本:

var firebase = require("firebase");
var once = require("once");
const apn = require('apn');

var config = {
apiKey: "<key>",
authDomain: "<domain>",
databaseURL: "<url>",
projectId: "<id>",
storageBucket: "<bucket>",
messagingSenderId: "<ID>"
};
firebase.initializeApp(config);

let options = {
 token: {
    key: "<p8 file>",

    keyId: "<key>",
    teamId: "<team>"
 },
 production: false
};

  let apnProvider = new apn.Provider(options);



  // Prepare the notifications
  let notification = new apn.Notification();
  notification.expiry = Math.floor(Date.now() / 1000) + 24 * 3600; // will      expire in 24 hours from now
  notification.badge = 3;
  notification.sound = "default";
  notification.topic = "<My bundle ID>";
  notification.payload = {'messageFrom': 'me'};


 var author;
 var dtoken;
 var spotter;
 var comment;
 var database = firebase.database();



 var postref = database.ref("posts").orderByChild("gen_notif").equalTo("yes").on("value", function (snapshot) {
     var key;
     var deviceToken;
     console.log("-------------------Post Ids----------------------")
     snapshot.forEach(function (childSnapshot) {
         key = childSnapshot.key
         author = childSnapshot.val()["author"];
         console.log(key)
         var newref = database.ref("posts/" + childSnapshot.key + "/comment").on('child_added', function(snapy){
             console.log(snapy.val())
             console.log("-----------------comment Keys----------------------")
             snapy.forEach(function(s){
                 var spotuserkey = s.key
                 comment = s.val()
                 console.log(spotuserkey)
                 var spotuser = database.ref("users/"+ spotuserkey +"/credentials/name").on('value', function(spottersnap){
                     console.log("-----------------User Key-----------------------")
                     spotuser = spottersnap.val()
                     console.log(spotuser)

                     var tokenref = database.ref("device/"+author+"/token").once('value', function(snap){
                         console.log("-----------------device token---------------------")
                         deviceToken = snap.val() 
                         console.log(deviceToken)

                          notification.alert = {
                             "title": "You Got Spotted",
                             "body": spotuser + " Spot you " + comment 
                         };

                         apnProvider.send(notification, deviceToken).then( result => {
                               console.log(result["failed"][0]["response"]);
                         });


                     })//tokenref end

                 })//spotteref end

             }); //snapy forEach end

         })//newref end    

     }); //snapshot forEach end

 }); //postref end

  apnProvider.shutdown();

【问题讨论】:

    标签: ios node.js firebase-realtime-database apple-push-notifications


    【解决方案1】:

    要初始化您的应用,我认为最好使用您可以通过 Firebase 控制台下载的 serviceAccountKey 文件,而不是直接在代码中写入您的密钥:

    const functions = require('firebase-functions');
    
    var admin = require('firebase-admin');
    
    var serviceAccount = require("./serviceAccountKey.json");
    
    admin.initializeApp({
      credential: admin.credential.cert(serviceAccount),
      databaseURL: "YOUR_DATABASE_URL_GOES_HERE"
    });
    

    当然,您需要在此处替换为您自己的数据库 url,并确保您的 serviceAccountKey.json 文件位于您的函数文件夹中。

    但是我认为你需要稍微重构你的数据库,以便更容易在服务器端检索你的 id,例如它可能看起来像这样:

    root/
    |     ....
    |  
    |
    |___ posts/
    |      |___ postID
    |              |___ authorId : ...
    |              |___ caption : ...
    |              |___ comments
    |                     |___ commentID 
    |                           |___ senderId: ...
    |                           |___ text: ...
    |               ...
    |
    |
    |___ users/
    |      |___ userID
    |              |___ name : ...
    |              |___ email : ...
    |              |___ notificationTokens
    |                       |___ token1 : true                       
    |                       |___ token2 : true
    |                        ...
    

    然后,只要您的 cmets 节点中有新的写入事件,您就可以创建使用该函数在通知节点内写入通知对象:

    exports.createPostCommentedNotification = functions.database.ref('/posts/{postID}/comments/{commentID}').onWrite(event => {
    
       const data = event.data;
    
       if(data == undefined || !data.val()) { return; }
    
        const postID = event.params.postID; 
        const commentID = event.params.commentID; 
    
        const getCommentSender = admin.database().ref(`/posts/${postID}/comments/${commentID}`).once('value');
    
        const getPostAuthor = admin.database().ref(`/posts/${postID}`).once('value');
    
        return Promise.all([getCommentSender, getPostAuthor]).then(results => {
    
           const commentSenderData = results[0];
           const postAuthorData = results[1];
    
           const commentSenderId = commentSenderData.val().senderId;
           const postAuthorId = postAuthorData.val().authorId;
    
           if(commentSenderId == postAuthorId) { 
              return;
           };
    
           const notificationID = admin.database().ref().push().key;
           const timestamp = Date.now()
    
           const getSenderProfilePromise = admin.auth().getUser(commentSenderId);
    
           return Promise.all([getSenderProfilePromise]).then(results => {
    
                // Note that when you create a user account you would need to set the displayName of the user using the updateProfile() method, otherwise you would need to retrieve the senderName in a different way:)
    
                const senderData = results[0]
                const senderName = senderData.providerData[0].displayName
    
                var notificationData = {
                 senderName: senderName,
                 notificationTimestamp: timestamp
                };
    
                var updates = {};
    
                updates['/notifications/' + postAuthorId + '/' + notificationID] = notificationData;
    
                admin.database().ref().update(updates);
    
                });
          });
    });
    

    然后您将创建另一个函数来实际发送推送通知,使用您的用户令牌,只要有新的通知对象添加到通知节点,如下所示:

    exports.sendPushNotifications = functions.database.ref('/notifications/{receiverId}/{notificationId}').onWrite(event => {
    
      const data = event.data;
    
      if(data == undefined || !data.val()) { return; }
    
      const receiverId = event.params.receiverId;
      const notificationId = event.params.notificationId;
    
      const getDeviceTokensPromise = admin.database().ref(`/users/${receiverId}/notificationTokens`).once('value');
    
      const getMessageContentPromise = admin.database().ref(`/notifications/${receiverId}/${notificationId}/notificationType`).once('value');
    
      const getSenderPromise = admin.database().ref(`/notifications/${receiverId}/${notificationId}/senderName`).once('value');
    
      return Promise.all([getDeviceTokensPromise, getSenderPromise]).then(results => {
    
        const tokensSnapshot = results[0];
        const senderSnapshot = results[1];
    
        const sender = senderSnapshot.val()
    
        if (!tokensSnapshot.hasChildren()) {
          return console.log('There are no notification tokens to send to.');
        }
    
        const payload = {
          notification: {
            title: `${sender}`,
            body: 'Someone commented on your post',
            badge: '1'
          }
        };
    
        var options = {
          priority: "high",
          timeToLive: 60 * 60 * 24,
          mutable_content : true,
          content_available : true,
          category : 'reminder'
        };
    
        const tokens = Object.keys(tokensSnapshot.val());
    
        return admin.messaging().sendToDevice(tokens, payload, options).then(response => {
    
    
          const tokensToRemove = [];
          response.results.forEach((result, index) => {
            const error = result.error;
            if (error) {
              console.error('Failure sending notification to', tokens[index], error);
    
              if (error.code === 'messaging/invalid-registration-token' ||
                  error.code === 'messaging/registration-token-not-registered') {
                tokensToRemove.push(tokensSnapshot.ref.child(tokens[index]).remove());
              }
            }
          });
          return Promise.all(tokensToRemove);
        });
      });
    });
    

    如果您有任何问题,请告诉我!

    【讨论】:

    • 这看起来很完美。我在我的脚本中犯了一个错误。感谢您的帮助。一旦我完成整合,我会更新你。干杯队友
    • 太棒了!是的,如果一切正常,请告诉我,并随时问我是否可以帮助您;)祝您的项目好运!
    • 嘿,我可能遇到了一个问题。我们在这里使用云功能。这似乎对我不起作用。我收到Firebase config variables are not available. Please use the latest version of the Firebase CLI to deploy this function.。虽然我已经安装了 firebase CLI。我错过了什么吗?
    • 我正在使用 cloud9,它似乎没有安装 firebase CLI。希望你能解决我的两个问题
    • 嗯,在终端输入 npm -v 会得到什么?
    猜你喜欢
    • 1970-01-01
    • 2021-05-04
    • 1970-01-01
    • 2014-04-23
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多