Fix usage of hardcoded strings for sw message types (#730)

Co-authored-by: ekzyis <ek@stacker.news>
This commit is contained in:
ekzyis 2024-01-03 21:02:01 +01:00 committed by GitHub
parent 31cef5a7b7
commit 562b243111
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 18 additions and 12 deletions

View File

@ -8,7 +8,13 @@ const applicationServerKey = process.env.NEXT_PUBLIC_VAPID_PUBKEY
const ServiceWorkerContext = createContext() const ServiceWorkerContext = createContext()
// message types for communication between app and service worker // message types for communication between app and service worker
export const STORE_OS = 'STORE_OS' export const MESSAGE_PORT = 'MESSAGE_PORT' // message to exchange message channel on which service worker will send messages back to app
export const ACTION_PORT = 'ACTION_PORT' // message to exchange action channel on which service worker will send actions back to app
export const SYNC_SUBSCRIPTION = 'SYNC_SUBSCRIPTION' // trigger onPushSubscriptionChange event in service worker manually
export const RESUBSCRIBE = 'RESUBSCRIBE' // trigger resubscribing to push notifications (sw -> app)
export const DELETE_SUBSCRIPTION = 'DELETE_SUBSCRIPTION' // delete subscription in IndexedDB (app -> sw)
export const STORE_SUBSCRIPTION = 'STORE_SUBSCRIPTION' // store subscription in IndexedDB (app -> sw)
export const STORE_OS = 'STORE_OS' // store OS in service worker
export const ServiceWorkerProvider = ({ children }) => { export const ServiceWorkerProvider = ({ children }) => {
const [registration, setRegistration] = useState(null) const [registration, setRegistration] = useState(null)
@ -77,7 +83,7 @@ export const ServiceWorkerProvider = ({ children }) => {
// Send subscription to service worker to save it so we can use it later during `pushsubscriptionchange` // Send subscription to service worker to save it so we can use it later during `pushsubscriptionchange`
// see https://medium.com/@madridserginho/how-to-handle-webpush-api-pushsubscriptionchange-event-in-modern-browsers-6e47840d756f // see https://medium.com/@madridserginho/how-to-handle-webpush-api-pushsubscriptionchange-event-in-modern-browsers-6e47840d756f
navigator.serviceWorker.controller.postMessage({ navigator.serviceWorker.controller.postMessage({
action: 'STORE_SUBSCRIPTION', action: STORE_SUBSCRIPTION,
subscription: pushSubscription subscription: pushSubscription
}) })
logger.info('sent STORE_SUBSCRIPTION to service worker', { endpoint }) logger.info('sent STORE_SUBSCRIPTION to service worker', { endpoint })
@ -98,7 +104,7 @@ export const ServiceWorkerProvider = ({ children }) => {
await deletePushSubscription({ variables: { endpoint } }) await deletePushSubscription({ variables: { endpoint } })
// also delete push subscription in IndexedDB so we can tell if the user disabled push subscriptions // also delete push subscription in IndexedDB so we can tell if the user disabled push subscriptions
// or we lost the push subscription due to a bug // or we lost the push subscription due to a bug
navigator.serviceWorker.controller.postMessage({ action: 'DELETE_SUBSCRIPTION' }) navigator.serviceWorker.controller.postMessage({ action: DELETE_SUBSCRIPTION })
logger.info('deleted push subscription from server', { endpoint }) logger.info('deleted push subscription from server', { endpoint })
} }
@ -143,16 +149,16 @@ export const ServiceWorkerProvider = ({ children }) => {
if (!registration) return if (!registration) return
// setup channel between app and service worker // setup channel between app and service worker
const channel = new MessageChannel() const channel = new MessageChannel()
navigator?.serviceWorker?.controller?.postMessage({ action: 'ACTION_PORT' }, [channel.port2]) navigator?.serviceWorker?.controller?.postMessage({ action: ACTION_PORT }, [channel.port2])
channel.port1.onmessage = (event) => { channel.port1.onmessage = (event) => {
if (event.data.action === 'RESUBSCRIBE') { if (event.data.action === RESUBSCRIBE) {
return subscribeToPushNotifications() return subscribeToPushNotifications()
} }
} }
// since (a lot of) browsers don't support the pushsubscriptionchange event, // since (a lot of) browsers don't support the pushsubscriptionchange event,
// we sync with server manually by checking on every page reload if the push subscription changed. // we sync with server manually by checking on every page reload if the push subscription changed.
// see https://medium.com/@madridserginho/how-to-handle-webpush-api-pushsubscriptionchange-event-in-modern-browsers-6e47840d756f // see https://medium.com/@madridserginho/how-to-handle-webpush-api-pushsubscriptionchange-event-in-modern-browsers-6e47840d756f
navigator?.serviceWorker?.controller?.postMessage?.({ action: 'SYNC_SUBSCRIPTION' }) navigator?.serviceWorker?.controller?.postMessage?.({ action: SYNC_SUBSCRIPTION })
logger.info('sent SYNC_SUBSCRIPTION to service worker') logger.info('sent SYNC_SUBSCRIPTION to service worker')
navigator?.serviceWorker?.controller?.postMessage?.({ action: STORE_OS, os: detectOS() }) navigator?.serviceWorker?.controller?.postMessage?.({ action: STORE_OS, os: detectOS() })
}, [registration]) }, [registration])

