FCM getToken()无法为范围错误注册ServiceWorker Flutter网站 [英] FCM getToken() Failed to register a ServiceWorker for scope error Flutter web
问题描述
在我的应用中,对于Web版本,我使用软件包 firebase 7.3.0
.我首先以单例实例化Firebase应用程序,然后像在应用程序中使用的所有其他Firebase服务一样实例化Messaging():
In my app, for the web version, I use package firebase 7.3.0
.
I first instantiate Firebase app with a singleton and then instantiate Messaging() as I have done with all other Firebase services I use in my app :
App firebase = FirebaseWeb.instance.app;
var firebaseMessaging = messaging();
我有 subscribeToTopic()
方法,该方法首先调用 getMessagingToken()
方法,因为它需要返回的令牌,但是 getMessagingToken()
会抛出错误:
I have subscribeToTopic()
method which first calls getMessagingToken()
method as it needs the returned token, but getMessagingToken()
throws the error:
PlatformPushNotificationWeb.getMessagingToken() getToken error: FirebaseError: Messaging: We are unable to register the default service worker. Failed to register a ServiceWorker for scope ('http://localhost:5000/firebase-cloud-messaging-push-scope') with script ('http://localhost:5000/firebase-messaging-sw.js'): A bad HTTP response code (404) was received when fetching the script. (messaging/failed-service-worker-registration). (messaging/failed-service-worker-registration)
Future<String> getMessagingToken() async {
String token;
await firebaseMessaging.requestPermission().timeout(Duration(seconds: 5)).then((value) {
print('PlatformPushNotificationWeb.getMessagingToken() requestPermission result is $value');
}).catchError((e) => print('PlatformPushNotificationWeb.getMessagingToken() requestPermission error: $e'));
await firebaseMessaging.getToken().then((value) {
print(' PlatformPushNotificationWeb.getMessagingToken() token is $value');
token = value;
}).catchError((e) => print('PlatformPushNotificationWeb.getMessagingToken() getToken error: $e'));
return token;
}
我检查了一下,并在我的 index.html
firebase-messaging中找到了
I checked and in my index.html
firebase-messaging is present:
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>fixit cloud biking</title>
<!-- <meta name="google-signin-client_id" content="YOUR_GOOGLE_SIGN_IN_OAUTH_CLIENT_ID.apps.googleusercontent.com">-->
<meta name="google-signin-client_id" content="xxxxxxxxxx.apps.googleusercontent.com">
<!-- <meta http-equiv="Content-Security-Policy" content="upgrade-insecure-requests">-->
</head>
<!--<body>-->
<body id="app-container">
<script src="main.dart.js?version=45" type="application/javascript"></script>
<!-- The core Firebase JS SDK is always required and must be listed first -->
<script src="https://www.gstatic.com/firebasejs/7.15.5/firebase-app.js"></script>
<!-- TODO: Add SDKs for Firebase products that you want to use
https://firebase.google.com/docs/web/setup#available-libraries -->
<script src="https://www.gstatic.com/firebasejs/7.15.5/firebase-auth.js"></script>
<script src="https://www.gstatic.com/firebasejs/7.15.5/firebase-analytics.js"></script>
<script src="https://www.gstatic.com/firebasejs/7.15.5/firebase-messaging.js"></script>
<script src="https://www.gstatic.com/firebasejs/7.15.5/firebase-storage.js"></script>
<script src="https://www.gstatic.com/firebasejs/7.15.5/firebase-database.js"></script>
<script src="https://www.gstatic.com/firebasejs/7.15.5/firebase-remote-config.js"></script>
</body>
</html>
现在,错误显示'http://localhost:5000/firebase-messaging-sw.js'
而不是 firebase-messaging.js
作为index.html 文件.我注意到
Messaging()不能直接通过firebase应用程序实例获得,因为它可以用于其他服务,因为Storage可以是
firebase.storage()`.我是否想设置其他用于消息传递的内容?
Now, the error says 'http://localhost:5000/firebase-messaging-sw.js'
not firebase-messaging.js
as the library in the index.htmlfile. I noticed that
Messaging()is not directly available through firebase app instance as it would be for other services, for Storage would be
firebase.storage()`.
Am I missing to setup something else for messaging?
推荐答案
Found this article https://medium.com/@rody.davis.jr/how-to-send-push-notifications-on-flutter-web-fcm-b3e64f1e2b76 and discovered that indeed there is a bit more setup for Firebase Cloud Messaging on web.
在 index.html
中,有一个脚本要添加:
In index.html
there is a script to add:
<script>
if ("serviceWorker" in navigator) {
window.addEventListener("load", function () {
// navigator.serviceWorker.register("/flutter_service_worker.js");
navigator.serviceWorker.register("/firebase-messaging-sw.js");
});
}
</script>
在项目 web
文件夹中创建一个新文件 firebase-messaging-sw.js
,在其中导入firebase程序包(匹配 index.html
版本),初始化Firebase应用程序,然后设置BackgroundMessageHandler.如果我使用单例初始化Firebase应用程序,则实例化 messaging()
会引发语法错误,因此需要使用所有参数对其进行初始化,否则在后台消息上将不起作用.
In project web
folder create a new file firebase-messaging-sw.js
where you import the firebase packages (match index.html
versions), initialize Firebase app , and set the BackgroundMessageHandler.
If I initialize Firebase app with the singleton then instantiating messaging()
throws a syntax error, so it needs to be initialized with all parameters, otherwise on background messages won't work.
importScripts("https://www.gstatic.com/firebasejs/7.15.5/firebase-app.js");
importScripts("https://www.gstatic.com/firebasejs/7.15.5/firebase-messaging.js");
//Using singleton breaks instantiating messaging()
// App firebase = FirebaseWeb.instance.app;
firebase.initializeApp({
apiKey: 'api-key',
authDomain: 'project-id.firebaseapp.com',
databaseURL: 'https://project-id.firebaseio.com',
projectId: 'project-id',
storageBucket: 'project-id.appspot.com',
messagingSenderId: 'sender-id',
appId: 'app-id',
measurementId: 'G-measurement-id',
});
const messaging = firebase.messaging();
messaging.setBackgroundMessageHandler(function (payload) {
const promiseChain = clients
.matchAll({
type: "window",
includeUncontrolled: true
})
.then(windowClients => {
for (let i = 0; i < windowClients.length; i++) {
const windowClient = windowClients[i];
windowClient.postMessage(payload);
}
})
.then(() => {
return registration.showNotification("New Message");
});
return promiseChain;
});
self.addEventListener('notificationclick', function (event) {
console.log('notification received: ', event)
});
因此,现在, getToken()
和 subscribeToTopic()
和 onMessage()
可以正常工作.
So now, getToken()
and subscribeToTopic()
and onMessage()
work as expected.
在我的集团中,我在 onMessage()
上有一个侦听器,该监听器(在网络上)将流转换为 Stream< Map< Map,lt; String,Dynamic>>
firebase_messaging
(在设备上)从:
In my bloc I have a listener on onMessage()
which (on web) Stream I convert to a Stream<Map<String,Dynamic>>
as the firebase_messaging
(on device) returns from :
Stream<Map<String, dynamic>> onMessage() async* {
print('PlatformPushNotificationWeb.onMessage() started');
handleData(Payload payload, EventSink<Map<String, dynamic>> sink) {
Map<String,dynamic> message = {
'notification': {
'title': payload.notification.title,
'body': payload.notification.body,
'sound': true
},
'data': payload.data
};
sink.add(message);
}
final transformer = StreamTransformer<Payload, Map<String, dynamic>>.fromHandlers(
handleData: handleData);
yield* firebaseMessaging.onMessage.transform(transformer);
}
希望它对其他人有帮助.干杯.
Hope it helps others. Cheers.
这篇关于FCM getToken()无法为范围错误注册ServiceWorker Flutter网站的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!