123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130 |
- /* global self ReadableStream Response */
- self.addEventListener('install', () => {
- self.skipWaiting()
- })
- self.addEventListener('activate', event => {
- event.waitUntil(self.clients.claim())
- })
- const map = new Map()
- // This should be called once per download
- // Each event has a dataChannel that the data will be piped through
- self.onmessage = event => {
- // We send a heartbeat every x second to keep the
- // service worker alive if a transferable stream is not sent
- if (event.data === 'ping') {
- return
- }
- const data = event.data
- const downloadUrl = data.url || Math.random() + '/' + (typeof data === 'string' ? data : data.filename)
- const port = event.ports[0]
- const metadata = new Array(3) // [stream, data, port]
- metadata[1] = data
- metadata[2] = port
- // Note to self:
- // old streamsaver v1.2.0 might still use `readableStream`...
- // but v2.0.0 will always transfer the stream through MessageChannel #94
- if (event.data.readableStream) {
- metadata[0] = event.data.readableStream
- } else if (event.data.transferringReadable) {
- port.onmessage = evt => {
- port.onmessage = null
- metadata[0] = evt.data.readableStream
- }
- } else {
- metadata[0] = createStream(port)
- }
- map.set(downloadUrl, metadata)
- port.postMessage({ download: downloadUrl })
- }
- function createStream (port) {
- // ReadableStream is only supported by chrome 52
- return new ReadableStream({
- start (controller) {
- // When we receive data on the messageChannel, we write
- port.onmessage = ({ data }) => {
- if (data === 'end') {
- return controller.close()
- }
- if (data === 'abort') {
- controller.error('Aborted the download')
- return
- }
- controller.enqueue(data)
- }
- },
- cancel (reason) {
- console.log('user aborted', reason)
- port.postMessage({ abort: true })
- }
- })
- }
- self.onfetch = event => {
- const url = event.request.url
- // this only works for Firefox
- if (url.endsWith('/ping')) {
- return event.respondWith(new Response('pong'))
- }
- const hijacke = map.get(url)
- if (!hijacke) return null
- const [ stream, data, port ] = hijacke
- map.delete(url)
- // Not comfortable letting any user control all headers
- // so we only copy over the length & disposition
- const responseHeaders = new Headers({
- 'Content-Type': 'application/octet-stream; charset=utf-8',
- // To be on the safe side, The link can be opened in a iframe.
- // but octet-stream should stop it.
- 'Content-Security-Policy': "default-src 'none'",
- 'X-Content-Security-Policy': "default-src 'none'",
- 'Cross-Origin-Embedder-Policy': 'require-corp',
- 'X-WebKit-CSP': "default-src 'none'",
- 'X-XSS-Protection': '1; mode=block'
- })
- let headers = new Headers(data.headers || {})
- if (headers.has('Content-Length')) {
- responseHeaders.set('Content-Length', headers.get('Content-Length'))
- }
- if (headers.has('Content-Disposition')) {
- responseHeaders.set('Content-Disposition', headers.get('Content-Disposition'))
- }
- // data, data.filename and size should not be used anymore
- if (data.size) {
- console.warn('Depricated')
- responseHeaders.set('Content-Length', data.size)
- }
- let fileName = typeof data === 'string' ? data : data.filename
- if (fileName) {
- console.warn('Depricated')
- // Make filename RFC5987 compatible
- fileName = encodeURIComponent(fileName).replace(/['()]/g, escape).replace(/\*/g, '%2A')
- responseHeaders.set('Content-Disposition', "attachment; filename*=UTF-8''" + fileName)
- }
- event.respondWith(new Response(stream, { headers: responseHeaders }))
- port.postMessage({ debug: 'Download started' })
- }
|