-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpush-notifications.js
165 lines (152 loc) · 5.38 KB
/
push-notifications.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
// @flow
import log from 'meteor/xerdi:logging';
import webpush from "web-push";
class PushNotificationsPrototype {
url = '/packages/xerdi_push-notifications/service-worker.js';
registration: ServiceWorkerRegistration | undefined;
subscription: PushSubscription | null;
status = 'uninitialized';
notifications() {
return this.registration.getNotifications();
}
async registerAndSubscribe() {
if (!this.registration) {
await this.register();
}
if (!this.subscription) {
await this.subscribe();
}
return this.subscription;
}
async register() {
if (this.status === 'uninitialized' && navigator && navigator.serviceWorker) {
try {
this.registration = await navigator.serviceWorker.register(this.url);
this.status = 'registered';
} catch (err) {
log.error('failed to register service worker');
console.error(err);
this.status = 'unsupported';
}
}
}
async subscribe() {
if (this.status === 'registered') {
this.validateSession();
try {
const existingSubscription = await this.registration.pushManager.getSubscription();
this.subscription = existingSubscription || await this._subscribe();
this.status = 'subscribed';
} catch (err) {
log.error('failed to subscribe');
console.error(err);
this.status = 'unsupported';
}
}
if (this.subscription) {
const subInfo = this.subscription.toJSON();
Meteor.call('Notifications.authorizeSubscription', subInfo, function (err, data) {
if (err) {
log.error(err);
} else {
log.info(`Added ${data || 0} subscriptions`);
}
});
return subInfo;
}
}
async _subscribe() {
return await this.registration.pushManager.subscribe({
userVisibleOnly: true,
applicationServerKey: await this.serverKey()
});
}
async unsubscribe() {
if (!this.registration) {
this.subscription = null;
this.status = 'uninitialized';
log.warn('no service worker registration for push notifications');
return;
}
if (!this.subscription)
this.subscription = await this.registration.pushManager.getSubscription();
if (this.subscription) {
Meteor.call('Notifications.revokeSubscription', this.subscription.toJSON(), function (err, data) {
if (err)
log.error(err);
else {
log.info(`Pulled ${data || 0} subscriptions`);
}
});
await this.subscription.unsubscribe()
.catch(function (err) {
log.error('Failed to unsubscribe push manager');
log.error(err);
});
this.subscription = null;
} else {
log.warn('No subscription to stop');
}
if (this.registration) {
await this.registration.unregister()
.catch(function (err) {
log.error('Failed to unregister push manager');
log.error(err);
});
this.registration = null;
}
this.status = !!this.registration ? 'registered' : 'uninitialized';
}
sendNotification(connection, notification) {
return webpush.sendNotification(connection, notification);
}
async requestPermission() {
const self = this;
if (Notification && Notification.requestPermission) {
return await Notification.requestPermission()
.then(function (permission) {
return permission === 'granted';
})
.catch(function (err) {
self.status = 'unsupported';
log.error(err.message);
return false;
});
} else {
self.status = 'unsupported';
log.warn('Notifications not supported');
}
return false;
}
validateSession() {
if (!this.registration) {
throw new Meteor.Error('A Service Worker must be registered');
}
if (!Meteor.userId()) {
throw new Meteor.Error('You need to be logged in');
}
}
async serverKey() {
const self = this;
return await new Promise(function (resolve, reject) {
Meteor.call('Notifications.publicKey', function(err, data) {
if (err) {
reject(err);
} else {
resolve(self.toBase64(data));
}
});
});
}
toBase64(url) {
const padding = '='.repeat((4 - (url.length % 4)) % 4)
const base64 = (url + padding).replace(/\-/g, '+').replace(/_/g, '/')
const rawData = atob(base64)
const outputArray = new Uint8Array(rawData.length)
for (let i = 0; i < rawData.length; ++i) {
outputArray[i] = rawData.charCodeAt(i)
}
return outputArray;
}
}
export const PushNotifications = new PushNotificationsPrototype();