From c1f36193453aedfea085fc972a6dd68366e592f2 Mon Sep 17 00:00:00 2001 From: Jake Archibald Date: Mon, 3 Jun 2019 07:44:35 +0100 Subject: [PATCH] Fixing deploy (#1412) * Remove html from repo * Ignore html * Leaning on travis for deploy. Moving to gh-pages. --- .gitignore | 4 +- .travis.yml | 15 +- compile.sh | 5 - deploy.sh | 61 - deploy_key.enc | Bin 3248 -> 0 bytes docs/index.html | 12101 ------------------------------------------- docs/v1/index.html | 11511 ---------------------------------------- 7 files changed, 11 insertions(+), 23686 deletions(-) delete mode 100644 deploy.sh delete mode 100644 deploy_key.enc delete mode 100644 docs/index.html delete mode 100644 docs/v1/index.html diff --git a/.gitignore b/.gitignore index b9b9da67..92590d7d 100644 --- a/.gitignore +++ b/.gitignore @@ -1,2 +1,4 @@ node_modules -.tern-port \ No newline at end of file +.tern-port +docs/index.html +docs/v1/index.html diff --git a/.travis.yml b/.travis.yml index b8d83b34..12ad8864 100644 --- a/.travis.yml +++ b/.travis.yml @@ -1,9 +1,10 @@ language: generic -python: -- '2.7' script: -- bash "./deploy.sh" -env: - global: - - ENCRYPTION_LABEL: "8bb8090ad69a" - - COMMIT_AUTHOR_EMAIL: "plh+deploy@w3.org" +- bash "./compile.sh" +deploy: + local_dir: docs + provider: pages + skip_cleanup: true + github_token: $GITHUB_TOKEN # Set in the settings page of your repository, as a secure variable + on: + branch: master diff --git a/compile.sh b/compile.sh index a6847ceb..74f80e25 100755 --- a/compile.sh +++ b/compile.sh @@ -26,8 +26,3 @@ curlbikeshed() { } cd docs && curlbikeshed "index.html" - -if [ -d out ]; then - echo Copy the generated spec into out/index.html - cp index.html out/index.html -fi diff --git a/deploy.sh b/deploy.sh deleted file mode 100644 index b25d82db..00000000 --- a/deploy.sh +++ /dev/null @@ -1,61 +0,0 @@ -#!/bin/bash -set -e # Exit with nonzero exit code if anything fails - -# From https://gist.github.com/domenic/ec8b0fc8ab45f39403dd - -SOURCE_BRANCH="master" - -function doCompile { - chmod 755 ./compile.sh - ./compile.sh -} - -LAST_COMMITER_EMAIL=`git log -1 --pretty=format:'%aE'` -# Pull requests and commits to other branches shouldn't try to deploy, just build to verify -if [ "$TRAVIS_PULL_REQUEST" != "false" -o "$TRAVIS_BRANCH" != "$SOURCE_BRANCH" -o "$LAST_COMMITER_EMAIL" = "plh+deploy@w3.org" ]; then - echo "Skipping deploy; just doing a build." - doCompile - exit 0 -fi - -# Save some useful information -REPO=`git config remote.origin.url` -SSH_REPO=${REPO/https:\/\/github.com\//git@github.com:} -SHA=`git rev-parse --verify HEAD` - -# Clone the existing branch for this repo into out/ -# Create a new empty branch if gh-pages doesn't exist yet (should only happen on first deply) -git clone $REPO out -cd out - -# Run our compile script -doCompile - -# Now let's go have some fun with the cloned repo - -git config user.name "Travis CI" -git config user.email "$COMMIT_AUTHOR_EMAIL" - -# If there are no changes to the compiled out (e.g. this is a README update) then just bail. -if git diff --quiet; then - echo "No changes to the output on this push; exiting." - exit 0 -fi - -# Commit the "changes", i.e. the new version. -# The delta will show diffs between new and old versions. -git add -A . -git commit -m "Building Spec: ${SHA}" - -# Get the deploy key by using Travis's stored variables to decrypt deploy_key.enc -ENCRYPTED_KEY_VAR="encrypted_${ENCRYPTION_LABEL}_key" -ENCRYPTED_IV_VAR="encrypted_${ENCRYPTION_LABEL}_iv" -ENCRYPTED_KEY=${!ENCRYPTED_KEY_VAR} -ENCRYPTED_IV=${!ENCRYPTED_IV_VAR} -openssl aes-256-cbc -K $ENCRYPTED_KEY -iv $ENCRYPTED_IV -in ../deploy_key.enc -out ../deploy_key -d -chmod 600 ../deploy_key -eval `ssh-agent -s` -ssh-add ../deploy_key - -# Now that we're all set up, we can push. -git push $SSH_REPO diff --git a/deploy_key.enc b/deploy_key.enc deleted file mode 100644 index f7b27fbb1fe2db86e1e0fc30016f08d289ad6fc7..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3248 zcmV;h3{UgaRvH1gIWFD@eWKX!p^rA$#K*;aD5Ylq$xGKfq8)%b$vwLZjOC(4nWaA4 z1tyyeX0L?axf=u){wicQLcx(S!T5pPw=c9=J1NgR&UV?X+0hDAxGvk+F>>*n z{B#SKox|wP8LOxzd0Q11glAh>!Nuwi6q4A6;ONbM;E%-*yfd}`eBp8Gwm(7-figFb zLPB-f3|U;XyYE;-YIi0J?GW`HW#oS>YlVtmBT({LA$gfF%ae*mjW6d(DpLy7kGsGc z7A*+h62KS0lB^rpgkTEu9+=BxaqUt73Nr`(oF?RAsu@YF(@uDj1C>o-*1<>ot^s?$ zUfd68+greub@NG<7=E20P-C={wd0Pr>-B4^7KMzM47Oq^?)p3HrDw7 z=yMPr(CN9oIe#kW5vZ)hlC}W1J>-!yw6~G>#|q@$>Y#(Sc8Md%yFvkkmE7eu6T&Um ze^|f?(u)YLUc|AZuN@vp7Kd7!p0#_m^HWFQSY96Xj7623&ffA67bJvwM{}%*WIBS7 z4ds|u!MIBL+@JTp*>9kfw8ZZ-|8(_TVp0hfgckcTS&xz8tHRXiSzg0BkpWSjs zn6@1YWgCYMmmgIAz@pqdSFK@EE~}Z>T%Vh#ySDSwdj;5gJ_N`~4tBssdL40REC7Z1 zhdh;}&G@KC-UigjUMzl?uz5KCXQC4A_LU=oS6&+-@>n6K_-AQW-_%yw=oIjGY6H|C z;lUuM28ovjR?4cc#|nU&^I+^yl?`A*;g`ot{JTqVBPxVw0voZZJiYw+Z1XXxCOoy- zPORzT`@;)^tVp#VL-H*l8_LN-vdK!B9tlcFi%*U3)<|W>V!4c@7fEW?@53!rvIL(G zIj~0V0$j}yz(6f(ZXYV7ySJQ5hB0pH=GJcXj_*3(+gMEc(aMo9dGs{j z(+d;c;wepBr>V|B>1i$xx6p!+&_aO4aTdHB#wiG*SCyKTvuk|j)CFhVgbPfNXEVk{ zur2~oyg0Q?U)MiQ0_!zAZihW51{W zs9KpKe#J?tS#5;8(0ZfQ%PeQ7uk=Sa|*1n&zROK%Ak&6 z@hHr}G?aNwkaZG|dP%_y09?6sKk3k0KOVt>LAby_UV~9$f#Q1l;t=Ys-a+ zdVRF<>>^AlCA@e9Q2;1a1cuecqZ8t9Zmu}Yteuzs*&^9#M4*yKxe@y(>lMcOIa7@G z2Omld7@A70?+&gj;bDZkf$kyPx4oZ!i7}Wv>68|1scJwo4ZumzqcTcg#C}gEjvBt_ zOxb#RjfRY^IL0N}A-xUd&Gx!v$B`;=^Jm4#wi#j4Y!LiVQ9WHJ*SM?R!elVDT!zM& zgW#rG)$4tpynUtmF>YJeE1v_HXE3ajKW}L=Ocd8){PfPi`9=l;IqrCMd;o5r+ULd+ zkQ6%%5(wp{fx1OvLH@3rvJ&}|{sqc0tZoJ`2P0d5+W}j}ZEOg%x>4`m{!Msec*_DH zjjZ&3+#|wjBD1J7JKK4oo)=}9NAli=UxS9X$Y8a(-O(U0 znjA-Ja;eT@X?A97Ufvc+tQDQt9-0)q8w$r7?3G`qE`TEm>jJ@T)xVe`uQleqA6gP{ z2Z|)Nn#>y+un^Ox=N%hBYfuZM&L5*%mM;m8ED0;wYiY;q)zO7Uy00Ac%|QaKA~O?R zP>B_*J$0v+dIg?@_z8GH-J>RqilEnHTBNSf9Oy<#hsNLe+`{^I9LL$Nc-PoxLEIx` z1(_OD>-Q_Ci%KR|;OgS|$yhhAx?x_Wg;%P=G%q@A-pC{}c%#1$1f-1`z#@bawh3K= z$v9mdFrx0XVq@ftEv7d%>1u%5ayk!YAiqvoaw-8k46Nt3XEZ5)nm!hHs1Orxn({N> zWe;CG=WOk>^e~bt-gE*XPzTY2uc;4s_(t+R;DOmWD52YZmw!1z~cy^2W#s*nacL2aE`T#_)-GnpB*CTM%i zGVId%_dI{P&G9|^HL0k5KNHK)JA^tngdtYrYH>~yMB9{Cuc%p@F%P*N0lw_ zoOXJCC*#!BF->}2yeTrUNJg>84vx<38-wSV87-!DpY}&cfTxU8k@^lRLT6;HjjJnM zp(8w~sUjILvfK`fSLKY%LOWYuZ`c9E^|8ew&tXPr`K|4&$4}!410j$y^;d_?7~+R1 zFZd4iOLJ;oYi^LpvR42-Kh8eUsF?-gUZ|p$Irgp=1d*EIi+H$k-x8?E>cP1)-@!u2 zu&;f}hJK9XwXjhl+jpK4Vg!YG8e15sc9ww_&Pm4|u?_4x@8}&M4Xe_~#bCoS@wAL`+H+=W+7Q$Lh>Abh16n(T z>U%R086?tLCGRgAdM6axewN?HId8Oy+Tt-kTlfHge4vO_%ohb~`|jc_B-a_Lm{I{x zOQ>3_0A}uas?0A4L6T`WS8(y+Ml10S=h>o#7FiOwV~nh-;Csa7Ogm+oeo2@-!4A!F z>SI|kNqNA}V2(S})l&T)5gjF%|GqWt1FJF%eLDH&+c`f_UhQU&;JUu`xqbsC16JF( zWzbDtK0JG#8jDb1#&qzEmm>h&iwllchb?`H)j(@^Q4psLWoVxYLvtNroQ5#(1xDlQ zU0cQz)XmR`(i14b$0sq(0%9Td;7Yt{v$&ug3I#rv9tFOr>6l0nTsJB{_wKK&cw)nb zV>Bg;sne{06a8N2`j2jGuv57SzK&^3mh&b%w_#&(?OiFBI;aF-Islwr)uj2eTNFe= z6RP53`7hj3wK{$Ff$z8NFi2AdbsLD)A&Pv2f&390TKVIiVFKJ$=ikI~U|-wknGN=F z_y_aM-c&p#Y8j|O(xRlUGXp^lah#C0q8^nOmDdi=x?^xmO5(iI;+4;&@d=g@9HHX) zdUfn42RY&M{IeR{vM8+3mFJmqe352LA+j_I2m%osk2yW(j8 zv+Fo9yjkSBTtjP?EwvA&LVY{?`|~h0(f|girYxpZ)18imYuX&{^GCfO(P*V+Ct7Yk zBUtM9dmHVvz=LAndQ8Zs>emnV@RuKYE1Ho2BHOzoT&bOKGU22+ot&&8PYWSxX=ulK z)yX!Aa{BqzrG>dMbDZ$aXBHUgDsKyYTIKRF`MAWDQk#Mmb5pkVX5H0rDNM9*=^F}aGub;`3b2F`PKA%HL?M##oYiWd?-rf zAmg6L!Qm$iyG)uRq+MZ*x_vY;;zWi{5~ElL2kmb^ItBJ+-#cVpj>p|m4)pB%qxcU9 z(o2X`!;aKZSs?{R`IF2p=oZfLRqEb^Of$81bfb{xbJOyrnk90?aD+qg4D*Y=fNQYerB2 diff --git a/docs/index.html b/docs/index.html deleted file mode 100644 index 74629424..00000000 --- a/docs/index.html +++ /dev/null @@ -1,12101 +0,0 @@ - - - - - Service Workers Nightly - - - - - - - - - - - - -
-

-

Service Workers Nightly

-

Editor’s Draft,

-
-
-
This version: -
https://w3c.github.io/ServiceWorker/ -
Latest published version: -
https://www.w3.org/TR/service-workers/ -
Issue Tracking: -
GitHub -
Editors: -
(Google) -
(Microsoft‚ represented Samsung until April 2018) -
(Google) -
(Google) -
Tests: -
web-platform-tests service-workers/ (ongoing work) -
-
-
- -
-
-
-

Abstract

-

This specification describes a method that enables applications to take advantage of persistent background processing, including hooks to enable bootstrapping of web applications while offline.

-

The core of this system is an event-driven Web Worker, which responds to events dispatched from documents and other sources. A system for managing installation, versions, and upgrades is provided.

-

The service worker is a generic entry point for event-driven background processing in the Web Platform that is extensible by other specifications.

-
-

Status of this document

-
-

This section describes the status of this document at the time of its publication. Other documents may supersede this document. A list of current W3C publications and the latest revision of this technical report can be found in the W3C technical reports index at https://www.w3.org/TR/.

-

This document was published by the Service Workers Working Group as an Editors Draft. This document is intended to become a W3C Recommendation.

-

-

This is a living document. Readers need to be aware that this specification may include unimplemented features, and details that may change. Service Workers 1 is a version that is advancing toward a W3C Recommendation.

-

-

Feedback and comments on this specification are welcome, please send them to public-webapps@w3.org (subscribe, archives) with [service-workers] at the start of your email’s subject.

-

Publication as an Editors Draft does not imply endorsement by the W3C Membership. This is a draft document and may - be updated, replaced or obsoleted by other documents at any time. It is inappropriate to cite - this document as other than work in progress.

-

This document was produced by a group operating under the W3C Patent Policy. W3C maintains a public list of any patent disclosures made in connection with the deliverables of the group; that page also includes instructions for disclosing a patent. An individual who has actual knowledge of a patent which the individual believes contains Essential Claim(s) must disclose the information in accordance with section 6 of the W3C Patent Policy.

-

This document is governed by the 1 March 2019 W3C Process Document.

-
-
- -
-
-

1. Motivations

-

This section is non-normative.

-

Web Applications traditionally assume that the network is reachable. This assumption pervades the platform. HTML documents are loaded over HTTP and traditionally fetch all of their sub-resources via subsequent HTTP requests. This places web content at a disadvantage versus other technology stacks.

-

The service worker is designed first to redress this balance by providing a Web Worker context, which can be started by a runtime when navigations are about to occur. This event-driven worker is registered against an origin and a path (or pattern), meaning it can be consulted when navigations occur to that location. Events that correspond to network requests are dispatched to the worker and the responses generated by the worker may override default network stack behavior. This puts the service worker, conceptually, between the network and a document renderer, allowing the service worker to provide content for documents, even while offline.

-

Web developers familiar with previous attempts to solve the offline problem have reported a deficit of flexibility in those solutions. As a result, the service worker is highly procedural, providing a maximum of flexibility at the price of additional complexity for developers. Part of this complexity arises from the need to keep service workers responsive in the face of a single-threaded execution model. As a result, APIs exposed by service workers are almost entirely asynchronous, a pattern familiar in other JavaScript contexts but accentuated here by the need to avoid blocking document and resource loading.

-

Developers using the HTML5 Application Cache have also reported that several attributes of the design contribute to unrecoverable errors. A key design principle of the service worker is that errors should always be recoverable. Many details of the update process of service workers are designed to avoid these hazards.

-

Service workers are started and kept alive by their relationship to events, not documents. This design borrows heavily from developer and vendor experience with Shared Workers and Chrome Background Pages. A key lesson from these systems is the necessity to time-limit the execution of background processing contexts, both to conserve resources and to ensure that background context loss and restart is top-of-mind for developers. As a result, service workers bear more than a passing resemblance to Chrome Event Pages, the successor to Background Pages. Service workers may be started by user agents without an attached document and may be killed by the user agent at nearly any time. Conceptually, service workers can be thought of as Shared Workers that can start, process events, and die without ever handling messages from documents. Developers are advised to keep in mind that service workers may be started and killed many times a second.

-

Service workers are generic, event-driven, time-limited script contexts that run at an origin. These properties make them natural endpoints for a range of runtime services that may outlive the context of a particular document, e.g. handling push notifications, background data synchronization, responding to resource requests from other origins, or receiving centralized updates to expensive-to-calculate data (e.g., geolocation or gyroscope).

-
-
-

2. Model

-
-

2.1. Service Worker

-

A service worker is a type of web worker. A service worker executes in the registering service worker client's origin.

-

A service worker has an associated state, which is one of parsed, installing, installed, activating, activated, and redundant. It is initially parsed.

-

A service worker has an associated script url (a URL).

-

A service worker has an associated type which is either "classic" or "module". Unless stated otherwise, it is "classic".

-

A service worker has an associated containing service worker registration (a service worker registration), which contains itself.

-

A service worker has an associated global object (a ServiceWorkerGlobalScope object or null).

-

A service worker has an associated script resource (a script), which represents its own script resource. It is initially set to null.

-

A script resource has an associated has ever been evaluated flag. It is initially unset.

-

A script resource has an associated HTTPS state (an HTTPS state value). It is initially "none".

-

A script resource has an associated referrer policy (a referrer policy). It is initially the empty string.

-

A service worker has an associated script resource map which is an ordered map where the keys are URLs and the values are responses.

-

A service worker has an associated set of used scripts (a set) whose item is a URL. It is initially a new set.

-

Note: The set of used scripts is only used to prune unused resources from a new worker’s map after installation, that were populated based on the old worker’s map during the update check.

-

A service worker has an associated skip waiting flag. Unless stated otherwise it is unset.

-

A service worker has an associated classic scripts imported flag. It is initially unset.

-

A service worker has an associated set of event types to handle (a set) whose item is an event listener’s event type. It is initially a new set.

-

A service worker has an associated set of extended events (a set) whose item is an ExtendableEvent. It is initially a new set.

-
-

2.1.1. Lifetime

-

The lifetime of a service worker is tied to the execution lifetime of events and not references held by service worker clients to the ServiceWorker object.

-

A user agent may terminate service workers at any time it:

-
    -
  • -

    Has no event to handle.

    -
  • -

    Detects abnormal operation: such as infinite loops and tasks exceeding imposed time limits (if any) while handling the events.

    -
-
-
-

2.1.2. Events

-

Service Workers specification defines lifecycle events (each of which is an event), install and activate. Service Workers and other specifications that extend Service Workers define set of events called functional events (each of which is an event) including fetch. (See the list of lifecycle events and functional events.)

-
-
-
-

2.2. Service Worker Registration

-

A service worker registration is a tuple of a scope url and a set of service workers, an installing worker, a waiting worker, and an active worker. A user agent may enable many service worker registrations at a single origin so long as the scope url of the service worker registration differs. A service worker registration of an identical scope url when one already exists in the user agent causes the existing service worker registration to be replaced.

-

A service worker registration has an associated scope url (a URL).

-

A service worker registration has an associated installing worker (a service worker or null) whose state is installing. It is initially set to null.

-

A service worker registration has an associated waiting worker (a service worker or null) whose state is installed. It is initially set to null.

-

A service worker registration has an associated active worker (a service worker or null) whose state is either activating or activated. It is initially set to null.

-

A service worker registration has an associated last update check time. It is initially set to null.

-

A service worker registration has an associated update via cache mode, which is "imports", "all", or "none". It is initially set to "imports".

-

A service worker registration has an associated uninstalling flag. It is initially unset.

-

A service worker registration has one or more task queues that back up the tasks from its active worker’s event loop’s corresponding task queues. (The target task sources for this back up operation are the handle fetch task source and the handle functional event task source.) The user agent dumps the active worker’s tasks to the service worker registration's task queues when the active worker is terminated and re-queues those tasks to the active worker’s event loop’s corresponding task queues when the active worker spins off. Unlike the task queues owned by event loops, the service worker registration's task queues are not processed by any event loops in and of itself.

-

A service worker registration has an associated NavigationPreloadManager object.

-

A service worker registration has an associated navigation preload enabled flag. It is initially unset.

-

A service worker registration has an associated navigation preload header value, which is a byte sequence. It is initially set to `true`.

-
-

2.2.1. Lifetime

-

A user agent must persistently keep a list of registered service worker registrations unless otherwise they are explicitly unregistered. A user agent has a scope to registration map that stores the entries of the tuple of service worker registration's scope url, serialized, and the corresponding service worker registration. The lifetime of service worker registrations is beyond that of the ServiceWorkerRegistration objects which represent them within the lifetime of their corresponding service worker clients.

-
-
-
-

2.3. Service Worker Client

-

A service worker client is an environment.

-

A service worker client has an associated discarded flag. It is initially unset.

-

Each service worker client has the following environment discarding steps:

-
    -
  1. -

    Set client’s discarded flag.

    -
-

Note: Implementations can discard clients whose discarded flag is set.

-

A service worker client has an algorithm defined as the origin that returns the service worker client's origin if the service worker client is an environment settings object, and the service worker client's creation URL’s origin otherwise.

-

A window client is a service worker client whose global object is a Window object.

-

A dedicated worker client is a service worker client whose global object is a DedicatedWorkerGlobalScope object.

-

A shared worker client is a service worker client whose global object is a SharedWorkerGlobalScope object.

-

A worker client is either a dedicated worker client or a shared worker client.

-
-
-

2.4. Selection and Use

-

A service worker client independently selects and uses a service worker registration for its own loading and its subresources. The selection of a service worker registration, upon a non-subresource request, is a process of either matching a service worker registration from scope to registration map or inheriting an existing service worker registration from its parent or owner context depending on the request’s url.

-

When the request’s url is not local, a service worker client matches a service worker registration from scope to registration map. That is, the service worker client attempts to consult a service worker registration whose scope url matches its creation URL.

-

When the request’s url is local, if the service worker client's responsible browsing context is a nested browsing context or the service worker client is a worker client, the service worker client inherits the service worker registration from its parent browsing context’s environment or from the environment of a Document in the service worker client's global object's owner set, respectively, if it exists.

-

If the selection was successful, the selected service worker registration's active worker starts to control the service worker client. Otherwise, the flow returns to fetch where it falls back to the default behavior. When a service worker client is controlled by an active worker, it is considered that the service worker client is using the active worker’s containing service worker registration.

-
-
-

2.5. Task Sources

-

The following additional task sources are used by service workers.

-
-
The handle fetch task source -
-

This task source is used for dispatching fetch events to service workers.

-
The handle functional event task source -
-

This task source is used for features that dispatch other functional events, e.g. push events, to service workers.

-

Note: A user agent may use a separate task source for each functional event type in order to avoid a head-of-line blocking phenomenon for certain functional events.

-
-
-
-

2.6. User Agent Shutdown

-

A user agent must maintain the state of its stored service worker registrations across restarts with the following rules:

- -

To attain this, the user agent must invoke Handle User Agent Shutdown when it terminates.

-
-
-
-

3. Client Context

-
- Bootstrapping with a service worker: -
// scope defaults to the path the script sits in
-// "/" in this example
-navigator.serviceWorker.register("/serviceworker.js").then(registration => {
-  console.log("success!");
-  if (registration.installing) {
-    registration.installing.postMessage("Howdy from your installing page.");
-  }
-}, err => {
-  console.error("Installing the worker failed!", err);
-});
-
-
-
-

3.1. ServiceWorker

-
[SecureContext, Exposed=(Window,Worker)]
-interface ServiceWorker : EventTarget {
-  readonly attribute USVString scriptURL;
-  readonly attribute ServiceWorkerState state;
-  void postMessage(any message, sequence<object> transfer);
-  void postMessage(any message, optional PostMessageOptions options);
-
-  // event
-  attribute EventHandler onstatechange;
-};
-ServiceWorker includes AbstractWorker;
-
-enum ServiceWorkerState {
-  "installing",
-  "installed",
-  "activating",
-  "activated",
-  "redundant"
-};
-
-

A ServiceWorker object represents a service worker. Each ServiceWorker object is associated with a service worker. Multiple separate objects implementing the ServiceWorker interface across documents and workers can all be associated with the same service worker simultaneously.

-

A ServiceWorker object has an associated ServiceWorkerState object which is itself associated with service worker's state.

-
-

3.1.1. scriptURL

-

The scriptURL attribute must return the service worker's serialized script url.

-
- For example, consider a document created by a navigation to https://example.com/app.html which matches via the following registration call which has been previously executed: -
// Script on the page https://example.com/app.html
-navigator.serviceWorker.register("/service_worker.js");
-
-

The value of navigator.serviceWorker.controller.scriptURL will be "https://example.com/service_worker.js".

-
-
-
-

3.1.2. state

-

The state attribute must return the value (in ServiceWorkerState enumeration) to which it was last set.

-
-
-

3.1.3. postMessage(message, transfer)

-

The postMessage(message, transfer) method must run these steps:

-
    -
  1. -

    Let options be «[ "transfer" → transfer ]».

    -
  2. -

    Invoke postMessage(message, options) with message and options as the arguments.

    -
-
-
-

3.1.4. postMessage(message, options)

-

The postMessage(message, options) method must run these steps:

-
    -
  1. -

    If the state attribute value of the context object is "redundant", throw an "InvalidStateError" DOMException.

    -
  2. -

    Let serviceWorker be the service worker represented by the context object.

    -
  3. -

    Invoke Run Service Worker algorithm with serviceWorker as the argument.

    -
  4. -

    Let incumbentSettings be the incumbent settings object, and incumbentGlobal its global object.

    -
  5. -

    Let serializeWithTransferResult be StructuredSerializeWithTransfer(message, options.transfer). Rethrow any exceptions.

    -
  6. -

    Queue a task on the DOM manipulation task source to run the following steps:

    -
      -
    1. -

      Let source be determined by switching on the type of incumbentGlobal:

      -
      -
      ServiceWorkerGlobalScope -
      a new ServiceWorker object that represents incumbentGlobal’s service worker. -
      Window -
      a new WindowClient object that represents incumbentGlobal’s relevant settings object. -
      Otherwise -
      a new Client object that represents incumbentGlobal’s associated worker -
      -
    2. -

      Let origin be the Unicode serialization of incumbentSettings’s origin.

      -
    3. -

      Let destination be the ServiceWorkerGlobalScope object associated with serviceWorker.

      -
    4. -

      Let deserializeRecord be StructuredDeserializeWithTransfer(serializeWithTransferResult, destination’s Realm).

      -

      If this throws an exception, catch it, fire an event named messageerror at destination, using MessageEvent, with the origin attribute initialized to origin and the source attribute initialized to source, and then abort these steps.

      -
    5. -

      Let messageClone be deserializeRecord.[[Deserialized]].

      -
    6. -

      Let newPorts be a new frozen array consisting of all MessagePort objects in deserializeRecord.[[TransferredValues]], if any, maintaining their relative order.

      -
    7. -

      Let e be the result of creating an event named message, using ExtendableMessageEvent, with the origin attribute initialized to origin, the source attribute initialized to source, the data attribute initialized to messageClone, and the ports attribute initialized to newPorts.

      -
    8. -

      Dispatch e at destination.

      -
    9. -

      Invoke Update Service Worker Extended Events Set with serviceWorker and e.

      -
    -
-
-
-

3.1.5. Event handler

-

The following is the event handler (and its corresponding event handler event type) that must be supported, as event handler IDL attributes, by all objects implementing ServiceWorker interface:

- - - - - -
event handler - event handler event type -
onstatechange - statechange -
-
-
-
-

3.2. ServiceWorkerRegistration

-
[SecureContext, Exposed=(Window,Worker)]
-interface ServiceWorkerRegistration : EventTarget {
-  readonly attribute ServiceWorker? installing;
-  readonly attribute ServiceWorker? waiting;
-  readonly attribute ServiceWorker? active;
-  [SameObject] readonly attribute NavigationPreloadManager navigationPreload;
-
-  readonly attribute USVString scope;
-  readonly attribute ServiceWorkerUpdateViaCache updateViaCache;
-
-  [NewObject] Promise<void> update();
-  [NewObject] Promise<boolean> unregister();
-
-  // event
-  attribute EventHandler onupdatefound;
-};
-
-enum ServiceWorkerUpdateViaCache {
-  "imports",
-  "all",
-  "none"
-};
-
-

A ServiceWorkerRegistration object represents a service worker registration. Each ServiceWorkerRegistration object is associated with a service worker registration (a service worker registration). Multiple separate objects implementing the ServiceWorkerRegistration interface across documents and workers can all be associated with the same service worker registration simultaneously.

-
- -

installing attribute must return the value to which it was last set.

-

Note: The ServiceWorker objects returned from this attribute getter that represent the same service worker are the same objects.

-
-
- -

waiting attribute must return the value to which it was last set.

-

Note: The ServiceWorker objects returned from this attribute getter that represent the same service worker are the same objects.

-
-
- -

active attribute must return the value to which it was last set.

-

Note: The ServiceWorker objects returned from this attribute getter that represent the same service worker are the same objects.

-
-
-

3.2.4. navigationPreload

-

The navigationPreload attribute must return service worker registration's NavigationPreloadManager object.

-
-
-

3.2.5. scope

-

The scope attribute must return service worker registration's serialized scope url.

-
In the example in §3.1.1 scriptURL, the value of registration.scope, obtained from navigator.serviceWorker.ready.then(registration => console.log(registration.scope)) for example, will be "https://example.com/".
-
-
-

3.2.6. updateViaCache

-

The updateViaCache attribute must return service worker registration's update via cache mode.

-
-
-

3.2.7. update()

-

update() method must run these steps:

-
    -
  1. -

    Let registration be the service worker registration.

    -
  2. -

    Let newestWorker be the result of running Get Newest Worker algorithm passing registration as its argument.

    -
  3. -

    If newestWorker is null, return a promise rejected with an "InvalidStateError" DOMException and abort these steps.

    -
  4. -

    If the context object’s relevant settings object’s global object globalObject is a ServiceWorkerGlobalScope object, and globalObject’s associated service worker's state is installing, return a promise rejected with an "InvalidStateError" DOMException and abort these steps.

    -
  5. -

    Let promise be a promise.

    -
  6. -

    Let job be the result of running Create Job with update, registration’s scope url, newestWorker’s script url, promise, and the context object’s relevant settings object.

    -
  7. -

    Set job’s worker type to newestWorker’s type.

    -
  8. -

    Invoke Schedule Job with job.

    -
  9. -

    Return promise.

    -
-
-
- -

Note: The unregister() method unregisters the service worker registration. It is important to note that the currently controlled service worker client's active service worker’s containing service worker registration is effective until all the service worker clients (including itself) using this service worker registration unload. That is, the unregister() method only affects subsequent navigations.

-

unregister() method must run these steps:

-
    -
  1. -

    Let promise be a promise.

    -
  2. -

    Let job be the result of running Create Job with unregister, the scope url of the service worker registration, null, promise, and the context object’s relevant settings object.

    -
  3. -

    Invoke Schedule Job with job.

    -
  4. -

    Return promise.

    -
-
-
-

3.2.9. Event handler

-

The following is the event handler (and its corresponding event handler event type) that must be supported, as event handler IDL attributes, by all objects implementing ServiceWorkerRegistration interface:

- - - - - -
event handler - event handler event type -
onupdatefound - updatefound -
-
-
-
- -
partial interface Navigator {
-  [SecureContext, SameObject] readonly attribute ServiceWorkerContainer serviceWorker;
-};
-
-partial interface WorkerNavigator {
-  [SecureContext, SameObject] readonly attribute ServiceWorkerContainer serviceWorker;
-};
-
-

The serviceWorker attribute must return the ServiceWorkerContainer object that is associated with the context object.

-
-
-

3.4. ServiceWorkerContainer

-
[SecureContext, Exposed=(Window,Worker)]
-interface ServiceWorkerContainer : EventTarget {
-  readonly attribute ServiceWorker? controller;
-  readonly attribute Promise<ServiceWorkerRegistration> ready;
-
-  [NewObject] Promise<ServiceWorkerRegistration> register(USVString scriptURL, optional RegistrationOptions options);
-
-  [NewObject] Promise<any> getRegistration(optional USVString clientURL = "");
-  [NewObject] Promise<FrozenArray<ServiceWorkerRegistration>> getRegistrations();
-
-  void startMessages();
-
-
-  // events
-  attribute EventHandler oncontrollerchange;
-  attribute EventHandler onmessage; // event.source of message events is ServiceWorker object
-  attribute EventHandler onmessageerror;
-};
-
-
dictionary RegistrationOptions {
-  USVString scope;
-  WorkerType type = "classic";
-  ServiceWorkerUpdateViaCache updateViaCache = "imports";
-};
-
-

The user agent must create a ServiceWorkerContainer object when a Navigator object or a WorkerNavigator object is created and associate it with that object.

-

A ServiceWorkerContainer provides capabilities to register, unregister, and update the service worker registrations, and provides access to the state of the service worker registrations and their associated service workers.

-

A ServiceWorkerContainer has an associated service worker client, which is a service worker client whose global object is associated with the Navigator object or the WorkerNavigator object that the ServiceWorkerContainer is retrieved from.

-

A ServiceWorkerContainer object has an associated ready promise (a promise). It is initially set to a new promise.

-

A ServiceWorkerContainer object has a task source called the client message queue, initially empty. A client message queue can be enabled or disabled, and is initially disabled. When a ServiceWorkerContainer object’s client message queue is enabled, the event loop must use it as one of its task sources. When the ServiceWorkerContainer object’s relevant global object is a Window object, all tasks queued on its client message queue must be associated with its relevant settings object’s responsible document.

-
- -

controller attribute must run these steps:

-
    -
  1. -

    Let client be the context object’s service worker client.

    -
  2. -

    Return the ServiceWorker object that represents client’s active service worker.

    -
-

Note: navigator.serviceWorker.controller returns null if the request is a force refresh (shift+refresh). The ServiceWorker objects returned from this attribute getter that represent the same service worker are the same objects.

-
-
- -

ready attribute must run these steps:

-
    -
  1. -

    Let readyPromise be the context object's ready promise.

    -
  2. -

    If readyPromise is pending, run the following substeps in parallel:

    -
      -
    1. -

      Let registration be the result of running Match Service Worker Registration with the context object's service worker client's creation URL.

      -
    2. -

      If registration is not null, and registration’s active worker is not null, queue a task on readyPromise’s relevant settings object's responsible event loop, using the DOM manipulation task source, to resolve readyPromise with the ServiceWorkerRegistration object that represents registration in readyPromise’s relevant Realm.

      -
    -
  3. -

    Return readyPromise.

    -
-

Note: The returned ready promise will never reject. If it does not resolve in this algorithm, it will eventually resolve when a matching service worker registration is registered and its active worker is set. (See the relevant Activate algorithm step.)

-
-
- -

Note: The register(scriptURL, options) method creates or updates a service worker registration for the given scope url. If successful, a service worker registration ties the provided scriptURL to a scope url, which is subsequently used for navigation matching.

-

register(scriptURL, options) method must run these steps:

-
    -
  1. -

    Let p be a promise.

    -
  2. -

    Let client be the context object’s service worker client.

    -
  3. -

    Let scriptURL be the result of parsing scriptURL with the context object’s relevant settings object’s API base URL.

    -
  4. -

    Let scopeURL be null.

    -
  5. -

    If options.scope is present, set scopeURL to the result of parsing options.scope with the context object’s relevant settings object’s API base URL.

    -
  6. -

    Invoke Start Register with scopeURL, scriptURL, p, client, client’s creation URL, options.type, and options.updateViaCache.

    -
  7. -

    Return p.

    -
-
-
- -

getRegistration(clientURL) method must run these steps:

-
    -
  1. -

    Let client be the context object’s service worker client.

    -
  2. -

    Let clientURL be the result of parsing clientURL with the context object’s relevant settings object’s API base URL.

    -
  3. -

    If clientURL is failure, return a promise rejected with a TypeError.

    -
  4. -

    Set clientURL’s fragment to null.

    -
  5. -

    If the origin of clientURL is not client’s origin, return a promise rejected with a "SecurityError" DOMException.

    -
  6. -

    Let promise be a new promise.

    -
  7. -

    Run the following substeps in parallel:

    -
      -
    1. -

      Let registration be the result of running Match Service Worker Registration algorithm with clientURL as its argument.

      -
    2. -

      If registration is not null, then:

      -
        -
      1. -

        Resolve promise with the ServiceWorkerRegistration object which represents registration.

        -
      -
    3. -

      Else:

      -
        -
      1. -

        Resolve promise with undefined.

        -
      -
    -
  8. -

    Return promise.

    -
-
-
- -

getRegistrations() method must run these steps:

-
    -
  1. -

    Let client be the context object's service worker client.

    -
  2. -

    Let promise be a new promise.

    -
  3. -

    Run the following steps in parallel:

    -
      -
    1. -

      Let registrations be a new list.

      -
    2. -

      For each scoperegistration of scope to registration map:

      -
        -
      1. -

        If the origin of the result of parsing scope is the same as client’s origin, and registration’s uninstalling flag is unset, then append registration to registrations.

        -
      -
    3. -

      Queue a task on promise’s relevant settings object's responsible event loop, using the DOM manipulation task source, to run the following steps:

      -
        -
      1. -

        Let registrationObjects be a new list.

        -
      2. -

        For each registration of registrations, append the ServiceWorkerRegistration object associated with registration to registrationObjects.

        -
      3. -

        Resolve promise with a new frozen array of registrationObjects in promise’s relevant Realm.

        -
      -
    -
  4. -

    Return promise.

    -
-
-
- -

startMessages() method must enable the context object’s client message queue if it is not enabled.

-
-
-

3.4.7. Event handlers

-

The following are the event handlers (and their corresponding event handler event types) that must be supported, as event handler IDL attributes, by all objects implementing the ServiceWorkerContainer interface:

- - - - - - - -
event handler - event handler event type -
oncontrollerchange - controllerchange -
onmessage - message -
onmessageerror - messageerror -
-

The first time the context object’s onmessage IDL attribute is set, its client message queue must be enabled.

-
-
-
-

3.5. Events

-

The following event is dispatched on ServiceWorker object:

- - - - - -
Event name - Interface - Dispatched when… -
statechange - Event - The state attribute of the ServiceWorker object is changed. -
-

The following event is dispatched on ServiceWorkerRegistration object:

- - - - - -
Event name - Interface - Dispatched when… -
updatefound - Event - The service worker registration's installing worker changes. (See step 8 of the Install algorithm.) -
-

The following events are dispatched on ServiceWorkerContainer object:

- - - - - -
Event name - Interface - Dispatched when… -
controllerchange - Event - The service worker client's active service worker changes. (See step 9.2 of the Activate algorithm. The skip waiting flag of a service worker causes activation of the service worker registration to occur while service worker clients are using the service worker registration, navigator.serviceWorker.controller immediately reflects the active worker as the service worker that controls the service worker client.) -
-
-
-
- -
[SecureContext, Exposed=(Window,Worker)]
-interface NavigationPreloadManager {
-  Promise<void> enable();
-  Promise<void> disable();
-  Promise<void> setHeaderValue(ByteString value);
-  Promise<NavigationPreloadState> getState();
-};
-
-dictionary NavigationPreloadState {
-  boolean enabled = false;
-  ByteString headerValue;
-};
-
-
- -

The enable() method, when invoked, must return a new promise promise and run the following steps in parallel:

-
    -
  1. -

    Let registration be the context object's associated service worker registration.

    -
  2. -

    If registration’s active worker is null, reject promise with an "InvalidStateError" DOMException, and abort these steps.

    -
  3. -

    Set registration’s navigation preload enabled flag.

    -
  4. -

    Resolve promise with undefined.

    -
-
-
- -

The disable() method, when invoked, must return a new promise promise and run the following steps in parallel:

-
    -
  1. -

    Let registration be the context object's associated service worker registration.

    -
  2. -

    If registration’s active worker is null, reject promise with an "InvalidStateError" DOMException, and abort these steps.

    -
  3. -

    Unset registration’s navigation preload enabled flag.

    -
  4. -

    Resolve promise with undefined.

    -
-
-
- -

The setHeaderValue(value) method, when invoked, must return a new promise promise and run the following steps in parallel:

-
    -
  1. -

    Let registration be the context object's associated service worker registration.

    -
  2. -

    If registration’s active worker is null, reject promise with an "InvalidStateError" DOMException, and abort these steps.

    -
  3. -

    Set registration’s navigation preload header value to value.

    -
  4. -

    Resolve promise with undefined.

    -
-
-
- -

The getState() method, when invoked, must return a new promise promise and run the following steps in parallel:

-
    -
  1. -

    Let registration be the context object's associated service worker registration.

    -
  2. -

    Let state be a new NavigationPreloadState dictionary.

    -
  3. -

    If registration’s navigation preload enabled flag is set, set state’s enabled dictionary member to true.

    -
  4. -

    Set state’s headerValue dictionary member to the registration’s navigation preload header value.

    -
  5. -

    Resolve promise with state.

    -
-
-
-
-

4. Execution Context

-
- Serving Cached Resources: -
// caching.js
-self.addEventListener("install", event => {
-  event.waitUntil(
-    // Open a cache of resources.
-    caches.open("shell-v1").then(cache => {
-      // Begins the process of fetching them.
-      // The coast is only clear when all the resources are ready.
-      return cache.addAll([
-        "/app.html",
-        "/assets/v1/base.css",
-        "/assets/v1/app.js",
-        "/assets/v1/logo.png",
-        "/assets/v1/intro_video.webm"
-      ]);
-    })
-  );
-});
-
-self.addEventListener("fetch", event => {
-  // No "fetch" events are dispatched to the service worker until it
-  // successfully installs and activates.
-
-  // All operations on caches are async, including matching URLs, so we use
-  // promises heavily. e.respondWith() even takes promises to enable this:
-  event.respondWith(
-    caches.match(e.request).then(response => {
-      return response || fetch(e.request);
-    }).catch(() => {
-      return caches.match("/fallback.html");
-    })
-  );
-});
-
-
-
-

4.1. ServiceWorkerGlobalScope

-
[Global=(Worker,ServiceWorker), Exposed=ServiceWorker]
-interface ServiceWorkerGlobalScope : WorkerGlobalScope {
-  [SameObject] readonly attribute Clients clients;
-  [SameObject] readonly attribute ServiceWorkerRegistration registration;
-
-  [NewObject] Promise<void> skipWaiting();
-
-  attribute EventHandler oninstall;
-  attribute EventHandler onactivate;
-  attribute EventHandler onfetch;
-
-  // event
-  attribute EventHandler onmessage; // event.source of the message events is Client object
-  attribute EventHandler onmessageerror;
-};
-
-

A ServiceWorkerGlobalScope object represents the global execution context of a service worker. A ServiceWorkerGlobalScope object has an associated service worker (a service worker). A ServiceWorkerGlobalScope object has an associated force bypass cache for importscripts flag. It is initially unset.

-

Note: ServiceWorkerGlobalScope object provides generic, event-driven, time-limited script execution contexts that run at an origin. Once successfully registered, a service worker is started, kept alive and killed by their relationship to events, not service worker clients. Any type of synchronous requests must not be initiated inside of a service worker.

-
-

4.1.1. clients

-

clients attribute must return the Clients object that is associated with the context object.

-
-
-

4.1.2. registration

-

The registration attribute must return the ServiceWorkerRegistration object that represents the service worker's containing service worker registration.

-
-
-

4.1.3. skipWaiting()

-

Note: The skipWaiting() method allows this service worker to progress from the registration's waiting position to active even while service worker clients are using the registration.

-

skipWaiting() method must run these steps:

-
    -
  1. -

    Let promise be a new promise.

    -
  2. -

    Run the following substeps in parallel:

    -
      -
    1. -

      Set service worker's skip waiting flag.

      -
    2. -

      Invoke Try Activate with service worker's containing service worker registration.

      -
    3. -

      Resolve promise with undefined.

      -
    -
  3. -

    Return promise.

    -
-
-
-

4.1.4. Event handlers

-

The following are the event handlers (and their corresponding event handler event types) that must be supported, as event handler IDL attributes, by all objects implementing the ServiceWorkerGlobalScope interface:

- - - - - - - - - -
event handler - event handler event type -
oninstall - install -
onactivate - activate -
onfetch - fetch -
onmessage - message -
onmessageerror - messageerror -
-
-
-
-

4.2. Client

-
[Exposed=ServiceWorker]
-interface Client {
-  readonly attribute USVString url;
-  readonly attribute FrameType frameType;
-  readonly attribute DOMString id;
-  readonly attribute ClientType type;
-  void postMessage(any message, sequence<object> transfer);
-  void postMessage(any message, optional PostMessageOptions options);
-};
-
-[Exposed=ServiceWorker]
-interface WindowClient : Client {
-  readonly attribute VisibilityState visibilityState;
-  readonly attribute boolean focused;
-  [SameObject] readonly attribute FrozenArray<USVString> ancestorOrigins;
-  [NewObject] Promise<WindowClient> focus();
-  [NewObject] Promise<WindowClient?> navigate(USVString url);
-};
-
-enum FrameType {
-  "auxiliary",
-  "top-level",
-  "nested",
-  "none"
-};
-
-

A Client object has an associated service worker client (a service worker client).

-

A Client object has an associated frame type, which is one of "auxiliary", "top-level", "nested", and "none". Unless stated otherwise it is "none".

-

A WindowClient object has an associated browsing context, which is its service worker client's global object's browsing context.

-

A WindowClient object has an associated visibility state, which is one of visibilityState attribute value.

-

A WindowClient object has an associated focus state, which is either true or false (initially false).

-

A WindowClient object has an associated ancestor origins array.

-
-

4.2.1. url

-

The url attribute must return the context object’s associated service worker client's serialized creation URL.

-
-
-

4.2.2. frameType

-

The frameType attribute must return the context object's frame type.

-
-
-

4.2.3. id

-

The id attribute must return its associated service worker client's id.

-
-
-

4.2.4. type

-

The type attribute must run these steps:

-
    -
  1. -

    Let client be context object's service worker client.

    -
  2. -

    If client is an environment settings object, then:

    -
      -
    1. -

      If client is a window client, return "window".

      -
    2. -

      Else if client is a dedicated worker client, return "worker".

      -
    3. -

      Else if client is a shared worker client, return "sharedworker".

      -
    -
  3. -

    Else:

    -
      -
    1. -

      Return "window".

      -
    -
-
-
-

4.2.5. postMessage(message, transfer)

-

The postMessage(message, transfer) method must run these steps:

-
    -
  1. -

    Let options be «[ "transfer" → transfer ]».

    -
  2. -

    Invoke postMessage(message, options) with message and options as the arguments.

    -
-
-
-

4.2.6. postMessage(message, options)

-

The postMessage(message, options) method must run these steps:

-
    -
  1. -

    Let sourceSettings be the context object’s relevant settings object.

    -
  2. -

    Let destination be the ServiceWorkerContainer object whose service worker client is the context object’s service worker client, or null if no match is found.

    -
  3. -

    If destination is null, throw an "InvalidStateError" DOMException.

    -
  4. -

    Let serializeWithTransferResult be StructuredSerializeWithTransfer(message, options.transfer). Rethrow any exceptions.

    -
  5. -

    Add a task that runs the following steps to destination’s client message queue:

    -
      -
    1. -

      Let origin be the Unicode serialization of sourceSettings’s origin.

      -
    2. -

      Let source be a ServiceWorker object, which represents the service worker associated with sourceSettings’s global object.

      -
    3. -

      Let deserializeRecord be StructuredDeserializeWithTransfer(serializeWithTransferResult, destination’s relevant Realm).

      -

      If this throws an exception, catch it, fire an event named messageerror at destination, using MessageEvent, with the origin attribute initialized to origin and the source attribute initialized to source, and then abort these steps.

      -
    4. -

      Let messageClone be deserializeRecord.[[Deserialized]].

      -
    5. -

      Let newPorts be a new frozen array consisting of all MessagePort objects in deserializeRecord.[[TransferredValues]], if any.

      -
    6. -

      Dispatch an event named message at destination, using MessageEvent, with the origin attribute initialized to origin, the source attribute initialized to source, the data attribute initialized to messageClone, and the ports attribute initialized to newPorts.

      -
    -
-
-
-

4.2.7. visibilityState

-

The visibilityState attribute must return the context object’s visibility state.

-
-
-

4.2.8. focused

-

The focused attribute must return the context object’s focus state.

-
-
-

4.2.9. ancestorOrigins

-

The ancestorOrigins attribute must return the context object’s associated ancestor origins array.

-
-
-

4.2.10. focus()

-

The focus() method must run these steps:

-
    -
  1. -

    If this algorithm is not triggered by user activation, return a promise rejected with an "InvalidAccessError" DOMException.

    -
  2. -

    Let serviceWorkerEventLoop be the current global object's event loop.

    -
  3. -

    Let promise be a new promise.

    -
  4. -

    Queue a task to run the following steps on the context object's associated service worker client's responsible event loop using the user interaction task source:

    -
      -
    1. -

      Run the focusing steps with the context object's browsing context.

      -
    2. -

      Let frameType be the result of running Get Frame Type with the context object's browsing context.

      -
    3. -

      Let visibilityState be the context object's browsing context's active document's visibilityState attribute value.

      -
    4. -

      Let focusState be the result of running the has focus steps with the context object's browsing context's active document.

      -
    5. -

      Let ancestorOriginsList be the context object's browsing context's active document's relevant global object's Location object’s ancestor origins list's associated list.

      -
    6. -

      Queue a task to run the following steps on serviceWorkerEventLoop using the DOM manipulation task source:

      -
        -
      1. -

        Let windowClient be the result of running Create Window Client with the context object's associated service worker client, frameType, visibilityState, focusState, and ancestorOriginsList.

        -
      2. -

        If windowClient’s focus state is true, resolve promise with windowClient.

        -
      3. -

        Else, reject promise with a TypeError.

        -
      -
    -
  5. -

    Return promise.

    -
-
-
-

4.2.11. navigate(url)

-

The navigate(url) method must run these steps:

-
    -
  1. -

    Let url be the result of parsing url with the context object’s relevant settings object’s API base URL.

    -
  2. -

    If url is failure, return a promise rejected with a TypeError.

    -
  3. -

    If url is about:blank, return a promise rejected with a TypeError.

    -
  4. -

    If the context object’s associated service worker client's active service worker is not the context object’s relevant global object’s service worker, return a promise rejected with a TypeError.

    -
  5. -

    Let serviceWorkerEventLoop be the current global object's event loop.

    -
  6. -

    Let promise be a new promise.

    -
  7. -

    Queue a task to run the following steps on the context object's associated service worker client's responsible event loop using the user interaction task source:

    -
      -
    1. -

      Let browsingContext be the context object's browsing context.

      -
    2. -

      If browsingContext has discarded its Document, queue a task to reject promise with a TypeError, on serviceWorkerEventLoop using the DOM manipulation task source, and abort these steps.

      -
    3. -

      HandleNavigate: Navigate browsingContext to url with exceptions enabled. The source browsing context must be browsingContext.

      -
    4. -

      If the algorithm steps invoked in the step labeled HandleNavigate throws an exception, queue a task to reject promise with the exception, on serviceWorkerEventLoop using the DOM manipulation task source, and abort these steps.

      -
    5. -

      Let frameType be the result of running Get Frame Type with browsingContext.

      -
    6. -

      Let visibilityState be browsingContext’s active document’s visibilityState attribute value.

      -
    7. -

      Let focusState be the result of running the has focus steps with browsingContext’s active document.

      -
    8. -

      Let ancestorOriginsList be browsingContext’s active document's relevant global object's Location object’s ancestor origins list's associated list.

      -
    9. -

      Queue a task to run the following steps on serviceWorkerEventLoop using the DOM manipulation task source:

      -
        -
      1. -

        If browsingContext’s Window object’s environment settings object’s creation URL’s origin is not the same as the service worker's origin, resolve promise with null and abort these steps.

        -
      2. -

        Let windowClient be the result of running Create Window Client with the context object's service worker client, frameType, visibilityState, focusState, and ancestorOriginsList.

        -
      3. -

        Resolve promise with windowClient.

        -
      -
    -
  8. -

    Return promise.

    -
-
-
-
-

4.3. Clients

-
[Exposed=ServiceWorker]
-interface Clients {
-  // The objects returned will be new instances every time
-  [NewObject] Promise<any> get(DOMString id);
-  [NewObject] Promise<FrozenArray<Client>> matchAll(optional ClientQueryOptions options);
-  [NewObject] Promise<WindowClient?> openWindow(USVString url);
-  [NewObject] Promise<void> claim();
-};
-
-
dictionary ClientQueryOptions {
-  boolean includeUncontrolled = false;
-  ClientType type = "window";
-};
-
-
enum ClientType {
-  "window",
-  "worker",
-  "sharedworker",
-  "all"
-};
-
-

The user agent must create a Clients object when a ServiceWorkerGlobalScope object is created and associate it with that object.

-
-

4.3.1. get(id)

-

The get(id) method must run these steps:

-
    -
  1. -

    Let promise be a new promise.

    -
  2. -

    Run these substeps in parallel:

    -
      -
    1. -

      For each service worker client client whose origin is the same as the associated service worker's origin:

      -
        -
      1. -

        If client’s id is not id, continue.

        -
      2. -

        Wait for either client’s execution ready flag to be set or for client’s discarded flag to be set.

        -
      3. -

        If client’s execution ready flag is set, then invoke Resolve Get Client Promise with client and promise, and abort these steps.

        -
      -
    2. -

      Resolve promise with undefined.

      -
    -
  3. -

    Return promise.

    -
-
-
-

4.3.2. matchAll(options)

-

The matchAll(options) method must run these steps:

-
    -
  1. -

    Let promise be a new promise.

    -
  2. -

    Run the following steps in parallel:

    -
      -
    1. -

      Let targetClients be a new list.

      -
    2. -

      For each service worker client client whose origin is the same as the associated service worker's origin:

      -
        -
      1. -

        If client’s execution ready flag is unset or client’s discarded flag is set, continue.

        -
      2. -

        If client is not a secure context, continue.

        -
      3. -

        If options["includeUncontrolled"] is false, and if client’s active service worker is not the associated service worker, continue.

        -
      4. -

        Add client to targetClients.

        -
      -
    3. -

      Let matchedWindowData be a new list.

      -
    4. -

      Let matchedClients be a new list.

      -
    5. -

      For each service worker client client in targetClients:

      -
        -
      1. -

        If options["type"] is "window" or "all", and client is not an environment settings object or is a window client, then:

        -
          -
        1. -

          Let windowData be «[ "client" → client, "ancestorOriginsList" → a new list ]».

          -
        2. -

          Let browsingContext be null.

          -
        3. -

          Let isClientEnumerable be true.

          -
        4. -

          If client is an environment settings object, set browsingContext to client’s global object's browsing context.

          -
        5. -

          Else, set browsingContext to client’s target browsing context.

          -
        6. -

          Queue a task task to run the following substeps on browsingContext’s event loop using the user interaction task source:

          -
            -
          1. -

            If browsingContext has been discarded, then set isClientEnumerable to false and abort these steps.

            -
          2. -

            If client is a window client and client’s responsible document is not browsingContext’s active document, then set isClientEnumerable to false and abort these steps.

            -
          3. -

            Set windowData["frameType"] to the result of running Get Frame Type with browsingContext.

            -
          4. -

            Set windowData["visibilityState"] to browsingContext’s active document's visibilityState attribute value.

            -
          5. -

            Set windowData["focusState"] to the result of running the has focus steps with browsingContext’s active document as the argument.

            -
          6. -

            If client is a window client, then set windowData["ancestorOriginsList"] to browsingContext’s active document's relevant global object's Location object’s ancestor origins list's associated list.

            -
          -
        7. -

          Wait for task to have executed.

          -

          Note: Wait is a blocking wait, but implementers may run the iterations in parallel as long as the state is not broken.

          -
        8. -

          If isClientEnumerable is true, then:

          -
            -
          1. -

            Add windowData to matchedWindowData.

            -
          -
        -
      2. -

        Else if options["type"] is "worker" or "all" and client is a dedicated worker client, or options["type"] is "sharedworker" or "all" and client is a shared worker client, then:

        -
          -
        1. -

          Add client to matchedClients.

          -
        -
      -
    6. -

      Queue a task to run the following steps on promise’s relevant settings object's responsible event loop using the DOM manipulation task source:

      -
        -
      1. -

        Let clientObjects be a new list.

        -
      2. -

        For each windowData in matchedWindowData:

        -
          -
        1. -

          Let windowClient be the result of running Create Window Client algorithm with windowData["client"], windowData["frameType"], windowData["visibilityState"], windowData["focusState"], and windowData["ancestorOriginsList"] as the arguments.

          -
        2. -

          Append windowClient to clientObjects.

          -
        -
      3. -

        For each client in matchedClients:

        -
          -
        1. -

          Let clientObject be the result of running Create Client algorithm with client as the argument.

          -
        2. -

          Append clientObject to clientObjects.

          -
        -
      4. -

        Sort clientObjects such that:

        - -

        Note: Window clients are always placed before worker clients.

        -
      5. -

        Resolve promise with a new frozen array of clientObjects in promise’s relevant Realm.

        -
      -
    -
  3. -

    Return promise.

    -
-
-
-

4.3.3. openWindow(url)

-

The openWindow(url) method must run these steps:

-
    -
  1. -

    Let url be the result of parsing url with the context object’s relevant settings object’s API base URL.

    -
  2. -

    If url is failure, return a promise rejected with a TypeError.

    -
  3. -

    If url is about:blank, return a promise rejected with a TypeError.

    -
  4. -

    If this algorithm is not triggered by user activation, return a promise rejected with an "InvalidAccessError" DOMException.

    -
  5. -

    Let serviceWorkerEventLoop be the current global object's event loop.

    -
  6. -

    Let promise be a new promise.

    -
  7. -

    Run these substeps in parallel:

    -
      -
    1. -

      Let newContext be a new top-level browsing context.

      -
    2. -

      Queue a task to run the following steps on newContext’s Window object’s environment settings object's responsible event loop using the user interaction task source:

      -
        -
      1. -

        HandleNavigate: Navigate newContext to url with exceptions enabled and replacement enabled.

        -
      2. -

        If the algorithm steps invoked in the step labeled HandleNavigate throws an exception, queue a task to reject promise with the exception, on serviceWorkerEventLoop using the DOM manipulation task source, and abort these steps.

        -
      3. -

        Let frameType be the result of running Get Frame Type with newContext.

        -
      4. -

        Let visibilityState be newContext’s active document’s visibilityState attribute value.

        -
      5. -

        Let focusState be the result of running the has focus steps with newContext’s active document as the argument.

        -
      6. -

        Let ancestorOriginsList be newContext’s active document’s relevant global object’s Location object’s ancestor origins list's associated list.

        -
      7. -

        Queue a task to run the following steps on serviceWorkerEventLoop using the DOM manipulation task source:

        -
          -
        1. -

          If newContext’s Window object’s environment settings object's creation URL's origin is not the same as the service worker's origin, resolve promise with null and abort these steps.

          -
        2. -

          Let client be the result of running Create Window Client with newContext’s Window object’s environment settings object, frameType, visibilityState, focusState, and ancestorOriginsList as the arguments.

          -
        3. -

          Resolve promise with client.

          -
        -
      -
    -
  8. -

    Return promise.

    -
-
-
-

4.3.4. claim()

-

The claim() method must run these steps:

-
    -
  1. -

    If the service worker is not an active worker, return a promise rejected with an "InvalidStateError" DOMException.

    -
  2. -

    Let promise be a new promise.

    -
  3. -

    Run the following substeps in parallel:

    -
      -
    1. -

      For each service worker client client whose origin is the same as the service worker's origin:

      -
        -
      1. -

        If client’s execution ready flag is unset or client’s discarded flag is set, continue.

        -
      2. -

        If client is not a secure context, continue.

        -
      3. -

        Let registration be the result of running Match Service Worker Registration algorithm passing client’s creation URL as the argument.

        -
      4. -

        If registration is not the service worker's containing service worker registration, continue.

        -
      5. -

        If client’s active service worker is not the service worker, then:

        -
          -
        1. -

          Invoke Handle Service Worker Client Unload with client as the argument.

          -
        2. -

          Set client’s active service worker to service worker.

          -
        3. -

          Invoke Notify Controller Change algorithm with client as the argument.

          -
        -
      -
    2. -

      Resolve promise with undefined.

      -
    -
  4. -

    Return promise.

    -
-
-
-
-

4.4. ExtendableEvent

-
[Constructor(DOMString type, optional ExtendableEventInit eventInitDict), Exposed=ServiceWorker]
-interface ExtendableEvent : Event {
-  void waitUntil(Promise<any> f);
-};
-
-
dictionary ExtendableEventInit : EventInit {
-  // Defined for the forward compatibility across the derived events
-};
-
-

An ExtendableEvent object has an associated extend lifetime promises (an array of promises). It is initially an empty array.

-

An ExtendableEvent object has an associated pending promises count (the number of pending promises in the extend lifetime promises). It is initially set to zero.

-

An ExtendableEvent object has an associated timed out flag. It is initially unset, and is set after an optional user agent imposed delay if the pending promises count is greater than zero.

-

An ExtendableEvent object is said to be active when its timed out flag is unset and either its pending promises count is greater than zero or its dispatch flag is set.

-

Service workers have two lifecycle events, install and activate. Service workers use the ExtendableEvent interface for activate event and install event.

-

Service worker extensions that define event handlers may also use or extend the ExtendableEvent interface.

-
-

4.4.1. event.waitUntil(f)

-

waitUntil() method extends the lifetime of the event.

-

waitUntil(f) method must run these steps:

-
    -
  1. -

    If the isTrusted attribute is false, throw an "InvalidStateError" DOMException.

    -
  2. -

    If not active, throw an "InvalidStateError" DOMException.

    -

    Note: If no lifetime extension promise has been added in the task that called the event handlers, calling waitUntil() in subsequent asynchronous tasks will throw.

    -
  3. -

    Add f to the extend lifetime promises.

    -
  4. -

    Increment the pending promises count by one.

    -

    Note: The pending promises count is incremented even if the given promise has already been settled. The corresponding count decrement is done in the microtask queued by the reaction to the promise.

    -
  5. -

    Upon fulfillment or rejection of f, queue a microtask to run these substeps:

    -
      -
    1. -

      Decrement the pending promises count by one.

      -
    2. -

      Let registration be the context object's relevant global object's associated service worker's containing service worker registration.

      -
    3. -

      If registration’s uninstalling flag is set, invoke Try Clear Registration with registration.

      -
    4. -

      If registration is not null, invoke Try Activate with registration.

      -
    -
-

The user agent should not terminate a service worker if Service Worker Has No Pending Events returns false for that service worker.

-
-

Service workers and extensions that define event handlers may define their own behaviors, allowing the extend lifetime promises to suggest operation length, and the rejected state of any of the promise in extend lifetime promises to suggest operation failure.

-

Note: Service workers delay treating the installing worker as installed until all the promises in the install event’s extend lifetime promises resolve successfully. (See the relevant Install algorithm step.) If any of the promises rejects, the installation fails. This is primarily used to ensure that a service worker is not considered installed until all of the core caches it depends on are populated. Likewise, service workers delay treating the active worker as activated until all the promises in the activate event’s extend lifetime promises settle. (See the relevant Activate algorithm step.) This is primarily used to ensure that any functional events are not dispatched to the service worker until it upgrades database schemas and deletes the outdated cache entries.

-
-
-

4.5. FetchEvent

-
[Constructor(DOMString type, FetchEventInit eventInitDict), Exposed=ServiceWorker]
-interface FetchEvent : ExtendableEvent {
-  [SameObject] readonly attribute Request request;
-  readonly attribute Promise<any> preloadResponse;
-  readonly attribute DOMString clientId;
-  readonly attribute DOMString resultingClientId;
-  readonly attribute DOMString replacesClientId;
-
-  void respondWith(Promise<Response> r);
-};
-
-
dictionary FetchEventInit : ExtendableEventInit {
-  required Request request;
-  Promise<any> preloadResponse;
-  DOMString clientId = "";
-  DOMString resultingClientId = "";
-  DOMString replacesClientId = "";
-};
-
-

Service workers have an essential functional event fetch. For fetch event, service workers use the FetchEvent interface which extends the ExtendableEvent interface.

-

Each event using FetchEvent interface has an associated potential response (a response), initially set to null, and the following associated flags that are initially unset:

-
    -
  • -

    wait to respond flag

    -
  • -

    respond-with entered flag

    -
  • -

    respond-with error flag

    -
-
-

4.5.1. event.request

-

request attribute must return the value it was initialized to.

-
-
-

4.5.2. event.preloadResponse

-

preloadResponse attribute must return the value it was initialized to. When an event is created the attribute must be initialized to a promise resolved with undefined.

-
-
-

4.5.3. event.clientId

-

clientId attribute must return the value it was initialized to. When an event is created the attribute must be initialized to the empty string.

-
-
-

4.5.4. event.resultingClientId

-

resultingClientId attribute must return the value it was initialized to. When an event is created the attribute must be initialized to the empty string.

-
-
-

4.5.5. event.replacesClientId

-

replacesClientId attribute must return the value it was initialized to. When an event is created the attribute must be initialized to the empty string.

-
-
-

4.5.6. event.respondWith(r)

-

Note: Developers can set the argument r with either a promise that resolves with a Response object or a Response object (which is automatically cast to a promise). Otherwise, a network error is returned to Fetch. Renderer-side security checks about tainting for cross-origin content are tied to the types of filtered responses defined in Fetch.

-

respondWith(r) method must run these steps:

-
    -
  1. -

    If the dispatch flag is unset, throw an "InvalidStateError" DOMException.

    -
  2. -

    If the respond-with entered flag is set, throw an "InvalidStateError" DOMException.

    -
  3. -

    Add r to the extend lifetime promises.

    -
  4. -

    Increment the pending promises count by one.

    -

    Note: The pending promises count is incremented even if the given promise has already been settled. The corresponding count decrement is done in the microtask queued by the reaction to the promise.

    -
  5. -

    Upon fulfillment or rejection of r, queue a microtask to run these substeps:

    -
      -
    1. -

      Decrement the pending promises count by one.

      -
    2. -

      Let registration be the context object's relevant global object's associated service worker's containing service worker registration.

      -
    3. -

      If registration’s uninstalling flag is set, invoke Try Clear Registration with registration.

      -
    4. -

      If registration is not null, invoke Try Activate with registration.

      -
    -

    Note: event.respondWith(r) extends the lifetime of the event by default as if event.waitUntil(r) is called.

    -
  6. -

    Set the stop propagation flag and stop immediate propagation flag.

    -
  7. -

    Set the respond-with entered flag.

    -
  8. -

    Set the wait to respond flag.

    -
  9. -

    Let targetRealm be the relevant Realm of the context object.

    -
  10. -

    Run the following substeps in parallel:

    -
      -
    1. -

      Wait until r settles.

      -
    2. -

      If r rejected, then:

      -
        -
      1. -

        Set the respond-with error flag.

        -
      -
    3. -

      If r resolved with response, then:

      -
        -
      1. -

        If response is a Response object, then:

        -
          -
        1. -

          If response is disturbed or locked, then:

          -
            -
          1. -

            Set the respond-with error flag.

            -
          -
        2. -

          Else:

          -
            -
          1. -

            Let bytes be an empty byte sequence.

            -
          2. -

            Let end-of-body be false.

            -
          3. -

            Let done be false.

            -
          4. -

            Let potentialResponse be a copy of response’s associated response, except for its body.

            -
          5. -

            If response’s body is non-null, run these substeps:

            -
              -
            1. -

              Let reader be the result of getting a reader from response’s body's stream.

              -
            2. -

              Let highWaterMark be a non-negative, non-NaN number, chosen by the user agent.

              -
            3. -

              Let sizeAlgorithm be an algorithm that accepts a chunk object and returns a non-negative, non-NaN, non-infinite number, chosen by the user agent.

              -
            4. -

              Let pull be an action that runs these subsubsteps:

              -
                -
              1. -

                Let promise be the result of reading a chunk from response’s body's stream with reader.

                -
              2. -

                When promise is fulfilled with an object whose done property is false and whose value property is a Uint8Array object, append the bytes represented by the value property to bytes and perform ! DetachArrayBuffer with the ArrayBuffer object wrapped by the value property.

                -
              3. -

                When promise is fulfilled with an object whose done property is true, set end-of-body to true.

                -
              4. -

                When promise is fulfilled with a value that matches with neither of the above patterns, or promise is rejected, error newStream with a TypeError.

                -
              -
            5. -

              Let cancel be an action that cancels response’s body's stream with reader.

              -
            6. -

              Let newStream be the result of construct a ReadableStream object with highWaterMark, sizeAlgorithm, pull, and cancel in targetRealm.

              -
            7. -

              Set potentialResponse’s body to a new body whose stream is newStream.

              -
            8. -

              Run these subsubsteps repeatedly in parallel while done is false:

              -
                -
              1. -

                If newStream is errored, then set done to true.

                -
              2. -

                Otherwise, if bytes is empty and end-of-body is true, then close newStream and set done to true.

                -
              3. -

                Otherwise, if bytes is not empty, run these subsubsubsteps:

                -
                  -
                1. -

                  Let chunk be a subsequence of bytes starting from the beginning of bytes.

                  -
                2. -

                  Remove chunk from bytes.

                  -
                3. -

                  Let buffer be an ArrayBuffer object created in targetRealm and containing chunk.

                  -
                4. -

                  Enqueue a Uint8Array object created in targetRealm and wrapping buffer to newStream.

                  -
                -
              -
            -

            Note: These substeps are meant to produce the observable equivalent of "piping" response’s body's stream into potentialResponse.

            -
          6. -

            Set the potential response to potentialResponse.

            -
          -
        -
      2. -

        Else:

        -
          -
        1. -

          Set the respond-with error flag.

          -
        -

        Note: If the respond-with error flag is set, a network error is returned to Fetch through Handle Fetch algorithm. (See the step 21.1.) Otherwise, the value response is returned to Fetch through Handle Fetch algorithm. (See the step 22.1.)

        -
      -
    4. -

      Unset the wait to respond flag.

      -
    -
-
-
-
-

4.6. ExtendableMessageEvent

-
[Constructor(DOMString type, optional ExtendableMessageEventInit eventInitDict), Exposed=ServiceWorker]
-interface ExtendableMessageEvent : ExtendableEvent {
-  readonly attribute any data;
-  readonly attribute USVString origin;
-  readonly attribute DOMString lastEventId;
-  [SameObject] readonly attribute (Client or ServiceWorker or MessagePort)? source;
-  readonly attribute FrozenArray<MessagePort> ports;
-};
-
-
dictionary ExtendableMessageEventInit : ExtendableEventInit {
-  any data = null;
-  USVString origin = "";
-  DOMString lastEventId = "";
-  (Client or ServiceWorker or MessagePort)? source = null;
-  sequence<MessagePort> ports = [];
-};
-
-

Service workers define the extendable message event to allow extending the lifetime of the event. For the message event, service workers use the ExtendableMessageEvent interface which extends the ExtendableEvent interface.

-
-

4.6.1. event.data

-

The data attribute must return the value it was initialized to. When the object is created, this attribute must be initialized to null. It represents the message being sent.

-
-
-

4.6.2. event.origin

-

The origin attribute must return the value it was initialized to. When the object is created, this attribute must be initialized to the empty string. It represents the origin of the service worker client that sent the message.

-
-
-

4.6.3. event.lastEventId

-

The lastEventId attribute must return the value it was initialized to. When the object is created, this attribute must be initialized to the empty string.

-
-
-

4.6.4. event.source

-

The source attribute must return the value it was initialized to. When the object is created, this attribute must be initialized to null. It represents the Client object from which the message is sent.

-
-
-

4.6.5. event.ports

-

The ports attribute must return the value it was initialized to. When the object is created, this attribute must be initialized to the empty array. It represents the MessagePort array being sent.

-
-
-
-

4.7. Events

-

The following events are dispatched on ServiceWorkerGlobalScope object:

- - - - - - - - - - - - - - - -
Event name - Interface - Category - Dispatched when… -
install - ExtendableEvent - Lifecycle - The service worker's containing service worker registration’s installing worker changes. (See step 11.2 of the Install algorithm.) -
activate - ExtendableEvent - Lifecycle - The service worker's containing service worker registration’s active worker changes. (See step 12.2 of the Activate algorithm.) -
fetch - FetchEvent - Functional - The http fetch invokes Handle Fetch with request. As a result of performing Handle Fetch, the service worker returns a response to the http fetch. The response, represented by a Response object, can be retrieved from a Cache object or directly from network using self.fetch(input, init) method. (A custom Response object can be another option.) -
push - PushEvent - Functional - (See Firing a push event.) -
notificationclick - NotificationEvent - Functional - (See Activating a notification.) -
notificationclose - NotificationEvent - Functional - (See Closing a notification.) -
sync - SyncEvent - Functional - (See Firing a sync event.) -
canmakepayment - CanMakePaymentEvent - Functional - (See Handling a CanMakePaymentEvent.) -
paymentrequest - PaymentRequestEvent - Functional - (See Handling a PaymentRequestEvent.) -
message - ExtendableMessageEvent - Legacy - When it receives a message. -
messageerror - MessageEvent - Legacy - When it was sent a message that cannot be deserialized. -
-
-
-
-

5. Caches

-

To allow authors to fully manage their content caches for offline use, the Window and the WorkerGlobalScope provide the asynchronous caching methods that open and manipulate Cache objects. An origin can have multiple, named Cache objects, whose contents are entirely under the control of scripts. Caches are not shared across origins, and they are completely isolated from the browser’s HTTP cache.

-
-

5.1. Constructs

-

A request response list is a list of pairs consisting of a request (a request) and a response (a response).

-

The relevant request response list is the instance that the context object represents.

-

A name to cache map is an ordered map whose entry consists of a key (a string that represents the name of a request response list) and a value (a request response list).

-

Each origin has an associated name to cache map.

-

The relevant name to cache map is the instance of the context object's associated global object's environment settings object's origin.

-
-
-

5.2. Understanding Cache Lifetimes

-

The Cache instances are not part of the browser’s HTTP cache. The Cache objects are exactly what authors have to manage themselves. The Cache objects do not get updated unless authors explicitly request them to be. The Cache objects do not expire unless authors delete the entries. The Cache objects do not disappear just because the service worker script is updated. That is, caches are not updated automatically. Updates must be manually managed. This implies that authors should version their caches by name and make sure to use the caches only from the version of the service worker that can safely operate on.

-
-
-

5.3. self.caches

-
partial interface mixin WindowOrWorkerGlobalScope {
-  [SecureContext, SameObject] readonly attribute CacheStorage caches;
-};
-
-
-

5.3.1. caches

-

caches attribute must return this object’s associated CacheStorage object.

-
-
-
-

5.4. Cache

-
[SecureContext, Exposed=(Window,Worker)]
-interface Cache {
-  [NewObject] Promise<any> match(RequestInfo request, optional CacheQueryOptions options);
-  [NewObject] Promise<FrozenArray<Response>> matchAll(optional RequestInfo request, optional CacheQueryOptions options);
-  [NewObject] Promise<void> add(RequestInfo request);
-  [NewObject] Promise<void> addAll(sequence<RequestInfo> requests);
-  [NewObject] Promise<void> put(RequestInfo request, Response response);
-  [NewObject] Promise<boolean> delete(RequestInfo request, optional CacheQueryOptions options);
-  [NewObject] Promise<FrozenArray<Request>> keys(optional RequestInfo request, optional CacheQueryOptions options);
-};
-
-
dictionary CacheQueryOptions {
-  boolean ignoreSearch = false;
-  boolean ignoreMethod = false;
-  boolean ignoreVary = false;
-};
-
-

A Cache object represents a request response list. Multiple separate objects implementing the Cache interface across documents and workers can all be associated with the same request response list simultaneously.

-

A cache batch operation is a struct that consists of:

- -
-

5.4.1. match(request, options)

-

match(request, options) method must run these steps:

-
    -
  1. -

    Let promise be a new promise.

    -
  2. -

    Run these substeps in parallel:

    -
      -
    1. -

      Let p be the result of running the algorithm specified in matchAll(request, options) method with request and options.

      -
    2. -

      Wait until p settles.

      -
    3. -

      If p rejects with an exception, then:

      -
        -
      1. -

        Reject promise with that exception.

        -
      -
    4. -

      Else if p resolves with an array, responses, then:

      -
        -
      1. -

        If responses is an empty array, then:

        -
          -
        1. -

          Resolve promise with undefined.

          -
        -
      2. -

        Else:

        -
          -
        1. -

          Resolve promise with the first element of responses.

          -
        -
      -
    -
  3. -

    Return promise.

    -
-
-
-

5.4.2. matchAll(request, options)

-

matchAll(request, options) method must run these steps:

-
    -
  1. -

    Let r be null.

    -
  2. -

    If the optional argument request is not omitted, then:

    -
      -
    1. -

      If request is a Request object, then:

      -
        -
      1. -

        Set r to request’s request.

        -
      2. -

        If r’s method is not `GET` and options.ignoreMethod is false, return a promise resolved with an empty array.

        -
      -
    2. -

      Else if request is a string, then:

      -
        -
      1. -

        Set r to the associated request of the result of invoking the initial value of Request as constructor with request as its argument. If this throws an exception, return a promise rejected with that exception.

        -
      -
    -
  3. -

    Let realm be the context object's relevant realm.

    -
  4. -

    Let promise be a new promise.

    -
  5. -

    Run these substeps in parallel:

    -
      -
    1. -

      Let responses be an empty list.

      -
    2. -

      If the optional argument request is omitted, then:

      -
        -
      1. -

        For each requestResponse of the relevant request response list:

        -
          -
        1. -

          Add a copy of requestResponse’s response to responses.

          -
        -
      -
    3. -

      Else:

      -
        -
      1. -

        Let requestResponses be the result of running Query Cache with r and options.

        -
      2. -

        For each requestResponse of requestResponses:

        -
          -
        1. -

          Add a copy of requestResponse’s response to responses.

          -
        -
      -
    4. -

      Queue a task, on promise’s relevant settings object's responsible event loop using the DOM manipulation task source, to perform the following steps:

      -
        -
      1. -

        Let responseList be a list.

        -
      2. -

        For each response of responses:

        -
          -
        1. -

          Add a new Response object associated with response and a new Headers object whose guard is "immutable" to responseList.

          -
        -
      3. -

        Resolve promise with a frozen array created from responseList, in realm.

        -
      -
    -
  6. -

    Return promise.

    -
-
-
-

5.4.3. add(request)

-

add(request) method must run these steps:

-
    -
  1. -

    Let requests be an array containing only request.

    -
  2. -

    Let responseArrayPromise be the result of running the algorithm specified in addAll(requests) passing requests as the argument.

    -
  3. -

    Return the result of transforming responseArrayPromise with a fulfillment handler that returns undefined.

    -
-
-
-

5.4.4. addAll(requests)

-

addAll(requests) method must run these steps:

-
    -
  1. -

    Let responsePromises be an empty list.

    -
  2. -

    Let requestList be an empty list.

    -
  3. -

    For each request whose type is Request in requests:

    -
      -
    1. -

      Let r be request’s request.

      -
    2. -

      If r’s url's scheme is not one of "http" and "https", or r’s method is not `GET`, return a promise rejected with a TypeError.

      -
    -
  4. -

    For each request in requests:

    -
      -
    1. -

      Let r be the associated request of the result of invoking the initial value of Request as constructor with request as its argument. If this throws an exception, return a promise rejected with that exception.

      -
    2. -

      If r’s url's scheme is not one of "http" and "https", then:

      -
        -
      1. -

        Terminate all the ongoing fetches initiated by requests with the aborted flag set.

        -
      2. -

        Return a promise rejected with a TypeError.

        -
      -
    3. -

      If r’s client's global object is a ServiceWorkerGlobalScope object, set request’s service-workers mode to "none".

      -
    4. -

      Set r’s initiator to "fetch" and destination to "subresource".

      -
    5. -

      Add r to requestList.

      -
    6. -

      Let responsePromise be a new promise.

      -
    7. -

      Run the following substeps in parallel:

      -
        -
      • -

        Fetch r.

        -
      • -

        To process response for response, run these substeps:

        -
          -
        1. -

          If response’s type is "error", or response’s status is not an ok status or is 206, reject responsePromise with a TypeError.

          -
        2. -

          Else if response’s header list contains a header named `Vary`, then:

          -
            -
          1. -

            Let fieldValues be the list containing the elements corresponding to the field-values of the Vary header.

            -
          2. -

            For each fieldValue of fieldValues:

            -
              -
            1. -

              If fieldValue matches "*", then:

              -
                -
              1. -

                Reject responsePromise with a TypeError.

                -
              2. -

                Terminate all the ongoing fetches initiated by requests with the aborted flag set.

                -
              3. -

                Abort these steps.

                -
              -
            -
          -
        -
      • -

        To process response end-of-body for response, run these substeps:

        -
          -
        1. -

          If response’s aborted flag is set, reject responsePromise with an "AbortError" DOMException and abort these steps.

          -
        2. -

          Resolve responsePromise with response.

          -
        -

        Note: The cache commit is allowed when the response’s body is fully received.

        -
      -
    8. -

      Add responsePromise to responsePromises.

      -
    -
  5. -

    Let p be waiting for all of responsePromises.

    -
  6. -

    Return the result of transforming p with a fulfillment handler that, when called with argument responses, performs the following substeps:

    -
      -
    1. -

      Let operations be an empty list.

      -
    2. -

      Let index be zero.

      -
    3. -

      For each response in responses:

      -
        -
      1. -

        Let operation be a cache batch operation.

        -
      2. -

        Set operation’s type to "put".

        -
      3. -

        Set operation’s request to requestList[index].

        -
      4. -

        Set operation’s response to response.

        -
      5. -

        Append operation to operations.

        -
      6. -

        Increment index by one.

        -
      -
    4. -

      Let realm be the context object's relevant realm.

      -
    5. -

      Let cacheJobPromise be a new promise.

      -
    6. -

      Run the following substeps in parallel:

      -
        -
      1. -

        Let errorData be null.

        -
      2. -

        Invoke Batch Cache Operations with operations. If this throws an exception, set errorData to the exception.

        -
      3. -

        Queue a task, on cacheJobPromise’s relevant settings object's responsible event loop using the DOM manipulation task source, to perform the following substeps:

        -
          -
        1. -

          If errorData is null, resolve cacheJobPromise with undefined.

          -
        2. -

          Else, reject cacheJobPromise with a new exception with errorData and a user agent-defined message, in realm.

          -
        -
      -
    7. -

      Return cacheJobPromise.

      -
    -
-
-
-

5.4.5. put(request, response)

-

put(request, response) method must run these steps:

-
    -
  1. -

    Let innerRequest be null.

    -
  2. -

    If request is a Request object, then set innerRequest to request’s request.

    -
  3. -

    Else:

    -
      -
    1. -

      Let requestObj be the result of invoking Request's constructor with request as its argument. If this throws an exception, return a promise rejected with exception.

      -
    2. -

      Set innerRequest to requestObj’s request.

      -
    -
  4. -

    If innerRequest’s url's scheme is not one of "http" and "https", or innerRequest’s method is not `GET`, return a promise rejected with a TypeError.

    -
  5. -

    Let innerResponse be response’s response.

    -
  6. -

    If innerResponse’s status is 206, return a promise rejected with a TypeError.

    -
  7. -

    If innerResponse’s header list contains a header named `Vary`, then:

    -
      -
    1. -

      Let fieldValues be the list containing the items corresponding to the Vary header’s field-values.

      -
    2. -

      For each fieldValue in fieldValues:

      -
        -
      1. -

        If fieldValue matches "*", return a promise rejected with a TypeError.

        -
      -
    -
  8. -

    If innerResponse’s body is disturbed or locked, return a promise rejected with a TypeError.

    -
  9. -

    Let clonedResponse be a clone of innerResponse.

    -
  10. -

    Let bodyReadPromise be a promise resolved with undefined.

    -
  11. -

    If innerResponse’s body is non-null, run these substeps:

    -
      -
    1. -

      Let stream be innerResponse’s body's stream.

      -
    2. -

      Let reader be the result of getting a reader for stream.

      -
    3. -

      Set bodyReadPromise to the result of reading all bytes from stream with reader.

      -
    -

    Note: This ensures that innerResponse’s body is locked, and we have a full buffered copy of the body in clonedResponse. An implementation could optimize by streaming directly to disk rather than memory.

    -
  12. -

    Let operations be an empty list.

    -
  13. -

    Let operation be a cache batch operation.

    -
  14. -

    Set operation’s type to "put".

    -
  15. -

    Set operation’s request to innerRequest.

    -
  16. -

    Set operation’s response to clonedResponse.

    -
  17. -

    Append operation to operations.

    -
  18. -

    Let realm be the context object's relevant realm.

    -
  19. -

    Return the result of the fulfillment of bodyReadPromise:

    -
      -
    1. -

      Let cacheJobPromise be a new promise.

      -
    2. -

      Return cacheJobPromise and run these steps in parallel:

      -
        -
      1. -

        Let errorData be null.

        -
      2. -

        Invoke Batch Cache Operations with operations. If this throws an exception, set errorData to the exception.

        -
      3. -

        Queue a task, on cacheJobPromise’s relevant settings object's responsible event loop using the DOM manipulation task source, to perform the following substeps:

        -
          -
        1. -

          If errorData is null, resolve cacheJobPromise with undefined.

          -
        2. -

          Else, reject cacheJobPromise with a new exception with errorData and a user agent-defined message, in realm.

          -
        -
      -
    -
-
-
-

5.4.6. delete(request, options)

-

delete(request, options) method must run these steps:

-
    -
  1. -

    Let r be null.

    -
  2. -

    If request is a Request object, then:

    -
      -
    1. -

      Set r to request’s request.

      -
    2. -

      If r’s method is not `GET` and options.ignoreMethod is false, return a promise resolved with false.

      -
    -
  3. -

    Else if request is a string, then:

    -
      -
    1. -

      Set r to the associated request of the result of invoking the initial value of Request as constructor with request as its argument. If this throws an exception, return a promise rejected with that exception.

      -
    -
  4. -

    Let operations be an empty list.

    -
  5. -

    Let operation be a cache batch operation.

    -
  6. -

    Set operation’s type to "delete".

    -
  7. -

    Set operation’s request to r.

    -
  8. -

    Set operation’s options to options.

    -
  9. -

    Append operation to operations.

    -
  10. -

    Let realm be the context object's relevant realm.

    -
  11. -

    Let cacheJobPromise be a new promise.

    -
  12. -

    Run the following substeps in parallel:

    -
      -
    1. -

      Let errorData be null.

      -
    2. -

      Let requestResponses be the result of running Batch Cache Operations with operations. If this throws an exception, set errorData to the exception.

      -
    3. -

      Queue a task, on cacheJobPromise’s relevant settings object's responsible event loop using the DOM manipulation task source, to perform the following substeps:

      -
        -
      1. -

        If errorData is null, then:

        -
          -
        1. -

          If requestResponses is not empty, resolve cacheJobPromise with true.

          -
        2. -

          Else, resolve cacheJobPromise with false.

          -
        -
      2. -

        Else, reject cacheJobPromise with a new exception with errorData and a user agent-defined message, in realm.

        -
      -
    -
  13. -

    Return cacheJobPromise.

    -
-
-
-

5.4.7. keys(request, options)

-

keys(request, options) method must run these steps:

-
    -
  1. -

    Let r be null.

    -
  2. -

    If the optional argument request is not omitted, then:

    -
      -
    1. -

      If request is a Request object, then:

      -
        -
      1. -

        Set r to request’s request.

        -
      2. -

        If r’s method is not `GET` and options.ignoreMethod is false, return a promise resolved with an empty array.

        -
      -
    2. -

      Else if request is a string, then:

      -
        -
      1. -

        Set r to the associated request of the result of invoking the initial value of Request as constructor with request as its argument. If this throws an exception, return a promise rejected with that exception.

        -
      -
    -
  3. -

    Let realm be the context object's relevant realm.

    -
  4. -

    Let promise be a new promise.

    -
  5. -

    Run these substeps in parallel:

    -
      -
    1. -

      Let requests be an empty list.

      -
    2. -

      If the optional argument request is omitted, then:

      -
        -
      1. -

        For each requestResponse of the relevant request response list:

        -
          -
        1. -

          Add requestResponse’s request to requests.

          -
        -
      -
    3. -

      Else:

      -
        -
      1. -

        Let requestResponses be the result of running Query Cache with r and options.

        -
      2. -

        For each requestResponse of requestResponses:

        -
          -
        1. -

          Add requestResponse’s request to requests.

          -
        -
      -
    4. -

      Queue a task, on promise’s relevant settings object's responsible event loop using the DOM manipulation task source, to perform the following steps:

      -
        -
      1. -

        Let requestList be a list.

        -
      2. -

        For each request of requests:

        -
          -
        1. -

          Add a new Request object associated with request and a new associated Headers object whose guard is "immutable" to requestList.

          -
        -
      3. -

        Resolve promise with a frozen array created from requestList, in realm.

        -
      -
    -
  6. -

    Return promise.

    -
-
-
-
-

5.5. CacheStorage

-
[SecureContext, Exposed=(Window,Worker)]
-interface CacheStorage {
-  [NewObject] Promise<any> match(RequestInfo request, optional MultiCacheQueryOptions options);
-  [NewObject] Promise<boolean> has(DOMString cacheName);
-  [NewObject] Promise<Cache> open(DOMString cacheName);
-  [NewObject] Promise<boolean> delete(DOMString cacheName);
-  [NewObject] Promise<sequence<DOMString>> keys();
-};
-
-dictionary MultiCacheQueryOptions : CacheQueryOptions {
-  DOMString cacheName;
-};
-
-

Note: CacheStorage interface is designed to largely conform to ECMAScript 6 Map objects but entirely async, and with additional convenience methods. The methods, clear, forEach, entries and values, are intentionally excluded from the scope of the first version resorting to the ongoing discussion about the async iteration by TC39.

-

The user agent must create a CacheStorage object when a Window object or a WorkerGlobalScope object is created and associate it with that global object.

-

A CacheStorage object represents a name to cache map of its associated global object's environment settings object’s origin. Multiple separate objects implementing the CacheStorage interface across documents and workers can all be associated with the same name to cache map simultaneously.

-
-

5.5.1. match(request, options)

-

match(request, options) method must run these steps:

-
    -
  1. -

    If options.cacheName is present, then:

    -
      -
    1. -

      Return a new promise promise and run the following substeps in parallel:

      -
        -
      1. -

        For each cacheNamecache of the relevant name to cache map:

        -
          -
        1. -

          If options.cacheName matches cacheName, then:

          -
            -
          1. -

            Resolve promise with the result of running the algorithm specified in match(request, options) method of Cache interface with request and options (providing cache as thisArgument to the [[Call]] internal method of match(request, options).)

            -
          2. -

            Abort these steps.

            -
          -
        -
      2. -

        Resolve promise with undefined.

        -
      -
    -
  2. -

    Else:

    -
      -
    1. -

      Let promise be a promise resolved with undefined.

      -
    2. -

      For each cacheNamecache of the relevant name to cache map:

      -
        -
      1. -

        Set promise to the result of transforming itself with a fulfillment handler that, when called with argument response, performs the following substeps:

        -
          -
        1. -

          If response is not undefined, return response.

          -
        2. -

          Return the result of running the algorithm specified in match(request, options) method of Cache interface with request and options as the arguments (providing cache as thisArgument to the [[Call]] internal method of match(request, options).)

          -
        -
      -
    3. -

      Return promise.

      -
    -
-
-
-

5.5.2. has(cacheName)

-

has(cacheName) method must run these steps:

-
    -
  1. -

    Let promise be a new promise.

    -
  2. -

    Run the following substeps in parallel:

    -
      -
    1. -

      For each keyvalue of the relevant name to cache map:

      -
        -
      1. -

        If cacheName matches key, resolve promise with true and abort these steps.

        -
      -
    2. -

      Resolve promise with false.

      -
    -
  3. -

    Return promise.

    -
-
-
-

5.5.3. open(cacheName)

-

open(cacheName) method must run these steps:

-
    -
  1. -

    Let promise be a new promise.

    -
  2. -

    Run the following substeps in parallel:

    -
      -
    1. -

      For each keyvalue of the relevant name to cache map:

      -
        -
      1. -

        If cacheName matches key, then:

        -
          -
        1. -

          Resolve promise with a new Cache object that represents value.

          -
        2. -

          Abort these steps.

          -
        -
      -
    2. -

      Let cache be a new request response list.

      -
    3. -

      Set the relevant name to cache map[cacheName] to cache. If this cache write operation failed due to exceeding the granted quota limit, reject promise with a "QuotaExceededError" DOMException and abort these steps.

      -
    4. -

      Resolve promise with a new Cache object that represents cache.

      -
    -
  3. -

    Return promise.

    -
-
-
-

5.5.4. delete(cacheName)

-

delete(cacheName) method must run these steps:

-
    -
  1. -

    Let promise be the result of running the algorithm specified in has(cacheName) method with cacheName.

    -
  2. -

    Return the result of transforming promise with a fulfillment handler that, when called with argument cacheExists, performs the following substeps:

    -
      -
    1. -

      If cacheExists is false, then:

      -
        -
      1. -

        Return false.

        -
      -
    2. -

      Let cacheJobPromise be a new promise.

      -
    3. -

      Run the following substeps in parallel:

      -
        -
      1. -

        Remove the relevant name to cache map[cacheName].

        -
      2. -

        Resolve cacheJobPromise with true.

        -
      -

      Note: After this step, the existing DOM objects (i.e. the currently referenced Cache, Request, and Response objects) should remain functional.

      -
    4. -

      Return cacheJobPromise.

      -
    -
-
-
-

5.5.5. keys()

-

keys() method must run these steps:

-
    -
  1. -

    Let promise be a new promise.

    -
  2. -

    Run the following substeps in parallel:

    -
      -
    1. -

      Let cacheKeys be the result of getting the keys of the relevant name to cache map.

      -

      Note: The items in the result ordered set are in the order that their corresponding entry was added to the name to cache map.

      -
    2. -

      Resolve promise with cacheKeys.

      -
    -
  3. -

    Return promise.

    -
-
-
-
-
-

6. Security Considerations

-
-

6.1. Secure Context

-

Service workers must execute in secure contexts. Service worker clients must also be secure contexts to register a service worker registration, to get access to the service worker registrations and the service workers, to do messaging with the service workers, and to be manipulated by the service workers.

-

Note: This effectively means that service workers and their service worker clients need to be hosted over HTTPS. A user agent can allow localhost (see the requirements), 127.0.0.0/8, and ::1/128 for development purposes. The primary reason for this restriction is to protect users from the risks associated with insecure contexts.

-
-
-

6.2. Content Security Policy

-

Whenever a user agent invokes Run Service Worker algorithm with a service worker serviceWorker:

-
    -
  • -

    If serviceWorker’s script resource was delivered with a Content-Security-Policy HTTP header containing the value policy, the user agent must enforce policy for serviceWorker.

    -
  • -

    If serviceWorker’s script resource was delivered with a Content-Security-Policy-Report-Only HTTP header containing the value policy, the user agent must monitor policy for serviceWorker.

    -
-

The primary reason for this restriction is to mitigate a broad class of content injection vulnerabilities, such as cross-site scripting (XSS).

-
-
-

6.3. Origin Relativity

-
-

6.3.1. Origin restriction

-

This section is non-normative.

-

A service worker executes in the registering service worker client's origin. One of the advanced concerns that major applications would encounter is whether they can be hosted from a CDN. By definition, these are servers in other places, often on other origins. Therefore, service workers cannot be hosted on CDNs. But they can include resources via importScripts(). The reason for this restriction is that service workers create the opportunity for a bad actor to turn a bad day into a bad eternity.

-
-
-

6.3.2. importScripts(urls)

-

When the importScripts(urls) method is called on a ServiceWorkerGlobalScope object, the user agent must import scripts into worker global scope, given this ServiceWorkerGlobalScope object and urls, and with the following steps to perform the fetch given the request request:

-
    -
  1. -

    Let serviceWorker be request’s client's global object's service worker.

    -
  2. -

    Let map be serviceWorker’s script resource map.

    -
  3. -

    Let url be request’s url.

    -
  4. -

    If serviceWorker’s state is not parsed or installing:

    -
      -
    1. -

      Return map[url] if it exists and a network error otherwise.

      -
    -
  5. -

    If map[url] exists:

    -
      -
    1. -

      Append url to serviceWorker’s set of used scripts.

      -
    2. -

      Return map[url].

      -
    -
  6. -

    Let registration be serviceWorker’s containing service worker registration.

    -
  7. -

    Set request’s service-workers mode to "none".

    -
  8. -

    Set request’s cache mode to "no-cache" if any of the following are true:

    - -
  9. -

    Let response be the result of fetching request.

    -
  10. -

    If response’s cache state is not "local", set registration’s last update check time to the current time.

    -
  11. -

    If response’s unsafe response is a bad import script response, then return a network error.

    -
  12. -

    Set map[url] to response.

    -
  13. -

    Append url to serviceWorker’s set of used scripts.

    -
  14. -

    Set serviceWorker’s classic scripts imported flag.

    -
  15. -

    Return response.

    -
-
-
-
-

6.4. Cross-Origin Resources and CORS

-

This section is non-normative.

-

Applications tend to cache items that come from a CDN or other origin. It is possible to request many of them directly using <script>, <img>, <video> and <link> elements. It would be hugely limiting if this sort of runtime collaboration broke when offline. Similarly, it is possible to fetch many sorts of off-origin resources when appropriate CORS headers are set. Service workers enable this by allowing Caches to fetch and cache off-origin items. Some restrictions apply, however. First, unlike same-origin resources which are managed in the Cache as Response objects whose corresponding responses are basic filtered response, the objects stored are Response objects whose corresponding responses are either CORS filtered responses or opaque filtered responses. They can be passed to event.respondWith(r) method in the same manner as the Response objects whose corresponding responses are basic filtered responses, but cannot be meaningfully created programmatically. These limitations are necessary to preserve the security invariants of the platform. Allowing Caches to store them allows applications to avoid re-architecting in most cases.

-
-
-

6.5. Implementer Concerns

-

This section is non-normative.

-

The implementers are encouraged to note:

-
    -
  • -

    Plug-ins should not load via service workers. As plug-ins may get their security origins from their own urls, the embedding service worker cannot handle it. For this reason, the Handle Fetch algorithm makes the potential-navigation-or-subresource request (whose context is either <embed> or <object>) immediately fallback to the network without dispatching fetch event.

    -
  • -

    Some of the legacy networking stack code may need to be carefully audited to understand the ramifications of interactions with service workers.

    -
-
-
-

6.6. Privacy

-

Service workers introduce new persistent storage features including scope to registration map (for service worker registrations and their service workers), request response list and name to cache map (for caches), and script resource map (for script resources). In order to protect users from any potential unsanctioned tracking threat, these persistent storages should be cleared when users intend to clear them and should maintain and interoperate with existing user controls e.g. purging all existing persistent storages.

-
-
-
-

7. Extensibility

-

Service Workers specification is extensible from other specifications.

-
-

7.1. Define API bound to Service Worker Registration

-

Specifications may define an API tied to a service worker registration by using partial interface definition to the ServiceWorkerRegistration interface where it may define the specification specific attributes and methods:

-
partial interface ServiceWorkerRegistration {
-  // e.g. define an API namespace
-  readonly attribute APISpaceType APISpace;
-  // e.g. define a method
-  Promise<T> methodName(/* list of arguments */);
-};
-
-
-
-

7.2. Define Functional Event

-

Specifications may define a functional event by extending ExtendableEvent interface:

-
// e.g. define FunctionalEvent interface
-interface FunctionalEvent : ExtendableEvent {
-  // add a functional event’s own attributes and methods
-};
-
-
-
-

7.3. Define Event Handler

-

Specifications may define an event handler attribute for the corresponding functional event using partial interface definition to the ServiceWorkerGlobalScope interface:

-
partial interface ServiceWorkerGlobalScope {
-  attribute EventHandler onfunctionalevent;
-};
-
-
-
-

7.4. Firing Functional Events

-

To request a functional event dispatch to the active worker of a service worker registration, specifications should invoke Fire Functional Event.

-
-
-
-

Appendix A: Algorithms

-

The following definitions are the user agent’s internal data structures used throughout the specification.

-

A scope to registration map is an ordered map where the keys are scope urls, serialized, and the values are service worker registrations.

-

A job is an abstraction of one of register, update, and unregister request for a service worker registration.

-
- A job has a job type, which is one of register, update, and unregister. -

A job has a scope url (a URL).

-

A job has a script url (a URL).

-

A job has a worker type ("classic" or "module").

-

A job has an update via cache mode, which is "imports", "all", or "none".

-

A job has a client (a service worker client). It is initially null.

-

A job has a referrer (a URL or null).

-

A job has a job promise (a promise). It is initially null.

-

A job has a containing job queue (a job queue or null). It is initially null.

-

A job has a list of equivalent jobs (a list of jobs). It is initially the empty list.

-

A job has a force bypass cache flag. It is initially unset.

-
-

Two jobs are equivalent when their job type is the same and:

- -

A job queue is a thread safe queue used to synchronize the set of concurrent jobs. The job queue contains jobs as its items. A job queue is initially empty.

-

A scope to job queue map is an ordered map where the keys are scope urls, serialized, and the values are job queues.

-

A bad import script response is a response for which any of the following conditions are met:

- -
-

Create Job

-
-
Input -
-

jobType, a job type

-
-

scopeURL, a URL

-
-

scriptURL, a URL

-
-

promise, a promise

-
-

client, a service worker client

-
Output -
-

job, a job

-
-
    -
  1. -

    Let job be a new job.

    -
  2. -

    Set job’s job type to jobType.

    -
  3. -

    Set job’s scope url to scopeURL.

    -
  4. -

    Set job’s script url to scriptURL.

    -
  5. -

    Set job’s job promise to promise.

    -
  6. -

    Set job’s client to client.

    -
  7. -

    If client is not null, set job’s referrer to client’s creation URL.

    -
  8. -

    Return job.

    -
-
-
-

Schedule Job

-
-
Input -
-

job, a job

-
Output -
-

none

-
-
    -
  1. -

    Let jobQueue be null.

    -
  2. -

    Let jobScope be job’s scope url, serialized.

    -
  3. -

    If scope to job queue map[jobScope] does not exist, set scope to job queue map[jobScope] to a new job queue.

    -
  4. -

    Set jobQueue to scope to job queue map[jobScope].

    -
  5. -

    If jobQueue is empty, then:

    -
      -
    1. -

      Set job’s containing job queue to jobQueue, and enqueue job to jobQueue.

      -
    2. -

      Invoke Run Job with jobQueue.

      -
    -
  6. -

    Else:

    -
      -
    1. -

      Let lastJob be the element at the back of jobQueue.

      -
    2. -

      If job is equivalent to lastJob and lastJob’s job promise has not settled, append job to lastJob’s list of equivalent jobs.

      -
    3. -

      Else, set job’s containing job queue to jobQueue, and enqueue job to jobQueue.

      -
    -
-
-
-

Run Job

-
-
Input -
-

jobQueue, a job queue

-
Output -
-

none

-
-
    -
  1. -

    Assert: jobQueue is not empty.

    -
  2. -

    Queue a task to run these steps:

    -
      -
    1. -

      Let job be the first item in jobQueue.

      -
    2. -

      If job’s job type is register, run Register with job in parallel.

      -
    3. -

      Else if job’s job type is update, run Update with job in parallel.

      -

      Note: For a register job and an update job, the user agent delays queuing a task for running the job until after a DOMContentLoaded event has been dispatched to the document that initiated the job.

      -
    4. -

      Else if job’s job type is unregister, run Unregister with job in parallel.

      -
    -
-
-
-

Finish Job

-
-
Input -
-

job, a job

-
Output -
-

none

-
-
    -
  1. -

    Let jobQueue be job’s containing job queue.

    -
  2. -

    Assert: the first item in jobQueue is job.

    -
  3. -

    Dequeue from jobQueue.

    -
  4. -

    If jobQueue is not empty, invoke Run Job with jobQueue.

    -
-
-
-

Resolve Job Promise

-
-
Input -
-

job, a job

-
-

value, any

-
Output -
-

none

-
-
    -
  1. -

    Let convertedValue be null.

    -
  2. -

    If job’s client is not null, queue a task, on job’s client's responsible event loop using the DOM manipulation task source, to run the following substeps:

    -
      -
    1. -

      If job’s job type is either register or update, set convertedValue to the ServiceWorkerRegistration object that represents value, in job’s client's Realm.

      -
    2. -

      Else, set convertedValue to value, in job’s client's Realm.

      -
    3. -

      Resolve job’s job promise with convertedValue.

      -
    -
  3. -

    For each equivalentJob in job’s list of equivalent jobs:

    -
      -
    1. -

      If equivalentJob’s client is null, continue to the next iteration of the loop.

      -
    2. -

      Queue a task, on equivalentJob’s client's responsible event loop using the DOM manipulation task source, to run the following substeps:

      -
        -
      1. -

        If equivalentJob’s job type is either register or update, set convertedValue to the ServiceWorkerRegistration object that represents value, in equivalentJob’s client's Realm.

        -
      2. -

        Else, set convertedValue to value, in equivalentJob’s client's Realm.

        -
      3. -

        Resolve equivalentJob’s job promise with convertedValue.

        -
      -
    -
-
-
-

Reject Job Promise

-
-
Input -
-

job, a job

-
-

errorData, the information necessary to create an exception

-
Output -
-

none

-
-
    -
  1. -

    If job’s client is not null, queue a task, on job’s client's responsible event loop using the DOM manipulation task source, to reject job’s job promise with a new exception with errorData and a user agent-defined message, in job’s client's Realm.

    -
  2. -

    For each equivalentJob in job’s list of equivalent jobs:

    -
      -
    1. -

      If equivalentJob’s client is null, continue.

      -
    2. -

      Queue a task, on equivalentJob’s client's responsible event loop using the DOM manipulation task source, to reject equivalentJob’s job promise with a new exception with errorData and a user agent-defined message, in equivalentJob’s client's Realm.

      -
    -
-
-
-

Start Register

-
-
Input -
-

scopeURL, a URL or failure or null

-
-

scriptURL, a URL or failure

-
-

promise, a promise

-
-

client, a service worker client

-
-

referrer, a URL

-
-

workerType, a worker type

-
-

updateViaCache, an update via cache mode

-
Output -
-

none

-
-
    -
  1. -

    If scriptURL is failure, reject promise with a TypeError and abort these steps.

    -
  2. -

    Set scriptURL’s fragment to null.

    -

    Note: The user agent does not store the fragment of the script’s url. This means that the fragment does not have an effect on identifying service workers.

    -
  3. -

    If scriptURL’s scheme is not one of "http" and "https", reject promise with a TypeError and abort these steps.

    -
  4. -

    If any of the strings in scriptURL’s path contains either ASCII case-insensitive "%2f" or ASCII case-insensitive "%5c", reject promise with a TypeError and abort these steps.

    -
  5. -

    If scopeURL is null, set scopeURL to the result of parsing the string "./" with scriptURL.

    -

    Note: The scope url for the registration is set to the location of the service worker script by default.

    -
  6. -

    If scopeURL is failure, reject promise with a TypeError and abort these steps.

    -
  7. -

    Set scopeURL’s fragment to null.

    -

    Note: The user agent does not store the fragment of the scope url. This means that the fragment does not have an effect on identifying service worker registrations.

    -
  8. -

    If scopeURL’s scheme is not one of "http" and "https", reject promise with a TypeError and abort these steps.

    -
  9. -

    If any of the strings in scopeURL’s path contains either ASCII case-insensitive "%2f" or ASCII case-insensitive "%5c", reject promise with a TypeError and abort these steps.

    -
  10. -

    Let job be the result of running Create Job with register, scopeURL, scriptURL, promise, and client.

    -
  11. -

    Set job’s worker type to workerType.

    -
  12. -

    Set job’s update via cache mode to updateViaCache.

    -
  13. -

    Set job’s referrer to referrer.

    -
  14. -

    Invoke Schedule Job with job.

    -
-
-
-

Register

-
-
Input -
-

job, a job

-
Output -
-

none

-
-
    -
  1. -

    If the result of running potentially trustworthy origin with the origin of job’s script url as the argument is Not Trusted, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and "SecurityError" DOMException.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  2. -

    If the origin of job’s script url is not job’s referrer's origin, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and "SecurityError" DOMException.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  3. -

    If the origin of job’s scope url is not job’s referrer's origin, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and "SecurityError" DOMException.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  4. -

    Let registration be the result of running the Get Registration algorithm passing job’s scope url as the argument.

    -
  5. -

    If registration is not null, then:

    -
      -
    1. -

      If registration’s uninstalling flag is set, unset it.

      -
    2. -

      Let newestWorker be the result of running the Get Newest Worker algorithm passing registration as the argument.

      -
    3. -

      If newestWorker is not null, job’s script url equals newestWorker’s script url, and job’s update via cache mode's value equals registration’s update via cache mode, then:

      -
        -
      1. -

        Invoke Resolve Job Promise with job and registration.

        -
      2. -

        Invoke Finish Job with job and abort these steps.

        -
      -
    -
  6. -

    Else:

    -
      -
    1. -

      Invoke Set Registration algorithm with job’s scope url and job’s update via cache mode.

      -
    -
  7. -

    Invoke Update algorithm passing job as the argument.

    -
-
-
-

Update

-
-
Input -
-

job, a job

-
Output -
-

none

-
-
    -
  1. -

    Let registration be the result of running the Get Registration algorithm passing job’s scope url as the argument.

    -
  2. -

    If registration is null or registration’s uninstalling flag is set, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and TypeError.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  3. -

    Let newestWorker be the result of running Get Newest Worker algorithm passing registration as the argument.

    -
  4. -

    If job’s job type is update, and newestWorker is not null and its script url does not equal job’s script url, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and TypeError.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  5. -

    Let httpsState be "none".

    -
  6. -

    Let referrerPolicy be the empty string.

    -
  7. -

    Let hasUpdatedResources be false.

    -
  8. -

    Let updatedResourceMap be an ordered map where the keys are URLs and the values are responses.

    -
  9. -

    Switching on job’s worker type, run these substeps with the following options:

    -
    -
    "classic" -
    -

    Fetch a classic worker script given job’s serialized script url, job’s client, "serviceworker", and the to-be-created environment settings object for this service worker.

    -
    "module" -
    -

    Fetch a module worker script graph given job’s serialized script url, job’s client, "serviceworker", "omit", and the to-be-created environment settings object for this service worker.

    -
    -

    To perform the fetch given request, run the following steps:

    -
      -
    1. -

      Append `Service-Worker`/`script` to request’s header list.

      -

      Note: See the definition of the Service-Worker header in Appendix B: Extended HTTP headers.

      -
    2. -

      Set request’s cache mode to "no-cache" if any of the following are true:

      - -

      Note: Even if the cache mode is not set to "no-cache", the user agent obeys Cache-Control header’s max-age value in the network layer to determine if it should bypass the browser cache.

      -
    3. -

      Set request’s service-workers mode to "none".

      -
    4. -

      If the is top-level flag is unset, then return the result of fetching request.

      -
    5. -

      Set request’s redirect mode to "error".

      -
    6. -

      Fetch request, and asynchronously wait to run the remaining steps as part of fetch’s process response for the response response.

      -
    7. -

      Extract a MIME type from the response’s header list. If this MIME type (ignoring parameters) is not a JavaScript MIME type, then:

      -
        -
      1. -

        Invoke Reject Job Promise with job and "SecurityError" DOMException.

        -
      2. -

        Asynchronously complete these steps with a network error.

        -
      -
    8. -

      Let serviceWorkerAllowed be the result of extracting header list values given `Service-Worker-Allowed` and response’s header list.

      -

      Note: See the definition of the Service-Worker-Allowed header in Appendix B: Extended HTTP headers.

      -
    9. -

      Set httpsState to response’s HTTPS state.

      -
    10. -

      Set referrerPolicy to the result of parse a referrer policy from a Referrer-Policy header of response.

      -
    11. -

      If serviceWorkerAllowed is failure, then:

      -
        -
      1. -

        Asynchronously complete these steps with a network error.

        -
      -
    12. -

      Let scopeURL be registration’s scope url.

      -
    13. -

      Let maxScopeString be null.

      -
    14. -

      If serviceWorkerAllowed is null, then:

      -
        -
      1. -

        Set maxScopeString to "/" concatenated with the strings, except the last string that denotes the script’s file name, in job’s script url's path (including empty strings), separated from each other by "/".

        -
      -
    15. -

      Else:

      -
        -
      1. -

        Let maxScope be the result of parsing serviceWorkerAllowed with job’s script url.

        -
      2. -

        Set maxScopeString to "/" concatenated with the strings in maxScope’s path (including empty strings), separated from each other by "/".

        -
      -
    16. -

      Let scopeString be "/" concatenated with the strings in scopeURL’s path (including empty strings), separated from each other by "/".

      -
    17. -

      If scopeString starts with maxScopeString, do nothing.

      -
    18. -

      Else:

      -
        -
      1. -

        Invoke Reject Job Promise with job and "SecurityError" DOMException.

        -
      2. -

        Asynchronously complete these steps with a network error.

        -
      -
    19. -

      Let url be request’s url.

      -
    20. -

      Set updatedResourceMap[url] to response.

      -
    21. -

      If response’s cache state is not "local", set registration’s last update check time to the current time.

      -
    22. -

      Let map be newestWorker’s script resource map if newestWorker is not null, and null otherwise.

      -
    23. -

      If map is null or map[url]'s body is not byte-for-byte identical with response’s body, set hasUpdatedResources to true.

      -
    24. -

      Else if newestWorker’s classic scripts imported flag is set, then:

      -

      Note: The following checks to see if an imported script has been updated, since the main script has not changed.

      -
        -
      1. -

        For each importUrlstoredResponse of newestWorker’s script resource map:

        -
          -
        1. -

          If importUrl is request’s url, then continue.

          -
        2. -

          Let importRequest be a new request whose url is importUrl, client is job’s client, destination is "script", parser metadata is "not parser-inserted", synchronous flag is set, and whose use-URL-credentials flag is set.

          -
        3. -

          Set importRequest’s cache mode to "no-cache" if any of the following are true:

          - -
        4. -

          Let fetchedResponse be the result of fetching importRequest.

          -
        5. -

          Set updatedResourceMap[importRequest’s url] to fetchedResponse.

          -
        6. -

          Set fetchedResponse to fetchedResponse’s unsafe response.

          -
        7. -

          If fetchedResponse’s cache state is not "local", set registration’s last update check time to the current time.

          -
        8. -

          If fetchedResponse is a bad import script response, continue.

          -

          Note: Bad responses for importScripts() are ignored for the purpose of the byte-to-byte check. Only good responses for the incumbent worker and good responses for the potential update worker are considered. See issue #1374 for some rationale.

          -
        9. -

          If fetchedResponse’s body is not byte-for-byte identical with storedResponse’s unsafe response's body, set hasUpdatedResources to true.

          -

          Note: The control does not break the loop in this step to continue with all the imported scripts to populate the cache.

          -
        -
      -
    25. -

      Asynchronously complete these steps with response.

      -
    -

    If the algorithm asynchronously completes with null, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and TypeError.

      -

      Note: This will do nothing if Reject Job Promise was previously invoked with "SecurityError" DOMException.

      -
    2. -

      If newestWorker is null, invoke Clear Registration algorithm passing registration as its argument.

      -
    3. -

      Invoke Finish Job with job and abort these steps.

      -
    -

    Else, continue the rest of these steps after the algorithm’s asynchronous completion, with script being the asynchronous completion value.

    -
  10. -

    If hasUpdatedResources is false, then:

    -
      -
    1. -

      Invoke Resolve Job Promise with job and registration.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  11. -

    Let worker be a new service worker.

    -
  12. -

    Set worker’s script url to job’s script url, worker’s script resource to script, worker’s type to job’s worker type, and worker’s script resource map to updatedResourceMap.

    -
  13. -

    Append url to worker’s set of used scripts.

    -
  14. -

    Set worker’s script resource’s HTTPS state to httpsState.

    -
  15. -

    Set worker’s script resource’s referrer policy to referrerPolicy.

    -
  16. -

    Invoke Run Service Worker algorithm given worker, with the force bypass cache for importscripts flag set if job’s force bypass cache flag is set, and with the following callback steps given evaluationStatus:

    -
      -
    1. -

      If evaluationStatus is an abrupt completion or evaluationStatus.[[Value]] is empty, then:

      -
        -
      1. -

        Invoke Reject Job Promise with job and TypeError.

        -
      2. -

        If newestWorker is null, invoke Clear Registration algorithm passing registration as its argument.

        -
      3. -

        Invoke Finish Job with job.

        -
      -
    2. -

      Else, invoke Install algorithm with job, worker, and registration as its arguments.

      -
    -
-
-
-

Soft Update

-

The user agent may call this as often as it likes to check for updates.

-
-
Input -
-

registration, a service worker registration

-
-

force bypass cache flag, an optional flag unset by default

-

Note: Implementers may use the force bypass cache flag to aid debugging (e.g. invocations from developer tools), and other specifications that extend service workers may also use the flag on their own needs.

-
Output -
-

None

-
-
    -
  1. -

    Let newestWorker be the result of running Get Newest Worker algorithm passing registration as its argument.

    -
  2. -

    If newestWorker is null, abort these steps.

    -
  3. -

    Let job be the result of running Create Job with update, registration’s scope url, newestWorker’s script url, null, and null.

    -
  4. -

    Set job’s worker type to newestWorker’s type.

    -
  5. -

    Set job’s force bypass cache flag if the force bypass cache flag is set.

    -
  6. -

    Invoke Schedule Job with job.

    -
-
-
-

Install

-
-
Input -
-

job, a job

-
-

worker, a service worker

-
-

registration, a service worker registration

-
Output -
-

none

-
-
    -
  1. -

    Let installFailed be false.

    -
  2. -

    Let newestWorker be the result of running Get Newest Worker algorithm passing registration as its argument.

    -
  3. -

    Run the Update Registration State algorithm passing registration, "installing" and worker as the arguments.

    -
  4. -

    Run the Update Worker State algorithm passing registration’s installing worker and installing as the arguments.

    -
  5. -

    Assert: job’s job promise is not null.

    -
  6. -

    Invoke Resolve Job Promise with job and registration.

    -
  7. -

    Queue a task to fire an event named updatefound at all the ServiceWorkerRegistration objects for all the service worker clients whose creation URL matches registration’s scope url and all the service workers whose containing service worker registration is registration.

    -
  8. -

    Let installingWorker be registration’s installing worker.

    -
  9. -

    Invoke Run Service Worker algorithm given installingWorker, and with the force bypass cache for importscripts flag set if job’s force bypass cache flag is set.

    -
  10. -

    Queue a task task to run the following substeps:

    -
      -
    1. -

      Let e be the result of creating an event with ExtendableEvent.

      -
    2. -

      Initialize e’s type attribute to install.

      -
    3. -

      Dispatch e at installingWorker’s global object.

      -
    4. -

      WaitForAsynchronousExtensions: Run the following substeps in parallel:

      -
        -
      1. -

        Wait until e is not active.

        -
      2. -

        If e’s timed out flag is set, or the result of waiting for all of e’s extend lifetime promises rejected, set installFailed to true.

        -
      -
    -

    If task is discarded or the script has been aborted by the termination of installingWorker, set installFailed to true.

    -
  11. -

    Wait for task to have executed or been discarded.

    -
  12. -

    Wait for the step labeled WaitForAsynchronousExtensions to complete.

    -
  13. -

    If installFailed is true, then:

    -
      -
    1. -

      Run the Update Worker State algorithm passing registration’s installing worker and redundant as the arguments.

      -
    2. -

      Run the Update Registration State algorithm passing registration, "installing" and null as the arguments.

      -
    3. -

      If newestWorker is null, invoke Clear Registration algorithm passing registration as its argument.

      -
    4. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  14. -

    Let map be registration’s installing worker's script resource map.

    -
  15. -

    Let usedSet be registration’s installing worker's set of used scripts.

    -
  16. -

    For each url of map:

    -
      -
    1. -

      If usedSet does not contain url, then remove map[url].

      -
    -
  17. -

    If registration’s waiting worker is not null, then:

    -
      -
    1. -

      Terminate registration’s waiting worker.

      -
    2. -

      Run the Update Worker State algorithm passing registration’s waiting worker and redundant as the arguments.

      -
    -
  18. -

    Run the Update Registration State algorithm passing registration, "waiting" and registration’s installing worker as the arguments.

    -
  19. -

    Run the Update Registration State algorithm passing registration, "installing" and null as the arguments.

    -
  20. -

    Run the Update Worker State algorithm passing registration’s waiting worker and installed as the arguments.

    -
  21. -

    Invoke Finish Job with job.

    -
  22. -

    Wait for all the tasks queued by Update Worker State invoked in this algorithm to have executed.

    -
  23. -

    Invoke Try Activate with registration.

    -

    Note: If Try Activate does not trigger Activate here, Activate is tried again when the last client controlled by the existing active worker is unloaded, skipWaiting() is asynchronously called, or the extend lifetime promises for the existing active worker settle.

    -
-
-
-

Activate

-
-
Input -
-

registration, a service worker registration

-
Output -
-

None

-
-
    -
  1. -

    If registration’s waiting worker is null, abort these steps.

    -
  2. -

    If registration’s active worker is not null, then:

    -
      -
    1. -

      Terminate registration’s active worker.

      -
    2. -

      Run the Update Worker State algorithm passing registration’s active worker and redundant as the arguments.

      -
    -
  3. -

    Run the Update Registration State algorithm passing registration, "active" and registration’s waiting worker as the arguments.

    -
  4. -

    Run the Update Registration State algorithm passing registration, "waiting" and null as the arguments.

    -
  5. -

    Run the Update Worker State algorithm passing registration’s active worker and activating as the arguments.

    -

    Note: Once an active worker is activating, neither a runtime script error nor a force termination of the active worker prevents the active worker from getting activated.

    -
  6. -

    Let matchedClients be a list of service worker clients whose creation URL matches registration’s scope url.

    -
  7. -

    For each client of matchedClients, queue a task on client’s responsible event loop, using the DOM manipulation task source, to run the following substeps:

    -
      -
    1. -

      Let readyPromise be client’s global object's ServiceWorkerContainer object’s ready promise.

      -
    2. -

      If readyPromise is pending, resolve readyPromise with the ServiceWorkerRegistration object that represents registration in readyPromise’s relevant Realm.

      -
    -
  8. -

    For each client of matchedClients:

    -
      -
    1. -

      If client is a window client, unassociate client’s responsible document from its application cache, if it has one.

      -
    2. -

      Else if client is a shared worker client, unassociate client’s global object from its application cache, if it has one.

      -
    -

    Note: Resources will now use the service worker registration instead of the existing application cache.

    -
  9. -

    For each service worker client client who is using registration:

    -
      -
    1. -

      Set client’s active worker to registration’s active worker.

      -
    2. -

      Invoke Notify Controller Change algorithm with client as the argument.

      -
    -
  10. -

    Let activeWorker be registration’s active worker.

    -
  11. -

    Invoke Run Service Worker algorithm with activeWorker as the argument.

    -
  12. -

    Queue a task task to run the following substeps:

    -
      -
    1. -

      Let e be the result of creating an event with ExtendableEvent.

      -
    2. -

      Initialize e’s type attribute to activate.

      -
    3. -

      Dispatch e at activeWorker’s global object.

      -
    4. -

      WaitForAsynchronousExtensions: Wait, in parallel, until e is not active.

      -
    -
  13. -

    Wait for task to have executed or been discarded, or the script to have been aborted by the termination of activeWorker.

    -
  14. -

    Wait for the step labeled WaitForAsynchronousExtensions to complete.

    -
  15. -

    Run the Update Worker State algorithm passing registration’s active worker and activated as the arguments.

    -
-
-
-

Try Activate

-
-
Input -
-

registration, a service worker registration

-
Output -
-

None

-
-
    -
  1. -

    If registration’s waiting worker is null, return.

    -
  2. -

    If registration’s active worker is not null and registration’s active worker's state is activating, return.

    -

    Note: If the existing active worker is still in activating state, the activation of the waiting worker is delayed.

    -
  3. -

    Invoke Activate with registration if either of the following is true:

    - -
-
-
-

Run Service Worker

-
-
Input -
-

serviceWorker, a service worker

-
-

force bypass cache for importscripts flag, an optional flag unset by default

-
-

optional callbackSteps, an algorithm which takes a Completion

-
-
    -
  1. -

    Let script be serviceWorker’s script resource.

    -
  2. -

    Assert: script is not null.

    -
  3. -

    If serviceWorker is already running, this algorithm must have been invoked previously. If callbackSteps is provided, run them with the same value as the previous time, and abort these steps.

    -
  4. -

    Create a separate parallel execution environment (i.e. a separate thread or process or equivalent construct), and run the following substeps in that context:

    -
      -
    1. -

      Call the JavaScript InitializeHostDefinedRealm() abstract operation with the following customizations:

      - -
    2. -

      Set serviceWorker’s global object to workerGlobalScope.

      -
    3. -

      Let workerEventLoop be a newly created event loop.

      -
    4. -

      Let settingsObject be a new environment settings object whose algorithms are defined as follows:

      -
      -
      The realm execution context -
      -

      Return realmExecutionContext.

      -
      The global object -
      -

      Return workerGlobalScope.

      -
      The responsible event loop -
      -

      Return workerEventLoop.

      -
      The referrer policy -
      -

      Return workerGlobalScope’s referrer policy.

      -
      The API URL character encoding -
      -

      Return UTF-8.

      -
      The API base URL -
      -

      Return serviceWorker’s script url.

      -
      The origin -
      -

      Return its registering service worker client's origin.

      -
      The creation URL -
      -

      Return workerGlobalScope’s url.

      -
      The HTTPS state -
      -

      Return workerGlobalScope’s HTTPS state.

      -
      -
    5. -

      Set workerGlobalScope’s url to serviceWorker’s script url.

      -
    6. -

      Set workerGlobalScope’s HTTPS state to serviceWorker’s script resource’s HTTPS state.

      -
    7. -

      Set workerGlobalScope’s referrer policy to serviceWorker’s script resource’s referrer policy.

      -
    8. -

      Set workerGlobalScope’s type to serviceWorker’s type.

      -
    9. -

      Set workerGlobalScope’s force bypass cache for importscripts flag if the force bypass cache for importscripts flag is set.

      -
    10. -

      Create a new WorkerLocation object and associate it with workerGlobalScope.

      -
    11. -

      If serviceWorker is an active worker, and there are any tasks queued in serviceWorker’s containing service worker registration’s task queues, queue them to serviceWorker’s event loop’s task queues in the same order using their original task sources.

      -
    12. -

      Let evaluationStatus be the result of running the classic script script if script is a classic script, otherwise, the result of running the module script script if script is a module script.

      -

      Note: In addition to the usual possibilities of returning a value or failing due to an exception, this could be prematurely aborted by the terminate service worker algorithm.

      -
    13. -

      If callbackSteps is provided, run them with evaluationStatus on the original thread that invoked this algorithm, while continuing in parallel with these steps.

      -
    14. -

      If script’s has ever been evaluated flag is unset, then:

      -
        -
      1. -

        For each eventType of settingsObject’s global object's associated list of event listeners' event types:

        -
          -
        1. -

          Append eventType to workerGlobalScope’s associated service worker's set of event types to handle.

          -
        -

        Note: If the global object’s associated list of event listeners does not have any event listener added at this moment, the service worker’s set of event types to handle remains an empty set. The user agents are encouraged to show a warning that the event listeners must be added on the very first evaluation of the worker script.

        -
      2. -

        Set script’s has ever been evaluated flag.

        -
      -
    15. -

      Run the responsible event loop specified by settingsObject until it is destroyed.

      -
    16. -

      Empty workerGlobalScope’s list of active timers.

      -
    -
-
-
-

Terminate Service Worker

-
-
Input -
-

serviceWorker, a service worker

-
Output -
-

None

-
-
    -
  1. -

    If serviceWorker is not running, abort these steps.

    -
  2. -

    Let serviceWorkerGlobalScope be serviceWorker’s global object.

    -
  3. -

    Set serviceWorkerGlobalScope’s closing flag to true.

    -
  4. -

    Remove all the items from serviceWorker’s set of extended events.

    -
  5. -

    If there are any tasks, whose task source is either the handle fetch task source or the handle functional event task source, queued in serviceWorkerGlobalScope’s event loop’s task queues, queue them to serviceWorker’s containing service worker registration’s corresponding task queues in the same order using their original task sources, and discard all the tasks (including tasks whose task source is neither the handle fetch task source nor the handle functional event task source) from serviceWorkerGlobalScope’s event loop’s task queues without processing them.

    -

    Note: This effectively means that the fetch events and the other functional events such as push events are backed up by the registration’s task queues while the other tasks including message events are discarded.

    -
  6. -

    Abort the script currently running in serviceWorker.

    -
-
-
-

Handle Fetch

-

The Handle Fetch algorithm is the entry point for the fetch handling handed to the service worker context.

-
-
Input -
-

request, a request

-
Output -
-

response, a response

-
-
    -
  1. -

    Let handleFetchFailed be false.

    -
  2. -

    Let respondWithEntered be false.

    -
  3. -

    Let eventCanceled be false.

    -
  4. -

    Let response be null.

    -
  5. -

    Let registration be null.

    -
  6. -

    Let client be request’s client.

    -
  7. -

    Let reservedClient be request’s reserved client.

    -
  8. -

    Let preloadResponse be a new promise.

    -
  9. -

    Let fetchInstance be the instance of the fetch algorithm representing the ongoing fetch.

    -
  10. -

    Assert: request’s destination is not "serviceworker".

    -
  11. -

    If request is a potential-navigation-or-subresource request, then:

    -
      -
    1. -

      Return null.

      -
    -
  12. -

    Else if request is a non-subresource request, then:

    -

    Note: If the non-subresource request is under the scope of a service worker registration, application cache is completely bypassed regardless of whether the non-subresource request uses the service worker registration.

    -
      -
    1. -

      If reservedClient is not null and is an environment settings object, then:

      -
        -
      1. -

        If reservedClient is not a secure context, return null.

        -
      -
    2. -

      Else:

      -
        -
      1. -

        If request’s url is not a potentially trustworthy URL, return null.

        -
      -
    3. -

      If request is a navigation request and the navigation triggering it was initiated with a shift+reload or equivalent, return null.

      -
    4. -

      Set registration to the result of running Match Service Worker Registration algorithm passing request’s url as the argument.

      -
    5. -

      If registration is null or registration’s active worker is null, return null.

      -
    6. -

      If request’s destination is not "report", set reservedClient’s active service worker to registration’s active worker.

      -
    7. -

      If request is a navigation request, registration’s navigation preload enabled flag is set, request’s method is `GET`, and registration’s active worker's set of event types to handle contains fetch, then:

      -

      Note: If the above is true except registration’s active worker's set of event types to handle does not contain fetch, then the user agent may wish to show a console warning, as the developer’s intent isn’t clear.

      -
        -
      1. -

        Let preloadRequest be the result of cloning the request request.

        -
      2. -

        Let preloadRequestHeaders be preloadRequest’s header list.

        -
      3. -

        Let preloadResponseObject be a new Response object associated with a new Headers object whose guard is "immutable".

        -
      4. -

        Append to preloadRequestHeaders a new header whose name is `Service-Worker-Navigation-Preload` and value is registration’s navigation preload header value.

        -
      5. -

        Set preloadRequest’s service-workers mode to "none".

        -
      6. -

        Run the following substeps in parallel:

        -
          -
        1. -

          Fetch preloadRequest and let preloadFetchInstance be the instance of the fetch algorithm.

          -

          To process response for navigationPreloadResponse, run these substeps:

          -
            -
          1. -

            If navigationPreloadResponse’s type is "error", reject preloadResponse with a TypeError and terminate these substeps.

            -
          2. -

            Associate preloadResponseObject with navigationPreloadResponse.

            -
          3. -

            Resolve preloadResponse with preloadResponseObject.

            -
          -
        2. -

          If fetchInstance is terminated, then terminate preloadFetchInstance with the aborted flag set.

          -
        -
      -
    8. -

      Else, resolve preloadResponse with undefined.

      -
    -

    Note: From this point, the service worker client starts to use its active service worker’s containing service worker registration.

    -
  13. -

    Else if request is a subresource request, then:

    -
      -
    1. -

      If client’s active service worker is non-null, set registration to client’s active service worker’s containing service worker registration.

      -
    2. -

      Else, return null.

      -
    -
  14. -

    Let activeWorker be registration’s active worker.

    -
  15. -

    If activeWorker’s set of event types to handle does not contain fetch, then:

    -
      -
    1. -

      Return null and continue running these steps in parallel.

      -
    2. -

      If request is a non-subresource request, or request is a subresource request and the time difference in seconds calculated by the current time minus registration’s last update check time is greater than 86400, invoke Soft Update algorithm with registration.

      -
    3. -

      Abort these steps.

      -
    -

    Note: To avoid unnecessary delays, the Handle Fetch enforces early return when no event listeners have been deterministically added in the service worker’s global during the very first script execution.

    -
  16. -

    If activeWorker’s state is activating, wait for activeWorker’s state to become activated.

    -
  17. -

    Invoke Run Service Worker algorithm with activeWorker as the argument.

    -
  18. -

    Queue a task task to run the following substeps:

    -
      -
    1. -

      Let e be the result of creating an event with FetchEvent.

      -
    2. -

      Let requestObject be a new Request object associated with request and a new associated Headers object whose guard is "immutable".

      -
    3. -

      Initialize e’s type attribute to fetch.

      -
    4. -

      Initialize e’s cancelable attribute to true.

      -
    5. -

      Initialize e’s request attribute to requestObject.

      -
    6. -

      Initialize e’s preloadResponse to preloadResponse.

      -
    7. -

      Initialize e’s clientId attribute to client’s id.

      -
    8. -

      If request is a non-subresource request and request’s destination is not "report", initialize e’s resultingClientId attribute to reservedClient’s id, and to the empty string otherwise.

      -
    9. -

      If request is a navigation request, initialize e’s replacesClientId attribute to request’s replaces client id, and to the empty string otherwise.

      -
    10. -

      Dispatch e at activeWorker’s global object.

      -
    11. -

      Invoke Update Service Worker Extended Events Set with activeWorker and e.

      -
    12. -

      If e’s respond-with entered flag is set, set respondWithEntered to true.

      -
    13. -

      If e’s wait to respond flag is set, then:

      -
        -
      1. -

        Wait until e’s wait to respond flag is unset.

        -
      2. -

        If e’s respond-with error flag is set, set handleFetchFailed to true.

        -
      3. -

        Else, set response to e’s potential response.

        -
      -
    14. -

      If e’s canceled flag is set, set eventCanceled to true.

      -
    15. -

      If fetchInstance is terminated, then queue a task to signal abort on requestObject’s signal.

      -
    -

    If task is discarded or the script has been aborted by the termination of activeWorker, set handleFetchFailed to true.

    -

    The task must use activeWorker’s event loop and the handle fetch task source.

    -
  19. -

    Wait for task to have executed or been discarded.

    -
  20. -

    If respondWithEntered is false, then:

    -
      -
    1. -

      If eventCanceled is true, return a network error and continue running these steps in parallel.

      -
    2. -

      Else, return null and continue running these steps in parallel.

      -
    3. -

      If request is a non-subresource request, or request is a subresource request and the time difference in seconds calculated by the current time minus registration’s last update check time is greater than 86400, invoke Soft Update algorithm with registration.

      -
    4. -

      Abort these steps.

      -
    -
  21. -

    If handleFetchFailed is true, then:

    -
      -
    1. -

      Return a network error and continue running these steps in parallel.

      -
    2. -

      If request is a non-subresource request, or request is a subresource request and the time difference in seconds calculated by the current time minus registration’s last update check time is greater than 86400, invoke Soft Update algorithm with registration.

      -
    -
  22. -

    Else:

    -
      -
    1. -

      Return response and continue running these steps in parallel.

      -
    2. -

      If request is a non-subresource request, or request is a subresource request and the time difference in seconds calculated by the current time minus registration’s last update check time is greater than 86400, invoke Soft Update algorithm with registration.

      -
    -
-
-
-

Fire Functional Event

-
-
Input -
-

eventName, a string

-
-

eventConstructor, an event constructor that extends ExtendableEvent

-
-

registration, a service worker registration

-
-

initialization, optional property initialization for event, constructed from eventConstructor

-
-

postDispatchSteps, optional steps to run on the active worker's event loop, with dispatchedEvent set to the instance of eventConstructor that was dispatched.

-
Output -
-

None

-
-
    -
  1. -

    Assert: scope to registration map contains a value equal to registration.

    -
  2. -

    Assert: registration’s active worker is not null.

    -
  3. -

    Let activeWorker be registration’s active worker.

    -
  4. -

    If activeWorker’s set of event types to handle does not contain eventName, then return and run the following steps in parallel:

    -
      -
    1. -

      If the time difference in seconds calculated by the current time minus registration’s last update check time is greater than 86400, invoke Soft Update algorithm with registration.

      -
    2. -

      Abort these steps.

      -
    -

    Note: To avoid unnecessary delays, the Handle Functional Event enforces early return when no event listeners have been deterministically added in the service worker’s global during the very first script execution.

    -
  5. -

    If activeWorker’s state is activating, wait for activeWorker’s state to become activated.

    -
  6. -

    Invoke Run Service Worker algorithm with activeWorker as the argument.

    -
  7. -

    Queue a task task to run these substeps:

    -
      -
    1. -

      Let event be the result of creating an event with eventConstructor and the relevant realm of activeWorker’s global object.

      -
    2. -

      If initialization is not null, then initialize event with initialization.

      -
    3. -

      Dispatch event on activeWorker’s global object.

      -
    4. -

      Invoke Update Service Worker Extended Events Set with activeWorker and event.

      -
    5. -

      If postDispatchSteps is not null, then run postDispatchSteps passing event as dispatchedEvent.

      -
    -

    The task must use activeWorker’s event loop and the handle functional event task source.

    -
  8. -

    Wait for task to have executed or been discarded.

    -
  9. -

    If the time difference in seconds calculated by the current time minus registration’s last update check time is greater than 86400, invoke Soft Update algorithm with registration.

    -
-
- To fire an "amazingthing" event (which is of type AmazingThingEvent) on a particular serviceWorkerRegistration, and initialize the event object’s properties, the prose would be: -
    -
  1. -

    Fire Functional Event "amazingthing" using AmazingThingEvent on serviceWorkerRegistration with the following properties:

    -
    -
    propertyName -
    -

    value

    -
    anotherPropertyName -
    -

    anotherValue

    -
    -

    Then run the following steps with dispatchedEvent:

    -
      -
    1. -

      Do whatever you need to with dispatchedEvent on the service worker’s event loop.

      -
    -
-

Note that the initialization steps and post-dispatch steps are optional. If they aren’t needed, the prose would be:

-
    -
  1. -

    Fire Functional Event "whatever" using ExtendableEvent on serviceWorkerRegistration.

    -
-
-
-
-

Handle Service Worker Client Unload

-

The user agent must run these steps when a service worker client unloads by unloading or terminating.

-
-
Input -
-

client, a service worker client

-
Output -
-

None

-
-
    -
  1. -

    Run the following steps atomically.

    -
  2. -

    Let registration be the service worker registration used by client.

    -
  3. -

    If registration is null, abort these steps.

    -
  4. -

    If any other service worker client is using registration, abort these steps.

    -
  5. -

    If registration’s uninstalling flag is set, invoke Try Clear Registration with registration.

    -
  6. -

    If registration is not null, invoke Try Activate with registration.

    -
-
-
-

Handle User Agent Shutdown

-
-
Input -
-

None

-
Output -
-

None

-
-
    -
  1. -

    For each scoperegistration of scope to registration map:

    -
      -
    1. -

      If registration’s installing worker installingWorker is not null, then:

      -
        -
      1. -

        If registration’s waiting worker is null and registration’s active worker is null, invoke Clear Registration with registration and continue to the next iteration of the loop.

        -
      2. -

        Else, set installingWorker to null.

        -
      -
    2. -

      If registration’s waiting worker is not null, run the following substep in parallel:

      -
        -
      1. -

        Invoke Activate with registration.

        -
      -
    -
-
-
-

Update Service Worker Extended Events Set

-
-
Input -
-

worker, a service worker

-
-

event, an event

-
Output -
-

None

-
-
    -
  1. -

    Assert: event’s dispatch flag is unset.

    -
  2. -

    For each item of worker’s set of extended events:

    -
      -
    1. -

      If item is not active, remove item from worker’s set of extended events.

      -
    -
  3. -

    If event is active, append event to worker’s set of extended events.

    -
-
-
-

Unregister

-
-
Input -
-

job, a job

-
Output -
-

none

-
-
    -
  1. -

    If the origin of job’s scope url is not job’s client's origin, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and "SecurityError" DOMException.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  2. -

    Let registration be the result of running Get Registration algorithm passing job’s scope url as the argument.

    -
  3. -

    If registration is null, then:

    -
      -
    1. -

      Invoke Resolve Job Promise with job and false.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  4. -

    Set registration’s uninstalling flag.

    -
  5. -

    Invoke Resolve Job Promise with job and true.

    -
  6. -

    Invoke Try Clear Registration with registration.

    -

    Note: If Try Clear Registration does not trigger Clear Registration here, Clear Registration is tried again when the last client using the registration is unloaded or the extend lifetime promises for the registration’s service workers settle.

    -
  7. -

    Invoke Finish Job with job.

    -
-
-
-

Set Registration

-
-
Input -
-

scope, a URL

-
-

updateViaCache, an update via cache mode

-
Output -
-

registration, a service worker registration

-
-
    -
  1. -

    Run the following steps atomically.

    -
  2. -

    Let scopeString be serialized scope with the exclude fragment flag set.

    -
  3. -

    Let registration be a new service worker registration whose scope url is set to scope and update via cache mode is set to updateViaCache.

    -
  4. -

    Set scope to registration map[scopeString] to registration.

    -
  5. -

    Return registration.

    -
-
-
-

Clear Registration

-
-
Input -
-

registration, a service worker registration

-
Output -
-

None

-
-
    -
  1. -

    Run the following steps atomically.

    -
  2. -

    If registration’s installing worker is not null, then:

    -
      -
    1. -

      Terminate registration’s installing worker.

      -
    2. -

      Run the Update Worker State algorithm passing registration’s installing worker and redundant as the arguments.

      -
    3. -

      Run the Update Registration State algorithm passing registration, "installing" and null as the arguments.

      -
    -
  3. -

    If registration’s waiting worker is not null, then:

    -
      -
    1. -

      Terminate registration’s waiting worker.

      -
    2. -

      Run the Update Worker State algorithm passing registration’s waiting worker and redundant as the arguments.

      -
    3. -

      Run the Update Registration State algorithm passing registration, "waiting" and null as the arguments.

      -
    -
  4. -

    If registration’s active worker is not null, then:

    -
      -
    1. -

      Terminate registration’s active worker.

      -
    2. -

      Run the Update Worker State algorithm passing registration’s active worker and redundant as the arguments.

      -
    3. -

      Run the Update Registration State algorithm passing registration, "active" and null as the arguments.

      -
    -
  5. -

    Let scopeString be registration’s serialized scope url.

    -
  6. -

    Remove scope to registration map[scopeString].

    -
-
-
-

Try Clear Registration

-
-
Input -
-

registration, a service worker registration

-
Output -
-

None

-
-
    -
  1. -

    Invoke Clear Registration with registration if no service worker client is using registration and all of the following conditions are true:

    - -
-
-
-

Update Registration State

-
-
Input -
-

registration, a service worker registration

-
-

target, a string (one of "installing", "waiting", and "active")

-
-

source, a service worker or null

-
Output -
-

None

-
-
    -
  1. -

    Let registrationObjects be an array containing all the ServiceWorkerRegistration objects associated with registration.

    -
  2. -

    If target is "installing", then:

    -
      -
    1. -

      Set registration’s installing worker to source.

      -
    2. -

      For each registrationObject in registrationObjects:

      -
        -
      1. -

        Queue a task to set the installing attribute of registrationObject to the ServiceWorker object that represents registration’s installing worker, or null if registration’s installing worker is null.

        -
      -
    -
  3. -

    Else if target is "waiting", then:

    -
      -
    1. -

      Set registration’s waiting worker to source.

      -
    2. -

      For each registrationObject in registrationObjects:

      -
        -
      1. -

        Queue a task to set the waiting attribute of registrationObject to the ServiceWorker object that represents registration’s waiting worker, or null if registration’s waiting worker is null.

        -
      -
    -
  4. -

    Else if target is "active", then:

    -
      -
    1. -

      Set registration’s active worker to source.

      -
    2. -

      For each registrationObject in registrationObjects:

      -
        -
      1. -

        Queue a task to set the active attribute of registrationObject to the ServiceWorker object that represents registration’s active worker, or null if registration’s active worker is null.

        -
      -
    -

    The task must use registrationObject’s relevant settings object’s responsible event loop and the DOM manipulation task source.

    -
-
-
-

Update Worker State

-
-
Input -
-

worker, a service worker

-
-

state, a service worker's state

-
Output -
-

None

-
-
    -
  1. -

    Set worker’s state to state.

    -
  2. -

    Let workerObjects be an array containing all the ServiceWorker objects associated with worker.

    -
  3. -

    For each workerObject in workerObjects:

    -
      -
    1. -

      Queue a task to run these substeps:

      -
        -
      1. -

        Set the state attribute of workerObject to the value (in ServiceWorkerState enumeration) corresponding to the first matching statement, switching on worker’s state:

        -
        -
        installing -
        -

        "installing"

        -

        Note: The service worker in this state is considered an installing worker. During this state, waitUntil() can be called inside the oninstall event handler to extend the life of the installing worker until the passed promise resolves successfully. This is primarily used to ensure that the service worker is not active until all of the core caches are populated.

        -
        installed -
        -

        "installed"

        -

        Note: The service worker in this state is considered a waiting worker.

        -
        activating -
        -

        "activating"

        -

        Note: The service worker in this state is considered an active worker. During this state, waitUntil() can be called inside the onactivate event handler to extend the life of the active worker until the passed promise resolves successfully. No functional events are dispatched until the state becomes activated.

        -
        activated -
        -

        "activated"

        -

        Note: The service worker in this state is considered an active worker ready to handle functional events.

        -
        redundant -
        -

        "redundant"

        -

        Note: A new service worker is replacing the current service worker, or the current service worker is being discarded due to an install failure.

        -
        -
      2. -

        Fire an event named statechange at workerObject.

        -
      -
    -

    The task must use workerObject’s relevant settings object’s responsible event loop and the DOM manipulation task source.

    -
-
-
-

Notify Controller Change

-
-
Input -
-

client, a service worker client

-
Output -
-

None

-
-
    -
  1. -

    Assert: client is not null.

    -
  2. -

    If client is an environment settings object, queue a task to fire an event named controllerchange at the ServiceWorkerContainer object that client is associated with.

    -
-

The task must use client’s responsible event loop and the DOM manipulation task source.

-
-
-

Match Service Worker Registration

-
-
Input -
-

clientURL, a URL

-
Output -
-

registration, a service worker registration

-
-
    -
  1. -

    Run the following steps atomically.

    -
  2. -

    Let clientURLString be serialized clientURL.

    -
  3. -

    Let matchingScopeString be the empty string.

    -
  4. -

    Let scopeStringSet be the result of getting the keys from scope to registration map.

    -
  5. -

    Set matchingScopeString to the longest value in scopeStringSet which the value of clientURLString starts with, if it exists.

    -

    Note: The URL string matching in this step is prefix-based rather than path-structural. E.g. a client URL string with "https://example.com/prefix-of/resource.html" will match a registration for a scope with "https://example.com/prefix". The URL string comparison is safe for the same-origin security as HTTP(S) URLs are always serialized with a trailing slash at the end of the origin part of the URLs.

    -
  6. -

    Let matchingScope be null.

    -
  7. -

    If matchingScopeString is not the empty string, then:

    -
      -
    1. -

      Set matchingScope to the result of parsing matchingScopeString.

      -
    2. -

      Assert: matchingScope’s origin and clientURL’s origin are same origin.

      -
    -
  8. -

    Let registration be the result of running Get Registration algorithm passing matchingScope as the argument.

    -
  9. -

    If registration is not null and registration’s uninstalling flag is set, return null.

    -
  10. -

    Return registration.

    -
-
-
-

Get Registration

-
-
Input -
-

scope, a URL

-
Output -
-

registration, a service worker registration

-
-
    -
  1. -

    Run the following steps atomically.

    -
  2. -

    Let scopeString be the empty string.

    -
  3. -

    If scope is not null, set scopeString to serialized scope with the exclude fragment flag set.

    -
  4. -

    Let registration be null.

    -
  5. -

    For each keyvalue of scope to registration map:

    -
      -
    1. -

      If scopeString matches key, set registration to value.

      -
    -
  6. -

    Return registration.

    -
-
-
-

Get Newest Worker

-
-
Input -
-

registration, a service worker registration

-
Output -
-

newestWorker, a service worker

-
-
    -
  1. -

    Run the following steps atomically.

    -
  2. -

    Let newestWorker be null.

    -
  3. -

    If registration’s installing worker is not null, set newestWorker to registration’s installing worker.

    -
  4. -

    Else if registration’s waiting worker is not null, set newestWorker to registration’s waiting worker.

    -
  5. -

    Else if registration’s active worker is not null, set newestWorker to registration’s active worker.

    -
  6. -

    Return newestWorker.

    -
-
-
-

Service Worker Has No Pending Events

-
-
Input -
-

worker, a service worker

-
Output -
-

True or false, a boolean

-
-
    -
  1. -

    For each event of worker’s set of extended events:

    -
      -
    1. -

      If event is active, return false.

      -
    -
  2. -

    Return true.

    -
-
-
-

Create Client

-
-
Input -
-

client, a service worker client

-
Output -
-

clientObject, a Client object

-
-
    -
  1. -

    Let clientObject be a new Client object.

    -
  2. -

    Set clientObject’s service worker client to client.

    -
  3. -

    Return clientObject.

    -
-
-
-

Create Window Client

-
-
Input -
-

client, a service worker client

-
-

frameType, a string

-
-

visibilityState, a string

-
-

focusState, a boolean

-
-

ancestorOriginsList, a list

-
Output -
-

windowClient, a WindowClient object

-
-
    -
  1. -

    Let windowClient be a new WindowClient object.

    -
  2. -

    Set windowClient’s service worker client to client.

    -
  3. -

    Set windowClient’s frame type to frameType.

    -
  4. -

    Set windowClient’s visibility state to visibilityState.

    -
  5. -

    Set windowClient’s focus state to focusState.

    -
  6. -

    Set windowClient’s ancestor origins array to a frozen array created from ancestorOriginsList.

    -
  7. -

    Return windowClient.

    -
-
-
-

Get Frame Type

-
-
Input -
-

browsingContext, a browsing context

-
Output -
-

frameType, a string

-
-
    -
  1. -

    Return the value by switching on the type of browsingContext:

    -
    -
    Nested browsing context -
    -

    "nested"

    -
    Auxiliary browsing context -
    -

    "auxiliary"

    -
    Otherwise -
    -

    "top-level"

    -
    -
-
-
-

Resolve Get Client Promise

-
-
Input -
-

client, a service worker client

-
-

promise, a promise

-
Output -
-

none

-
-
    -
  1. -

    If client is an environment settings object, then:

    -
      -
    1. -

      If client is not a secure context, queue a task to reject promise with a "SecurityError" DOMException, on promise’s relevant settings object's responsible event loop using the DOM manipulation task source, and abort these steps.

      -
    -
  2. -

    Else:

    -
      -
    1. -

      If client’s creation URL is not a potentially trustworthy URL, queue a task to reject promise with a "SecurityError" DOMException, on promise’s relevant settings object's responsible event loop using the DOM manipulation task source, and abort these steps.

      -
    -
  3. -

    If client is an environment settings object and is not a window client, then:

    -
      -
    1. -

      Let clientObject be the result of running Create Client algorithm with client as the argument.

      -
    2. -

      Queue a task to resolve promise with clientObject, on promise’s relevant settings object's responsible event loop using the DOM manipulation task source, and abort these steps.

      -
    -
  4. -

    Else:

    -
      -
    1. -

      Let browsingContext be null.

      -
    2. -

      If client is an environment settings object, set browsingContext to client’s global object's browsing context.

      -
    3. -

      Else, set browsingContext to client’s target browsing context.

      -
    4. -

      Queue a task to run the following steps on browsingContext’s event loop using the user interaction task source:

      -
        -
      1. -

        Let frameType be the result of running Get Frame Type with browsingContext.

        -
      2. -

        Let visibilityState be browsingContext’s active document's visibilityState attribute value.

        -
      3. -

        Let focusState be the result of running the has focus steps with browsingContext’s active document as the argument.

        -
      4. -

        Let ancestorOriginsList be the empty list.

        -
      5. -

        If client is a window client, set ancestorOriginsList to browsingContext’s active document's relevant global object's Location object’s ancestor origins list's associated list.

        -
      6. -

        Queue a task to run the following steps on promise’s relevant settings object's responsible event loop using the DOM manipulation task source:

        -
          -
        1. -

          If client’s discarded flag is set, resolve promise with undefined and abort these steps.

          -
        2. -

          Let windowClient be the result of running Create Window Client with client, frameType, visibilityState, focusState, and ancestorOriginsList.

          -
        3. -

          Resolve promise with windowClient.

          -
        -
      -
    -
-
-
-

Query Cache

-
-
Input -
-

requestQuery, a request

-
-

options, a CacheQueryOptions object, optional

-
-

targetStorage, a request response list, optional

-
Output -
-

resultList, a request response list

-
-
    -
  1. -

    Let resultList be an empty list.

    -
  2. -

    Let storage be null.

    -
  3. -

    If the optional argument targetStorage is omitted, set storage to the relevant request response list.

    -
  4. -

    Else, set storage to targetStorage.

    -
  5. -

    For each requestResponse of storage:

    -
      -
    1. -

      Let cachedRequest be requestResponse’s request.

      -
    2. -

      Let cachedResponse be requestResponse’s response.

      -
    3. -

      If Request Matches Cached Item with requestQuery, cachedRequest, cachedResponse, and options returns true, then:

      -
        -
      1. -

        Let requestCopy be a copy of cachedRequest.

        -
      2. -

        Let responseCopy be a copy of cachedResponse.

        -
      3. -

        Add requestCopy/responseCopy to resultList.

        -
      -
    -
  6. -

    Return resultList.

    -
-
-
-

Request Matches Cached Item

-
-
Input -
-

requestQuery, a request

-
-

request, a request

-
-

response, a response or null, optional, defaulting to null

-
-

options, a CacheQueryOptions object, optional

-
Output -
-

a boolean

-
-
    -
  1. -

    If options.ignoreMethod is false and request’s method is not `GET`, return false.

    -
  2. -

    Let queryURL be requestQuery’s url.

    -
  3. -

    Let cachedURL be request’s url.

    -
  4. -

    If options.ignoreSearch is true, then:

    -
      -
    1. -

      Set cachedURL’s query to the empty string.

      -
    2. -

      Set queryURL’s query to the empty string.

      -
    -
  5. -

    If queryURL does not equal cachedURL with the exclude fragment flag set, then return false.

    -
  6. -

    If response is null, options.ignoreVary is true, or response’s header list does not contain `Vary`, then return true.

    -
  7. -

    Let fieldValues be the list containing the elements corresponding to the field-values of the Vary header for the value of the header with name `Vary`.

    -
  8. -

    For each fieldValue in fieldValues:

    -
      -
    1. -

      If fieldValue matches "*", or the combined value given fieldValue and request’s header list does not match the combined value given fieldValue and requestQuery’s header list, then return false.

      -
    -
  9. -

    Return true.

    -
-
-
-

Batch Cache Operations

-
-
Input -
-

operations, a list of cache batch operation objects

-
Output -
-

resultList, a request response list

-
-
    -
  1. -

    Let cache be the relevant request response list.

    -
  2. -

    Let backupCache be a new request response list that is a copy of cache.

    -
  3. -

    Let addedItems be an empty list.

    -
  4. -

    Try running the following substeps atomically:

    -
      -
    1. -

      Let resultList be an empty list.

      -
    2. -

      For each operation in operations:

      -
        -
      1. -

        If operation’s type matches neither "delete" nor "put", throw a TypeError.

        -
      2. -

        If operation’s type matches "delete" and operation’s response is not null, throw a TypeError.

        -
      3. -

        If the result of running Query Cache with operation’s request, operation’s options, and addedItems is not empty, throw an "InvalidStateError" DOMException.

        -
      4. -

        Let requestResponses be an empty list.

        -
      5. -

        If operation’s type matches "delete", then:

        -
          -
        1. -

          Set requestResponses to the result of running Query Cache with operation’s request and operation’s options.

          -
        2. -

          For each requestResponse in requestResponses:

          -
            -
          1. -

            Remove the item whose value matches requestResponse from cache.

            -
          -
        -
      6. -

        Else if operation’s type matches "put", then:

        -
          -
        1. -

          If operation’s response is null, throw a TypeError.

          -
        2. -

          Let r be operation’s request's associated request.

          -
        3. -

          If r’s url's scheme is not one of "http" and "https", throw a TypeError.

          -
        4. -

          If r’s method is not `GET`, throw a TypeError.

          -
        5. -

          If operation’s options is not null, throw a TypeError.

          -
        6. -

          Set requestResponses to the result of running Query Cache with operation’s request.

          -
        7. -

          For each requestResponse of requestResponses:

          -
            -
          1. -

            Remove the item whose value matches requestResponse from cache.

            -
          -
        8. -

          Append operation’s request/operation’s response to cache.

          -
        9. -

          If the cache write operation in the previous two steps failed due to exceeding the granted quota limit, throw a "QuotaExceededError" DOMException.

          -
        10. -

          Append operation’s request/operation’s response to addedItems.

          -
        -
      7. -

        Append operation’s request/operation’s response to resultList.

        -
      -
    3. -

      Return resultList.

      -
    -
  5. -

    And then, if an exception was thrown, then:

    -
      -
    1. -

      Remove all the items from the relevant request response list.

      -
    2. -

      For each requestResponse of backupCache:

      -
        -
      1. -

        Append requestResponse to the relevant request response list.

        -
      -
    3. -

      Throw the exception.

      -
    -

    Note: When an exception is thrown, the implementation does undo (roll back) any changes made to the cache storage during the batch operation job.

    -
-
-
-
-

Appendix B: Extended HTTP headers

-
-

Service Worker Script Request

-

An HTTP request to fetch a service worker's script resource will include the following header:

-
-
`Service-Worker` -
-

Indicates this request is a service worker's script resource request.

-

Note: This header helps administrators log the requests and detect threats.

-
-
-
-

Service Worker Script Response

-

An HTTP response to a service worker's script resource request can include the following header:

-
-
`Service-Worker-Allowed` -
-

Indicates the user agent will override the path restriction, which limits the maximum allowed scope url that the script can control, to the given value.

-

Note: The value is a URL. If a relative URL is given, it is parsed against the script’s URL.

-
-
- Default scope: -
// Maximum allowed scope defaults to the path the script sits in
-// "/js" in this example
-navigator.serviceWorker.register("/js/sw.js").then(() => {
-  console.log("Install succeeded with the default scope '/js'.");
-});
-
-
-
- Upper path without Service-Worker-Allowed header: -
// Set the scope to an upper path of the script location
-// Response has no Service-Worker-Allowed header
-navigator.serviceWorker.register("/js/sw.js", { scope: "/" }).catch(() => {
-  console.error("Install failed due to the path restriction violation.");
-});
-
-
-
- Upper path with Service-Worker-Allowed header: -
// Set the scope to an upper path of the script location
-// Response included "Service-Worker-Allowed : /"
-navigator.serviceWorker.register("/js/sw.js", { scope: "/" }).then(() => {
-  console.log("Install succeeded as the max allowed scope was overriden to '/'.");
-});
-
-
-
- A path restriction voliation even with Service-Worker-Allowed header: -
// Set the scope to an upper path of the script location
-// Response included "Service-Worker-Allowed : /foo"
-navigator.serviceWorker.register("/foo/bar/sw.js", { scope: "/" }).catch(() => {
-  console.error("Install failed as the scope is still out of the overriden maximum allowed scope.");
-});
-
-
-
-
-

Syntax

-

ABNF for the values of the headers used by the service worker's script resource requests and responses:

-
Service-Worker = %x73.63.72.69.70.74 ; "script", case-sensitive
-
-

Note: The validation of the Service-Worker-Allowed header’s values is done by URL parsing algorithm (in Update algorithm) instead of using ABNF.

-
-
-
-

8. Acknowledgements

-

Deep thanks go to Andrew Betts for organizing and hosting a small workshop of like-minded individuals including: Jake Archibald, Jackson Gabbard, Tobie Langel, Robin Berjon, Patrick Lauke, Christian Heilmann. From the clarity of the day’s discussions and the use-cases outlined there, much has become possible. Further thanks to Andrew for raising consciousness about the offline problem. His organization of EdgeConf and inclusion of Offline as a persistent topic there has created many opportunities and connections that have enabled this work to progress.

-

Anne van Kesteren has generously lent his encyclopedic knowledge of Web Platform arcana and standards development experience throughout the development of the service worker. This specification would be incomplete without his previous work in describing the real-world behavior of URLs, HTTP Fetch, Promises, and DOM. Similarly, this specification would not be possible without Ian Hickson’s rigorous Web Worker spec. Much thanks to him.

-

In no particular order, deep gratitude for design guidance and discussion goes to: Jungkee Song, Alec Flett, David Barrett-Kahn, Aaron Boodman, Michael Nordman, Tom Ashworth, Kinuko Yasuda, Darin Fisher, Jonas Sicking, Jesús Leganés Combarro, Mark Christian, Dave Hermann, Yehuda Katz, François Remy, Ilya Grigorik, Will Chan, Domenic Denicola, Nikhil Marathe, Yves Lafon, Adam Barth, Greg Simon, Devdatta Akhawe, Dominic Cooney, Jeffrey Yasskin, Joshua Bell, Boris Zbarsky, Matt Falkenhagen, Tobie Langel, Gavin Peters, Ben Kelly, Hiroki Nakagawa, Jake Archibald, Josh Soref, Jinho Bang, Yutaka Hirano, Michael(tm) Smith, isonmad, Ali Alabbas, Philip Jägenstedt, Mike Pennisi, and Eric Willigers.

-

Jason Weber, Chris Wilson, Paul Kinlan, Ehsan Akhgari, and Daniel Austin have provided valuable, well-timed feedback on requirements and the standardization process.

-

The authors would also like to thank Dimitri Glazkov for his scripts and formatting tools which have been essential in the production of this specification. The authors are also grateful for his considerable guidance.

-

Thanks also to Vivian Cromwell, Greg Simon, Alex Komoroske, Wonsuk Lee, and Seojin Kim for their considerable professional support.

-
-
-
-

Conformance

-

Document conventions

-

Conformance requirements are expressed with a combination of - descriptive assertions and RFC 2119 terminology. The key words “MUST”, - “MUST NOT”, “REQUIRED”, “SHALL”, “SHALL NOT”, “SHOULD”, “SHOULD NOT”, - “RECOMMENDED”, “MAY”, and “OPTIONAL” in the normative parts of this - document are to be interpreted as described in RFC 2119. - However, for readability, these words do not appear in all uppercase - letters in this specification.

-

All of the text of this specification is normative except sections - explicitly marked as non-normative, examples, and notes. [RFC2119]

-

Examples in this specification are introduced with the words “for example” - or are set apart from the normative text with class="example", - like this:

-
- -

This is an example of an informative example.

-
-

Informative notes begin with the word “Note” and are set apart from the - normative text with class="note", like this:

-

Note, this is an informative note.

-

Conformant Algorithms

-

Requirements phrased in the imperative as part of algorithms (such as - "strip any leading space characters" or "return false and abort these - steps") are to be interpreted with the meaning of the key word ("must", - "should", "may", etc) used in introducing the algorithm.

-

Conformance requirements phrased as algorithms or specific steps can be - implemented in any manner, so long as the end result is equivalent. In - particular, the algorithms defined in this specification are intended to - be easy to understand and are not intended to be performant. Implementers - are encouraged to optimize.

-
- - . - -

Index

-

Terms defined by this specification

- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -

Terms defined by reference

-
    -
  • - [csp-3] defines the following terms: -
      -
    • enforced -
    • monitored -
    -
  • - [DOM] defines the following terms: -
      -
    • Document -
    • Event -
    • EventInit -
    • EventTarget -
    • cancelable -
    • canceled flag -
    • context object -
    • creating an event -
    • dispatch -
    • dispatch flag -
    • event -
    • event listener -
    • fire an event -
    • isTrusted -
    • signal abort -
    • stop immediate propagation flag -
    • stop propagation flag -
    • type -
    -
  • - [ECMASCRIPT] defines the following terms: -
      -
    • abrupt completion -
    • completion -
    • detacharraybuffer -
    • execution context -
    • initializehostdefinedrealm -
    • map objects -
    • promise -
    -
  • - [FETCH] defines the following terms: -
      -
    • "report" -
    • Headers -
    • Request -
    • RequestInfo -
    • Response -
    • aborted flag -
    • append -
    • basic filtered response -
    • body (for response) -
    • cache mode -
    • cache state -
    • cancel -
    • client -
    • clone (for response) -
    • closed -
    • construct a readablestream object -
    • contains -
    • cors filtered response -
    • destination -
    • disturbed -
    • enqueue -
    • errored -
    • extract a mime type -
    • extracting header list values -
    • fetch -
    • fetch(input, init) -
    • filtered response -
    • get a reader -
    • guard -
    • header -
    • header list (for response) -
    • http fetch -
    • https state -
    • https state value -
    • initiator -
    • is local -
    • locked -
    • method -
    • name -
    • navigation request -
    • network error -
    • non-subresource request -
    • ok status -
    • opaque filtered response -
    • parser metadata -
    • potential-navigation-or-subresource request -
    • process response -
    • process response end-of-body -
    • read a chunk -
    • read all bytes -
    • redirect mode -
    • replaces client id -
    • request (for Request) -
    • reserved client -
    • response (for Response) -
    • service-workers mode -
    • signal -
    • status -
    • stream -
    • subresource request -
    • synchronous flag -
    • terminated -
    • type -
    • url -
    • use-url-credentials flag -
    • value -
    -
  • - [HTML] defines the following terms: -
      -
    • AbstractWorker -
    • DOMContentLoaded -
    • DedicatedWorkerGlobalScope -
    • EventHandler -
    • Location -
    • MessageEvent -
    • MessagePort -
    • Navigator -
    • PostMessageOptions -
    • SharedWorkerGlobalScope -
    • StructuredDeserializeWithTransfer -
    • StructuredSerializeWithTransfer -
    • Window -
    • WindowOrWorkerGlobalScope -
    • Worker -
    • WorkerGlobalScope -
    • WorkerLocation -
    • WorkerNavigator -
    • WorkerType -
    • active document -
    • active service worker -
    • ancestor origins list -
    • api base url -
    • api url character encoding -
    • application cache -
    • auxiliary browsing context -
    • browsing context -
    • classic script -
    • closing -
    • creation url -
    • current global object -
    • data -
    • discard a document -
    • dom manipulation task source -
    • environment -
    • environment discarding steps -
    • environment settings object -
    • event handler -
    • event handler event type -
    • event handler idl attribute -
    • event loop -
    • exceptions enabled flag -
    • execution ready flag -
    • fetch a classic worker script -
    • fetch a classic worker-imported script -
    • fetch a module worker script graph -
    • focusing steps -
    • global object -
    • has focus steps -
    • https state (for environment settings object) -
    • id -
    • import scripts into worker global scope -
    • importScripts(urls) -
    • in parallel -
    • incumbent settings object -
    • is top-level -
    • list of active timers -
    • message -
    • module script -
    • navigate -
    • nested browsing context -
    • origin (for environment settings object) -
    • owner set -
    • parent browsing context -
    • perform the fetch -
    • ports -
    • queue a microtask -
    • queue a task -
    • realm (for global object) -
    • realm execution context -
    • referrer policy (for environment settings object) -
    • relevant global object -
    • relevant realm -
    • relevant settings object -
    • replacement enabled -
    • responsible browsing context -
    • responsible document -
    • responsible event loop -
    • run a classic script -
    • run a module script -
    • same origin -
    • script -
    • shared worker -
    • source -
    • source browsing context -
    • target browsing context -
    • task -
    • task queues -
    • task source -
    • top-level browsing context -
    • triggered by user activation -
    • type -
    • unload a document -
    • unsafe response -
    • url -
    • user interaction task source -
    • web worker -
    -
  • - [INFRA] defines the following terms: -
      -
    • append (for set) -
    • ascii case-insensitive -
    • byte sequence -
    • contain -
    • continue -
    • dequeue -
    • enqueue -
    • entry -
    • exist -
    • for each (for map) -
    • get the keys -
    • is not empty -
    • item -
    • key -
    • list -
    • ordered map -
    • ordered set -
    • pair -
    • queue -
    • remove (for map) -
    • set -
    • struct -
    • value -
    -
  • - [MIMESNIFF] defines the following terms: -
      -
    • javascript mime type -
    -
  • - [page-visibility] defines the following terms: -
      -
    • VisibilityState -
    • visibilityState -
    -
  • - [promises-guide] defines the following terms: -
      -
    • a new promise -
    • a promise rejected with -
    • a promise resolved with -
    • reject -
    • transforming -
    • upon fulfillment -
    • upon rejection -
    • waiting for all -
    -
  • - [push] defines the following terms: -
      -
    • push -
    -
  • - [referrer-policy] defines the following terms: -
      -
    • referrer policy -
    -
  • - [rfc7230] defines the following terms: -
      -
    • field-value -
    -
  • - [rfc7231] defines the following terms: -
      -
    • vary -
    -
  • - [secure-contexts] defines the following terms: -
      -
    • potentially trustworthy origin -
    • potentially trustworthy url -
    • secure contexts -
    -
  • - [STREAMS] defines the following terms: -
      -
    • chunk -
    -
  • - [URL] defines the following terms: -
      -
    • equal -
    • fragment -
    • origin -
    • path -
    • query -
    • scheme -
    • url -
    • url parser -
    • url serializer -
    -
  • - [webappsec-referrer-policy] defines the following terms: -
      -
    • parse a referrer policy from a referrer-policy header -
    -
  • - [WebIDL] defines the following terms: -
      -
    • AbortError -
    • ByteString -
    • DOMException -
    • DOMString -
    • Exposed -
    • Global -
    • InvalidAccessError -
    • InvalidStateError -
    • NewObject -
    • QuotaExceededError -
    • SameObject -
    • SecureContext -
    • SecurityError -
    • USVString -
    • boolean -
    • create a frozen array -
    • created -
    • exception -
    • frozen array type -
    • message -
    • object -
    • partial interface -
    • present -
    • throw -
    -
-

References

-

Normative References

-
-
[CSP-3] -
Content Security Policy Level 3 URL: https://www.w3.org/TR/CSP3/ -
[DOM] -
Anne van Kesteren. DOM Standard. Living Standard. URL: https://dom.spec.whatwg.org/ -
[ECMASCRIPT] -
ECMAScript Language Specification. URL: https://tc39.github.io/ecma262/ -
[FETCH] -
Anne van Kesteren. Fetch Standard. Living Standard. URL: https://fetch.spec.whatwg.org/ -
[HTML] -
Anne van Kesteren; et al. HTML Standard. Living Standard. URL: https://html.spec.whatwg.org/multipage/ -
[INFRA] -
Anne van Kesteren; Domenic Denicola. Infra Standard. Living Standard. URL: https://infra.spec.whatwg.org/ -
[MIMESNIFF] -
Gordon P. Hemsley. MIME Sniffing Standard. Living Standard. URL: https://mimesniff.spec.whatwg.org/ -
[PAGE-VISIBILITY] -
Jatinder Mann; Arvind Jain. Page Visibility (Second Edition). 29 October 2013. REC. URL: https://www.w3.org/TR/page-visibility/ -
[PROMISES-GUIDE] -
Domenic Denicola. Writing Promise-Using Specifications. 16 February 2016. TAG Finding. URL: https://www.w3.org/2001/tag/doc/promises-guide -
[REFERRER-POLICY] -
Jochen Eisinger; Emily Stark. Referrer Policy. 26 January 2017. CR. URL: https://www.w3.org/TR/referrer-policy/ -
[RFC2119] -
S. Bradner. Key words for use in RFCs to Indicate Requirement Levels. March 1997. Best Current Practice. URL: https://tools.ietf.org/html/rfc2119 -
[RFC5234] -
D. Crocker, Ed.; P. Overell. Augmented BNF for Syntax Specifications: ABNF. January 2008. Internet Standard. URL: https://tools.ietf.org/html/rfc5234 -
[RFC7230] -
R. Fielding, Ed.; J. Reschke, Ed.. Hypertext Transfer Protocol (HTTP/1.1): Message Syntax and Routing. June 2014. Proposed Standard. URL: https://tools.ietf.org/html/rfc7230 -
[RFC7231] -
R. Fielding, Ed.; J. Reschke, Ed.. Hypertext Transfer Protocol (HTTP/1.1): Semantics and Content. June 2014. Proposed Standard. URL: https://tools.ietf.org/html/rfc7231 -
[SECURE-CONTEXTS] -
Mike West. Secure Contexts. 15 September 2016. CR. URL: https://www.w3.org/TR/secure-contexts/ -
[STREAMS] -
Adam Rice; Domenic Denicola; 吉野剛史 (Takeshi Yoshino). Streams Standard. Living Standard. URL: https://streams.spec.whatwg.org/ -
[URL] -
Anne van Kesteren. URL Standard. Living Standard. URL: https://url.spec.whatwg.org/ -
[WebIDL] -
Boris Zbarsky. Web IDL. 15 December 2016. ED. URL: https://heycam.github.io/webidl/ -
-

Informative References

-
-
[UNSANCTIONED-TRACKING] -
Unsanctioned Web Tracking. 17 July 2015. Finding of the W3C TAG. URL: https://www.w3.org/2001/tag/doc/unsanctioned-tracking/ -
-

IDL Index

-
[SecureContext, Exposed=(Window,Worker)]
-interface ServiceWorker : EventTarget {
-  readonly attribute USVString scriptURL;
-  readonly attribute ServiceWorkerState state;
-  void postMessage(any message, sequence<object> transfer);
-  void postMessage(any message, optional PostMessageOptions options);
-
-  // event
-  attribute EventHandler onstatechange;
-};
-ServiceWorker includes AbstractWorker;
-
-enum ServiceWorkerState {
-  "installing",
-  "installed",
-  "activating",
-  "activated",
-  "redundant"
-};
-
-[SecureContext, Exposed=(Window,Worker)]
-interface ServiceWorkerRegistration : EventTarget {
-  readonly attribute ServiceWorker? installing;
-  readonly attribute ServiceWorker? waiting;
-  readonly attribute ServiceWorker? active;
-  [SameObject] readonly attribute NavigationPreloadManager navigationPreload;
-
-  readonly attribute USVString scope;
-  readonly attribute ServiceWorkerUpdateViaCache updateViaCache;
-
-  [NewObject] Promise<void> update();
-  [NewObject] Promise<boolean> unregister();
-
-  // event
-  attribute EventHandler onupdatefound;
-};
-
-enum ServiceWorkerUpdateViaCache {
-  "imports",
-  "all",
-  "none"
-};
-
-partial interface Navigator {
-  [SecureContext, SameObject] readonly attribute ServiceWorkerContainer serviceWorker;
-};
-
-partial interface WorkerNavigator {
-  [SecureContext, SameObject] readonly attribute ServiceWorkerContainer serviceWorker;
-};
-
-[SecureContext, Exposed=(Window,Worker)]
-interface ServiceWorkerContainer : EventTarget {
-  readonly attribute ServiceWorker? controller;
-  readonly attribute Promise<ServiceWorkerRegistration> ready;
-
-  [NewObject] Promise<ServiceWorkerRegistration> register(USVString scriptURL, optional RegistrationOptions options);
-
-  [NewObject] Promise<any> getRegistration(optional USVString clientURL = "");
-  [NewObject] Promise<FrozenArray<ServiceWorkerRegistration>> getRegistrations();
-
-  void startMessages();
-
-
-  // events
-  attribute EventHandler oncontrollerchange;
-  attribute EventHandler onmessage; // event.source of message events is ServiceWorker object
-  attribute EventHandler onmessageerror;
-};
-
-dictionary RegistrationOptions {
-  USVString scope;
-  WorkerType type = "classic";
-  ServiceWorkerUpdateViaCache updateViaCache = "imports";
-};
-
-[SecureContext, Exposed=(Window,Worker)]
-interface NavigationPreloadManager {
-  Promise<void> enable();
-  Promise<void> disable();
-  Promise<void> setHeaderValue(ByteString value);
-  Promise<NavigationPreloadState> getState();
-};
-
-dictionary NavigationPreloadState {
-  boolean enabled = false;
-  ByteString headerValue;
-};
-
-[Global=(Worker,ServiceWorker), Exposed=ServiceWorker]
-interface ServiceWorkerGlobalScope : WorkerGlobalScope {
-  [SameObject] readonly attribute Clients clients;
-  [SameObject] readonly attribute ServiceWorkerRegistration registration;
-
-  [NewObject] Promise<void> skipWaiting();
-
-  attribute EventHandler oninstall;
-  attribute EventHandler onactivate;
-  attribute EventHandler onfetch;
-
-  // event
-  attribute EventHandler onmessage; // event.source of the message events is Client object
-  attribute EventHandler onmessageerror;
-};
-
-[Exposed=ServiceWorker]
-interface Client {
-  readonly attribute USVString url;
-  readonly attribute FrameType frameType;
-  readonly attribute DOMString id;
-  readonly attribute ClientType type;
-  void postMessage(any message, sequence<object> transfer);
-  void postMessage(any message, optional PostMessageOptions options);
-};
-
-[Exposed=ServiceWorker]
-interface WindowClient : Client {
-  readonly attribute VisibilityState visibilityState;
-  readonly attribute boolean focused;
-  [SameObject] readonly attribute FrozenArray<USVString> ancestorOrigins;
-  [NewObject] Promise<WindowClient> focus();
-  [NewObject] Promise<WindowClient?> navigate(USVString url);
-};
-
-enum FrameType {
-  "auxiliary",
-  "top-level",
-  "nested",
-  "none"
-};
-
-[Exposed=ServiceWorker]
-interface Clients {
-  // The objects returned will be new instances every time
-  [NewObject] Promise<any> get(DOMString id);
-  [NewObject] Promise<FrozenArray<Client>> matchAll(optional ClientQueryOptions options);
-  [NewObject] Promise<WindowClient?> openWindow(USVString url);
-  [NewObject] Promise<void> claim();
-};
-
-dictionary ClientQueryOptions {
-  boolean includeUncontrolled = false;
-  ClientType type = "window";
-};
-
-enum ClientType {
-  "window",
-  "worker",
-  "sharedworker",
-  "all"
-};
-
-[Constructor(DOMString type, optional ExtendableEventInit eventInitDict), Exposed=ServiceWorker]
-interface ExtendableEvent : Event {
-  void waitUntil(Promise<any> f);
-};
-
-dictionary ExtendableEventInit : EventInit {
-  // Defined for the forward compatibility across the derived events
-};
-
-[Constructor(DOMString type, FetchEventInit eventInitDict), Exposed=ServiceWorker]
-interface FetchEvent : ExtendableEvent {
-  [SameObject] readonly attribute Request request;
-  readonly attribute Promise<any> preloadResponse;
-  readonly attribute DOMString clientId;
-  readonly attribute DOMString resultingClientId;
-  readonly attribute DOMString replacesClientId;
-
-  void respondWith(Promise<Response> r);
-};
-
-dictionary FetchEventInit : ExtendableEventInit {
-  required Request request;
-  Promise<any> preloadResponse;
-  DOMString clientId = "";
-  DOMString resultingClientId = "";
-  DOMString replacesClientId = "";
-};
-
-[Constructor(DOMString type, optional ExtendableMessageEventInit eventInitDict), Exposed=ServiceWorker]
-interface ExtendableMessageEvent : ExtendableEvent {
-  readonly attribute any data;
-  readonly attribute USVString origin;
-  readonly attribute DOMString lastEventId;
-  [SameObject] readonly attribute (Client or ServiceWorker or MessagePort)? source;
-  readonly attribute FrozenArray<MessagePort> ports;
-};
-
-dictionary ExtendableMessageEventInit : ExtendableEventInit {
-  any data = null;
-  USVString origin = "";
-  DOMString lastEventId = "";
-  (Client or ServiceWorker or MessagePort)? source = null;
-  sequence<MessagePort> ports = [];
-};
-
-partial interface mixin WindowOrWorkerGlobalScope {
-  [SecureContext, SameObject] readonly attribute CacheStorage caches;
-};
-
-[SecureContext, Exposed=(Window,Worker)]
-interface Cache {
-  [NewObject] Promise<any> match(RequestInfo request, optional CacheQueryOptions options);
-  [NewObject] Promise<FrozenArray<Response>> matchAll(optional RequestInfo request, optional CacheQueryOptions options);
-  [NewObject] Promise<void> add(RequestInfo request);
-  [NewObject] Promise<void> addAll(sequence<RequestInfo> requests);
-  [NewObject] Promise<void> put(RequestInfo request, Response response);
-  [NewObject] Promise<boolean> delete(RequestInfo request, optional CacheQueryOptions options);
-  [NewObject] Promise<FrozenArray<Request>> keys(optional RequestInfo request, optional CacheQueryOptions options);
-};
-
-dictionary CacheQueryOptions {
-  boolean ignoreSearch = false;
-  boolean ignoreMethod = false;
-  boolean ignoreVary = false;
-};
-
-[SecureContext, Exposed=(Window,Worker)]
-interface CacheStorage {
-  [NewObject] Promise<any> match(RequestInfo request, optional MultiCacheQueryOptions options);
-  [NewObject] Promise<boolean> has(DOMString cacheName);
-  [NewObject] Promise<Cache> open(DOMString cacheName);
-  [NewObject] Promise<boolean> delete(DOMString cacheName);
-  [NewObject] Promise<sequence<DOMString>> keys();
-};
-
-dictionary MultiCacheQueryOptions : CacheQueryOptions {
-  DOMString cacheName;
-};
-
-
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/docs/v1/index.html b/docs/v1/index.html deleted file mode 100644 index 56f622b7..00000000 --- a/docs/v1/index.html +++ /dev/null @@ -1,11511 +0,0 @@ - - - - - Service Workers 1 - - - - - - - - - - - - - -
-

-

Service Workers 1

-

Editor’s Draft,

-
-
-
This version: -
https://w3c.github.io/ServiceWorker/v1/ -
Latest published version: -
https://www.w3.org/TR/service-workers/ -
Issue Tracking: -
GitHub -
Editors: -
(Google) -
(Microsoft‚ represented Samsung until April 2018) -
(Google) -
(Google) -
Tests: -
web-platform-tests service-workers/ (ongoing work) -
-
-
-
- V1 Branch -

This spec is a subset of the nightly version that is advancing toward a W3C Recommendation. For implementers and developers who seek all the latest features, Service Workers Nightly is a right document that is constantly reflecting new requirements.

-
-
- -
-
-
-

Abstract

-

This specification describes a method that enables applications to take advantage of persistent background processing, including hooks to enable bootstrapping of web applications while offline.

-

The core of this system is an event-driven Web Worker, which responds to events dispatched from documents and other sources. A system for managing installation, versions, and upgrades is provided.

-

The service worker is a generic entry point for event-driven background processing in the Web Platform that is extensible by other specifications.

-
-

Status of this document

-
-

This section describes the status of this document at the time of its publication. Other documents may supersede this document. A list of current W3C publications and the latest revision of this technical report can be found in the W3C technical reports index at https://www.w3.org/TR/.

-

This document was published by the Service Workers Working Group as an Editors Draft. This document is intended to become a W3C Recommendation.

-

-

Feedback and comments on this specification are welcome, please send them to public-webapps@w3.org (subscribe, archives) with [service-workers] at the start of your email’s subject.

-

Publication as an Editors Draft does not imply endorsement by the W3C Membership. This is a draft document and may - be updated, replaced or obsoleted by other documents at any time. It is inappropriate to cite - this document as other than work in progress.

-

This document was produced by a group operating under the W3C Patent Policy. W3C maintains a public list of any patent disclosures made in connection with the deliverables of the group; that page also includes instructions for disclosing a patent. An individual who has actual knowledge of a patent which the individual believes contains Essential Claim(s) must disclose the information in accordance with section 6 of the W3C Patent Policy.

-

This document is governed by the 1 February 2018 W3C Process Document.

-
-
- -
-
-

1. Motivations

-

This section is non-normative.

-

Web Applications traditionally assume that the network is reachable. This assumption pervades the platform. HTML documents are loaded over HTTP and traditionally fetch all of their sub-resources via subsequent HTTP requests. This places web content at a disadvantage versus other technology stacks.

-

The service worker is designed first to redress this balance by providing a Web Worker context, which can be started by a runtime when navigations are about to occur. This event-driven worker is registered against an origin and a path (or pattern), meaning it can be consulted when navigations occur to that location. Events that correspond to network requests are dispatched to the worker and the responses generated by the worker may override default network stack behavior. This puts the service worker, conceptually, between the network and a document renderer, allowing the service worker to provide content for documents, even while offline.

-

Web developers familiar with previous attempts to solve the offline problem have reported a deficit of flexibility in those solutions. As a result, the service worker is highly procedural, providing a maximum of flexibility at the price of additional complexity for developers. Part of this complexity arises from the need to keep service workers responsive in the face of a single-threaded execution model. As a result, APIs exposed by service workers are almost entirely asynchronous, a pattern familiar in other JavaScript contexts but accentuated here by the need to avoid blocking document and resource loading.

-

Developers using the HTML5 Application Cache have also reported that several attributes of the design contribute to unrecoverable errors. A key design principle of the service worker is that errors should always be recoverable. Many details of the update process of service workers are designed to avoid these hazards.

-

Service workers are started and kept alive by their relationship to events, not documents. This design borrows heavily from developer and vendor experience with Shared Workers and Chrome Background Pages. A key lesson from these systems is the necessity to time-limit the execution of background processing contexts, both to conserve resources and to ensure that background context loss and restart is top-of-mind for developers. As a result, service workers bear more than a passing resemblance to Chrome Event Pages, the successor to Background Pages. Service workers may be started by user agents without an attached document and may be killed by the user agent at nearly any time. Conceptually, service workers can be thought of as Shared Workers that can start, process events, and die without ever handling messages from documents. Developers are advised to keep in mind that service workers may be started and killed many times a second.

-

Service workers are generic, event-driven, time-limited script contexts that run at an origin. These properties make them natural endpoints for a range of runtime services that may outlive the context of a particular document, e.g. handling push notifications, background data synchronization, responding to resource requests from other origins, or receiving centralized updates to expensive-to-calculate data (e.g., geolocation or gyroscope).

-
-
-

2. Model

-
-

2.1. Service Worker

-

A service worker is a type of web worker. A service worker executes in the registering service worker client's origin.

-

A service worker has an associated state, which is one of parsed, installing, installed, activating, activated, and redundant. It is initially parsed.

-

A service worker has an associated script url (a URL).

-

A service worker has an associated type which is either "classic" or "module". Unless stated otherwise, it is "classic".

-

A service worker has an associated containing service worker registration (a service worker registration), which contains itself.

-

A service worker has an associated global object (a ServiceWorkerGlobalScope object or null).

-

A service worker has an associated script resource (a script), which represents its own script resource. It is initially set to null.

-

A script resource has an associated has ever been evaluated flag. It is initially unset.

-

A script resource has an associated HTTPS state (an HTTPS state value). It is initially "none".

-

A script resource has an associated referrer policy (a referrer policy). It is initially the empty string.

-

A service worker has an associated script resource map which is an ordered map where the keys are URLs and the values are responses.

-

A service worker has an associated skip waiting flag. Unless stated otherwise it is unset.

-

A service worker has an associated classic scripts imported flag. It is initially unset.

-

A service worker has an associated set of event types to handle (a set) whose item is an event listener’s event type. It is initially an empty set.

-

A service worker has an associated set of extended events (a set) whose item is an ExtendableEvent. It is initially an empty set.

-
-

2.1.1. Lifetime

-

The lifetime of a service worker is tied to the execution lifetime of events and not references held by service worker clients to the ServiceWorker object.

-

A user agent may terminate service workers at any time it:

-
    -
  • -

    Has no event to handle.

    -
  • -

    Detects abnormal operation: such as infinite loops and tasks exceeding imposed time limits (if any) while handling the events.

    -
-
-
-

2.1.2. Events

-

Service Workers specification defines lifecycle events (each of which is an event), install and activate. Service Workers and other specifications that extend Service Workers define set of events called functional events (each of which is an event) including fetch. (See the list of lifecycle events and functional events.)

-
-
-
-

2.2. Service Worker Registration

-

A service worker registration is a tuple of a scope url and a set of service workers, an installing worker, a waiting worker, and an active worker. A user agent may enable many service worker registrations at a single origin so long as the scope url of the service worker registration differs. A service worker registration of an identical scope url when one already exists in the user agent causes the existing service worker registration to be replaced.

-

A service worker registration has an associated scope url (a URL).

-

A service worker registration has an associated installing worker (a service worker or null) whose state is installing. It is initially set to null.

-

A service worker registration has an associated waiting worker (a service worker or null) whose state is installed. It is initially set to null.

-

A service worker registration has an associated active worker (a service worker or null) whose state is either activating or activated. It is initially set to null.

-

A service worker registration has an associated last update check time. It is initially set to null.

-

A service worker registration has an associated update via cache mode, which is "imports", "all", or "none". It is initially set to "imports".

-

A service worker registration has an associated uninstalling flag. It is initially unset.

-

A service worker registration has one or more task queues that back up the tasks from its active worker’s event loop’s corresponding task queues. (The target task sources for this back up operation are the handle fetch task source and the handle functional event task source.) The user agent dumps the active worker’s tasks to the service worker registration's task queues when the active worker is terminated and re-queues those tasks to the active worker’s event loop’s corresponding task queues when the active worker spins off. Unlike the task queues owned by event loops, the service worker registration's task queues are not processed by any event loops in and of itself.

-
-

2.2.1. Lifetime

-

A user agent must persistently keep a list of registered service worker registrations unless otherwise they are explicitly unregistered. A user agent has a scope to registration map that stores the entries of the tuple of service worker registration's scope url, serialized, and the corresponding service worker registration. The lifetime of service worker registrations is beyond that of the ServiceWorkerRegistration objects which represent them within the lifetime of their corresponding service worker clients.

-
-
-
-

2.3. Service Worker Client

-

A service worker client is an environment.

-

A service worker client has an associated discarded flag. It is initially unset.

-

Each service worker client has the following environment discarding steps:

-
    -
  1. -

    Set client’s discarded flag.

    -
-

Note: Implementations can discard clients whose discarded flag is set.

-

A service worker client has an algorithm defined as the origin that returns the service worker client's origin if the service worker client is an environment settings object, and the service worker client's creation URL’s origin otherwise.

-

A window client is a service worker client whose global object is a Window object.

-

A dedicated worker client is a service worker client whose global object is a DedicatedWorkerGlobalScope object.

-

A shared worker client is a service worker client whose global object is a SharedWorkerGlobalScope object.

-

A worker client is either a dedicated worker client or a shared worker client.

-
-
-

2.4. Selection and Use

-

A service worker client independently selects and uses a service worker registration for its own loading and its subresources. The selection of a service worker registration, upon a non-subresource request, is a process of either matching a service worker registration from scope to registration map or inheriting an existing service worker registration from its parent or owner context depending on the request’s url.

-

When the request’s url is not local, a service worker client matches a service worker registration from scope to registration map. That is, the service worker client attempts to consult a service worker registration whose scope url matches its creation URL.

-

When the request’s url is local, if the service worker client's responsible browsing context is a nested browsing context or the service worker client is a worker client, the service worker client inherits the service worker registration from its parent browsing context’s environment or from the environment of a Document in the service worker client's global object's owner set, respectively, if it exists.

-

If the selection was successful, the selected service worker registration's active worker starts to control the service worker client. Otherwise, the flow returns to fetch where it falls back to the default behavior. When a service worker client is controlled by an active worker, it is considered that the service worker client is using the active worker’s containing service worker registration.

-
-
-

2.5. Task Sources

-

The following additional task sources are used by service workers.

-
-
The handle fetch task source -
-

This task source is used for dispatching fetch events to service workers.

-
The handle functional event task source -
-

This task source is used for features that dispatch other functional events, e.g. push events, to service workers.

-

Note: A user agent may use a separate task source for each functional event type in order to avoid a head-of-line blocking phenomenon for certain functional events.

-
-
-
-

2.6. User Agent Shutdown

-

A user agent must maintain the state of its stored service worker registrations across restarts with the following rules:

- -

To attain this, the user agent must invoke Handle User Agent Shutdown when it terminates.

-
-
-
-

3. Client Context

-
- Bootstrapping with a service worker: -
// scope defaults to the path the script sits in
-// "/" in this example
-navigator.serviceWorker.register("/serviceworker.js").then(registration => {
-  console.log("success!");
-  if (registration.installing) {
-    registration.installing.postMessage("Howdy from your installing page.");
-  }
-}, err => {
-  console.error("Installing the worker failed!", err);
-});
-
-
-
-

3.1. ServiceWorker

-
[SecureContext, Exposed=(Window,Worker)]
-interface ServiceWorker : EventTarget {
-  readonly attribute USVString scriptURL;
-  readonly attribute ServiceWorkerState state;
-  void postMessage(any message, optional sequence<object> transfer = []);
-
-  // event
-  attribute EventHandler onstatechange;
-};
-ServiceWorker includes AbstractWorker;
-
-enum ServiceWorkerState {
-  "installing",
-  "installed",
-  "activating",
-  "activated",
-  "redundant"
-};
-
-

A ServiceWorker object represents a service worker. Each ServiceWorker object is associated with a service worker. Multiple separate objects implementing the ServiceWorker interface across documents and workers can all be associated with the same service worker simultaneously.

-

A ServiceWorker object has an associated ServiceWorkerState object which is itself associated with service worker's state.

-
-

3.1.1. scriptURL

-

The scriptURL attribute must return the service worker's serialized script url.

-
- For example, consider a document created by a navigation to https://example.com/app.html which matches via the following registration call which has been previously executed: -
// Script on the page https://example.com/app.html
-navigator.serviceWorker.register("/service_worker.js");
-
-

The value of navigator.serviceWorker.controller.scriptURL will be "https://example.com/service_worker.js".

-
-
-
-

3.1.2. state

-

The state attribute must return the value (in ServiceWorkerState enumeration) to which it was last set.

-
-
-

3.1.3. postMessage(message, transfer)

-

The postMessage(message, transfer) method must run these steps:

-
    -
  1. -

    If the state attribute value of the context object is "redundant", throw an "InvalidStateError" DOMException.

    -
  2. -

    Let serviceWorker be the service worker represented by the context object.

    -
  3. -

    Invoke Run Service Worker algorithm with serviceWorker as the argument.

    -
  4. -

    Let incumbentSettings be the incumbent settings object, and incumbentGlobal its global object.

    -
  5. -

    Let serializeWithTransferResult be StructuredSerializeWithTransfer(message, transfer). Rethrow any exceptions.

    -
  6. -

    Queue a task on the DOM manipulation task source to run the following steps:

    -
      -
    1. -

      Let source be determined by switching on the type of incumbentGlobal:

      -
      -
      ServiceWorkerGlobalScope -
      a new ServiceWorker object that represents incumbentGlobal’s service worker. -
      Window -
      a new WindowClient object that represents incumbentGlobal’s relevant settings object. -
      Otherwise -
      a new Client object that represents incumbentGlobal’s associated worker -
      -
    2. -

      Let origin be the Unicode serialization of incumbentSettings’s origin.

      -
    3. -

      Let destination be the ServiceWorkerGlobalScope object associated with serviceWorker.

      -
    4. -

      Let deserializeRecord be StructuredDeserializeWithTransfer(serializeWithTransferResult, destination’s Realm).

      -

      If this throws an exception, catch it, fire an event named messageerror at destination, using MessageEvent, with the origin attribute initialized to origin and the source attribute initialized to source, and then abort these steps.

      -
    5. -

      Let messageClone be deserializeRecord.[[Deserialized]].

      -
    6. -

      Let newPorts be a new frozen array consisting of all MessagePort objects in deserializeRecord.[[TransferredValues]], if any, maintaining their relative order.

      -
    7. -

      Let e be the result of creating an event named message, using ExtendableMessageEvent, with the origin attribute initialized to origin, the source attribute initialized to source, the data attribute initialized to messageClone, and the ports attribute initialized to newPorts.

      -
    8. -

      Dispatch e at destination.

      -
    9. -

      Invoke Update Service Worker Extended Events Set with serviceWorker and e.

      -
    -
-
-
-

3.1.4. Event handler

-

The following is the event handler (and its corresponding event handler event type) that must be supported, as event handler IDL attributes, by all objects implementing ServiceWorker interface:

- - - - - -
event handler - event handler event type -
onstatechange - statechange -
-
-
-
-

3.2. ServiceWorkerRegistration

-
[SecureContext, Exposed=(Window,Worker)]
-interface ServiceWorkerRegistration : EventTarget {
-  readonly attribute ServiceWorker? installing;
-  readonly attribute ServiceWorker? waiting;
-  readonly attribute ServiceWorker? active;
-
-  readonly attribute USVString scope;
-  readonly attribute ServiceWorkerUpdateViaCache updateViaCache;
-
-  [NewObject] Promise<void> update();
-  [NewObject] Promise<boolean> unregister();
-
-  // event
-  attribute EventHandler onupdatefound;
-};
-
-enum ServiceWorkerUpdateViaCache {
-  "imports",
-  "all",
-  "none"
-};
-
-

A ServiceWorkerRegistration object represents a service worker registration. Each ServiceWorkerRegistration object is associated with a service worker registration (a service worker registration). Multiple separate objects implementing the ServiceWorkerRegistration interface across documents and workers can all be associated with the same service worker registration simultaneously.

-
- -

installing attribute must return the value to which it was last set.

-

Note: The ServiceWorker objects returned from this attribute getter that represent the same service worker are the same objects.

-
-
- -

waiting attribute must return the value to which it was last set.

-

Note: The ServiceWorker objects returned from this attribute getter that represent the same service worker are the same objects.

-
-
- -

active attribute must return the value to which it was last set.

-

Note: The ServiceWorker objects returned from this attribute getter that represent the same service worker are the same objects.

-
-
-

3.2.4. scope

-

The scope attribute must return service worker registration's serialized scope url.

-
In the example in §3.1.1 scriptURL, the value of registration.scope, obtained from navigator.serviceWorker.ready.then(registration => console.log(registration.scope)) for example, will be "https://example.com/".
-
-
-

3.2.5. updateViaCache

-

The updateViaCache attribute must return service worker registration's update via cache mode.

-
-
-

3.2.6. update()

-

update() method must run these steps:

-
    -
  1. -

    Let registration be the service worker registration.

    -
  2. -

    Let newestWorker be the result of running Get Newest Worker algorithm passing registration as its argument.

    -
  3. -

    If newestWorker is null, return a promise rejected with an "InvalidStateError" DOMException and abort these steps.

    -
  4. -

    If the context object’s relevant settings object’s global object globalObject is a ServiceWorkerGlobalScope object, and globalObject’s associated service worker's state is installing, return a promise rejected with an "InvalidStateError" DOMException and abort these steps.

    -
  5. -

    Let promise be a promise.

    -
  6. -

    Let job be the result of running Create Job with update, registration’s scope url, newestWorker’s script url, promise, and the context object’s relevant settings object.

    -
  7. -

    Set job’s worker type to newestWorker’s type.

    -
  8. -

    Invoke Schedule Job with job.

    -
  9. -

    Return promise.

    -
-
-
- -

Note: The unregister() method unregisters the service worker registration. It is important to note that the currently controlled service worker client's active service worker’s containing service worker registration is effective until all the service worker clients (including itself) using this service worker registration unload. That is, the unregister() method only affects subsequent navigations.

-

unregister() method must run these steps:

-
    -
  1. -

    Let promise be a promise.

    -
  2. -

    Let job be the result of running Create Job with unregister, the scope url of the service worker registration, null, promise, and the context object’s relevant settings object.

    -
  3. -

    Invoke Schedule Job with job.

    -
  4. -

    Return promise.

    -
-
-
-

3.2.8. Event handler

-

The following is the event handler (and its corresponding event handler event type) that must be supported, as event handler IDL attributes, by all objects implementing ServiceWorkerRegistration interface:

- - - - - -
event handler - event handler event type -
onupdatefound - updatefound -
-
-
-
- -
partial interface Navigator {
-  [SecureContext, SameObject] readonly attribute ServiceWorkerContainer serviceWorker;
-};
-
-partial interface WorkerNavigator {
-  [SecureContext, SameObject] readonly attribute ServiceWorkerContainer serviceWorker;
-};
-
-

The serviceWorker attribute must return the ServiceWorkerContainer object that is associated with the context object.

-
-
-

3.4. ServiceWorkerContainer

-
[SecureContext, Exposed=(Window,Worker)]
-interface ServiceWorkerContainer : EventTarget {
-  readonly attribute ServiceWorker? controller;
-  readonly attribute Promise<ServiceWorkerRegistration> ready;
-
-  [NewObject] Promise<ServiceWorkerRegistration> register(USVString scriptURL, optional RegistrationOptions options);
-
-  [NewObject] Promise<any> getRegistration(optional USVString clientURL = "");
-  [NewObject] Promise<FrozenArray<ServiceWorkerRegistration>> getRegistrations();
-
-  void startMessages();
-
-
-  // events
-  attribute EventHandler oncontrollerchange;
-  attribute EventHandler onmessage; // event.source of message events is ServiceWorker object
-  attribute EventHandler onmessageerror;
-};
-
-
dictionary RegistrationOptions {
-  USVString scope;
-  WorkerType type = "classic";
-  ServiceWorkerUpdateViaCache updateViaCache = "imports";
-};
-
-

The user agent must create a ServiceWorkerContainer object when a Navigator object or a WorkerNavigator object is created and associate it with that object.

-

A ServiceWorkerContainer provides capabilities to register, unregister, and update the service worker registrations, and provides access to the state of the service worker registrations and their associated service workers.

-

A ServiceWorkerContainer has an associated service worker client, which is a service worker client whose global object is associated with the Navigator object or the WorkerNavigator object that the ServiceWorkerContainer is retrieved from.

-

A ServiceWorkerContainer object has an associated ready promise (a promise). It is initially set to a new promise.

-

A ServiceWorkerContainer object has a task source called the client message queue, initially empty. A client message queue can be enabled or disabled, and is initially disabled. When a ServiceWorkerContainer object’s client message queue is enabled, the event loop must use it as one of its task sources. When the ServiceWorkerContainer object’s relevant global object is a Window object, all tasks queued on its client message queue must be associated with its relevant settings object’s responsible document.

-
- -

controller attribute must run these steps:

-
    -
  1. -

    Let client be the context object’s service worker client.

    -
  2. -

    Return the ServiceWorker object that represents client’s active service worker.

    -
-

Note: navigator.serviceWorker.controller returns null if the request is a force refresh (shift+refresh). The ServiceWorker objects returned from this attribute getter that represent the same service worker are the same objects.

-
-
- -

ready attribute must run these steps:

-
    -
  1. -

    Let readyPromise be the context object's ready promise.

    -
  2. -

    If readyPromise is pending, run the following substeps in parallel:

    -
      -
    1. -

      Let registration be the result of running Match Service Worker Registration with the context object's service worker client's creation URL.

      -
    2. -

      If registration is not null, and registration’s active worker is not null, queue a task on readyPromise’s relevant settings object's responsible event loop, using the DOM manipulation task source, to resolve readyPromise with the ServiceWorkerRegistration object that represents registration in readyPromise’s relevant Realm.

      -
    -
  3. -

    Return readyPromise.

    -
-

Note: The returned ready promise will never reject. If it does not resolve in this algorithm, it will eventually resolve when a matching service worker registration is registered and its active worker is set. (See the relevant Activate algorithm step.)

-
-
- -

Note: The register(scriptURL, options) method creates or updates a service worker registration for the given scope url. If successful, a service worker registration ties the provided scriptURL to a scope url, which is subsequently used for navigation matching.

-

register(scriptURL, options) method must run these steps:

-
    -
  1. -

    Let p be a promise.

    -
  2. -

    Let client be the context object’s service worker client.

    -
  3. -

    Let scriptURL be the result of parsing scriptURL with the context object’s relevant settings object’s API base URL.

    -
  4. -

    If scriptURL is failure, reject p with a TypeError and abort these steps.

    -
  5. -

    Set scriptURL’s fragment to null.

    -

    Note: The user agent does not store the fragment of the script’s url. This means that the fragment does not have an effect on identifying service workers.

    -
  6. -

    If scriptURL’s scheme is not one of "http" and "https", reject p with a TypeError and abort these steps.

    -
  7. -

    If any of the strings in scriptURL’s path contains either ASCII case-insensitive "%2f" or ASCII case-insensitive "%5c", reject p with a TypeError and abort these steps.

    -
  8. -

    Let scopeURL be null.

    -
  9. -

    If options.scope is not present, set scopeURL to the result of parsing a string "./" with scriptURL.

    -

    Note: The scope url for the registration is set to the location of the service worker script by default.

    -
  10. -

    Else, set scopeURL to the result of parsing options.scope with the context object’s relevant settings object’s API base URL.

    -
  11. -

    If scopeURL is failure, reject p with a TypeError and abort these steps.

    -
  12. -

    Set scopeURL’s fragment to null.

    -

    Note: The user agent does not store the fragment of the scope url. This means that the fragment does not have an effect on identifying service worker registrations.

    -
  13. -

    If scopeURL’s scheme is not one of "http" and "https", reject p with a TypeError and abort these steps.

    -
  14. -

    If any of the strings in scopeURL’s path contains either ASCII case-insensitive "%2f" or ASCII case-insensitive "%5c", reject p with a TypeError and abort these steps.

    -
  15. -

    Let job be the result of running Create Job with register, scopeURL, scriptURL, p, and client.

    -
  16. -

    Set job’s worker type to options.type.

    -
  17. -

    Set job’s update via cache mode to options.updateViaCache.

    -
  18. -

    Invoke Schedule Job with job.

    -
  19. -

    Return p.

    -
-
-
- -

getRegistration(clientURL) method must run these steps:

-
    -
  1. -

    Let client be the context object’s service worker client.

    -
  2. -

    Let clientURL be the result of parsing clientURL with the context object’s relevant settings object’s API base URL.

    -
  3. -

    If clientURL is failure, return a promise rejected with a TypeError.

    -
  4. -

    Set clientURL’s fragment to null.

    -
  5. -

    If the origin of clientURL is not client’s origin, return a promise rejected with a "SecurityError" DOMException.

    -
  6. -

    Let promise be a new promise.

    -
  7. -

    Run the following substeps in parallel:

    -
      -
    1. -

      Let registration be the result of running Match Service Worker Registration algorithm with clientURL as its argument.

      -
    2. -

      If registration is not null, then:

      -
        -
      1. -

        Resolve promise with the ServiceWorkerRegistration object which represents registration.

        -
      -
    3. -

      Else:

      -
        -
      1. -

        Resolve promise with undefined.

        -
      -
    -
  8. -

    Return promise.

    -
-
-
- -

getRegistrations() method must run these steps:

-
    -
  1. -

    Let client be the context object's service worker client.

    -
  2. -

    Let promise be a new promise.

    -
  3. -

    Run the following steps in parallel:

    -
      -
    1. -

      Let registrations be a new list.

      -
    2. -

      For each scoperegistration of scope to registration map:

      -
        -
      1. -

        If the origin of the result of parsing scope is the same as client’s origin, and registration’s uninstalling flag is unset, then append registration to registrations.

        -
      -
    3. -

      Queue a task on promise’s relevant settings object's responsible event loop, using the DOM manipulation task source, to run the following steps:

      -
        -
      1. -

        Let registrationObjects be a new list.

        -
      2. -

        For each registration of registrations, append the ServiceWorkerRegistration object associated with registration to registrationObjects.

        -
      3. -

        Resolve promise with a new frozen array of registrationObjects in promise’s relevant Realm.

        -
      -
    -
  4. -

    Return promise.

    -
-
-
- -

startMessages() method must enable the context object’s client message queue if it is not enabled.

-
-
-

3.4.7. Event handlers

-

The following are the event handlers (and their corresponding event handler event types) that must be supported, as event handler IDL attributes, by all objects implementing the ServiceWorkerContainer interface:

- - - - - - - -
event handler - event handler event type -
oncontrollerchange - controllerchange -
onmessage - message -
onmessageerror - messageerror -
-

The first time the context object’s onmessage IDL attribute is set, its client message queue must be enabled.

-
-
-
-

3.5. Events

-

The following event is dispatched on ServiceWorker object:

- - - - - -
Event name - Interface - Dispatched when… -
statechange - Event - The state attribute of the ServiceWorker object is changed. -
-

The following event is dispatched on ServiceWorkerRegistration object:

- - - - - -
Event name - Interface - Dispatched when… -
updatefound - Event - The service worker registration's installing worker changes. (See step 8 of the Install algorithm.) -
-

The following events are dispatched on ServiceWorkerContainer object:

- - - - - -
Event name - Interface - Dispatched when… -
controllerchange - Event - The service worker client's active service worker changes. (See step 9.2 of the Activate algorithm. The skip waiting flag of a service worker causes activation of the service worker registration to occur while service worker clients are using the service worker registration, navigator.serviceWorker.controller immediately reflects the active worker as the service worker that controls the service worker client.) -
-
-
-
-

4. Execution Context

-
- Serving Cached Resources: -
// caching.js
-self.addEventListener("install", event => {
-  event.waitUntil(
-    // Open a cache of resources.
-    caches.open("shell-v1").then(cache => {
-      // Begins the process of fetching them.
-      // The coast is only clear when all the resources are ready.
-      return cache.addAll([
-        "/app.html",
-        "/assets/v1/base.css",
-        "/assets/v1/app.js",
-        "/assets/v1/logo.png",
-        "/assets/v1/intro_video.webm"
-      ]);
-    })
-  );
-});
-
-self.addEventListener("fetch", event => {
-  // No "fetch" events are dispatched to the service worker until it
-  // successfully installs and activates.
-
-  // All operations on caches are async, including matching URLs, so we use
-  // promises heavily. e.respondWith() even takes promises to enable this:
-  event.respondWith(
-    caches.match(e.request).then(response => {
-      return response || fetch(e.request);
-    }).catch(() => {
-      return caches.match("/fallback.html");
-    })
-  );
-});
-
-
-
-

4.1. ServiceWorkerGlobalScope

-
[Global=(Worker,ServiceWorker), Exposed=ServiceWorker]
-interface ServiceWorkerGlobalScope : WorkerGlobalScope {
-  [SameObject] readonly attribute Clients clients;
-  [SameObject] readonly attribute ServiceWorkerRegistration registration;
-
-  [NewObject] Promise<void> skipWaiting();
-
-  attribute EventHandler oninstall;
-  attribute EventHandler onactivate;
-  attribute EventHandler onfetch;
-
-  // event
-  attribute EventHandler onmessage; // event.source of the message events is Client object
-  attribute EventHandler onmessageerror;
-};
-
-

A ServiceWorkerGlobalScope object represents the global execution context of a service worker. A ServiceWorkerGlobalScope object has an associated service worker (a service worker). A ServiceWorkerGlobalScope object has an associated force bypass cache for importscripts flag. It is initially unset.

-

Note: ServiceWorkerGlobalScope object provides generic, event-driven, time-limited script execution contexts that run at an origin. Once successfully registered, a service worker is started, kept alive and killed by their relationship to events, not service worker clients. Any type of synchronous requests must not be initiated inside of a service worker.

-
-

4.1.1. clients

-

clients attribute must return the Clients object that is associated with the context object.

-
-
-

4.1.2. registration

-

The registration attribute must return the ServiceWorkerRegistration object that represents the service worker's containing service worker registration.

-
-
-

4.1.3. skipWaiting()

-

Note: The skipWaiting() method allows this service worker to progress from the registration's waiting position to active even while service worker clients are using the registration.

-

skipWaiting() method must run these steps:

-
    -
  1. -

    Let promise be a new promise.

    -
  2. -

    Run the following substeps in parallel:

    -
      -
    1. -

      Set service worker's skip waiting flag.

      -
    2. -

      Invoke Try Activate with service worker's containing service worker registration.

      -
    3. -

      Resolve promise with undefined.

      -
    -
  3. -

    Return promise.

    -
-
-
-

4.1.4. Event handlers

-

The following are the event handlers (and their corresponding event handler event types) that must be supported, as event handler IDL attributes, by all objects implementing the ServiceWorkerGlobalScope interface:

- - - - - - - - - -
event handler - event handler event type -
oninstall - install -
onactivate - activate -
onfetch - fetch -
onmessage - message -
onmessageerror - messageerror -
-
-
-
-

4.2. Client

-
[Exposed=ServiceWorker]
-interface Client {
-  readonly attribute USVString url;
-  readonly attribute FrameType frameType;
-  readonly attribute DOMString id;
-  readonly attribute ClientType type;
-  void postMessage(any message, optional sequence<object> transfer = []);
-};
-
-[Exposed=ServiceWorker]
-interface WindowClient : Client {
-  readonly attribute VisibilityState visibilityState;
-  readonly attribute boolean focused;
-  [SameObject] readonly attribute FrozenArray<USVString> ancestorOrigins;
-  [NewObject] Promise<WindowClient> focus();
-  [NewObject] Promise<WindowClient?> navigate(USVString url);
-};
-
-enum FrameType {
-  "auxiliary",
-  "top-level",
-  "nested",
-  "none"
-};
-
-

A Client object has an associated service worker client (a service worker client).

-

A Client object has an associated frame type, which is one of "auxiliary", "top-level", "nested", and "none". Unless stated otherwise it is "none".

-

A WindowClient object has an associated browsing context, which is its service worker client's global object's browsing context.

-

A WindowClient object has an associated visibility state, which is one of visibilityState attribute value.

-

A WindowClient object has an associated focus state, which is either true or false (initially false).

-

A WindowClient object has an associated ancestor origins array.

-
-

4.2.1. url

-

The url attribute must return the context object’s associated service worker client's serialized creation URL.

-
-
-

4.2.2. frameType

-

The frameType attribute must return the context object's frame type.

-
-
-

4.2.3. id

-

The id attribute must return its associated service worker client's id.

-
-
-

4.2.4. type

-

The type attribute must run these steps:

-
    -
  1. -

    Let client be context object's service worker client.

    -
  2. -

    If client is an environment settings object, then:

    -
      -
    1. -

      If client is a window client, return "window".

      -
    2. -

      Else if client is a dedicated worker client, return "worker".

      -
    3. -

      Else if client is a shared worker client, return "sharedworker".

      -
    -
  3. -

    Else:

    -
      -
    1. -

      Return "window".

      -
    -
-
-
-

4.2.5. postMessage(message, transfer)

-

The postMessage(message, transfer) method must run these steps:

-
    -
  1. -

    Let sourceSettings be the context object’s relevant settings object.

    -
  2. -

    Let destination be the ServiceWorkerContainer object whose service worker client is the context object’s service worker client, or null if no match is found.

    -
  3. -

    If destination is null, throw an "InvalidStateError" DOMException.

    -
  4. -

    Let serializeWithTransferResult be StructuredSerializeWithTransfer(message, transfer). Rethrow any exceptions.

    -
  5. -

    Add a task that runs the following steps to destination’s client message queue:

    -
      -
    1. -

      Let origin be the Unicode serialization of sourceSettings’s origin.

      -
    2. -

      Let source be a ServiceWorker object, which represents the service worker associated with sourceSettings’s global object.

      -
    3. -

      Let deserializeRecord be StructuredDeserializeWithTransfer(serializeWithTransferResult, destination’s relevant Realm).

      -

      If this throws an exception, catch it, fire an event named messageerror at destination, using MessageEvent, with the origin attribute initialized to origin and the source attribute initialized to source, and then abort these steps.

      -
    4. -

      Let messageClone be deserializeRecord.[[Deserialized]].

      -
    5. -

      Let newPorts be a new frozen array consisting of all MessagePort objects in deserializeRecord.[[TransferredValues]], if any.

      -
    6. -

      Dispatch an event named message at destination, using MessageEvent, with the origin attribute initialized to origin, the source attribute initialized to source, the data attribute initialized to messageClone, and the ports attribute initialized to newPorts.

      -
    -
-
-
-

4.2.6. visibilityState

-

The visibilityState attribute must return the context object’s visibility state.

-
-
-

4.2.7. focused

-

The focused attribute must return the context object’s focus state.

-
-
-

4.2.8. ancestorOrigins

-

The ancestorOrigins attribute must return the context object’s associated ancestor origins array.

-
-
-

4.2.9. focus()

-

The focus() method must run these steps:

-
    -
  1. -

    If this algorithm is not triggered by user activation, return a promise rejected with an "InvalidAccessError" DOMException.

    -
  2. -

    Let serviceWorkerEventLoop be the current global object's event loop.

    -
  3. -

    Let promise be a new promise.

    -
  4. -

    Queue a task to run the following steps on the context object's associated service worker client's responsible event loop using the user interaction task source:

    -
      -
    1. -

      Run the focusing steps with the context object's browsing context.

      -
    2. -

      Let frameType be the result of running Get Frame Type with the context object's browsing context.

      -
    3. -

      Let visibilityState be the context object's browsing context's active document's visibilityState attribute value.

      -
    4. -

      Let focusState be the result of running the has focus steps with the context object's browsing context's active document.

      -
    5. -

      Let ancestorOriginsList be the context object's browsing context's active document's relevant global object's Location object’s ancestor origins list's associated list.

      -
    6. -

      Queue a task to run the following steps on serviceWorkerEventLoop using the DOM manipulation task source:

      -
        -
      1. -

        Let windowClient be the result of running Create Window Client with the context object's associated service worker client, frameType, visibilityState, focusState, and ancestorOriginsList.

        -
      2. -

        If windowClient’s focus state is true, resolve promise with windowClient.

        -
      3. -

        Else, reject promise with a TypeError.

        -
      -
    -
  5. -

    Return promise.

    -
-
-
-

4.2.10. navigate(url)

-

The navigate(url) method must run these steps:

-
    -
  1. -

    Let url be the result of parsing url with the context object’s relevant settings object’s API base URL.

    -
  2. -

    If url is failure, return a promise rejected with a TypeError.

    -
  3. -

    If url is about:blank, return a promise rejected with a TypeError.

    -
  4. -

    If the context object’s associated service worker client's active service worker is not the context object’s relevant global object’s service worker, return a promise rejected with a TypeError.

    -
  5. -

    Let serviceWorkerEventLoop be the current global object's event loop.

    -
  6. -

    Let promise be a new promise.

    -
  7. -

    Queue a task to run the following steps on the context object's associated service worker client's responsible event loop using the user interaction task source:

    -
      -
    1. -

      Let browsingContext be the context object's browsing context.

      -
    2. -

      If browsingContext has discarded its Document, queue a task to reject promise with a TypeError, on serviceWorkerEventLoop using the DOM manipulation task source, and abort these steps.

      -
    3. -

      HandleNavigate: Navigate browsingContext to url with exceptions enabled. The source browsing context must be browsingContext.

      -
    4. -

      If the algorithm steps invoked in the step labeled HandleNavigate throws an exception, queue a task to reject promise with the exception, on serviceWorkerEventLoop using the DOM manipulation task source, and abort these steps.

      -
    5. -

      Let frameType be the result of running Get Frame Type with browsingContext.

      -
    6. -

      Let visibilityState be browsingContext’s active document’s visibilityState attribute value.

      -
    7. -

      Let focusState be the result of running the has focus steps with browsingContext’s active document.

      -
    8. -

      Let ancestorOriginsList be browsingContext’s active document's relevant global object's Location object’s ancestor origins list's associated list.

      -
    9. -

      Queue a task to run the following steps on serviceWorkerEventLoop using the DOM manipulation task source:

      -
        -
      1. -

        If browsingContext’s Window object’s environment settings object’s creation URL’s origin is not the same as the service worker's origin, resolve promise with null and abort these steps.

        -
      2. -

        Let windowClient be the result of running Create Window Client with the context object's service worker client, frameType, visibilityState, focusState, and ancestorOriginsList.

        -
      3. -

        Resolve promise with windowClient.

        -
      -
    -
  8. -

    Return promise.

    -
-
-
-
-

4.3. Clients

-
[Exposed=ServiceWorker]
-interface Clients {
-  // The objects returned will be new instances every time
-  [NewObject] Promise<any> get(DOMString id);
-  [NewObject] Promise<FrozenArray<Client>> matchAll(optional ClientQueryOptions options);
-  [NewObject] Promise<WindowClient?> openWindow(USVString url);
-  [NewObject] Promise<void> claim();
-};
-
-
dictionary ClientQueryOptions {
-  boolean includeUncontrolled = false;
-  ClientType type = "window";
-};
-
-
enum ClientType {
-  "window",
-  "worker",
-  "sharedworker",
-  "all"
-};
-
-

The user agent must create a Clients object when a ServiceWorkerGlobalScope object is created and associate it with that object.

-
-

4.3.1. get(id)

-

The get(id) method must run these steps:

-
    -
  1. -

    Let promise be a new promise.

    -
  2. -

    Run these substeps in parallel:

    -
      -
    1. -

      For each service worker client client whose origin is the same as the associated service worker's origin:

      -
        -
      1. -

        If client’s id is not id, continue.

        -
      2. -

        Wait for either client’s execution ready flag to be set or for client’s discarded flag to be set.

        -
      3. -

        If client’s execution ready flag is set, then invoke Resolve Get Client Promise with client and promise, and abort these steps.

        -
      -
    2. -

      Resolve promise with undefined.

      -
    -
  3. -

    Return promise.

    -
-
-
-

4.3.2. matchAll(options)

-

The matchAll(options) method must run these steps:

-
    -
  1. -

    Let promise be a new promise.

    -
  2. -

    Run the following steps in parallel:

    -
      -
    1. -

      Let targetClients be a new list.

      -
    2. -

      For each service worker client client whose origin is the same as the associated service worker's origin:

      -
        -
      1. -

        If client’s execution ready flag is unset or client’s discarded flag is set, continue.

        -
      2. -

        If client is not a secure context, continue.

        -
      3. -

        If options["includeUncontrolled"] is false, and if client’s active service worker is not the associated service worker, continue.

        -
      4. -

        Add client to targetClients.

        -
      -
    3. -

      Let matchedWindowData be a new list.

      -
    4. -

      Let matchedClients be a new list.

      -
    5. -

      For each service worker client client in targetClients:

      -
        -
      1. -

        If options["type"] is "window" or "all", and client is not an environment settings object or is a window client, then:

        -
          -
        1. -

          Let windowData be «[ "client" → client, "ancestorOriginsList" → a new list ]».

          -
        2. -

          Let browsingContext be null.

          -
        3. -

          Let isClientEnumerable be true.

          -
        4. -

          If client is an environment settings object, set browsingContext to client’s global object's browsing context.

          -
        5. -

          Else, set browsingContext to client’s target browsing context.

          -
        6. -

          Queue a task task to run the following substeps on browsingContext’s event loop using the user interaction task source:

          -
            -
          1. -

            If browsingContext has been discarded, then set isClientEnumerable to false and abort these steps.

            -
          2. -

            If client is a window client and client’s responsible document is not browsingContext’s active document, then set isClientEnumerable to false and abort these steps.

            -
          3. -

            Set windowData["frameType"] to the result of running Get Frame Type with browsingContext.

            -
          4. -

            Set windowData["visibilityState"] to browsingContext’s active document's visibilityState attribute value.

            -
          5. -

            Set windowData["focusState"] to the result of running the has focus steps with browsingContext’s active document as the argument.

            -
          6. -

            If client is a window client, then set windowData["ancestorOriginsList"] to browsingContext’s active document's relevant global object's Location object’s ancestor origins list's associated list.

            -
          -
        7. -

          Wait for task to have executed.

          -

          Note: Wait is a blocking wait, but implementers may run the iterations in parallel as long as the state is not broken.

          -
        8. -

          If isClientEnumerable is true, then:

          -
            -
          1. -

            Add windowData to matchedWindowData.

            -
          -
        -
      2. -

        Else if options["type"] is "worker" or "all" and client is a dedicated worker client, or options["type"] is "sharedworker" or "all" and client is a shared worker client, then:

        -
          -
        1. -

          Add client to matchedClients.

          -
        -
      -
    6. -

      Queue a task to run the following steps on promise’s relevant settings object's responsible event loop using the DOM manipulation task source:

      -
        -
      1. -

        Let clientObjects be a new list.

        -
      2. -

        For each windowData in matchedWindowData:

        -
          -
        1. -

          Let windowClient be the result of running Create Window Client algorithm with windowData["client"], windowData["frameType"], windowData["visibilityState"], windowData["focusState"], and windowData["ancestorOriginsList"] as the arguments.

          -
        2. -

          Append windowClient to clientObjects.

          -
        -
      3. -

        For each client in matchedClients:

        -
          -
        1. -

          Let clientObject be the result of running Create Client algorithm with client as the argument.

          -
        2. -

          Append clientObject to clientObjects.

          -
        -
      4. -

        Sort clientObjects such that:

        - -

        Note: Window clients are always placed before worker clients.

        -
      5. -

        Resolve promise with a new frozen array of clientObjects in promise’s relevant Realm.

        -
      -
    -
  3. -

    Return promise.

    -
-
-
-

4.3.3. openWindow(url)

-

The openWindow(url) method must run these steps:

-
    -
  1. -

    Let url be the result of parsing url with the context object’s relevant settings object’s API base URL.

    -
  2. -

    If url is failure, return a promise rejected with a TypeError.

    -
  3. -

    If url is about:blank, return a promise rejected with a TypeError.

    -
  4. -

    If this algorithm is not triggered by user activation, return a promise rejected with an "InvalidAccessError" DOMException.

    -
  5. -

    Let serviceWorkerEventLoop be the current global object's event loop.

    -
  6. -

    Let promise be a new promise.

    -
  7. -

    Run these substeps in parallel:

    -
      -
    1. -

      Let newContext be a new top-level browsing context.

      -
    2. -

      Queue a task to run the following steps on newContext’s Window object’s environment settings object's responsible event loop using the user interaction task source:

      -
        -
      1. -

        HandleNavigate: Navigate newContext to url with exceptions enabled and replacement enabled.

        -
      2. -

        If the algorithm steps invoked in the step labeled HandleNavigate throws an exception, queue a task to reject promise with the exception, on serviceWorkerEventLoop using the DOM manipulation task source, and abort these steps.

        -
      3. -

        Let frameType be the result of running Get Frame Type with newContext.

        -
      4. -

        Let visibilityState be newContext’s active document’s visibilityState attribute value.

        -
      5. -

        Let focusState be the result of running the has focus steps with newContext’s active document as the argument.

        -
      6. -

        Let ancestorOriginsList be newContext’s active document’s relevant global object’s Location object’s ancestor origins list's associated list.

        -
      7. -

        Queue a task to run the following steps on serviceWorkerEventLoop using the DOM manipulation task source:

        -
          -
        1. -

          If newContext’s Window object’s environment settings object's creation URL's origin is not the same as the service worker's origin, resolve promise with null and abort these steps.

          -
        2. -

          Let client be the result of running Create Window Client with newContext’s Window object’s environment settings object, frameType, visibilityState, focusState, and ancestorOriginsList as the arguments.

          -
        3. -

          Resolve promise with client.

          -
        -
      -
    -
  8. -

    Return promise.

    -
-
-
-

4.3.4. claim()

-

The claim() method must run these steps:

-
    -
  1. -

    If the service worker is not an active worker, return a promise rejected with an "InvalidStateError" DOMException.

    -
  2. -

    Let promise be a new promise.

    -
  3. -

    Run the following substeps in parallel:

    -
      -
    1. -

      For each service worker client client whose origin is the same as the service worker's origin:

      -
        -
      1. -

        If client’s execution ready flag is unset or client’s discarded flag is set, continue.

        -
      2. -

        If client is not a secure context, continue.

        -
      3. -

        Let registration be the result of running Match Service Worker Registration algorithm passing client’s creation URL as the argument.

        -
      4. -

        If registration is not the service worker's containing service worker registration, continue.

        -
      5. -

        If client’s active service worker is not the service worker, then:

        -
          -
        1. -

          Invoke Handle Service Worker Client Unload with client as the argument.

          -
        2. -

          Set client’s active service worker to service worker.

          -
        3. -

          Invoke Notify Controller Change algorithm with client as the argument.

          -
        -
      -
    2. -

      Resolve promise with undefined.

      -
    -
  4. -

    Return promise.

    -
-
-
-
-

4.4. ExtendableEvent

-
[Constructor(DOMString type, optional ExtendableEventInit eventInitDict), Exposed=ServiceWorker]
-interface ExtendableEvent : Event {
-  void waitUntil(Promise<any> f);
-};
-
-
dictionary ExtendableEventInit : EventInit {
-  // Defined for the forward compatibility across the derived events
-};
-
-

An ExtendableEvent object has an associated extend lifetime promises (an array of promises). It is initially an empty array.

-

An ExtendableEvent object has an associated pending promises count (the number of pending promises in the extend lifetime promises). It is initially set to zero.

-

An ExtendableEvent object has an associated timed out flag. It is initially unset, and is set after an optional user agent imposed delay if the pending promises count is greater than zero.

-

An ExtendableEvent object is said to be active when its timed out flag is unset and either its pending promises count is greater than zero or its dispatch flag is set.

-

Service workers have two lifecycle events, install and activate. Service workers use the ExtendableEvent interface for activate event and install event.

-

Service worker extensions that define event handlers may also use or extend the ExtendableEvent interface.

-
-

4.4.1. event.waitUntil(f)

-

waitUntil() method extends the lifetime of the event.

-

waitUntil(f) method must run these steps:

-
    -
  1. -

    If the isTrusted attribute is false, throw an "InvalidStateError" DOMException.

    -
  2. -

    If not active, throw an "InvalidStateError" DOMException.

    -

    Note: If no lifetime extension promise has been added in the task that called the event handlers, calling waitUntil() in subsequent asynchronous tasks will throw.

    -
  3. -

    Add f to the extend lifetime promises.

    -
  4. -

    Increment the pending promises count by one.

    -

    Note: The pending promises count is incremented even if the given promise has already been settled. The corresponding count decrement is done in the microtask queued by the reaction to the promise.

    -
  5. -

    Upon fulfillment or rejection of f, queue a microtask to run these substeps:

    -
      -
    1. -

      Decrement the pending promises count by one.

      -
    2. -

      Let registration be the context object's relevant global object's associated service worker's containing service worker registration.

      -
    3. -

      If registration’s uninstalling flag is set, invoke Try Clear Registration with registration.

      -
    4. -

      If registration is not null, invoke Try Activate with registration.

      -
    -
-

The user agent should not terminate a service worker if Service Worker Has No Pending Events returns false for that service worker.

-
-

Service workers and extensions that define event handlers may define their own behaviors, allowing the extend lifetime promises to suggest operation length, and the rejected state of any of the promise in extend lifetime promises to suggest operation failure.

-

Note: Service workers delay treating the installing worker as installed until all the promises in the install event’s extend lifetime promises resolve successfully. (See the relevant Install algorithm step.) If any of the promises rejects, the installation fails. This is primarily used to ensure that a service worker is not considered installed until all of the core caches it depends on are populated. Likewise, service workers delay treating the active worker as activated until all the promises in the activate event’s extend lifetime promises settle. (See the relevant Activate algorithm step.) This is primarily used to ensure that any functional events are not dispatched to the service worker until it upgrades database schemas and deletes the outdated cache entries.

-
-
-

4.5. FetchEvent

-
[Constructor(DOMString type, FetchEventInit eventInitDict), Exposed=ServiceWorker]
-interface FetchEvent : ExtendableEvent {
-  [SameObject] readonly attribute Request request;
-  readonly attribute DOMString clientId;
-
-  void respondWith(Promise<Response> r);
-};
-
-
dictionary FetchEventInit : ExtendableEventInit {
-  required Request request;
-  DOMString clientId = "";
-};
-
-

Service workers have an essential functional event fetch. For fetch event, service workers use the FetchEvent interface which extends the ExtendableEvent interface.

-

Each event using FetchEvent interface has an associated potential response (a response), initially set to null, and the following associated flags that are initially unset:

-
    -
  • -

    wait to respond flag

    -
  • -

    respond-with entered flag

    -
  • -

    respond-with error flag

    -
-
-

4.5.1. event.request

-

request attribute must return the value it was initialized to.

-
-
-

4.5.2. event.clientId

-

clientId attribute must return the value it was initialized to. When an event is created the attribute must be initialized to the empty string.

-
-
-

4.5.3. event.respondWith(r)

-

Note: Developers can set the argument r with either a promise that resolves with a Response object or a Response object (which is automatically cast to a promise). Otherwise, a network error is returned to Fetch. Renderer-side security checks about tainting for cross-origin content are tied to the types of filtered responses defined in Fetch.

-

respondWith(r) method must run these steps:

-
    -
  1. -

    If the dispatch flag is unset, throw an "InvalidStateError" DOMException.

    -
  2. -

    If the respond-with entered flag is set, throw an "InvalidStateError" DOMException.

    -
  3. -

    Add r to the extend lifetime promises.

    -
  4. -

    Increment the pending promises count by one.

    -

    Note: The pending promises count is incremented even if the given promise has already been settled. The corresponding count decrement is done in the microtask queued by the reaction to the promise.

    -
  5. -

    Upon fulfillment or rejection of r, queue a microtask to run these substeps:

    -
      -
    1. -

      Decrement the pending promises count by one.

      -
    2. -

      Let registration be the context object's relevant global object's associated service worker's containing service worker registration.

      -
    3. -

      If registration’s uninstalling flag is set, invoke Try Clear Registration with registration.

      -
    4. -

      If registration is not null, invoke Try Activate with registration.

      -
    -

    Note: event.respondWith(r) extends the lifetime of the event by default as if event.waitUntil(r) is called.

    -
  6. -

    Set the stop propagation flag and stop immediate propagation flag.

    -
  7. -

    Set the respond-with entered flag.

    -
  8. -

    Set the wait to respond flag.

    -
  9. -

    Let targetRealm be the relevant Realm of the context object.

    -
  10. -

    Run the following substeps in parallel:

    -
      -
    1. -

      Wait until r settles.

      -
    2. -

      If r rejected, then:

      -
        -
      1. -

        Set the respond-with error flag.

        -
      -
    3. -

      If r resolved with response, then:

      -
        -
      1. -

        If response is a Response object, then:

        -
          -
        1. -

          If response is disturbed or locked, then:

          -
            -
          1. -

            Set the respond-with error flag.

            -
          -
        2. -

          Else:

          -
            -
          1. -

            Let bytes be an empty byte sequence.

            -
          2. -

            Let end-of-body be false.

            -
          3. -

            Let done be false.

            -
          4. -

            Let potentialResponse be a copy of response’s associated response, except for its body.

            -
          5. -

            If response’s body is non-null, run these substeps:

            -
              -
            1. -

              Let reader be the result of getting a reader from response’s body's stream.

              -
            2. -

              Let strategy be an object created in targetRealm. The user agent may choose any object.

              -
            3. -

              Let pull be an action that runs these subsubsteps:

              -
                -
              1. -

                Let promise be the result of reading a chunk from response’s body's stream with reader.

                -
              2. -

                When promise is fulfilled with an object whose done property is false and whose value property is a Uint8Array object, append the bytes represented by the value property to bytes and perform ! DetachArrayBuffer with the ArrayBuffer object wrapped by the value property.

                -
              3. -

                When promise is fulfilled with an object whose done property is true, set end-of-body to true.

                -
              4. -

                When promise is fulfilled with a value that matches with neither of the above patterns, or promise is rejected, error newStream with a TypeError.

                -
              -
            4. -

              Let cancel be an action that cancels response’s body's stream with reader.

              -
            5. -

              Let newStream be the result of construct a ReadableStream object with strategy, pull and cancel in targetRealm.

              -
            6. -

              Set potentialResponse’s body to a new body whose stream is newStream.

              -
            7. -

              Run these subsubsteps repeatedly in parallel while done is false:

              -
                -
              1. -

                If newStream is errored, then set done to true.

                -
              2. -

                Otherwise, if bytes is empty and end-of-body is true, then close newStream and set done to true.

                -
              3. -

                Otherwise, if bytes is not empty, run these subsubsubsteps:

                -
                  -
                1. -

                  Let chunk be a subsequence of bytes starting from the beginning of bytes.

                  -
                2. -

                  Remove chunk from bytes.

                  -
                3. -

                  Let buffer be an ArrayBuffer object created in targetRealm and containing chunk.

                  -
                4. -

                  Enqueue a Uint8Array object created in targetRealm and wrapping buffer to newStream.

                  -
                -
              -
            -

            Note: These substeps are meant to produce the observable equivalent of "piping" response’s body's stream into potentialResponse.

            -
          6. -

            Set the potential response to potentialResponse.

            -
          -
        -
      2. -

        Else:

        -
          -
        1. -

          Set the respond-with error flag.

          -
        -

        Note: If the respond-with error flag is set, a network error is returned to Fetch through Handle Fetch algorithm. (See the step 21.1.) Otherwise, the value response is returned to Fetch through Handle Fetch algorithm. (See the step 22.1.)

        -
      -
    4. -

      Unset the wait to respond flag.

      -
    -
-
-
-
-

4.6. ExtendableMessageEvent

-
[Constructor(DOMString type, optional ExtendableMessageEventInit eventInitDict), Exposed=ServiceWorker]
-interface ExtendableMessageEvent : ExtendableEvent {
-  readonly attribute any data;
-  readonly attribute USVString origin;
-  readonly attribute DOMString lastEventId;
-  [SameObject] readonly attribute (Client or ServiceWorker or MessagePort)? source;
-  readonly attribute FrozenArray<MessagePort> ports;
-};
-
-
dictionary ExtendableMessageEventInit : ExtendableEventInit {
-  any data = null;
-  USVString origin = "";
-  DOMString lastEventId = "";
-  (Client or ServiceWorker or MessagePort)? source = null;
-  sequence<MessagePort> ports = [];
-};
-
-

Service workers define the extendable message event to allow extending the lifetime of the event. For the message event, service workers use the ExtendableMessageEvent interface which extends the ExtendableEvent interface.

-
-

4.6.1. event.data

-

The data attribute must return the value it was initialized to. When the object is created, this attribute must be initialized to null. It represents the message being sent.

-
-
-

4.6.2. event.origin

-

The origin attribute must return the value it was initialized to. When the object is created, this attribute must be initialized to the empty string. It represents the origin of the service worker client that sent the message.

-
-
-

4.6.3. event.lastEventId

-

The lastEventId attribute must return the value it was initialized to. When the object is created, this attribute must be initialized to the empty string.

-
-
-

4.6.4. event.source

-

The source attribute must return the value it was initialized to. When the object is created, this attribute must be initialized to null. It represents the Client object from which the message is sent.

-
-
-

4.6.5. event.ports

-

The ports attribute must return the value it was initialized to. When the object is created, this attribute must be initialized to the empty array. It represents the MessagePort array being sent.

-
-
-
-

4.7. Events

-

The following events are dispatched on ServiceWorkerGlobalScope object:

- - - - - - - - - - - - - - - -
Event name - Interface - Category - Dispatched when… -
install - ExtendableEvent - Lifecycle - The service worker's containing service worker registration’s installing worker changes. (See step 11.2 of the Install algorithm.) -
activate - ExtendableEvent - Lifecycle - The service worker's containing service worker registration’s active worker changes. (See step 12.2 of the Activate algorithm.) -
fetch - FetchEvent - Functional - The http fetch invokes Handle Fetch with request. As a result of performing Handle Fetch, the service worker returns a response to the http fetch. The response, represented by a Response object, can be retrieved from a Cache object or directly from network using self.fetch(input, init) method. (A custom Response object can be another option.) -
push - PushEvent - Functional - (See Firing a push event.) -
notificationclick - NotificationEvent - Functional - (See Activating a notification.) -
notificationclose - NotificationEvent - Functional - (See Closing a notification.) -
sync - SyncEvent - Functional - (See Firing a sync event.) -
canmakepayment - CanMakePaymentEvent - Functional - (See Handling a CanMakePaymentEvent.) -
paymentrequest - PaymentRequestEvent - Functional - (See Handling a PaymentRequestEvent.) -
message - ExtendableMessageEvent - Legacy - When it receives a message. -
messageerror - MessageEvent - Legacy - When it was sent a message that cannot be deserialized. -
-
-
-
-

5. Caches

-

To allow authors to fully manage their content caches for offline use, the Window and the WorkerGlobalScope provide the asynchronous caching methods that open and manipulate Cache objects. An origin can have multiple, named Cache objects, whose contents are entirely under the control of scripts. Caches are not shared across origins, and they are completely isolated from the browser’s HTTP cache.

-
-

5.1. Constructs

-

A request response list is a list of pairs consisting of a request (a request) and a response (a response).

-

The relevant request response list is the instance that the context object represents.

-

A name to cache map is an ordered map whose entry consists of a key (a string that represents the name of a request response list) and a value (a request response list).

-

Each origin has an associated name to cache map.

-

The relevant name to cache map is the instance of the context object's associated global object's environment settings object's origin.

-
-
-

5.2. Understanding Cache Lifetimes

-

The Cache instances are not part of the browser’s HTTP cache. The Cache objects are exactly what authors have to manage themselves. The Cache objects do not get updated unless authors explicitly request them to be. The Cache objects do not expire unless authors delete the entries. The Cache objects do not disappear just because the service worker script is updated. That is, caches are not updated automatically. Updates must be manually managed. This implies that authors should version their caches by name and make sure to use the caches only from the version of the service worker that can safely operate on.

-
-
-

5.3. self.caches

-
partial interface WindowOrWorkerGlobalScope {
-  [SecureContext, SameObject] readonly attribute CacheStorage caches;
-};
-
-
-

5.3.1. caches

-

caches attribute must return this object’s associated CacheStorage object.

-
-
-
-

5.4. Cache

-
[SecureContext, Exposed=(Window,Worker)]
-interface Cache {
-  [NewObject] Promise<any> match(RequestInfo request, optional CacheQueryOptions options);
-  [NewObject] Promise<FrozenArray<Response>> matchAll(optional RequestInfo request, optional CacheQueryOptions options);
-  [NewObject] Promise<void> add(RequestInfo request);
-  [NewObject] Promise<void> addAll(sequence<RequestInfo> requests);
-  [NewObject] Promise<void> put(RequestInfo request, Response response);
-  [NewObject] Promise<boolean> delete(RequestInfo request, optional CacheQueryOptions options);
-  [NewObject] Promise<FrozenArray<Request>> keys(optional RequestInfo request, optional CacheQueryOptions options);
-};
-
-
dictionary CacheQueryOptions {
-  boolean ignoreSearch = false;
-  boolean ignoreMethod = false;
-  boolean ignoreVary = false;
-};
-
-

A Cache object represents a request response list. Multiple separate objects implementing the Cache interface across documents and workers can all be associated with the same request response list simultaneously.

-

A cache batch operation is a struct that consists of:

- -
-

5.4.1. match(request, options)

-

match(request, options) method must run these steps:

-
    -
  1. -

    Let promise be a new promise.

    -
  2. -

    Run these substeps in parallel:

    -
      -
    1. -

      Let p be the result of running the algorithm specified in matchAll(request, options) method with request and options.

      -
    2. -

      Wait until p settles.

      -
    3. -

      If p rejects with an exception, then:

      -
        -
      1. -

        Reject promise with that exception.

        -
      -
    4. -

      Else if p resolves with an array, responses, then:

      -
        -
      1. -

        If responses is an empty array, then:

        -
          -
        1. -

          Resolve promise with undefined.

          -
        -
      2. -

        Else:

        -
          -
        1. -

          Resolve promise with the first element of responses.

          -
        -
      -
    -
  3. -

    Return promise.

    -
-
-
-

5.4.2. matchAll(request, options)

-

matchAll(request, options) method must run these steps:

-
    -
  1. -

    Let r be null.

    -
  2. -

    If the optional argument request is not omitted, then:

    -
      -
    1. -

      If request is a Request object, then:

      -
        -
      1. -

        Set r to request’s request.

        -
      2. -

        If r’s method is not `GET` and options.ignoreMethod is false, return a promise resolved with an empty array.

        -
      -
    2. -

      Else if request is a string, then:

      -
        -
      1. -

        Set r to the associated request of the result of invoking the initial value of Request as constructor with request as its argument. If this throws an exception, return a promise rejected with that exception.

        -
      -
    -
  3. -

    Let realm be the context object's relevant realm.

    -
  4. -

    Let promise be a new promise.

    -
  5. -

    Run these substeps in parallel:

    -
      -
    1. -

      Let responses be an empty list.

      -
    2. -

      If the optional argument request is omitted, then:

      -
        -
      1. -

        For each requestResponse of the relevant request response list:

        -
          -
        1. -

          Add a copy of requestResponse’s response to responses.

          -
        -
      -
    3. -

      Else:

      -
        -
      1. -

        Let requestResponses be the result of running Query Cache with r and options.

        -
      2. -

        For each requestResponse of requestResponses:

        -
          -
        1. -

          Add a copy of requestResponse’s response to responses.

          -
        -
      -
    4. -

      Queue a task, on promise’s relevant settings object's responsible event loop using the DOM manipulation task source, to perform the following steps:

      -
        -
      1. -

        Let responseList be a list.

        -
      2. -

        For each response of responses:

        -
          -
        1. -

          Add a new Response object associated with response and a new Headers object whose guard is "immutable" to responseList.

          -
        -
      3. -

        Resolve promise with a frozen array created from responseList, in realm.

        -
      -
    -
  6. -

    Return promise.

    -
-
-
-

5.4.3. add(request)

-

add(request) method must run these steps:

-
    -
  1. -

    Let requests be an array containing only request.

    -
  2. -

    Let responseArrayPromise be the result of running the algorithm specified in addAll(requests) passing requests as the argument.

    -
  3. -

    Return the result of transforming responseArrayPromise with a fulfillment handler that returns undefined.

    -
-
-
-

5.4.4. addAll(requests)

-

addAll(requests) method must run these steps:

-
    -
  1. -

    Let responsePromises be an empty list.

    -
  2. -

    Let requestList be an empty list.

    -
  3. -

    For each request whose type is Request in requests:

    -
      -
    1. -

      Let r be request’s request.

      -
    2. -

      If r’s url's scheme is not one of "http" and "https", or r’s method is not `GET`, return a promise rejected with a TypeError.

      -
    -
  4. -

    For each request in requests:

    -
      -
    1. -

      Let r be the associated request of the result of invoking the initial value of Request as constructor with request as its argument. If this throws an exception, return a promise rejected with that exception.

      -
    2. -

      If r’s url's scheme is not one of "http" and "https", then:

      -
        -
      1. -

        Terminate all the ongoing fetches initiated by requests with the aborted flag set.

        -
      2. -

        Return a promise rejected with a TypeError.

        -
      -
    3. -

      If r’s client's global object is a ServiceWorkerGlobalScope object, set request’s service-workers mode to "foreign".

      -
    4. -

      Set r’s initiator to "fetch" and destination to "subresource".

      -
    5. -

      Add r to requestList.

      -
    6. -

      Let responsePromise be a new promise.

      -
    7. -

      Run the following substeps in parallel:

      -
        -
      • -

        Fetch r.

        -
      • -

        To process response for response, run these substeps:

        -
          -
        1. -

          If response’s type is "error", or response’s status is not an ok status or is 206, reject responsePromise with a TypeError.

          -
        2. -

          Else if response’s header list contains a header named `Vary`, then:

          -
            -
          1. -

            Let fieldValues be the list containing the elements corresponding to the field-values of the Vary header.

            -
          2. -

            For each fieldValue of fieldValues:

            -
              -
            1. -

              If fieldValue matches "*", then:

              -
                -
              1. -

                Reject responsePromise with a TypeError.

                -
              2. -

                Terminate all the ongoing fetches initiated by requests with the aborted flag set.

                -
              3. -

                Abort these steps.

                -
              -
            -
          -
        -
      • -

        To process response end-of-body for response, run these substeps:

        -
          -
        1. -

          If response’s aborted flag is set, reject responsePromise with an "AbortError" DOMException and abort these steps.

          -
        2. -

          Resolve responsePromise with response.

          -
        -

        Note: The cache commit is allowed when the response’s body is fully received.

        -
      -
    8. -

      Add responsePromise to responsePromises.

      -
    -
  5. -

    Let p be waiting for all of responsePromises.

    -
  6. -

    Return the result of transforming p with a fulfillment handler that, when called with argument responses, performs the following substeps:

    -
      -
    1. -

      Let operations be an empty list.

      -
    2. -

      Let index be zero.

      -
    3. -

      For each response in responses:

      -
        -
      1. -

        Let operation be a cache batch operation.

        -
      2. -

        Set operation’s type to "put".

        -
      3. -

        Set operation’s request to requestList[index].

        -
      4. -

        Set operation’s response to response.

        -
      5. -

        Append operation to operations.

        -
      6. -

        Increment index by one.

        -
      -
    4. -

      Let realm be the context object's relevant realm.

      -
    5. -

      Let cacheJobPromise be a new promise.

      -
    6. -

      Run the following substeps in parallel:

      -
        -
      1. -

        Let errorData be null.

        -
      2. -

        Invoke Batch Cache Operations with operations. If this throws an exception, set errorData to the exception.

        -
      3. -

        Queue a task, on cacheJobPromise’s relevant settings object's responsible event loop using the DOM manipulation task source, to perform the following substeps:

        -
          -
        1. -

          If errorData is null, resolve cacheJobPromise with undefined.

          -
        2. -

          Else, reject cacheJobPromise with a new exception with errorData and a user agent-defined message, in realm.

          -
        -
      -
    7. -

      Return cacheJobPromise.

      -
    -
-
-
-

5.4.5. put(request, response)

-

put(request, response) method must run these steps:

-
    -
  1. -

    Let r be null.

    -
  2. -

    If request is a Request object, then:

    -
      -
    1. -

      Set r to request’s request.

      -
    2. -

      If r’s url's scheme is not one of "http" and "https", or r’s method is not `GET`, return a promise rejected with a TypeError.

      -
    -
  3. -

    Else if request is a string, then:

    -
      -
    1. -

      Set r to the associated request of the result of invoking the initial value of Request as constructor with request as its argument. If this throws an exception, return a promise rejected with that exception.

      -
    2. -

      If r’s url's scheme is not one of "http" and "https", return a promise rejected with a TypeError.

      -
    -
  4. -

    If response’s associated response's status is 206, return a promise rejected with a TypeError.

    -
  5. -

    If response’s associated response's header list contains a header named `Vary`, then:

    -
      -
    1. -

      Let fieldValues be the list containing the items corresponding to the Vary header’s field-values.

      -
    2. -

      For each fieldValue in fieldValues:

      -
        -
      1. -

        If fieldValue matches "*", return a promise rejected with a TypeError.

        -
      -
    -
  6. -

    If response is disturbed or locked, return a promise rejected with a TypeError.

    -
  7. -

    Let clonedResponse be the result of cloning response’s associated response.

    -
  8. -

    If response’s body is non-null, run these substeps:

    -
      -
    1. -

      Let dummyStream be an empty ReadableStream object.

      -
    2. -

      Set response’s body to a new body whose stream is dummyStream.

      -
    3. -

      Let reader be the result of getting a reader from dummyStream.

      -
    4. -

      Read all bytes from dummyStream with reader.

      -
    -
  9. -

    Let operations be an empty list.

    -
  10. -

    Let operation be a cache batch operation.

    -
  11. -

    Set operation’s type to "put".

    -
  12. -

    Set operation’s request to r.

    -
  13. -

    Set operation’s response to clonedResponse.

    -
  14. -

    Append operation to operations.

    -
  15. -

    Let realm be the context object's relevant realm.

    -
  16. -

    Let cacheJobPromise be a new promise.

    -
  17. -

    Run the following substeps in parallel:

    -
      -
    1. -

      Let errorData be null.

      -
    2. -

      Invoke Batch Cache Operations with operations. If this throws an exception, set errorData to the exception.

      -
    3. -

      Queue a task, on cacheJobPromise’s relevant settings object's responsible event loop using the DOM manipulation task source, to perform the following substeps:

      -
        -
      1. -

        If errorData is null, resolve cacheJobPromise with undefined.

        -
      2. -

        Else, reject cacheJobPromise with a new exception with errorData and a user agent-defined message, in realm.

        -
      -
    -
  18. -

    Return cacheJobPromise.

    -
-
-
-

5.4.6. delete(request, options)

-

delete(request, options) method must run these steps:

-
    -
  1. -

    Let r be null.

    -
  2. -

    If request is a Request object, then:

    -
      -
    1. -

      Set r to request’s request.

      -
    2. -

      If r’s method is not `GET` and options.ignoreMethod is false, return a promise resolved with false.

      -
    -
  3. -

    Else if request is a string, then:

    -
      -
    1. -

      Set r to the associated request of the result of invoking the initial value of Request as constructor with request as its argument. If this throws an exception, return a promise rejected with that exception.

      -
    -
  4. -

    Let operations be an empty list.

    -
  5. -

    Let operation be a cache batch operation.

    -
  6. -

    Set operation’s type to "delete".

    -
  7. -

    Set operation’s request to r.

    -
  8. -

    Set operation’s options to options.

    -
  9. -

    Append operation to operations.

    -
  10. -

    Let realm be the context object's relevant realm.

    -
  11. -

    Let cacheJobPromise be a new promise.

    -
  12. -

    Run the following substeps in parallel:

    -
      -
    1. -

      Let errorData be null.

      -
    2. -

      Let requestResponses be the result of running Batch Cache Operations with operations. If this throws an exception, set errorData to the exception.

      -
    3. -

      Queue a task, on cacheJobPromise’s relevant settings object's responsible event loop using the DOM manipulation task source, to perform the following substeps:

      -
        -
      1. -

        If errorData is null, then:

        -
          -
        1. -

          If requestResponses is not empty, resolve cacheJobPromise with true.

          -
        2. -

          Else, resolve cacheJobPromise with false.

          -
        -
      2. -

        Else, reject cacheJobPromise with a new exception with errorData and a user agent-defined message, in realm.

        -
      -
    -
  13. -

    Return cacheJobPromise.

    -
-
-
-

5.4.7. keys(request, options)

-

keys(request, options) method must run these steps:

-
    -
  1. -

    Let r be null.

    -
  2. -

    If the optional argument request is not omitted, then:

    -
      -
    1. -

      If request is a Request object, then:

      -
        -
      1. -

        Set r to request’s request.

        -
      2. -

        If r’s method is not `GET` and options.ignoreMethod is false, return a promise resolved with an empty array.

        -
      -
    2. -

      Else if request is a string, then:

      -
        -
      1. -

        Set r to the associated request of the result of invoking the initial value of Request as constructor with request as its argument. If this throws an exception, return a promise rejected with that exception.

        -
      -
    -
  3. -

    Let realm be the context object's relevant realm.

    -
  4. -

    Let promise be a new promise.

    -
  5. -

    Run these substeps in parallel:

    -
      -
    1. -

      Let requests be an empty list.

      -
    2. -

      If the optional argument request is omitted, then:

      -
        -
      1. -

        For each requestResponse of the relevant request response list:

        -
          -
        1. -

          Add requestResponse’s request to requests.

          -
        -
      -
    3. -

      Else:

      -
        -
      1. -

        Let requestResponses be the result of running Query Cache with r and options.

        -
      2. -

        For each requestResponse of requestResponses:

        -
          -
        1. -

          Add requestResponse’s request to requests.

          -
        -
      -
    4. -

      Queue a task, on promise’s relevant settings object's responsible event loop using the DOM manipulation task source, to perform the following steps:

      -
        -
      1. -

        Let requestList be a list.

        -
      2. -

        For each request of requests:

        -
          -
        1. -

          Add a new Request object associated with request and a new associated Headers object whose guard is "immutable" to requestList.

          -
        -
      3. -

        Resolve promise with a frozen array created from requestList, in realm.

        -
      -
    -
  6. -

    Return promise.

    -
-
-
-
-

5.5. CacheStorage

-
[SecureContext, Exposed=(Window,Worker)]
-interface CacheStorage {
-  [NewObject] Promise<any> match(RequestInfo request, optional MultiCacheQueryOptions options);
-  [NewObject] Promise<boolean> has(DOMString cacheName);
-  [NewObject] Promise<Cache> open(DOMString cacheName);
-  [NewObject] Promise<boolean> delete(DOMString cacheName);
-  [NewObject] Promise<sequence<DOMString>> keys();
-};
-
-dictionary MultiCacheQueryOptions : CacheQueryOptions {
-  DOMString cacheName;
-};
-
-

Note: CacheStorage interface is designed to largely conform to ECMAScript 6 Map objects but entirely async, and with additional convenience methods. The methods, clear, forEach, entries and values, are intentionally excluded from the scope of the first version resorting to the ongoing discussion about the async iteration by TC39.

-

The user agent must create a CacheStorage object when a Window object or a WorkerGlobalScope object is created and associate it with that global object.

-

A CacheStorage object represents a name to cache map of its associated global object's environment settings object’s origin. Multiple separate objects implementing the CacheStorage interface across documents and workers can all be associated with the same name to cache map simultaneously.

-
-

5.5.1. match(request, options)

-

match(request, options) method must run these steps:

-
    -
  1. -

    If options.cacheName is present, then:

    -
      -
    1. -

      Return a new promise promise and run the following substeps in parallel:

      -
        -
      1. -

        For each cacheNamecache of the relevant name to cache map:

        -
          -
        1. -

          If options.cacheName matches cacheName, then:

          -
            -
          1. -

            Resolve promise with the result of running the algorithm specified in match(request, options) method of Cache interface with request and options (providing cache as thisArgument to the [[Call]] internal method of match(request, options).)

            -
          2. -

            Abort these steps.

            -
          -
        -
      2. -

        Resolve promise with undefined.

        -
      -
    -
  2. -

    Else:

    -
      -
    1. -

      Let promise be a promise resolved with undefined.

      -
    2. -

      For each cacheNamecache of the relevant name to cache map:

      -
        -
      1. -

        Set promise to the result of transforming itself with a fulfillment handler that, when called with argument response, performs the following substeps:

        -
          -
        1. -

          If response is not undefined, return response.

          -
        2. -

          Return the result of running the algorithm specified in match(request, options) method of Cache interface with request and options as the arguments (providing cache as thisArgument to the [[Call]] internal method of match(request, options).)

          -
        -
      -
    3. -

      Return promise.

      -
    -
-
-
-

5.5.2. has(cacheName)

-

has(cacheName) method must run these steps:

-
    -
  1. -

    Let promise be a new promise.

    -
  2. -

    Run the following substeps in parallel:

    -
      -
    1. -

      For each keyvalue of the relevant name to cache map:

      -
        -
      1. -

        If cacheName matches key, resolve promise with true and abort these steps.

        -
      -
    2. -

      Resolve promise with false.

      -
    -
  3. -

    Return promise.

    -
-
-
-

5.5.3. open(cacheName)

-

open(cacheName) method must run these steps:

-
    -
  1. -

    Let promise be a new promise.

    -
  2. -

    Run the following substeps in parallel:

    -
      -
    1. -

      For each keyvalue of the relevant name to cache map:

      -
        -
      1. -

        If cacheName matches key, then:

        -
          -
        1. -

          Resolve promise with a new Cache object that represents value.

          -
        2. -

          Abort these steps.

          -
        -
      -
    2. -

      Let cache be a new request response list.

      -
    3. -

      Set the relevant name to cache map[cacheName] to cache. If this cache write operation failed due to exceeding the granted quota limit, reject promise with a "QuotaExceededError" DOMException and abort these steps.

      -
    4. -

      Resolve promise with a new Cache object that represents cache.

      -
    -
  3. -

    Return promise.

    -
-
-
-

5.5.4. delete(cacheName)

-

delete(cacheName) method must run these steps:

-
    -
  1. -

    Let promise be the result of running the algorithm specified in has(cacheName) method with cacheName.

    -
  2. -

    Return the result of transforming promise with a fulfillment handler that, when called with argument cacheExists, performs the following substeps:

    -
      -
    1. -

      If cacheExists is false, then:

      -
        -
      1. -

        Return false.

        -
      -
    2. -

      Let cacheJobPromise be a new promise.

      -
    3. -

      Run the following substeps in parallel:

      -
        -
      1. -

        Remove the relevant name to cache map[cacheName].

        -
      2. -

        Resolve cacheJobPromise with true.

        -
      -

      Note: After this step, the existing DOM objects (i.e. the currently referenced Cache, Request, and Response objects) should remain functional.

      -
    4. -

      Return cacheJobPromise.

      -
    -
-
-
-

5.5.5. keys()

-

keys() method must run these steps:

-
    -
  1. -

    Let promise be a new promise.

    -
  2. -

    Run the following substeps in parallel:

    -
      -
    1. -

      Let cacheKeys be the result of getting the keys of the relevant name to cache map.

      -

      Note: The items in the result ordered set are in the order that their corresponding entry was added to the name to cache map.

      -
    2. -

      Resolve promise with cacheKeys.

      -
    -
  3. -

    Return promise.

    -
-
-
-
-
-

6. Security Considerations

-
-

6.1. Secure Context

-

Service workers must execute in secure contexts. Service worker clients must also be secure contexts to register a service worker registration, to get access to the service worker registrations and the service workers, to do messaging with the service workers, and to be manipulated by the service workers.

-

Note: This effectively means that service workers and their service worker clients need to be hosted over HTTPS. A user agent can allow localhost (see the requirements), 127.0.0.0/8, and ::1/128 for development purposes. The primary reason for this restriction is to protect users from the risks associated with insecure contexts.

-
-
-

6.2. Content Security Policy

-

Whenever a user agent invokes Run Service Worker algorithm with a service worker serviceWorker:

-
    -
  • -

    If serviceWorker’s script resource was delivered with a Content-Security-Policy HTTP header containing the value policy, the user agent must enforce policy for serviceWorker.

    -
  • -

    If serviceWorker’s script resource was delivered with a Content-Security-Policy-Report-Only HTTP header containing the value policy, the user agent must monitor policy for serviceWorker.

    -
-

The primary reason for this restriction is to mitigate a broad class of content injection vulnerabilities, such as cross-site scripting (XSS).

-
-
-

6.3. Origin Relativity

-
-

6.3.1. Origin restriction

-

This section is non-normative.

-

A service worker executes in the registering service worker client's origin. One of the advanced concerns that major applications would encounter is whether they can be hosted from a CDN. By definition, these are servers in other places, often on other origins. Therefore, service workers cannot be hosted on CDNs. But they can include resources via importScripts(). The reason for this restriction is that service workers create the opportunity for a bad actor to turn a bad day into a bad eternity.

-
-
-

6.3.2. importScripts(urls)

-

When the importScripts(urls) method is called on a ServiceWorkerGlobalScope object, the user agent must import scripts into worker global scope, given this ServiceWorkerGlobalScope object and urls, and with the following steps to perform the fetch given the request request:

-
    -
  1. -

    Let serviceWorker be request’s client's global object's service worker.

    -
  2. -

    If serviceWorker’s script resource map[request’s url] exists, return the entry's value.

    -
  3. -

    If serviceWorker’s state is not parsed or installing, return a network error.

    -
  4. -

    Let registration be serviceWorker’s containing service worker registration.

    -
  5. -

    Set request’s service-workers mode to "none".

    -
  6. -

    Set request’s cache mode to "no-cache" if any of the following are true:

    - -
  7. -

    Let response be the result of fetching request.

    -
  8. -

    Set response to response’s unsafe response.

    -
  9. -

    If response’s cache state is not "local", set registration’s last update check time to the current time.

    -
  10. -

    Extract a MIME type from the response’s header list. If this MIME type (ignoring parameters) is not a JavaScript MIME type, return a network error.

    -
  11. -

    If response’s type is not "error", and response’s status is an ok status, then:

    -
      -
    1. -

      Set serviceWorker’s script resource map[request’s url] to response.

      -
    2. -

      Set serviceWorker’s classic scripts imported flag.

      -
    -
  12. -

    Return response.

    -
-
-
-
-

6.4. Cross-Origin Resources and CORS

-

This section is non-normative.

-

Applications tend to cache items that come from a CDN or other origin. It is possible to request many of them directly using <script>, <img>, <video> and <link> elements. It would be hugely limiting if this sort of runtime collaboration broke when offline. Similarly, it is possible to fetch many sorts of off-origin resources when appropriate CORS headers are set. Service workers enable this by allowing Caches to fetch and cache off-origin items. Some restrictions apply, however. First, unlike same-origin resources which are managed in the Cache as Response objects whose corresponding responses are basic filtered response, the objects stored are Response objects whose corresponding responses are either CORS filtered responses or opaque filtered responses. They can be passed to event.respondWith(r) method in the same manner as the Response objects whose corresponding responses are basic filtered responses, but cannot be meaningfully created programmatically. These limitations are necessary to preserve the security invariants of the platform. Allowing Caches to store them allows applications to avoid re-architecting in most cases.

-
-
-

6.5. Implementer Concerns

-

This section is non-normative.

-

The implementers are encouraged to note:

-
    -
  • -

    Plug-ins should not load via service workers. As plug-ins may get their security origins from their own urls, the embedding service worker cannot handle it. For this reason, the Handle Fetch algorithm makes the potential-navigation-or-subresource request (whose context is either <embed> or <object>) immediately fallback to the network without dispatching fetch event.

    -
  • -

    Some of the legacy networking stack code may need to be carefully audited to understand the ramifications of interactions with service workers.

    -
-
-
-

6.6. Privacy

-

Service workers introduce new persistent storage features including scope to registration map (for service worker registrations and their service workers), request response list and name to cache map (for caches), and script resource map (for script resources). In order to protect users from any potential unsanctioned tracking threat, these persistent storages should be cleared when users intend to clear them and should maintain and interoperate with existing user controls e.g. purging all existing persistent storages.

-
-
-
-

7. Extensibility

-

Service Workers specification is extensible from other specifications.

-
-

7.1. Define API bound to Service Worker Registration

-

Specifications may define an API tied to a service worker registration by using partial interface definition to the ServiceWorkerRegistration interface where it may define the specification specific attributes and methods:

-
partial interface ServiceWorkerRegistration {
-  // e.g. define an API namespace
-  readonly attribute APISpaceType APISpace;
-  // e.g. define a method
-  Promise<T> methodName(/* list of arguments */);
-};
-
-
-
-

7.2. Define Functional Event

-

Specifications may define a functional event by extending ExtendableEvent interface:

-
// e.g. define FunctionalEvent interface
-interface FunctionalEvent : ExtendableEvent {
-  // add a functional event’s own attributes and methods
-};
-
-
-
-

7.3. Define Event Handler

-

Specifications may define an event handler attribute for the corresponding functional event using partial interface definition to the ServiceWorkerGlobalScope interface:

-
partial interface ServiceWorkerGlobalScope {
-  attribute EventHandler onfunctionalevent;
-};
-
-
-
-

7.4. Firing Functional Events

-

To request a functional event dispatch to the active worker of a service worker registration, specifications should invoke Fire Functional Event.

-
-
-
-

Appendix A: Algorithms

-

The following definitions are the user agent’s internal data structures used throughout the specification.

-

A scope to registration map is an ordered map where the keys are scope urls, serialized, and the values are service worker registrations.

-

A job is an abstraction of one of register, update, and unregister request for a service worker registration.

-
- A job has a job type, which is one of register, update, and unregister. -

A job has a scope url (a URL).

-

A job has a script url (a URL).

-

A job has a worker type ("classic" or "module").

-

A job has an update via cache mode, which is "imports", "all", or "none".

-

A job has a client (a service worker client). It is initially null.

-

A job has a job promise (a promise). It is initially null.

-

A job has a containing job queue (a job queue or null). It is initially null.

-

A job has a list of equivalent jobs (a list of jobs). It is initially the empty list.

-

A job has a force bypass cache flag. It is initially unset.

-
-

Two jobs are equivalent when their job type is the same and:

- -

A job queue is a thread safe queue used to synchronize the set of concurrent jobs. The job queue contains jobs as its items. A job queue is initially empty.

-

A scope to job queue map is an ordered map where the keys are scope urls, serialized, and the values are job queues.

-
-

Create Job

-
-
Input -
-

jobType, a job type

-
-

scopeURL, a URL

-
-

scriptURL, a URL

-
-

promise, a promise

-
-

client, a service worker client

-
Output -
-

job, a job

-
-
    -
  1. -

    Let job be a new job.

    -
  2. -

    Set job’s job type to jobType.

    -
  3. -

    Set job’s scope url to scopeURL.

    -
  4. -

    Set job’s script url to scriptURL.

    -
  5. -

    Set job’s job promise to promise.

    -
  6. -

    Set job’s client to client.

    -
  7. -

    Return job.

    -
-
-
-

Schedule Job

-
-
Input -
-

job, a job

-
Output -
-

none

-
-
    -
  1. -

    Let jobQueue be null.

    -
  2. -

    Let jobScope be job’s scope url, serialized.

    -
  3. -

    If scope to job queue map[jobScope] does not exist, set scope to job queue map[jobScope] to a new job queue.

    -
  4. -

    Set jobQueue to scope to job queue map[jobScope].

    -
  5. -

    If jobQueue is empty, then:

    -
      -
    1. -

      Set job’s containing job queue to jobQueue, and enqueue job to jobQueue.

      -
    2. -

      Invoke Run Job with jobQueue.

      -
    -
  6. -

    Else:

    -
      -
    1. -

      Let lastJob be the element at the back of jobQueue.

      -
    2. -

      If job is equivalent to lastJob and lastJob’s job promise has not settled, append job to lastJob’s list of equivalent jobs.

      -
    3. -

      Else, set job’s containing job queue to jobQueue, and enqueue job to jobQueue.

      -
    -
-
-
-

Run Job

-
-
Input -
-

jobQueue, a job queue

-
Output -
-

none

-
-
    -
  1. -

    Assert: jobQueue is not empty.

    -
  2. -

    Queue a task to run these steps:

    -
      -
    1. -

      Let job be the first item in jobQueue.

      -
    2. -

      If job’s job type is register, run Register with job in parallel.

      -
    3. -

      Else if job’s job type is update, run Update with job in parallel.

      -

      Note: For a register job and an update job, the user agent delays queuing a task for running the job until after a DOMContentLoaded event has been dispatched to the document that initiated the job.

      -
    4. -

      Else if job’s job type is unregister, run Unregister with job in parallel.

      -
    -
-
-
-

Finish Job

-
-
Input -
-

job, a job

-
Output -
-

none

-
-
    -
  1. -

    Let jobQueue be job’s containing job queue.

    -
  2. -

    Assert: the first item in jobQueue is job.

    -
  3. -

    Dequeue from jobQueue.

    -
  4. -

    If jobQueue is not empty, invoke Run Job with jobQueue.

    -
-
-
-

Resolve Job Promise

-
-
Input -
-

job, a job

-
-

value, any

-
Output -
-

none

-
-
    -
  1. -

    Let convertedValue be null.

    -
  2. -

    If job’s client is not null, queue a task, on job’s client's responsible event loop using the DOM manipulation task source, to run the following substeps:

    -
      -
    1. -

      If job’s job type is either register or update, set convertedValue to the ServiceWorkerRegistration object that represents value, in job’s client's Realm.

      -
    2. -

      Else, set convertedValue to value, in job’s client's Realm.

      -
    3. -

      Resolve job’s job promise with convertedValue.

      -
    -
  3. -

    For each equivalentJob in job’s list of equivalent jobs:

    -
      -
    1. -

      If equivalentJob’s client is null, continue to the next iteration of the loop.

      -
    2. -

      Queue a task, on equivalentJob’s client's responsible event loop using the DOM manipulation task source, to run the following substeps:

      -
        -
      1. -

        If equivalentJob’s job type is either register or update, set convertedValue to the ServiceWorkerRegistration object that represents value, in equivalentJob’s client's Realm.

        -
      2. -

        Else, set convertedValue to value, in equivalentJob’s client's Realm.

        -
      3. -

        Resolve equivalentJob’s job promise with convertedValue.

        -
      -
    -
-
-
-

Reject Job Promise

-
-
Input -
-

job, a job

-
-

errorData, the information necessary to create an exception

-
Output -
-

none

-
-
    -
  1. -

    If job’s client is not null, queue a task, on job’s client's responsible event loop using the DOM manipulation task source, to reject job’s job promise with a new exception with errorData and a user agent-defined message, in job’s client's Realm.

    -
  2. -

    For each equivalentJob in job’s list of equivalent jobs:

    -
      -
    1. -

      If equivalentJob’s client is null, continue.

      -
    2. -

      Queue a task, on equivalentJob’s client's responsible event loop using the DOM manipulation task source, to reject equivalentJob’s job promise with a new exception with errorData and a user agent-defined message, in equivalentJob’s client's Realm.

      -
    -
-
-
-

Register

-
-
Input -
-

job, a job

-
Output -
-

none

-
-
    -
  1. -

    If the result of running potentially trustworthy origin with the origin of job’s script url as the argument is Not Trusted, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and "SecurityError" DOMException.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  2. -

    If the origin of job’s script url is not job’s client's origin, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and "SecurityError" DOMException.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  3. -

    If the origin of job’s scope url is not job’s client's origin, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and "SecurityError" DOMException.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  4. -

    Let registration be the result of running the Get Registration algorithm passing job’s scope url as the argument.

    -
  5. -

    If registration is not null, then:

    -
      -
    1. -

      If registration’s uninstalling flag is set, unset it.

      -
    2. -

      Let newestWorker be the result of running the Get Newest Worker algorithm passing registration as the argument.

      -
    3. -

      If newestWorker is not null, job’s script url equals newestWorker’s script url, and job’s update via cache mode's value equals registration’s update via cache mode's value, then:

      -
        -
      1. -

        Invoke Resolve Job Promise with job and registration.

        -
      2. -

        Invoke Finish Job with job and abort these steps.

        -
      -
    -
  6. -

    Else:

    -
      -
    1. -

      Invoke Set Registration algorithm with job’s scope url and job’s update via cache mode.

      -
    -
  7. -

    Invoke Update algorithm passing job as the argument.

    -
-
-
-

Update

-
-
Input -
-

job, a job

-
Output -
-

none

-
-
    -
  1. -

    Let registration be the result of running the Get Registration algorithm passing job’s scope url as the argument.

    -
  2. -

    If registration is null or registration’s uninstalling flag is set, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and TypeError.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  3. -

    Let newestWorker be the result of running Get Newest Worker algorithm passing registration as the argument.

    -
  4. -

    If job’s job type is update, and newestWorker’s script url does not equal job’s script url, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and TypeError.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  5. -

    Let httpsState be "none".

    -
  6. -

    Let referrerPolicy be the empty string.

    -
  7. -

    Let hasUpdatedResources be false.

    -
  8. -

    Let updatedResourceMap be an ordered map where the keys are URLs and the values are responses.

    -
  9. -

    Switching on job’s worker type, run these substeps with the following options:

    -
    -
    "classic" -
    -

    Fetch a classic worker script given job’s serialized script url, job’s client, "serviceworker", and the to-be-created environment settings object for this service worker.

    -
    "module" -
    -

    Fetch a module worker script graph given job’s serialized script url, job’s client, "serviceworker", "omit", and the to-be-created environment settings object for this service worker.

    -
    -

    To perform the fetch given request, run the following steps:

    -
      -
    1. -

      Append `Service-Worker`/`script` to request’s header list.

      -

      Note: See the definition of the Service-Worker header in Appendix B: Extended HTTP headers.

      -
    2. -

      Set request’s cache mode to "no-cache" if any of the following are true:

      - -

      Note: Even if the cache mode is not set to "no-cache", the user agent obeys Cache-Control header’s max-age value in the network layer to determine if it should bypass the browser cache.

      -
    3. -

      Set request’s service-workers mode to "none".

      -
    4. -

      If the is top-level flag is unset, then return the result of fetching request.

      -
    5. -

      Set request’s redirect mode to "error".

      -
    6. -

      Fetch request, and asynchronously wait to run the remaining steps as part of fetch’s process response for the response response.

      -
    7. -

      Extract a MIME type from the response’s header list. If this MIME type (ignoring parameters) is not a JavaScript MIME type, then:

      -
        -
      1. -

        Invoke Reject Job Promise with job and "SecurityError" DOMException.

        -
      2. -

        Asynchronously complete these steps with a network error.

        -
      -
    8. -

      Let serviceWorkerAllowed be the result of extracting header list values given `Service-Worker-Allowed` and response’s header list.

      -

      Note: See the definition of the Service-Worker-Allowed header in Appendix B: Extended HTTP headers.

      -
    9. -

      Set httpsState to response’s HTTPS state.

      -
    10. -

      Set referrerPolicy to the result of parse a referrer policy from a Referrer-Policy header of response.

      -
    11. -

      If serviceWorkerAllowed is failure, then:

      -
        -
      1. -

        Asynchronously complete these steps with a network error.

        -
      -
    12. -

      Let scopeURL be registration’s scope url.

      -
    13. -

      Let maxScopeString be null.

      -
    14. -

      If serviceWorkerAllowed is null, then:

      -
        -
      1. -

        Set maxScopeString to "/" concatenated with the strings, except the last string that denotes the script’s file name, in job’s script url's path (including empty strings), separated from each other by "/".

        -
      -
    15. -

      Else:

      -
        -
      1. -

        Let maxScope be the result of parsing serviceWorkerAllowed with job’s script url.

        -
      2. -

        Set maxScopeString to "/" concatenated with the strings in maxScope’s path (including empty strings), separated from each other by "/".

        -
      -
    16. -

      Let scopeString be "/" concatenated with the strings in scopeURL’s path (including empty strings), separated from each other by "/".

      -
    17. -

      If scopeString starts with maxScopeString, do nothing.

      -
    18. -

      Else:

      -
        -
      1. -

        Invoke Reject Job Promise with job and "SecurityError" DOMException.

        -
      2. -

        Asynchronously complete these steps with a network error.

        -
      -
    19. -

      Set updatedResourceMap[request’s url] to response.

      -
    20. -

      If response’s cache state is not "local", set registration’s last update check time to the current time.

      -
    21. -

      If newestWorker is null, or newestWorker’s script resource map[request’s url]'s body is not byte-for-byte identical with response’s body, set hasUpdatedResources to true.

      -
    22. -

      Else if newestWorker’s classic scripts imported flag is set, then:

      -
        -
      1. -

        For each urlstoredResponse of newestWorker’s script resource map:

        -
          -
        1. -

          Let request be a new request whose url is url, client is job’s client, destination is "script", parser metadata is "not parser-inserted", synchronous flag is set, and whose use-URL-credentials flag is set.

          -
        2. -

          Set request’s cache mode to "no-cache" if any of the following are true:

          - -
        3. -

          Let fetchedResponse be the result of fetching request.

          -
        4. -

          Set fetchedResponse to fetchedResponse’s unsafe response.

          -
        5. -

          Set updatedResourceMap[request’s url] to fetchedResponse.

          -
        6. -

          If fetchedResponse’s cache state is not "local", set registration’s last update check time to the current time.

          -
        7. -

          Extract a MIME type from the fetchedResponse’s header list. If this MIME type (ignoring parameters) is not a JavaScript MIME type, asynchronously complete these steps with a network error.

          -
        8. -

          If fetchedResponse’s type is "error", or fetchedResponse’s status is not an ok status, asynchronously complete these steps with a network error.

          -
        9. -

          If fetchedResponse’s body is not byte-for-byte identical with storedResponse’s body, set hasUpdatedResources to true.

          -

          Note: The control does not break the loop in this step to continue with all the imported scripts to populate the cache.

          -
        -
      -
    23. -

      Return true.

      -
    -

    If the algorithm asynchronously completes with null, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and TypeError.

      -

      Note: This will do nothing if Reject Job Promise was previously invoked with "SecurityError" DOMException.

      -
    2. -

      If newestWorker is null, invoke Clear Registration algorithm passing registration as its argument.

      -
    3. -

      Invoke Finish Job with job and abort these steps.

      -
    -

    Else, continue the rest of these steps after the algorithm’s asynchronous completion, with script being the asynchronous completion value.

    -
  10. -

    If hasUpdatedResources is false, then:

    -
      -
    1. -

      Invoke Resolve Job Promise with job and registration.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  11. -

    Let worker be a new service worker.

    -
  12. -

    Set worker’s script url to job’s script url, worker’s script resource to script, and worker’s type to job’s worker type.

    -
  13. -

    For each urlresponse of updatedResourceMap:

    -
      -
    1. -

      Set worker’s script resource map[url] to response.

      -
    -
  14. -

    Set worker’s script resource’s HTTPS state to httpsState.

    -
  15. -

    Set worker’s script resource’s referrer policy to referrerPolicy.

    -
  16. -

    Invoke Run Service Worker algorithm given worker, and with the force bypass cache for importscripts flag set if job’s force bypass cache flag is set.

    -
  17. -

    If an uncaught runtime script error occurs during the above step, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and TypeError.

      -
    2. -

      If newestWorker is null, invoke Clear Registration algorithm passing registration as its argument.

      -
    3. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  18. -

    Invoke Install algorithm with job, worker, and registration as its arguments.

    -
-
-
-

Soft Update

-

The user agent may call this as often as it likes to check for updates.

-
-
Input -
-

registration, a service worker registration

-
-

force bypass cache flag, an optional flag unset by default

-

Note: Implementers may use the force bypass cache flag to aid debugging (e.g. invocations from developer tools), and other specifications that extend service workers may also use the flag on their own needs.

-
Output -
-

None

-
-
    -
  1. -

    Let newestWorker be the result of running Get Newest Worker algorithm passing registration as its argument.

    -
  2. -

    If newestWorker is null, abort these steps.

    -
  3. -

    Let job be the result of running Create Job with update, registration’s scope url, newestWorker’s script url, null, and null.

    -
  4. -

    Set job’s worker type to newestWorker’s type.

    -
  5. -

    Set job’s force bypass cache flag if the force bypass cache flag is set.

    -
  6. -

    Invoke Schedule Job with job.

    -
-
-
-

Install

-
-
Input -
-

job, a job

-
-

worker, a service worker

-
-

registration, a service worker registration

-
Output -
-

none

-
-
    -
  1. -

    Let installFailed be false.

    -
  2. -

    Let newestWorker be the result of running Get Newest Worker algorithm passing registration as its argument.

    -
  3. -

    Run the Update Registration State algorithm passing registration, "installing" and worker as the arguments.

    -
  4. -

    Run the Update Worker State algorithm passing registration’s installing worker and installing as the arguments.

    -
  5. -

    Assert: job’s job promise is not null.

    -
  6. -

    Invoke Resolve Job Promise with job and registration.

    -
  7. -

    Queue a task to fire an event named updatefound at all the ServiceWorkerRegistration objects for all the service worker clients whose creation URL matches registration’s scope url and all the service workers whose containing service worker registration is registration.

    -
  8. -

    Let installingWorker be registration’s installing worker.

    -
  9. -

    Invoke Run Service Worker algorithm given installingWorker, and with the force bypass cache for importscripts flag set if job’s force bypass cache flag is set.

    -
  10. -

    Queue a task task to run the following substeps:

    -
      -
    1. -

      Let e be the result of creating an event with ExtendableEvent.

      -
    2. -

      Initialize e’s type attribute to install.

      -
    3. -

      Dispatch e at installingWorker’s global object.

      -
    4. -

      WaitForAsynchronousExtensions: Run the following substeps in parallel:

      -
        -
      1. -

        Wait until e is not active.

        -
      2. -

        If e’s timed out flag is set, or the result of waiting for all of e’s extend lifetime promises rejected, set installFailed to true.

        -
      -
    -

    If task is discarded or the script has been aborted by the termination of installingWorker, set installFailed to true.

    -
  11. -

    Wait for task to have executed or been discarded.

    -
  12. -

    Wait for the step labeled WaitForAsynchronousExtensions to complete.

    -
  13. -

    If installFailed is true, then:

    -
      -
    1. -

      Run the Update Worker State algorithm passing registration’s installing worker and redundant as the arguments.

      -
    2. -

      Run the Update Registration State algorithm passing registration, "installing" and null as the arguments.

      -
    3. -

      If newestWorker is null, invoke Clear Registration algorithm passing registration as its argument.

      -
    4. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  14. -

    If registration’s waiting worker is not null, then:

    -
      -
    1. -

      Terminate registration’s waiting worker.

      -
    2. -

      Run the Update Worker State algorithm passing registration’s waiting worker and redundant as the arguments.

      -
    -
  15. -

    Run the Update Registration State algorithm passing registration, "waiting" and registration’s installing worker as the arguments.

    -
  16. -

    Run the Update Registration State algorithm passing registration, "installing" and null as the arguments.

    -
  17. -

    Run the Update Worker State algorithm passing registration’s waiting worker and installed as the arguments.

    -
  18. -

    Invoke Finish Job with job.

    -
  19. -

    Wait for all the tasks queued by Update Worker State invoked in this algorithm to have executed.

    -
  20. -

    Invoke Try Activate with registration.

    -

    Note: If Try Activate does not trigger Activate here, Activate is tried again when the last client controlled by the existing active worker is unloaded, skipWaiting() is asynchronously called, or the extend lifetime promises for the existing active worker settle.

    -
-
-
-

Activate

-
-
Input -
-

registration, a service worker registration

-
Output -
-

None

-
-
    -
  1. -

    If registration’s waiting worker is null, abort these steps.

    -
  2. -

    If registration’s active worker is not null, then:

    -
      -
    1. -

      Terminate registration’s active worker.

      -
    2. -

      Run the Update Worker State algorithm passing registration’s active worker and redundant as the arguments.

      -
    -
  3. -

    Run the Update Registration State algorithm passing registration, "active" and registration’s waiting worker as the arguments.

    -
  4. -

    Run the Update Registration State algorithm passing registration, "waiting" and null as the arguments.

    -
  5. -

    Run the Update Worker State algorithm passing registration’s active worker and activating as the arguments.

    -

    Note: Once an active worker is activating, neither a runtime script error nor a force termination of the active worker prevents the active worker from getting activated.

    -
  6. -

    Let matchedClients be a list of service worker clients whose creation URL matches registration’s scope url.

    -
  7. -

    For each client of matchedClients, queue a task on client’s responsible event loop, using the DOM manipulation task source, to run the following substeps:

    -
      -
    1. -

      Let readyPromise be client’s global object's ServiceWorkerContainer object’s ready promise.

      -
    2. -

      If readyPromise is pending, resolve readyPromise with the ServiceWorkerRegistration object that represents registration in readyPromise’s relevant Realm.

      -
    -
  8. -

    For each client of matchedClients:

    -
      -
    1. -

      If client is a window client, unassociate client’s responsible document from its application cache, if it has one.

      -
    2. -

      Else if client is a shared worker client, unassociate client’s global object from its application cache, if it has one.

      -
    -

    Note: Resources will now use the service worker registration instead of the existing application cache.

    -
  9. -

    For each service worker client client who is using registration:

    -
      -
    1. -

      Set client’s active worker to registration’s active worker.

      -
    2. -

      Invoke Notify Controller Change algorithm with client as the argument.

      -
    -
  10. -

    Let activeWorker be registration’s active worker.

    -
  11. -

    Invoke Run Service Worker algorithm with activeWorker as the argument.

    -
  12. -

    Queue a task task to run the following substeps:

    -
      -
    1. -

      Let e be the result of creating an event with ExtendableEvent.

      -
    2. -

      Initialize e’s type attribute to activate.

      -
    3. -

      Dispatch e at activeWorker’s global object.

      -
    4. -

      WaitForAsynchronousExtensions: Wait, in parallel, until e is not active.

      -
    -
  13. -

    Wait for task to have executed or been discarded, or the script to have been aborted by the termination of activeWorker.

    -
  14. -

    Wait for the step labeled WaitForAsynchronousExtensions to complete.

    -
  15. -

    Run the Update Worker State algorithm passing registration’s active worker and activated as the arguments.

    -
-
-
-

Try Activate

-
-
Input -
-

registration, a service worker registration

-
Output -
-

None

-
-
    -
  1. -

    If registration’s waiting worker is null, return.

    -
  2. -

    If registration’s active worker is not null and registration’s active worker's state is activating, return.

    -

    Note: If the existing active worker is still in activating state, the activation of the waiting worker is delayed.

    -
  3. -

    Invoke Activate with registration if either of the following is true:

    - -
-
-
-

Run Service Worker

-
-
Input -
-

serviceWorker, a service worker

-
-

force bypass cache for importscripts flag, an optional flag unset by default

-
Output -
-

None

-
-
    -
  1. -

    Let script be serviceWorker’s script resource.

    -
  2. -

    Assert: script is not null.

    -
  3. -

    If serviceWorker is already running, abort these steps.

    -
  4. -

    Create a separate parallel execution environment (i.e. a separate thread or process or equivalent construct), and run the rest of these steps in that context.

    -
  5. -

    Call the JavaScript InitializeHostDefinedRealm() abstract operation with the following customizations:

    - -
  6. -

    Set serviceWorker’s global object to workerGlobalScope.

    -
  7. -

    Let workerEventLoop be a newly created event loop.

    -
  8. -

    Let settingsObject be a new environment settings object whose algorithms are defined as follows:

    -
    -
    The realm execution context -
    -

    Return realmExecutionContext.

    -
    The global object -
    -

    Return workerGlobalScope.

    -
    The responsible event loop -
    -

    Return workerEventLoop.

    -
    The referrer policy -
    -

    Return workerGlobalScope’s referrer policy.

    -
    The API URL character encoding -
    -

    Return UTF-8.

    -
    The API base URL -
    -

    Return serviceWorker’s script url.

    -
    The origin -
    -

    Return its registering service worker client's origin.

    -
    The creation URL -
    -

    Return workerGlobalScope’s url.

    -
    The HTTPS state -
    -

    Return workerGlobalScope’s HTTPS state.

    -
    -
  9. -

    Set workerGlobalScope’s url to serviceWorker’s script url.

    -
  10. -

    Set workerGlobalScope’s HTTPS state to serviceWorker’s script resource’s HTTPS state.

    -
  11. -

    Set workerGlobalScope’s referrer policy to serviceWorker’s script resource’s referrer policy.

    -
  12. -

    Set workerGlobalScope’s type to serviceWorker’s type.

    -
  13. -

    Set workerGlobalScope’s force bypass cache for importscripts flag if the force bypass cache for importscripts flag is set.

    -
  14. -

    Create a new WorkerLocation object and associate it with workerGlobalScope.

    -
  15. -

    If serviceWorker is an active worker, and there are any tasks queued in serviceWorker’s containing service worker registration’s task queues, queue them to serviceWorker’s event loop’s task queues in the same order using their original task sources.

    -
  16. -

    If script is a classic script, then run the classic script script. Otherwise, it is a module script; run the module script script.

    -

    Note: In addition to the usual possibilities of returning a value or failing due to an exception, this could be prematurely aborted by the terminate service worker algorithm.

    -
  17. -

    If script’s has ever been evaluated flag is unset, then:

    -
      -
    1. -

      For each eventType of settingsObject’s global object's associated list of event listeners' event types:

      -
        -
      1. -

        Append eventType to workerGlobalScope’s associated service worker's set of event types to handle.

        -
      -
    -

    Note: If the global object’s associated list of event listeners does not have any event listener added at this moment, the service worker’s set of event types to handle remains an empty set. The user agents are encouraged to show a warning that the event listeners must be added on the very first evaluation of the worker script.

    -
      -
    1. -

      Set script’s has ever been evaluated flag.

      -
    -
  18. -

    Run the responsible event loop specified by settingsObject until it is destroyed.

    -
  19. -

    Empty workerGlobalScope’s list of active timers.

    -
-
-
-

Terminate Service Worker

-
-
Input -
-

serviceWorker, a service worker

-
Output -
-

None

-
-
    -
  1. -

    If serviceWorker is not running, abort these steps.

    -
  2. -

    Let serviceWorkerGlobalScope be serviceWorker’s global object.

    -
  3. -

    Set serviceWorkerGlobalScope’s closing flag to true.

    -
  4. -

    Remove all the items from serviceWorker’s set of extended events.

    -
  5. -

    If there are any tasks, whose task source is either the handle fetch task source or the handle functional event task source, queued in serviceWorkerGlobalScope’s event loop’s task queues, queue them to serviceWorker’s containing service worker registration’s corresponding task queues in the same order using their original task sources, and discard all the tasks (including tasks whose task source is neither the handle fetch task source nor the handle functional event task source) from serviceWorkerGlobalScope’s event loop’s task queues without processing them.

    -

    Note: This effectively means that the fetch events and the other functional events such as push events are backed up by the registration’s task queues while the other tasks including message events are discarded.

    -
  6. -

    Abort the script currently running in serviceWorker.

    -
-
-
-

Handle Fetch

-

The Handle Fetch algorithm is the entry point for the fetch handling handed to the service worker context.

-
-
Input -
-

request, a request

-
Output -
-

response, a response

-
-
    -
  1. -

    Let handleFetchFailed be false.

    -
  2. -

    Let respondWithEntered be false.

    -
  3. -

    Let eventCanceled be false.

    -
  4. -

    Let response be null.

    -
  5. -

    Let registration be null.

    -
  6. -

    Let client be request’s client.

    -
  7. -

    Let reservedClient be request’s reserved client.

    -
  8. -

    Assert: request’s destination is not "serviceworker".

    -
  9. -

    If request is a potential-navigation-or-subresource request, then:

    -
      -
    1. -

      Return null.

      -
    -
  10. -

    Else if request is a non-subresource request, then:

    -

    Note: If the non-subresource request is under the scope of a service worker registration, application cache is completely bypassed regardless of whether the non-subresource request uses the service worker registration.

    -
      -
    1. -

      If reservedClient is not null and is an environment settings object, then:

      -
        -
      1. -

        If reservedClient is not a secure context, return null.

        -
      -
    2. -

      Else:

      -
        -
      1. -

        If request’s url is not a potentially trustworthy URL, return null.

        -
      -
    3. -

      If request is a navigation request and the navigation triggering it was initiated with a shift+reload or equivalent, return null.

      -
    4. -

      Set registration to the result of running Match Service Worker Registration algorithm passing request’s url as the argument.

      -
    5. -

      If registration is null or registration’s active worker is null, return null.

      -
    6. -

      If request’s destination is not "report", set reservedClient’s active service worker to registration’s active worker.

      -
    -

    Note: From this point, the service worker client starts to use its active service worker’s containing service worker registration.

    -
  11. -

    Else if request is a subresource request, then:

    -
      -
    1. -

      If client’s active service worker is non-null, set registration to client’s active service worker’s containing service worker registration.

      -
    2. -

      Else, return null.

      -
    -
  12. -

    Let activeWorker be registration’s active worker.

    -
  13. -

    If activeWorker’s set of event types to handle does not contain fetch, then:

    -
      -
    1. -

      Return null and continue running these steps in parallel.

      -
    2. -

      If request is a non-subresource request, or request is a subresource request and the time difference in seconds calculated by the current time minus registration’s last update check time is greater than 86400, invoke Soft Update algorithm with registration.

      -
    3. -

      Abort these steps.

      -
    -

    Note: To avoid unnecessary delays, the Handle Fetch enforces early return when no event listeners have been deterministically added in the service worker’s global during the very first script execution.

    -
  14. -

    If activeWorker’s state is activating, wait for activeWorker’s state to become activated.

    -
  15. -

    Invoke Run Service Worker algorithm with activeWorker as the argument.

    -
  16. -

    Queue a task task to run the following substeps:

    -
      -
    1. -

      Let e be the result of creating an event with FetchEvent.

      -
    2. -

      Initialize e’s type attribute to fetch.

      -
    3. -

      Initialize e’s cancelable attribute to true.

      -
    4. -

      Initialize e’s request attribute to a new Request object associated with request and a new associated Headers object whose guard is "immutable".

      -
    5. -

      If request is a non-subresource request and request’s destination is not "report", initialize e’s clientId attribute to the empty string, and to client’s id otherwise.

      -
    6. -

      Dispatch e at activeWorker’s global object.

      -
    7. -

      Invoke Update Service Worker Extended Events Set with activeWorker and e.

      -
    8. -

      If e’s respond-with entered flag is set, set respondWithEntered to true.

      -
    9. -

      If e’s wait to respond flag is set, then:

      -
        -
      1. -

        Wait until e’s wait to respond flag is unset.

        -
      2. -

        If e’s respond-with error flag is set, set handleFetchFailed to true.

        -
      3. -

        Else, set response to e’s potential response.

        -
      -
    10. -

      If e’s canceled flag is set, set eventCanceled to true.

      -
    -

    If task is discarded or the script has been aborted by the termination of activeWorker, set handleFetchFailed to true.

    -

    The task must use activeWorker’s event loop and the handle fetch task source.

    -
  17. -

    Wait for task to have executed or been discarded.

    -
  18. -

    If respondWithEntered is false, then:

    -
      -
    1. -

      If eventCanceled is true, return a network error and continue running these steps in parallel.

      -
    2. -

      Else, return null and continue running these steps in parallel.

      -
    3. -

      If request is a non-subresource request, or request is a subresource request and the time difference in seconds calculated by the current time minus registration’s last update check time is greater than 86400, invoke Soft Update algorithm with registration.

      -
    4. -

      Abort these steps.

      -
    -
  19. -

    If handleFetchFailed is true, then:

    -
      -
    1. -

      Return a network error and continue running these steps in parallel.

      -
    2. -

      If request is a non-subresource request, or request is a subresource request and the time difference in seconds calculated by the current time minus registration’s last update check time is greater than 86400, invoke Soft Update algorithm with registration.

      -
    -
  20. -

    Else:

    -
      -
    1. -

      Return response and continue running these steps in parallel.

      -
    2. -

      If request is a non-subresource request, or request is a subresource request and the time difference in seconds calculated by the current time minus registration’s last update check time is greater than 86400, invoke Soft Update algorithm with registration.

      -
    -
-
-
-

Fire Functional Event

-
-
Input -
-

eventName, a string

-
-

eventConstructor, an event constructor that extends ExtendableEvent

-
-

registration, a service worker registration

-
-

initialization, optional property initialization for event, constructed from eventConstructor

-
-

postDispatchSteps, optional steps to run on the active worker's event loop, with dispatchedEvent set to the instance of eventConstructor that was dispatched.

-
Output -
-

None

-
-
    -
  1. -

    Assert: scope to registration map contains a value equal to registration.

    -
  2. -

    Assert: registration’s active worker is not null.

    -
  3. -

    Let activeWorker be registration’s active worker.

    -
  4. -

    If activeWorker’s set of event types to handle does not contain eventName, then return and run the following steps in parallel:

    -
      -
    1. -

      If the time difference in seconds calculated by the current time minus registration’s last update check time is greater than 86400, invoke Soft Update algorithm with registration.

      -
    2. -

      Abort these steps.

      -
    -

    Note: To avoid unnecessary delays, the Handle Functional Event enforces early return when no event listeners have been deterministically added in the service worker’s global during the very first script execution.

    -
  5. -

    If activeWorker’s state is activating, wait for activeWorker’s state to become activated.

    -
  6. -

    Invoke Run Service Worker algorithm with activeWorker as the argument.

    -
  7. -

    Queue a task task to run these substeps:

    -
      -
    1. -

      Let event be the result of creating an event with eventConstructor and the relevant realm of activeWorker’s global object.

      -
    2. -

      If initialization is not null, then initialize event with initialization.

      -
    3. -

      Dispatch event on activeWorker’s global object.

      -
    4. -

      Invoke Update Service Worker Extended Events Set with activeWorker and event.

      -
    5. -

      If postDispatchSteps is not null, then run postDispatchSteps passing event as dispatchedEvent.

      -
    -

    The task must use activeWorker’s event loop and the handle functional event task source.

    -
  8. -

    Wait for task to have executed or been discarded.

    -
  9. -

    If the time difference in seconds calculated by the current time minus registration’s last update check time is greater than 86400, invoke Soft Update algorithm with registration.

    -
-
- To fire an "amazingthing" event (which is of type AmazingThingEvent) on a particular serviceWorkerRegistration, and initialize the event object’s properties, the prose would be: -
    -
  1. -

    Fire Functional Event "amazingthing" using AmazingThingEvent on serviceWorkerRegistration with the following properties:

    -
    -
    propertyName -
    -

    value

    -
    anotherPropertyName -
    -

    anotherValue

    -
    -

    Then run the following steps with dispatchedEvent:

    -
      -
    1. -

      Do whatever you need to with dispatchedEvent on the service worker’s event loop.

      -
    -
-

Note that the initialization steps and post-dispatch steps are optional. If they aren’t needed, the prose would be:

-
    -
  1. -

    Fire Functional Event "whatever" using ExtendableEvent on serviceWorkerRegistration.

    -
-
-
-
-

Handle Service Worker Client Unload

-

The user agent must run these steps when a service worker client unloads by unloading or terminating.

-
-
Input -
-

client, a service worker client

-
Output -
-

None

-
-
    -
  1. -

    Run the following steps atomically.

    -
  2. -

    Let registration be the service worker registration used by client.

    -
  3. -

    If registration is null, abort these steps.

    -
  4. -

    If any other service worker client is using registration, abort these steps.

    -
  5. -

    If registration’s uninstalling flag is set, invoke Try Clear Registration with registration.

    -
  6. -

    If registration is not null, invoke Try Activate with registration.

    -
-
-
-

Handle User Agent Shutdown

-
-
Input -
-

None

-
Output -
-

None

-
-
    -
  1. -

    For each scoperegistration of scope to registration map:

    -
      -
    1. -

      If registration’s installing worker installingWorker is not null, then:

      -
        -
      1. -

        If registration’s waiting worker is null and registration’s active worker is null, invoke Clear Registration with registration and continue to the next iteration of the loop.

        -
      2. -

        Else, set installingWorker to null.

        -
      -
    2. -

      If registration’s waiting worker is not null, run the following substep in parallel:

      -
        -
      1. -

        Invoke Activate with registration.

        -
      -
    -
-
-
-

Update Service Worker Extended Events Set

-
-
Input -
-

worker, a service worker

-
-

event, an event

-
Output -
-

None

-
-
    -
  1. -

    Assert: event’s dispatch flag is unset.

    -
  2. -

    For each item of worker’s set of extended events:

    -
      -
    1. -

      If item is not active, remove item from worker’s set of extended events.

      -
    -
  3. -

    If event is active, append event to worker’s set of extended events.

    -
-
-
-

Unregister

-
-
Input -
-

job, a job

-
Output -
-

none

-
-
    -
  1. -

    If the origin of job’s scope url is not job’s client's origin, then:

    -
      -
    1. -

      Invoke Reject Job Promise with job and "SecurityError" DOMException.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  2. -

    Let registration be the result of running Get Registration algorithm passing job’s scope url as the argument.

    -
  3. -

    If registration is null, then:

    -
      -
    1. -

      Invoke Resolve Job Promise with job and false.

      -
    2. -

      Invoke Finish Job with job and abort these steps.

      -
    -
  4. -

    Set registration’s uninstalling flag.

    -
  5. -

    Invoke Resolve Job Promise with job and true.

    -
  6. -

    Invoke Try Clear Registration with registration.

    -

    Note: If Try Clear Registration does not trigger Clear Registration here, Clear Registration is tried again when the last client using the registration is unloaded or the extend lifetime promises for the registration’s service workers settle.

    -
  7. -

    Invoke Finish Job with job.

    -
-
-
-

Set Registration

-
-
Input -
-

scope, a URL

-
-

updateViaCache, an update via cache mode

-
Output -
-

registration, a service worker registration

-
-
    -
  1. -

    Run the following steps atomically.

    -
  2. -

    Let scopeString be serialized scope with the exclude fragment flag set.

    -
  3. -

    Let registration be a new service worker registration whose scope url is set to scope and update via cache mode is set to updateViaCache.

    -
  4. -

    Set scope to registration map[scopeString] to registration.

    -
  5. -

    Return registration.

    -
-
-
-

Clear Registration

-
-
Input -
-

registration, a service worker registration

-
Output -
-

None

-
-
    -
  1. -

    Run the following steps atomically.

    -
  2. -

    If registration’s installing worker is not null, then:

    -
      -
    1. -

      Terminate registration’s installing worker.

      -
    2. -

      Run the Update Worker State algorithm passing registration’s installing worker and redundant as the arguments.

      -
    3. -

      Run the Update Registration State algorithm passing registration, "installing" and null as the arguments.

      -
    -
  3. -

    If registration’s waiting worker is not null, then:

    -
      -
    1. -

      Terminate registration’s waiting worker.

      -
    2. -

      Run the Update Worker State algorithm passing registration’s waiting worker and redundant as the arguments.

      -
    3. -

      Run the Update Registration State algorithm passing registration, "waiting" and null as the arguments.

      -
    -
  4. -

    If registration’s active worker is not null, then:

    -
      -
    1. -

      Terminate registration’s active worker.

      -
    2. -

      Run the Update Worker State algorithm passing registration’s active worker and redundant as the arguments.

      -
    3. -

      Run the Update Registration State algorithm passing registration, "active" and null as the arguments.

      -
    -
  5. -

    Let scopeString be registration’s serialized scope url.

    -
  6. -

    Remove scope to registration map[scopeString].

    -
-
-
-

Try Clear Registration

-
-
Input -
-

registration, a service worker registration

-
Output -
-

None

-
-
    -
  1. -

    Invoke Clear Registration with registration if no service worker client is using registration and all of the following conditions are true:

    - -
-
-
-

Update Registration State

-
-
Input -
-

registration, a service worker registration

-
-

target, a string (one of "installing", "waiting", and "active")

-
-

source, a service worker or null

-
Output -
-

None

-
-
    -
  1. -

    Let registrationObjects be an array containing all the ServiceWorkerRegistration objects associated with registration.

    -
  2. -

    If target is "installing", then:

    -
      -
    1. -

      Set registration’s installing worker to source.

      -
    2. -

      For each registrationObject in registrationObjects:

      -
        -
      1. -

        Queue a task to set the installing attribute of registrationObject to the ServiceWorker object that represents registration’s installing worker, or null if registration’s installing worker is null.

        -
      -
    -
  3. -

    Else if target is "waiting", then:

    -
      -
    1. -

      Set registration’s waiting worker to source.

      -
    2. -

      For each registrationObject in registrationObjects:

      -
        -
      1. -

        Queue a task to set the waiting attribute of registrationObject to the ServiceWorker object that represents registration’s waiting worker, or null if registration’s waiting worker is null.

        -
      -
    -
  4. -

    Else if target is "active", then:

    -
      -
    1. -

      Set registration’s active worker to source.

      -
    2. -

      For each registrationObject in registrationObjects:

      -
        -
      1. -

        Queue a task to set the active attribute of registrationObject to the ServiceWorker object that represents registration’s active worker, or null if registration’s active worker is null.

        -
      -
    -

    The task must use registrationObject’s relevant settings object’s responsible event loop and the DOM manipulation task source.

    -
-
-
-

Update Worker State

-
-
Input -
-

worker, a service worker

-
-

state, a service worker's state

-
Output -
-

None

-
-
    -
  1. -

    Set worker’s state to state.

    -
  2. -

    Let workerObjects be an array containing all the ServiceWorker objects associated with worker.

    -
  3. -

    For each workerObject in workerObjects:

    -
      -
    1. -

      Queue a task to run these substeps:

      -
        -
      1. -

        Set the state attribute of workerObject to the value (in ServiceWorkerState enumeration) corresponding to the first matching statement, switching on worker’s state:

        -
        -
        installing -
        -

        "installing"

        -

        Note: The service worker in this state is considered an installing worker. During this state, waitUntil() can be called inside the oninstall event handler to extend the life of the installing worker until the passed promise resolves successfully. This is primarily used to ensure that the service worker is not active until all of the core caches are populated.

        -
        installed -
        -

        "installed"

        -

        Note: The service worker in this state is considered a waiting worker.

        -
        activating -
        -

        "activating"

        -

        Note: The service worker in this state is considered an active worker. During this state, waitUntil() can be called inside the onactivate event handler to extend the life of the active worker until the passed promise resolves successfully. No functional events are dispatched until the state becomes activated.

        -
        activated -
        -

        "activated"

        -

        Note: The service worker in this state is considered an active worker ready to handle functional events.

        -
        redundant -
        -

        "redundant"

        -

        Note: A new service worker is replacing the current service worker, or the current service worker is being discarded due to an install failure.

        -
        -
      2. -

        Fire an event named statechange at workerObject.

        -
      -
    -

    The task must use workerObject’s relevant settings object’s responsible event loop and the DOM manipulation task source.

    -
-
-
-

Notify Controller Change

-
-
Input -
-

client, a service worker client

-
Output -
-

None

-
-
    -
  1. -

    Assert: client is not null.

    -
  2. -

    If client is an environment settings object, queue a task to fire an event named controllerchange at the ServiceWorkerContainer object that client is associated with.

    -
-

The task must use client’s responsible event loop and the DOM manipulation task source.

-
-
-

Match Service Worker Registration

-
-
Input -
-

clientURL, a URL

-
Output -
-

registration, a service worker registration

-
-
    -
  1. -

    Run the following steps atomically.

    -
  2. -

    Let clientURLString be serialized clientURL.

    -
  3. -

    Let matchingScopeString be the empty string.

    -
  4. -

    Let scopeStringSet be the result of getting the keys from scope to registration map.

    -
  5. -

    Set matchingScopeString to the longest value in scopeStringSet which the value of clientURLString starts with, if it exists.

    -

    Note: The URL string matching in this step is prefix-based rather than path-structural. E.g. a client URL string with "https://example.com/prefix-of/resource.html" will match a registration for a scope with "https://example.com/prefix". The URL string comparison is safe for the same-origin security as HTTP(S) URLs are always serialized with a trailing slash at the end of the origin part of the URLs.

    -
  6. -

    Let matchingScope be null.

    -
  7. -

    If matchingScopeString is not the empty string, then:

    -
      -
    1. -

      Set matchingScope to the result of parsing matchingScopeString.

      -
    2. -

      Assert: matchingScope’s origin and clientURL’s origin are same origin.

      -
    -
  8. -

    Let registration be the result of running Get Registration algorithm passing matchingScope as the argument.

    -
  9. -

    If registration is not null and registration’s uninstalling flag is set, return null.

    -
  10. -

    Return registration.

    -
-
-
-

Get Registration

-
-
Input -
-

scope, a URL

-
Output -
-

registration, a service worker registration

-
-
    -
  1. -

    Run the following steps atomically.

    -
  2. -

    Let scopeString be the empty string.

    -
  3. -

    If scope is not null, set scopeString to serialized scope with the exclude fragment flag set.

    -
  4. -

    Let registration be null.

    -
  5. -

    For each keyvalue of scope to registration map:

    -
      -
    1. -

      If scopeString matches key, set registration to value.

      -
    -
  6. -

    Return registration.

    -
-
-
-

Get Newest Worker

-
-
Input -
-

registration, a service worker registration

-
Output -
-

newestWorker, a service worker

-
-
    -
  1. -

    Run the following steps atomically.

    -
  2. -

    Let newestWorker be null.

    -
  3. -

    If registration’s installing worker is not null, set newestWorker to registration’s installing worker.

    -
  4. -

    Else if registration’s waiting worker is not null, set newestWorker to registration’s waiting worker.

    -
  5. -

    Else if registration’s active worker is not null, set newestWorker to registration’s active worker.

    -
  6. -

    Return newestWorker.

    -
-
-
-

Service Worker Has No Pending Events

-
-
Input -
-

worker, a service worker

-
Output -
-

True or false, a boolean

-
-
    -
  1. -

    For each event of worker’s set of extended events:

    -
      -
    1. -

      If event is active, return false.

      -
    -
  2. -

    Return true.

    -
-
-
-

Create Client

-
-
Input -
-

client, a service worker client

-
Output -
-

clientObject, a Client object

-
-
    -
  1. -

    Let clientObject be a new Client object.

    -
  2. -

    Set clientObject’s service worker client to client.

    -
  3. -

    Return clientObject.

    -
-
-
-

Create Window Client

-
-
Input -
-

client, a service worker client

-
-

frameType, a string

-
-

visibilityState, a string

-
-

focusState, a boolean

-
-

ancestorOriginsList, a list

-
Output -
-

windowClient, a WindowClient object

-
-
    -
  1. -

    Let windowClient be a new WindowClient object.

    -
  2. -

    Set windowClient’s service worker client to client.

    -
  3. -

    Set windowClient’s frame type to frameType.

    -
  4. -

    Set windowClient’s visibility state to visibilityState.

    -
  5. -

    Set windowClient’s focus state to focusState.

    -
  6. -

    Set windowClient’s ancestor origins array to a frozen array created from ancestorOriginsList.

    -
  7. -

    Return windowClient.

    -
-
-
-

Get Frame Type

-
-
Input -
-

browsingContext, a browsing context

-
Output -
-

frameType, a string

-
-
    -
  1. -

    Return the value by switching on the type of browsingContext:

    -
    -
    Nested browsing context -
    -

    "nested"

    -
    Auxiliary browsing context -
    -

    "auxiliary"

    -
    Otherwise -
    -

    "top-level"

    -
    -
-
-
-

Resolve Get Client Promise

-
-
Input -
-

client, a service worker client

-
-

promise, a promise

-
Output -
-

none

-
-
    -
  1. -

    If client is an environment settings object, then:

    -
      -
    1. -

      If client is not a secure context, queue a task to reject promise with a "SecurityError" DOMException, on promise’s relevant settings object's responsible event loop using the DOM manipulation task source, and abort these steps.

      -
    -
  2. -

    Else:

    -
      -
    1. -

      If client’s creation URL is not a potentially trustworthy URL, queue a task to reject promise with a "SecurityError" DOMException, on promise’s relevant settings object's responsible event loop using the DOM manipulation task source, and abort these steps.

      -
    -
  3. -

    If client is an environment settings object and is not a window client, then:

    -
      -
    1. -

      Let clientObject be the result of running Create Client algorithm with client as the argument.

      -
    2. -

      Queue a task to resolve promise with clientObject, on promise’s relevant settings object's responsible event loop using the DOM manipulation task source, and abort these steps.

      -
    -
  4. -

    Else:

    -
      -
    1. -

      Let browsingContext be null.

      -
    2. -

      If client is an environment settings object, set browsingContext to client’s global object's browsing context.

      -
    3. -

      Else, set browsingContext to client’s target browsing context.

      -
    4. -

      Queue a task to run the following steps on browsingContext’s event loop using the user interaction task source:

      -
        -
      1. -

        Let frameType be the result of running Get Frame Type with browsingContext.

        -
      2. -

        Let visibilityState be browsingContext’s active document's visibilityState attribute value.

        -
      3. -

        Let focusState be the result of running the has focus steps with browsingContext’s active document as the argument.

        -
      4. -

        Let ancestorOriginsList be the empty list.

        -
      5. -

        If client is a window client, set ancestorOriginsList to browsingContext’s active document's relevant global object's Location object’s ancestor origins list's associated list.

        -
      6. -

        Queue a task to run the following steps on promise’s relevant settings object's responsible event loop using the DOM manipulation task source:

        -
          -
        1. -

          If client’s discarded flag is set, resolve promise with undefined and abort these steps.

          -
        2. -

          Let windowClient be the result of running Create Window Client with client, frameType, visibilityState, focusState, and ancestorOriginsList.

          -
        3. -

          Resolve promise with windowClient.

          -
        -
      -
    -
-
-
-

Query Cache

-
-
Input -
-

requestQuery, a request

-
-

options, a CacheQueryOptions object, optional

-
-

targetStorage, a request response list, optional

-
Output -
-

resultList, a request response list

-
-
    -
  1. -

    Let resultList be an empty list.

    -
  2. -

    Let storage be null.

    -
  3. -

    If the optional argument targetStorage is omitted, set storage to the relevant request response list.

    -
  4. -

    Else, set storage to targetStorage.

    -
  5. -

    For each requestResponse of storage:

    -
      -
    1. -

      Let cachedRequest be requestResponse’s request.

      -
    2. -

      Let cachedResponse be requestResponse’s response.

      -
    3. -

      If Request Matches Cached Item with requestQuery, cachedRequest, cachedResponse, and options returns true, then:

      -
        -
      1. -

        Let requestCopy be a copy of cachedRequest.

        -
      2. -

        Let responseCopy be a copy of cachedResponse.

        -
      3. -

        Add requestCopy/responseCopy to resultList.

        -
      -
    -
  6. -

    Return resultList.

    -
-
-
-

Request Matches Cached Item

-
-
Input -
-

requestQuery, a request

-
-

request, a request

-
-

response, a response or null, optional, defaulting to null

-
-

options, a CacheQueryOptions object, optional

-
Output -
-

a boolean

-
-
    -
  1. -

    If options.ignoreMethod is false and request’s method is not `GET`, return false.

    -
  2. -

    Let queryURL be requestQuery’s url.

    -
  3. -

    Let cachedURL be request’s url.

    -
  4. -

    If options.ignoreSearch is true, then:

    -
      -
    1. -

      Set cachedURL’s query to the empty string.

      -
    2. -

      Set queryURL’s query to the empty string.

      -
    -
  5. -

    If queryURL does not equal cachedURL with the exclude fragment flag set, then return false.

    -
  6. -

    If response is null, options.ignoreVary is true, or response’s header list does not contain `Vary`, then return true.

    -
  7. -

    Let fieldValues be the list containing the elements corresponding to the field-values of the Vary header for the value of the header with name `Vary`.

    -
  8. -

    For each fieldValue in fieldValues:

    -
      -
    1. -

      If fieldValue matches "*", or the combined value given fieldValue and request’s header list does not match the combined value given fieldValue and requestQuery’s header list, then return false.

      -
    -
  9. -

    Return true.

    -
-
-
-

Batch Cache Operations

-
-
Input -
-

operations, a list of cache batch operation objects

-
Output -
-

resultList, a request response list

-
-
    -
  1. -

    Let cache be the relevant request response list.

    -
  2. -

    Let backupCache be a new request response list that is a copy of cache.

    -
  3. -

    Let addedItems be an empty list.

    -
  4. -

    Try running the following substeps atomically:

    -
      -
    1. -

      Let resultList be an empty list.

      -
    2. -

      For each operation in operations:

      -
        -
      1. -

        If operation’s type matches neither "delete" nor "put", throw a TypeError.

        -
      2. -

        If operation’s type matches "delete" and operation’s response is not null, throw a TypeError.

        -
      3. -

        If the result of running Query Cache with operation’s request, operation’s options, and addedItems is not empty, throw an "InvalidStateError" DOMException.

        -
      4. -

        Let requestResponses be an empty list.

        -
      5. -

        If operation’s type matches "delete", then:

        -
          -
        1. -

          Set requestResponses to the result of running Query Cache with operation’s request and operation’s options.

          -
        2. -

          For each requestResponse in requestResponses:

          -
            -
          1. -

            Remove the item whose value matches requestResponse from cache.

            -
          -
        -
      6. -

        Else if operation’s type matches "put", then:

        -
          -
        1. -

          If operation’s response is null, throw a TypeError.

          -
        2. -

          Let r be operation’s request's associated request.

          -
        3. -

          If r’s url's scheme is not one of "http" and "https", throw a TypeError.

          -
        4. -

          If r’s method is not `GET`, throw a TypeError.

          -
        5. -

          If operation’s options is not null, throw a TypeError.

          -
        6. -

          Set requestResponses to the result of running Query Cache with operation’s request.

          -
        7. -

          For each requestResponse of requestResponses:

          -
            -
          1. -

            Remove the item whose value matches requestResponse from cache.

            -
          -
        8. -

          Append operation’s request/operation’s response to cache.

          -
        9. -

          If the cache write operation in the previous two steps failed due to exceeding the granted quota limit, throw a "QuotaExceededError" DOMException.

          -
        10. -

          Append operation’s request/operation’s response to addedItems.

          -
        -
      7. -

        Append operation’s request/operation’s response to resultList.

        -
      -
    3. -

      Return resultList.

      -
    -
  5. -

    And then, if an exception was thrown, then:

    -
      -
    1. -

      Remove all the items from the relevant request response list.

      -
    2. -

      For each requestResponse of backupCache:

      -
        -
      1. -

        Append requestResponse to the relevant request response list.

        -
      -
    3. -

      Throw the exception.

      -
    -

    Note: When an exception is thrown, the implementation does undo (roll back) any changes made to the cache storage during the batch operation job.

    -
-
-
-
-

Appendix B: Extended HTTP headers

-
-

Service Worker Script Request

-

An HTTP request to fetch a service worker's script resource will include the following header:

-
-
`Service-Worker` -
-

Indicates this request is a service worker's script resource request.

-

Note: This header helps administrators log the requests and detect threats.

-
-
-
-

Service Worker Script Response

-

An HTTP response to a service worker's script resource request can include the following header:

-
-
`Service-Worker-Allowed` -
-

Indicates the user agent will override the path restriction, which limits the maximum allowed scope url that the script can control, to the given value.

-

Note: The value is a URL. If a relative URL is given, it is parsed against the script’s URL.

-
-
- Default scope: -
// Maximum allowed scope defaults to the path the script sits in
-// "/js" in this example
-navigator.serviceWorker.register("/js/sw.js").then(() => {
-  console.log("Install succeeded with the default scope '/js'.");
-});
-
-
-
- Upper path without Service-Worker-Allowed header: -
// Set the scope to an upper path of the script location
-// Response has no Service-Worker-Allowed header
-navigator.serviceWorker.register("/js/sw.js", { scope: "/" }).catch(() => {
-  console.error("Install failed due to the path restriction violation.");
-});
-
-
-
- Upper path with Service-Worker-Allowed header: -
// Set the scope to an upper path of the script location
-// Response included "Service-Worker-Allowed : /"
-navigator.serviceWorker.register("/js/sw.js", { scope: "/" }).then(() => {
-  console.log("Install succeeded as the max allowed scope was overriden to '/'.");
-});
-
-
-
- A path restriction voliation even with Service-Worker-Allowed header: -
// Set the scope to an upper path of the script location
-// Response included "Service-Worker-Allowed : /foo"
-navigator.serviceWorker.register("/foo/bar/sw.js", { scope: "/" }).catch(() => {
-  console.error("Install failed as the scope is still out of the overriden maximum allowed scope.");
-});
-
-
-
-
-

Syntax

-

ABNF for the values of the headers used by the service worker's script resource requests and responses:

-
Service-Worker = %x73.63.72.69.70.74 ; "script", case-sensitive
-
-

Note: The validation of the Service-Worker-Allowed header’s values is done by URL parsing algorithm (in Update algorithm) instead of using ABNF.

-
-
-
-

8. Acknowledgements

-

Deep thanks go to Andrew Betts for organizing and hosting a small workshop of like-minded individuals including: Jake Archibald, Jackson Gabbard, Tobie Langel, Robin Berjon, Patrick Lauke, Christian Heilmann. From the clarity of the day’s discussions and the use-cases outlined there, much has become possible. Further thanks to Andrew for raising consciousness about the offline problem. His organization of EdgeConf and inclusion of Offline as a persistent topic there has created many opportunities and connections that have enabled this work to progress.

-

Anne van Kesteren has generously lent his encyclopedic knowledge of Web Platform arcana and standards development experience throughout the development of the service worker. This specification would be incomplete without his previous work in describing the real-world behavior of URLs, HTTP Fetch, Promises, and DOM. Similarly, this specification would not be possible without Ian Hickson’s rigorous Web Worker spec. Much thanks to him.

-

In no particular order, deep gratitude for design guidance and discussion goes to: Jungkee Song, Alec Flett, David Barrett-Kahn, Aaron Boodman, Michael Nordman, Tom Ashworth, Kinuko Yasuda, Darin Fisher, Jonas Sicking, Jesús Leganés Combarro, Mark Christian, Dave Hermann, Yehuda Katz, François Remy, Ilya Grigorik, Will Chan, Domenic Denicola, Nikhil Marathe, Yves Lafon, Adam Barth, Greg Simon, Devdatta Akhawe, Dominic Cooney, Jeffrey Yasskin, Joshua Bell, Boris Zbarsky, Matt Falkenhagen, Tobie Langel, Gavin Peters, Ben Kelly, Hiroki Nakagawa, Jake Archibald, Josh Soref, Jinho Bang, Yutaka Hirano, isonmad, Ali Alabbas, Philip Jägenstedt, Mike Pennisi, and Eric Willigers.

-

Jason Weber, Chris Wilson, Paul Kinlan, Ehsan Akhgari, and Daniel Austin have provided valuable, well-timed feedback on requirements and the standardization process.

-

The authors would also like to thank Dimitri Glazkov for his scripts and formatting tools which have been essential in the production of this specification. The authors are also grateful for his considerable guidance.

-

Thanks also to Vivian Cromwell, Greg Simon, Alex Komoroske, Wonsuk Lee, and Seojin Kim for their considerable professional support.

-
-
-
-

Conformance

-

Document conventions

-

Conformance requirements are expressed with a combination of - descriptive assertions and RFC 2119 terminology. The key words “MUST”, - “MUST NOT”, “REQUIRED”, “SHALL”, “SHALL NOT”, “SHOULD”, “SHOULD NOT”, - “RECOMMENDED”, “MAY”, and “OPTIONAL” in the normative parts of this - document are to be interpreted as described in RFC 2119. - However, for readability, these words do not appear in all uppercase - letters in this specification.

-

All of the text of this specification is normative except sections - explicitly marked as non-normative, examples, and notes. [RFC2119]

-

Examples in this specification are introduced with the words “for example” - or are set apart from the normative text with class="example", - like this:

-
- -

This is an example of an informative example.

-
-

Informative notes begin with the word “Note” and are set apart from the - normative text with class="note", like this:

-

Note, this is an informative note.

-

Conformant Algorithms

-

Requirements phrased in the imperative as part of algorithms (such as - "strip any leading space characters" or "return false and abort these - steps") are to be interpreted with the meaning of the key word ("must", - "should", "may", etc) used in introducing the algorithm.

-

Conformance requirements phrased as algorithms or specific steps can be - implemented in any manner, so long as the end result is equivalent. In - particular, the algorithms defined in this specification are intended to - be easy to understand and are not intended to be performant. Implementers - are encouraged to optimize.

-
- - . - -

Index

-

Terms defined by this specification

- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -

Terms defined by reference

-
    -
  • - [csp-3] defines the following terms: -
      -
    • enforced -
    • monitored -
    -
  • - [DOM] defines the following terms: -
      -
    • Document -
    • Event -
    • EventInit -
    • EventTarget -
    • cancelable -
    • canceled flag -
    • context object -
    • creating an event -
    • dispatch -
    • dispatch flag -
    • event -
    • event listener -
    • fire an event -
    • isTrusted -
    • stop immediate propagation flag -
    • stop propagation flag -
    • type -
    -
  • - [ECMASCRIPT] defines the following terms: -
      -
    • detacharraybuffer -
    • execution context -
    • initializehostdefinedrealm -
    • map objects -
    • promise -
    -
  • - [FETCH] defines the following terms: -
      -
    • "report" -
    • Headers -
    • ReadableStream -
    • Request -
    • RequestInfo -
    • Response -
    • aborted flag -
    • basic filtered response -
    • body (for response) -
    • cache mode -
    • cache state -
    • cancel -
    • client -
    • clone -
    • closed -
    • combined value -
    • construct a readablestream object -
    • contains -
    • cors filtered response -
    • destination -
    • disturbed -
    • empty -
    • enqueue -
    • errored -
    • extract a mime type -
    • extracting header list values -
    • fetch -
    • fetch(input, init) -
    • filtered response -
    • get a reader -
    • guard -
    • header -
    • header list (for response) -
    • http fetch -
    • https state -
    • https state value -
    • initiator -
    • is local -
    • locked -
    • method -
    • name -
    • navigation request -
    • network error -
    • non-subresource request -
    • ok status -
    • opaque filtered response -
    • parser metadata -
    • potential-navigation-or-subresource request -
    • process response -
    • process response end-of-body -
    • read a chunk -
    • read all bytes -
    • redirect mode -
    • request (for Request) -
    • reserved client -
    • response (for Response) -
    • service-workers mode -
    • status -
    • stream -
    • subresource request -
    • synchronous flag -
    • terminated -
    • type -
    • url -
    • use-url-credentials flag -
    • value -
    -
  • - [HTML] defines the following terms: -
      -
    • AbstractWorker -
    • DOMContentLoaded -
    • DedicatedWorkerGlobalScope -
    • EventHandler -
    • Location -
    • MessageEvent -
    • MessagePort -
    • Navigator -
    • SharedWorkerGlobalScope -
    • StructuredDeserializeWithTransfer -
    • StructuredSerializeWithTransfer -
    • Window -
    • WindowOrWorkerGlobalScope -
    • Worker -
    • WorkerGlobalScope -
    • WorkerLocation -
    • WorkerNavigator -
    • WorkerType -
    • a browsing context is discarded -
    • active document -
    • active service worker -
    • ancestor origins list -
    • api base url -
    • api url character encoding -
    • application cache -
    • auxiliary browsing context -
    • browsing context -
    • classic script -
    • closing -
    • creation url -
    • current global object -
    • data -
    • discard a document -
    • dom manipulation task source -
    • environment -
    • environment discarding steps -
    • environment settings object -
    • event handler -
    • event handler event type -
    • event handler idl attribute -
    • event loop -
    • exceptions enabled flag -
    • execution ready flag -
    • fetch a classic worker script -
    • fetch a module worker script graph -
    • focusing steps -
    • global object -
    • has focus steps -
    • https state (for environment settings object) -
    • id -
    • import scripts into worker global scope -
    • importScripts(urls) -
    • in parallel -
    • incumbent settings object -
    • is top-level -
    • list of active timers -
    • message -
    • module script -
    • navigate -
    • nested browsing context -
    • origin (for environment settings object) -
    • owner set -
    • parent browsing context -
    • perform the fetch -
    • ports -
    • queue a microtask -
    • queue a task -
    • realm (for global object) -
    • realm execution context -
    • referrer policy (for environment settings object) -
    • relevant global object -
    • relevant realm -
    • relevant settings object -
    • replacement enabled -
    • responsible browsing context -
    • responsible document -
    • responsible event loop -
    • run a classic script -
    • run a module script -
    • same origin -
    • script -
    • shared worker -
    • source -
    • source browsing context -
    • target browsing context -
    • task -
    • task queues -
    • task source -
    • top-level browsing context -
    • triggered by user activation -
    • type -
    • unload a document -
    • unsafe response -
    • url -
    • user interaction task source -
    • web worker -
    -
  • - [INFRA] defines the following terms: -
      -
    • append (for set) -
    • ascii case-insensitive -
    • contain -
    • continue -
    • dequeue -
    • enqueue -
    • entry -
    • exist -
    • for each (for map) -
    • get the keys -
    • is not empty -
    • item -
    • key -
    • list -
    • ordered map -
    • ordered set -
    • pair -
    • queue -
    • remove (for map) -
    • set -
    • struct -
    • value -
    -
  • - [MIMESNIFF] defines the following terms: -
      -
    • javascript mime type -
    -
  • - [page-visibility] defines the following terms: -
      -
    • VisibilityState -
    • visibilityState -
    -
  • - [promises-guide] defines the following terms: -
      -
    • a new promise -
    • a promise rejected with -
    • a promise resolved with -
    • transforming -
    • upon fulfillment -
    • upon rejection -
    • waiting for all -
    -
  • - [push] defines the following terms: -
      -
    • push -
    -
  • - [referrer-policy] defines the following terms: -
      -
    • referrer policy -
    -
  • - [rfc7230] defines the following terms: -
      -
    • field-value -
    -
  • - [rfc7231] defines the following terms: -
      -
    • vary -
    -
  • - [secure-contexts] defines the following terms: -
      -
    • potentially trustworthy origin -
    • potentially trustworthy url -
    • secure contexts -
    -
  • - [URL] defines the following terms: -
      -
    • equal -
    • fragment -
    • origin -
    • path -
    • query -
    • scheme -
    • url -
    • url parser -
    • url serializer -
    -
  • - [webappsec-referrer-policy] defines the following terms: -
      -
    • parse a referrer policy from a referrer-policy header -
    -
  • - [WebIDL] defines the following terms: -
      -
    • AbortError -
    • DOMException -
    • DOMString -
    • Exposed -
    • Global -
    • InvalidAccessError -
    • InvalidStateError -
    • NewObject -
    • QuotaExceededError -
    • SameObject -
    • SecureContext -
    • SecurityError -
    • USVString -
    • boolean -
    • create a frozen array -
    • created -
    • exception -
    • frozen array type -
    • message -
    • object -
    • partial interface -
    • present -
    • throw -
    -
-

References

-

Normative References

-
-
[CSP-3] -
Content Security Policy Level 3 URL: https://www.w3.org/TR/CSP3/ -
[DOM] -
Anne van Kesteren. DOM Standard. Living Standard. URL: https://dom.spec.whatwg.org/ -
[ECMASCRIPT] -
ECMAScript Language Specification. URL: https://tc39.github.io/ecma262/ -
[FETCH] -
Anne van Kesteren. Fetch Standard. Living Standard. URL: https://fetch.spec.whatwg.org/ -
[HTML] -
Anne van Kesteren; et al. HTML Standard. Living Standard. URL: https://html.spec.whatwg.org/multipage/ -
[INFRA] -
Anne van Kesteren; Domenic Denicola. Infra Standard. Living Standard. URL: https://infra.spec.whatwg.org/ -
[MIMESNIFF] -
Gordon P. Hemsley. MIME Sniffing Standard. Living Standard. URL: https://mimesniff.spec.whatwg.org/ -
[PAGE-VISIBILITY] -
Jatinder Mann; Arvind Jain. Page Visibility (Second Edition). 29 October 2013. REC. URL: https://www.w3.org/TR/page-visibility/ -
[PROMISES-GUIDE] -
Domenic Denicola. Writing Promise-Using Specifications. 16 February 2016. Finding of the W3C TAG. URL: https://www.w3.org/2001/tag/doc/promises-guide -
[REFERRER-POLICY] -
Jochen Eisinger; Emily Stark. Referrer Policy. 26 January 2017. CR. URL: https://www.w3.org/TR/referrer-policy/ -
[RFC2119] -
S. Bradner. Key words for use in RFCs to Indicate Requirement Levels. March 1997. Best Current Practice. URL: https://tools.ietf.org/html/rfc2119 -
[RFC5234] -
D. Crocker, Ed.; P. Overell. Augmented BNF for Syntax Specifications: ABNF. January 2008. Internet Standard. URL: https://tools.ietf.org/html/rfc5234 -
[RFC7230] -
R. Fielding, Ed.; J. Reschke, Ed.. Hypertext Transfer Protocol (HTTP/1.1): Message Syntax and Routing. June 2014. Proposed Standard. URL: https://tools.ietf.org/html/rfc7230 -
[RFC7231] -
R. Fielding, Ed.; J. Reschke, Ed.. Hypertext Transfer Protocol (HTTP/1.1): Semantics and Content. June 2014. Proposed Standard. URL: https://tools.ietf.org/html/rfc7231 -
[SECURE-CONTEXTS] -
Mike West. Secure Contexts. 15 September 2016. CR. URL: https://www.w3.org/TR/secure-contexts/ -
[URL] -
Anne van Kesteren. URL Standard. Living Standard. URL: https://url.spec.whatwg.org/ -
[WebIDL] -
Cameron McCormack; Boris Zbarsky; Tobie Langel. Web IDL. 15 December 2016. ED. URL: https://heycam.github.io/webidl/ -
-

Informative References

-
-
[UNSANCTIONED-TRACKING] -
Unsanctioned Web Tracking. 17 July 2015. Finding of the W3C TAG. URL: https://www.w3.org/2001/tag/doc/unsanctioned-tracking/ -
-

IDL Index

-
[SecureContext, Exposed=(Window,Worker)]
-interface ServiceWorker : EventTarget {
-  readonly attribute USVString scriptURL;
-  readonly attribute ServiceWorkerState state;
-  void postMessage(any message, optional sequence<object> transfer = []);
-
-  // event
-  attribute EventHandler onstatechange;
-};
-ServiceWorker includes AbstractWorker;
-
-enum ServiceWorkerState {
-  "installing",
-  "installed",
-  "activating",
-  "activated",
-  "redundant"
-};
-
-[SecureContext, Exposed=(Window,Worker)]
-interface ServiceWorkerRegistration : EventTarget {
-  readonly attribute ServiceWorker? installing;
-  readonly attribute ServiceWorker? waiting;
-  readonly attribute ServiceWorker? active;
-
-  readonly attribute USVString scope;
-  readonly attribute ServiceWorkerUpdateViaCache updateViaCache;
-
-  [NewObject] Promise<void> update();
-  [NewObject] Promise<boolean> unregister();
-
-  // event
-  attribute EventHandler onupdatefound;
-};
-
-enum ServiceWorkerUpdateViaCache {
-  "imports",
-  "all",
-  "none"
-};
-
-partial interface Navigator {
-  [SecureContext, SameObject] readonly attribute ServiceWorkerContainer serviceWorker;
-};
-
-partial interface WorkerNavigator {
-  [SecureContext, SameObject] readonly attribute ServiceWorkerContainer serviceWorker;
-};
-
-[SecureContext, Exposed=(Window,Worker)]
-interface ServiceWorkerContainer : EventTarget {
-  readonly attribute ServiceWorker? controller;
-  readonly attribute Promise<ServiceWorkerRegistration> ready;
-
-  [NewObject] Promise<ServiceWorkerRegistration> register(USVString scriptURL, optional RegistrationOptions options);
-
-  [NewObject] Promise<any> getRegistration(optional USVString clientURL = "");
-  [NewObject] Promise<FrozenArray<ServiceWorkerRegistration>> getRegistrations();
-
-  void startMessages();
-
-
-  // events
-  attribute EventHandler oncontrollerchange;
-  attribute EventHandler onmessage; // event.source of message events is ServiceWorker object
-  attribute EventHandler onmessageerror;
-};
-
-dictionary RegistrationOptions {
-  USVString scope;
-  WorkerType type = "classic";
-  ServiceWorkerUpdateViaCache updateViaCache = "imports";
-};
-
-[Global=(Worker,ServiceWorker), Exposed=ServiceWorker]
-interface ServiceWorkerGlobalScope : WorkerGlobalScope {
-  [SameObject] readonly attribute Clients clients;
-  [SameObject] readonly attribute ServiceWorkerRegistration registration;
-
-  [NewObject] Promise<void> skipWaiting();
-
-  attribute EventHandler oninstall;
-  attribute EventHandler onactivate;
-  attribute EventHandler onfetch;
-
-  // event
-  attribute EventHandler onmessage; // event.source of the message events is Client object
-  attribute EventHandler onmessageerror;
-};
-
-[Exposed=ServiceWorker]
-interface Client {
-  readonly attribute USVString url;
-  readonly attribute FrameType frameType;
-  readonly attribute DOMString id;
-  readonly attribute ClientType type;
-  void postMessage(any message, optional sequence<object> transfer = []);
-};
-
-[Exposed=ServiceWorker]
-interface WindowClient : Client {
-  readonly attribute VisibilityState visibilityState;
-  readonly attribute boolean focused;
-  [SameObject] readonly attribute FrozenArray<USVString> ancestorOrigins;
-  [NewObject] Promise<WindowClient> focus();
-  [NewObject] Promise<WindowClient?> navigate(USVString url);
-};
-
-enum FrameType {
-  "auxiliary",
-  "top-level",
-  "nested",
-  "none"
-};
-
-[Exposed=ServiceWorker]
-interface Clients {
-  // The objects returned will be new instances every time
-  [NewObject] Promise<any> get(DOMString id);
-  [NewObject] Promise<FrozenArray<Client>> matchAll(optional ClientQueryOptions options);
-  [NewObject] Promise<WindowClient?> openWindow(USVString url);
-  [NewObject] Promise<void> claim();
-};
-
-dictionary ClientQueryOptions {
-  boolean includeUncontrolled = false;
-  ClientType type = "window";
-};
-
-enum ClientType {
-  "window",
-  "worker",
-  "sharedworker",
-  "all"
-};
-
-[Constructor(DOMString type, optional ExtendableEventInit eventInitDict), Exposed=ServiceWorker]
-interface ExtendableEvent : Event {
-  void waitUntil(Promise<any> f);
-};
-
-dictionary ExtendableEventInit : EventInit {
-  // Defined for the forward compatibility across the derived events
-};
-
-[Constructor(DOMString type, FetchEventInit eventInitDict), Exposed=ServiceWorker]
-interface FetchEvent : ExtendableEvent {
-  [SameObject] readonly attribute Request request;
-  readonly attribute DOMString clientId;
-
-  void respondWith(Promise<Response> r);
-};
-
-dictionary FetchEventInit : ExtendableEventInit {
-  required Request request;
-  DOMString clientId = "";
-};
-
-[Constructor(DOMString type, optional ExtendableMessageEventInit eventInitDict), Exposed=ServiceWorker]
-interface ExtendableMessageEvent : ExtendableEvent {
-  readonly attribute any data;
-  readonly attribute USVString origin;
-  readonly attribute DOMString lastEventId;
-  [SameObject] readonly attribute (Client or ServiceWorker or MessagePort)? source;
-  readonly attribute FrozenArray<MessagePort> ports;
-};
-
-dictionary ExtendableMessageEventInit : ExtendableEventInit {
-  any data = null;
-  USVString origin = "";
-  DOMString lastEventId = "";
-  (Client or ServiceWorker or MessagePort)? source = null;
-  sequence<MessagePort> ports = [];
-};
-
-partial interface WindowOrWorkerGlobalScope {
-  [SecureContext, SameObject] readonly attribute CacheStorage caches;
-};
-
-[SecureContext, Exposed=(Window,Worker)]
-interface Cache {
-  [NewObject] Promise<any> match(RequestInfo request, optional CacheQueryOptions options);
-  [NewObject] Promise<FrozenArray<Response>> matchAll(optional RequestInfo request, optional CacheQueryOptions options);
-  [NewObject] Promise<void> add(RequestInfo request);
-  [NewObject] Promise<void> addAll(sequence<RequestInfo> requests);
-  [NewObject] Promise<void> put(RequestInfo request, Response response);
-  [NewObject] Promise<boolean> delete(RequestInfo request, optional CacheQueryOptions options);
-  [NewObject] Promise<FrozenArray<Request>> keys(optional RequestInfo request, optional CacheQueryOptions options);
-};
-
-dictionary CacheQueryOptions {
-  boolean ignoreSearch = false;
-  boolean ignoreMethod = false;
-  boolean ignoreVary = false;
-};
-
-[SecureContext, Exposed=(Window,Worker)]
-interface CacheStorage {
-  [NewObject] Promise<any> match(RequestInfo request, optional MultiCacheQueryOptions options);
-  [NewObject] Promise<boolean> has(DOMString cacheName);
-  [NewObject] Promise<Cache> open(DOMString cacheName);
-  [NewObject] Promise<boolean> delete(DOMString cacheName);
-  [NewObject] Promise<sequence<DOMString>> keys();
-};
-
-dictionary MultiCacheQueryOptions : CacheQueryOptions {
-  DOMString cacheName;
-};
-
-
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file