Skip to content

Instantly share code, notes, and snippets.

@adactio
Last active March 27, 2023 09:26
Show Gist options
  • Star 18 You must be signed in to star a gist
  • Fork 3 You must be signed in to fork a gist
  • Save adactio/4d588bb8a65fa11a3ea3 to your computer and use it in GitHub Desktop.
Save adactio/4d588bb8a65fa11a3ea3 to your computer and use it in GitHub Desktop.
A Service Worker to progressively enhance a blog by storing assets in a cache, and keeping limited caches of pages and images for offline browsing.
'use strict';
// Licensed under a CC0 1.0 Universal (CC0 1.0) Public Domain Dedication
// http://creativecommons.org/publicdomain/zero/1.0/
(function() {
// A cache for core files like CSS and JavaScript
var staticCacheName = 'static';
// A cache for pages to store for offline
var pagesCacheName = 'pages';
// A cache for images to store for offline
var imagesCacheName = 'images';
// Update 'version' if you need to refresh the caches
var version = 'v1::';
// Store core files in a cache (including a page to display when offline)
var updateStaticCache = function() {
return caches.open(version + staticCacheName)
.then(function (cache) {
return cache.addAll([
'/path/to/javascript.js',
'/path/to/stylesheet.css',
'/path/to/someimage.png',
'/path/to/someotherimage.png',
'/',
'/offline.html'
]);
});
};
// Put an item in a specified cache
var stashInCache = function (cacheName, request, response) {
caches.open(cacheName)
.then(function (cache) {
cache.put(request, response);
});
};
// Limit the number of items in a specified cache.
var trimCache = function (cacheName, maxItems) {
caches.open(cacheName)
.then(function (cache) {
cache.keys()
.then(function (keys) {
if (keys.length > maxItems) {
cache.delete(keys[0])
.then(trimCache(cacheName, maxItems));
}
});
});
};
// Remove caches whose name is no longer valid
var clearOldCaches = function() {
return caches.keys()
.then(function (keys) {
return Promise.all(keys
.filter(function (key) {
return key.indexOf(version) !== 0;
})
.map(function (key) {
return caches.delete(key);
})
);
})
};
self.addEventListener('install', function (event) {
event.waitUntil(updateStaticCache()
.then(function () {
return self.skipWaiting();
})
);
});
self.addEventListener('activate', function (event) {
event.waitUntil(clearOldCaches()
.then(function () {
return self.clients.claim();
})
);
});
// See: https://brandonrozek.com/2015/11/limiting-cache-service-workers-revisited/
self.addEventListener('message', function(event) {
if (event.data.command == 'trimCaches') {
trimCache(version + pagesCacheName, 35);
trimCache(version + imagesCacheName, 20);
}
});
self.addEventListener('fetch', function (event) {
var request = event.request;
// For non-GET requests, try the network first, fall back to the offline page
if (request.method !== 'GET') {
event.respondWith(
fetch(request)
.catch(function () {
return caches.match('/offline');
})
);
return;
}
// For HTML requests, try the network first, fall back to the cache, finally the offline page
if (request.headers.get('Accept').indexOf('text/html') !== -1) {
// Fix for Chrome bug: https://code.google.com/p/chromium/issues/detail?id=573937
if (request.mode != 'navigate') {
request = new Request(request.url, {
method: 'GET',
headers: request.headers,
mode: request.mode,
credentials: request.credentials,
redirect: request.redirect
});
}
event.respondWith(
fetch(request)
.then(function (response) {
// NETWORK
// Stash a copy of this page in the pages cache
var copy = response.clone();
var cacheName = version + pagesCacheName;
stashInCache(cacheName, request, copy);
return response;
})
.catch(function () {
// CACHE or FALLBACK
return caches.match(request)
.then(function (response) {
return response || caches.match('/offline.html');
})
})
);
return;
}
// For non-HTML requests, look in the cache first, fall back to the network
event.respondWith(
caches.match(request)
.then(function (response) {
// CACHE
return response || fetch(request)
.then(function (response) {
// NETWORK
// If the request is for an image, stash a copy of this image in the images cache
if (request.headers.get('Accept').indexOf('image') !== -1) {
var copy = response.clone();
var cacheName = version + imagesCacheName;
stashInCache(cacheName, request, copy);
}
return response;
})
.catch(function () {
// OFFLINE
// If the request is for an image, show an offline placeholder
if (request.headers.get('Accept').indexOf('image') !== -1) {
return new Response('<svg role="img" aria-labelledby="offline-title" viewBox="0 0 400 300" xmlns="http://www.w3.org/2000/svg"><title id="offline-title">Offline</title><g fill="none" fill-rule="evenodd"><path fill="#D8D8D8" d="M0 0h400v300H0z"/><text fill="#9B9B9B" font-family="Helvetica Neue,Arial,Helvetica,sans-serif" font-size="72" font-weight="bold"><tspan x="93" y="172">offline</tspan></text></g></svg>', { headers: { 'Content-Type': 'image/svg+xml' }});
}
});
})
);
});
})();
@oceangravity
Copy link

+1

@prtk418
Copy link

prtk418 commented Jan 30, 2018

i wanted to know what will happen if a cached file is updated in local / server version, will the cached file updated automatically?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment