获取PushManager订阅出错 - JavaScript

9
我有以下代码,注册一个服务工作者,并请求用户允许通知。在用户允许推送通知后,serviceWorkerRegistration.pushManager.getSubscription()返回的Promise被设置为null,当我关闭浏览器并再次强制调用此函数时,它可以正常运行而没有错误。我不理解原因。这是我的代码:
window.vapidPublicKey = new Uint8Array([4, 45, ...]); 
if (navigator.serviceWorker) {
  navigator.serviceWorker.register('/serviceworker.js')
  .then(function(reg) {
     console.log('Service worker change, registered the service worker');
  });
}
// Otherwise, no push notifications :(
else {
  console.error('Service worker is not supported in this browser');
}

navigator.serviceWorker.ready.then((serviceWorkerRegistration) => {
  console.log("ready");
  serviceWorkerRegistration.pushManager
  .subscribe({
    userVisibleOnly: true,
    applicationServerKey: window.vapidPublicKey
  });
});



// application.js

// Let's check if the browser supports notifications
if (!("Notification" in window)) {
  console.error("This browser does not support notifications.");
}

// Let's check whether notification permissions have already been granted
else if (Notification.permission === "granted") {
  console.log("Permission to receive notifications has been granted");
}

// Otherwise, we need to ask the user for permission
else if (Notification.permission !== 'denied') {
  Notification.requestPermission(function (permission) {
    // If the user accepts, let's create a notification
    if (permission === "granted") {
      console.log("Permission to receive notifications has been granted");
      saveSubscriptionToDatabase();
    }
  });
}

function saveSubscriptionToDatabase(){

      navigator.serviceWorker.ready
      .then((serviceWorkerRegistration) => {
        console.log(serviceWorkerRegistration.pushManager.getSubscription());
        serviceWorkerRegistration.pushManager.getSubscription()
        .then((subscription) => {
          $.post("/users/save_subscription", { subscription: subscription.toJSON() });
        });
      });

}

错误:
Uncaught (in promise) TypeError: Cannot read property 'toJSON' of null
at serviceWorkerRegistration.pushManager.getSubscription.then

更新:

因此,我添加了以下按钮:

 <input type="button" onclick="saveSubscriptionToDatabase();" value="test">

只要我从按钮点击调用函数,所有的JS和POST请求都能正常工作,但是当我从检查权限状态的条件语句中调用它时,它仍然像以前一样失败。
更新2:
我尝试从不同的角度重新编写代码。
function registerServiceWorker() {
  return navigator.serviceWorker.register('serviceworker.js')
  .then(function(registration) {
    console.log('Service worker successfully registered.');
    return registration;
  })
  .catch(function(err) {
    console.error('Unable to register service worker.', err);
  });
}


function askPermission() {
  return new Promise(function(resolve, reject) {
    const permissionResult = Notification.requestPermission(function(result) {
      resolve(result);
    });

    if (permissionResult) {
      permissionResult.then(resolve, reject);
    }
  })
  .then(function(permissionResult) {
    if (permissionResult !== 'granted') {
      throw new Error('We weren\'t granted permission.');
    }
  });
}


function subscribeUserToPush() {
  return navigator.serviceWorker.register('serviceworker.js')
  .then(function(registration) {
    const subscribeOptions = {
      userVisibleOnly: true,
      applicationServerKey: new Uint8Array("key goes here")

    };

    return registration.pushManager.subscribe(subscribeOptions);
  })
  .then(function(pushSubscription) {
    console.log("never called");
    console.log('Received PushSubscription: ', JSON.stringify(pushSubscription));
    return pushSubscription;
  });
}

registerServiceWorker();
askPermission();
subscribeUserToPush();
< p >在< code >subscribeUserToPush()命令链的最后一部分,我有< code > console.log(“never called”); 从未执行过,并且没有错误。 我在Chrome 67中测试此功能。 我确信< code > registration.pushManager.subscribe(subscribeOptions);已被调用。

也许您在此处拼写错误,但请在其他地方使用正确的拼写:vapidPublicKey。 - boateng
另一个尝试的方法是创建一个带有按钮的页面,点击该按钮将发送您的$.post请求。然后使用诸如Fiddler之类的工具对响应/请求标头、参数或任何其他错误进行故障排除。 - boateng
1个回答

5

经过数小时的尝试,我终于找到了问题,而且它非常简单。所以每次调用navigator.serviceWorker.register('serviceworker.js')时,你必须在文件名之前加上/。服务工作者将在没有/的情况下正常注册,但抓取订阅将失败。在文档中找到的示例不包括/

所以,代码应该像这样:

navigator.serviceWorker.register('/serviceworker.js')

唰,现在它可行了...

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接