View File

@ -1,7 +1,7 @@
import ServiceWorkerStorage from 'serviceworker-storage' import ServiceWorkerStorage from 'serviceworker-storage'
import { numWithUnits } from '../lib/format' import { numWithUnits } from '../lib/format'
import { CLEAR_NOTIFICATIONS, clearAppBadge, setAppBadge } from '../lib/badge' import { CLEAR_NOTIFICATIONS, clearAppBadge, setAppBadge } from '../lib/badge'
import { STORE_OS } from '../components/serviceworker' import { ACTION_PORT, DELETE_SUBSCRIPTION, MESSAGE_PORT, STORE_OS, STORE_SUBSCRIPTION, SYNC_SUBSCRIPTION } from '../components/serviceworker'
// we store existing push subscriptions to keep them in sync with server // we store existing push subscriptions to keep them in sync with server
const storage = new ServiceWorkerStorage('sw:storage', 1) const storage = new ServiceWorkerStorage('sw:storage', 1)
@ -243,7 +243,7 @@ export function onPushSubscriptionChange (sw) {
export function onMessage (sw) { export function onMessage (sw) {
return (event) => { return (event) => {
if (event.data.action === 'ACTION_PORT') { if (event.data.action === ACTION_PORT) {
actionChannelPort = event.ports[0] actionChannelPort = event.ports[0]
return return
} }
@ -251,18 +251,18 @@ export function onMessage (sw) {
os = event.data.os os = event.data.os
return return
} }
if (event.data.action === 'MESSAGE_PORT') { if (event.data.action === MESSAGE_PORT) {
messageChannelPort = event.ports[0] messageChannelPort = event.ports[0]
} }
log('[sw:message] received message', 'info', { action: event.data.action }) log('[sw:message] received message', 'info', { action: event.data.action })
if (event.data.action === 'STORE_SUBSCRIPTION') { if (event.data.action === STORE_SUBSCRIPTION) {
log('[sw:message] storing subscription in IndexedDB', 'info', { endpoint: event.data.subscription.endpoint }) log('[sw:message] storing subscription in IndexedDB', 'info', { endpoint: event.data.subscription.endpoint })
return event.waitUntil(storage.setItem('subscription', { ...event.data.subscription, swVersion: 2 })) return event.waitUntil(storage.setItem('subscription', { ...event.data.subscription, swVersion: 2 }))
} }
if (event.data.action === 'SYNC_SUBSCRIPTION') { if (event.data.action === SYNC_SUBSCRIPTION) {
return event.waitUntil(onPushSubscriptionChange(sw)(event, true)) return event.waitUntil(onPushSubscriptionChange(sw)(event, true))
} }
if (event.data.action === 'DELETE_SUBSCRIPTION') { if (event.data.action === DELETE_SUBSCRIPTION) {
return event.waitUntil(storage.removeItem('subscription')) return event.waitUntil(storage.removeItem('subscription'))
} }
if (event.data.action === CLEAR_NOTIFICATIONS) { if (event.data.action === CLEAR_NOTIFICATIONS) {