如何将数据(如用户ID)传递给Web工作者以从服务器获取其他推送通知数据?

Rya*_*tin 7 javascript html5 push-notification web-worker google-chrome-app

我想我错过了一些非常容易的东西.

我正在注册一个服务工作者,通过GCM从我的服务器接收推送通知.这很好用.

不幸的是,我不能传递任何数据和我的推送通知一起,因此,当服务人员接收推,它不知道有这回事.因此,在我显示通知之前,我会回调我的服务器以请求有关推送的其他信息,例如标题,消息和图标.

我的问题是,当我回到我的服务器获取更多信息时,在显示推送之前,我在服务工作者中没有关于请求其他信息的用户的任何信息.为了收集该信息,我需要将用户ID和令牌传递给我的服务器.

所以我的问题归结为:如何从我的core.js代码中获取用户id和令牌变量到服务工作者,以便我可以使用它们从我的服务器收集其他信息?

在我的网络应用程序中,我正在注册服务工作者,如下所示:

core.js

var uid = "89123891231239"; // Dummy global id variable
var token = "eo4Dr8qhrFY"; // Dummy global token variable

function setupPushNotifications(){
    console.log("Setting up push notifications");

    if ('serviceWorker' in navigator) {

        navigator.serviceWorker.register('js/pushWorker.js').then(function(reg) {

            console.log(reg);

            reg.pushManager.subscribe({
                userVisibleOnly: true
            }).then(function(sub) {
                console.log('endpoint:', sub.endpoint);
            });

        }).catch(function(error) {
            console.log('Error: ', error);
        });
    }
}
Run Code Online (Sandbox Code Playgroud)

然后,我有一个看起来像这样的服务工作者:

pushWorker.js

'use strict';

self.addEventListener('install', function(event) {
    self.skipWaiting();
    console.log('Installed', event);
});

self.addEventListener('activate', function(event) {
    console.log('Activated', event);
});

self.addEventListener('push', function(event) {  

    var uid = "EXISTS_IN_CORE.JS";
    var token = "ALSO_EXISTS_IN_CORE.JS";

    console.log("Push recieved - we need to know the uid and token here, from core.js");

    event.waitUntil(  
        fetch("https://oapi.co/o?f=getExtraPushData&uid=" + uid + "&t=" + token).then(function(response) {  
          if (response.status !== 200) {  
            console.log('Looks like there was a problem. Status Code: ' + response.status);  
            throw new Error();  
          }

          return response.json().then(function(data) {  
            if (data.error || !data.notification) {  
              console.error('The API returned an error.', data.error);  
              throw new Error();  
            }  

            var title = data.notification.title;  
            var message = data.notification.message;  
            var icon = data.notification.icon;  

            return self.registration.showNotification(title, {  
              body: message,  
              icon: icon,  
            });  
          });  
        }).catch(function(err) {  
          console.error('Unable to retrieve data', err);

          var title = 'An error occurred';
          var message = 'We were unable to get the information for this push message';  
          var icon = "https://oapi.co/occurrences_secure/img/stepNotify_1.png";  
          var notificationTag = 'notification-error';  
          return self.registration.showNotification(title, {  
              body: message,  
              icon: icon,  
              tag: notificationTag  
            });  
        })  
    );  
});
Run Code Online (Sandbox Code Playgroud)

我在这里读到了工人的局限性:从webWorker访问localStorage

我知道这个postMessage功能,但我无法弄清楚如何在实践中理解它.

任何帮助将不胜感激.

小智 11

你可以像这样轻松解决你的问题:

if ('serviceWorker' in navigator) {
    navigator.serviceWorker.register('pushWorker.js').then(function() {
        return navigator.serviceWorker.ready;
    }).then(function(reg) {
        console.log('Service Worker is ready', reg);
        reg.pushManager.subscribe({userVisibleOnly: true}).then(function(sub) {

            console.log('endpoint:', sub.endpoint);
            reg.active.postMessage(JSON.stringify({uid: uid, token: token}));
            console.log("Posted message");

        });
    }).catch(function(error) {
        console.log('Error : ', error);
    });
}
Run Code Online (Sandbox Code Playgroud)

这里的关键是......

reg.active.postMessage(JSON.stringify({uid: uid, token: token}));
Run Code Online (Sandbox Code Playgroud)

...因为在代码中的那一点,你将有权访问postMessage();

在您的工作代码中,请确保您具有以下内容:

self.addEventListener('message', function(event){
    var data = JSON.parse(event.data);

    console.log("SW Received Message:");
    console.log(data);

    self.userID = data.uid;
    self.userToken = data.token;

});
Run Code Online (Sandbox Code Playgroud)

当您最终需要调用服务器并请求其他推送信息时,这将使您可以全局访问您的userID和userToken信息.

  • 我在使用时收到错误消息:“注册失败 - 缺少 applicationServerKey,并且清单为空或丢失”。最后通过直接使用 postMessage 设法做到了同样的事情:/sf/answers/1995968551/ (4认同)
  • 不确定工作程序的生命周期:如果重新启动浏览器(但无法访问网页),网络工作程序也会重新启动吗?如果是这样,userId和userToken变量会丢失吗? (2认同)