In this repository, you'll find code examples to illustrate the Pragmatist's Guide presentation, given at Smashing Conference, New York City, 2017.
Clone this repository, then:
npm install
To run a local web server and see the examples in action, run:
npm start
You'll want to use a browser that supports Service Worker; I use Chrome but SW is also supported in Firefox and Opera to differing extents.
Note: These examples, run locally, rely on the localhost
exception to the SSL/TLS requirement for Service Workers.
- Mission 1: Offline message — Respond to
navigation
requests (i.e. requests for HTML documents). Try to fetch from the network, but if that fails, return aResponse
object that emulates a simple web page with anOh, dear
message. - Mission 2: Offline page — Same as before, but instead of returning a self-made Response on fetch failure, cache an offline HTML page during the
install
phase and return that upon fetch failure. - Mission 3: Network strategies — Respond to fetches for content (HTML) and static assets (images) differently. Use a network-first strategy for content and a cache-first strategy for images.
- Mission 4: Application shell — During the
install
phase, cache a bunch of static assets that we consider to be our application's "shell". Respond to fetches and look in the cache first for requests for those assets. Once the service worker is installed, you can go offline and continue to "request clouds" (images of clouds). - Mission 5: Cache naming and cleanup — Use cache-prefixing to manage versioning of a service worker and, during the
activate
phase, clean up (old) caches that don't match the new cache prefix. To version-bump, you'd want to change thecachePrefix
value. - Mission 6: Fancypants — Add a fallback offline image and use a JSON file as a source of URLs to pre-cache during
install
.
Service Worker is a type of Web Worker. Web Workers are able to execute on a background thread, staying out of the way of the main execution thread of the browser. Other APIs in the Service Worker/Web Worker universe include:
A Web Worker is created/instantiated by code running in one context, but they execute in a different context. Web Workers consist of a JavaScript file.
Service Workers execute in a different context than a web page in a window or tab.
Each window or tab in your browser is a unique browsing context (MDN glossary, HTML specification). JavaScript in an HTML document executes within the scope of a browsing context. The global scope object in a document in a browsing context is window
.
Service Workers, like other Web Workers, execute in a separate context from the clients (web pages in browsing contexts, e.g.) that they control. The global scope object in a Service Worker is ServiceWorkerGlobalScope (MDN, Spec)
Some of the methods and properties available in ServiceWorkerGlobalScope
that the presentation takes advantage of include (but are not limited to!):
fetch(request)
skipWaiting()
clients
(clients.claim()
specifically)caches
(CacheStorage
reference)Request
andResponse
constructors
A handy-dandy chart of what's shipping where (thanks, Jake!).
navigator.serviceWorker
is a ServiceWorkerContainer (MDN, spec). It:
...provides an object...including facilities to register, unregister and update service workers, and access the state of service workers and their registrations.
To register a ServiceWorker, the navigator.serviceWorker.register(scriptURL, options)
method is used within client code.
When a client registers a Service Worker, it does so against a scope, which is a path or pattern within which the Service Worker can listen for fetch
events. A service worker cannot respond to fetches outside of its scope.
Scope can be provided as a second (String) argument to ServiceWorkerContainer.register(scriptURL, options)
. If not provided, scope defaults to ./
, relative to the script's location.
A Service Worker may not have a scope above itself in the directory hierarchy (e.g. ../
or /
if the Service Worker is not at the top level itself) unless you use a Service-Worker-Allowed
header.
When the browser requests a resource that falls within an active Service Worker's scope, a fetch event is dispatched on the Service Worker and the SW may listen for it.
fetch
event handlers are invoked with a FetchEvent
. FetchEvent
objects contain two very useful things:
- a
Request
object (fetchEvent.request
) containing many details about the request in question - a
respondWith()
method that allows the SW to respond to the fetch with its ownResponse
If a Service Worker uses the fetchEvent.respondWith()
method, it should provide a Response
or a Promise
that will resolve to a Response
. That is, the browser spits out a request and is looking for a response in return.
The Fetch
API provides an interface for fetching resources from the network. It is similar to XMLHttpRequest
in ambition.
fetch(request)
returns a Promise
that resolves to a Response
if the fetching is successful.
A Request is chock full of info about a request for a resource. You can create a Request
object using the Request constructor, available in the SW's global scope. A typical instantiation of a Request
passes a string representing the URL for what's being requested, e.g.:
const theRequest = new Request('foo.html');
More often, you'll be dealing with a pre-existing request inside of a fetchEvent
, e.g., rather than creating your own. In this case, looking at details of the fetchEvent.request
can help you figure out how to handle it. Examples in this presentation include:
- Looking at the request's
Accept
headers to see if the request is for an image (e.g.:request.headers.get('Accept').indexOf('image') !== -1
) - Checking
request.mode
: it's value will benavigate
if this is a request for a web document/page
request.mode
isn't supported absolutely everywhere yet. An equivalent check is:
request.method === 'GET' && request.headers.get('Accept').includes('text/html')
A Response represents, unsurprisingly, a response to a request.
You can instantiate a Response
object by using the Response
constructor. The constructor takes two arguments: body
, the response's body, and init
, which is a confusingly-named catch-all settings argument.
In one presentation example, a Response is created that "looks like a web page". It does this like so:
new Response('<p>Oh, Dear!</p>',
{ headers: { 'Content-Type': 'text/html' } });
The body
is a chunk of HTML, and setting a Content-Type
header to text/html
makes the browser treat the response as an HTML document.
Response.ok
is a Boolean that will be true
if the Response's HTTP code was in the 200-299 range, i.e., it is OK and not an error.
Response.clone()
creates a clone of the object to deal with the reality that a Response's body can only be used once. This allows the same Response to be given to the browser to use immediately and be cached for later use.
A Promise is an object that may produce a value at some point. We generally hope that it will resolve to the type of value that we expect. A pending promise can be settled in one of two ways: it can be fulfilled, or resolved; or it can be rejected.
Chains of operations with Promises can be created using Promise.prototype.then()
and Promise.prototype.catch()
. Promise.all(promises)
returns a Promise that will resolve if all of the Promises in promises
resolve, or reject if any one of the Promises in promises
rejects.
In one example in the presentation, a function given as a catch
handler itself Throw
s an error. A subsequent catch
can be added to the promise chain to handle this, e.g.:
doThis().then(doThat).catch(() => {
// blah blah
throw Error('...');
}).catch(() => {
// This function gets invoked if the previous promise rejects or
// if it throws
});
A ServiceWorker has a state
(the value of which can be: parsed, installing, installed, activating, activated or redundant).
When the browser downloads a new service worker file—and this can be an entirely new service worker or a changed service worker file—the service worker is initially parsed
but moves automatically into installing
and is installed
(the install phase).
Later, the service worker moves through activating
and activated
states (activate phase). When this happens depends on if the new service worker file is an update to a previously-activated service worker or an entirely new service worker.
If it's an entirely new service worker, it will move into the activate phase automatically after the install phase completes. If it's a changed/updated service worker, it will not do so until all of the clients controlled by the previous service worker have closed.
The lifecycle events install
and activate
are both ExtendableEvent
s.
ExtendableEvent
objects have a waitUntil(promise)
method that allow you to make the lifetime of the event "stretch out" until the promise
provided resolves.
The install lifecycle phase has an associated event, install
. This phase is meant for setting the service worker up and pre-caching needed assets for later.
Using skipWaiting()
at the end of an install handler will cause the service worker to move into activation immediately without having to wait for clients to close.
The activate lifecycle phase has an associated event, activate
. This phase is meant for cleaning up after old versions of the service worker.
Using clients.claim()
at the end of an activate handler will cause the service worker to take effect immediately without having to wait for clients to reload.
A cache
is a map of Request
- Response
pairs. You can create as many caches as you like and name them whatever you please. These caches are distinct from the browser's built-in caching.
The CacheStorage
interface serves as a directory for all available caches and is available as caches
in ServiceWorkerGlobalScope.
caches.open('name')
: Returns a Promise that resolves to the Cache requested. If one doesn't exist byname
, a new one will be created.caches.delete(key)
: Returns a Promise that resolves totrue
if a Cache exists with namekey
and is successfully deleted. It will resolve tofalse
if there is no Cache by that name.caches.keys()
: Returns a Promise that resolves to an Array of keys (Strings) for every current Cache.
Remember, access a Cache by using caches.open(name)
.
cache.put(request, response)
: Add the given request-response pair to the Cache.cache.add(requestOrURL)
: Fetch the Request given and store it and its resulting Response in the Cache.cache.addAll(urls)
: Fetch all of the URLs inurls
Array and store the resulting request-response pairs in the Cache.
cache.match(requestOrURL)
: Look for a matchingResponse
forrequestOrURL
in this Cache object (only this cache). You need to access the Cache first withcaches.open()
.caches.match(requestOrURL)
: Look for a matchingResponse
across all Caches. Does not require opening a Cache first.
Important Note: Both cache.match()
and caches.match()
return a Promise that will resolve to undefined
if a match is not found. The Promise will not reject. Philosophically, this is because an "answer" (result) was obtained: the answer is that there isn't a match. Promises should only reject when there is a failure in actually obtaining an answer.
Network strategies allow you to optimize both online and offline performance by avoiding network round-trips and responding with cached items when the user is offline.
A network-first strategy is used for assets that change frequently and should be as fresh as possible. It takes the following steps when responding to a fetch event:
- Try to obtain a fresh copy of the resource from the network.
- If this is successful, store a copy of the resource in cache for potential later use before returning the response to the browser.
- If this fails, check to see if there is a cached copy of the resource and return that, if so.
- In the presentation, the network-first strategy has an additional fallback behavior: return an offline page response if the network is unavailable and there is no cached copy of the resource.
A cache-first strategy can be used for static assets that don't change much, reducing network usage. It takes the following steps when responding to a fetch event:
- Try to find the resource in cache.
- If that is successful, respond to the fetch with that cached resource.
- If there is no cached copy of the resource, fetch one from the network.
- If that is successful, cache a copy of the resource for later use before returning the response to the browser.
- See also
Response.ok
andResponse.clone()
Read-through caching is the technique of caching items as they're fetched in fetch handlers for potential later use, either offline or for cache-first strategies.
An application shell is those static assets like icons, images, CSS, scripts that are used on every page or nearly every page of a web site or app.
Application shell resources can be pre-cached during service-worker install and subsequent fetches for these resources can be handled in a cache-first manner.
The activate phase of a service worker can be used to clean up after old service worker versions.
One technique for versioning service workers involves using a unique version string or prefix every time you make changes to your service worker. This string can be prefixed to cache names to identify which version of a service worker a given cache is associated with.
When cleaning up after old caches, you can use the caches.keys()
method to retrieve an Array of all current cache keys (Strings). An example in the presentation then used Array.prototype.filter
to filter the set of keys down to those that don't match the current version string, that is, those that should be deleted. It then used Array.prototype.map
to map each key-to-delete to caches.delete(key)
, resulting in an Array of Promises for those delete operations.
The global property clients
is an interface to the collection of clients (e.g. browsing contexts) that this service worker controls. Using clients.claim()
at the end of an activate handler allows the newly-activated service worker to take control of its clients immediately instead of having to wait for a reload on each.