mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-06 00:55:37 +00:00
b57c44dc2d
--HG-- extra : rebase_source : c8636b7a77d23eb0c6392dac41825f62dde00db8
547 lines
14 KiB
JavaScript
547 lines
14 KiB
JavaScript
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
"use strict";
|
|
|
|
const { Ci } = require("chrome");
|
|
const { DebuggerServer } = require("devtools/server/main");
|
|
const Services = require("Services");
|
|
const { XPCOMUtils } = require("resource://gre/modules/XPCOMUtils.jsm");
|
|
const protocol = require("devtools/shared/protocol");
|
|
const { Arg, method, RetVal } = protocol;
|
|
const {
|
|
workerSpec,
|
|
pushSubscriptionSpec,
|
|
serviceWorkerRegistrationSpec,
|
|
} = require("devtools/shared/specs/worker");
|
|
|
|
loader.lazyRequireGetter(this, "ChromeUtils");
|
|
loader.lazyRequireGetter(this, "events", "sdk/event/core");
|
|
|
|
XPCOMUtils.defineLazyServiceGetter(
|
|
this, "wdm",
|
|
"@mozilla.org/dom/workers/workerdebuggermanager;1",
|
|
"nsIWorkerDebuggerManager"
|
|
);
|
|
|
|
XPCOMUtils.defineLazyServiceGetter(
|
|
this, "swm",
|
|
"@mozilla.org/serviceworkers/manager;1",
|
|
"nsIServiceWorkerManager"
|
|
);
|
|
|
|
XPCOMUtils.defineLazyServiceGetter(
|
|
this, "PushService",
|
|
"@mozilla.org/push/Service;1",
|
|
"nsIPushService"
|
|
);
|
|
|
|
function matchWorkerDebugger(dbg, options) {
|
|
if ("type" in options && dbg.type !== options.type) {
|
|
return false;
|
|
}
|
|
if ("window" in options) {
|
|
let window = dbg.window;
|
|
while (window !== null && window.parent !== window) {
|
|
window = window.parent;
|
|
}
|
|
|
|
if (window !== options.window) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
let WorkerActor = protocol.ActorClassWithSpec(workerSpec, {
|
|
initialize(conn, dbg) {
|
|
protocol.Actor.prototype.initialize.call(this, conn);
|
|
this._dbg = dbg;
|
|
this._attached = false;
|
|
this._threadActor = null;
|
|
this._transport = null;
|
|
},
|
|
|
|
form(detail) {
|
|
if (detail === "actorid") {
|
|
return this.actorID;
|
|
}
|
|
let form = {
|
|
actor: this.actorID,
|
|
consoleActor: this._consoleActor,
|
|
url: this._dbg.url,
|
|
type: this._dbg.type
|
|
};
|
|
if (this._dbg.type === Ci.nsIWorkerDebugger.TYPE_SERVICE) {
|
|
let registration = this._getServiceWorkerRegistrationInfo();
|
|
form.scope = registration.scope;
|
|
}
|
|
return form;
|
|
},
|
|
|
|
attach() {
|
|
if (this._dbg.isClosed) {
|
|
return { error: "closed" };
|
|
}
|
|
|
|
if (!this._attached) {
|
|
// Automatically disable their internal timeout that shut them down
|
|
// Should be refactored by having actors specific to service workers
|
|
if (this._dbg.type == Ci.nsIWorkerDebugger.TYPE_SERVICE) {
|
|
let worker = this._getServiceWorkerInfo();
|
|
if (worker) {
|
|
worker.attachDebugger();
|
|
}
|
|
}
|
|
this._dbg.addListener(this);
|
|
this._attached = true;
|
|
}
|
|
|
|
return {
|
|
type: "attached",
|
|
url: this._dbg.url
|
|
};
|
|
},
|
|
|
|
detach() {
|
|
if (!this._attached) {
|
|
return { error: "wrongState" };
|
|
}
|
|
|
|
this._detach();
|
|
|
|
return { type: "detached" };
|
|
},
|
|
|
|
destroy() {
|
|
protocol.Actor.prototype.destroy.call(this);
|
|
if (this._attached) {
|
|
this._detach();
|
|
}
|
|
},
|
|
|
|
disconnect() {
|
|
this.destroy();
|
|
},
|
|
|
|
connect(options) {
|
|
if (!this._attached) {
|
|
return { error: "wrongState" };
|
|
}
|
|
|
|
if (this._threadActor !== null) {
|
|
return {
|
|
type: "connected",
|
|
threadActor: this._threadActor
|
|
};
|
|
}
|
|
|
|
return DebuggerServer.connectToWorker(
|
|
this.conn, this._dbg, this.actorID, options
|
|
).then(({ threadActor, transport, consoleActor }) => {
|
|
this._threadActor = threadActor;
|
|
this._transport = transport;
|
|
this._consoleActor = consoleActor;
|
|
|
|
return {
|
|
type: "connected",
|
|
threadActor: this._threadActor,
|
|
consoleActor: this._consoleActor
|
|
};
|
|
}, (error) => {
|
|
return { error: error.toString() };
|
|
});
|
|
},
|
|
|
|
push() {
|
|
if (this._dbg.type !== Ci.nsIWorkerDebugger.TYPE_SERVICE) {
|
|
return { error: "wrongType" };
|
|
}
|
|
let registration = this._getServiceWorkerRegistrationInfo();
|
|
let originAttributes = ChromeUtils.originAttributesToSuffix(
|
|
this._dbg.principal.originAttributes);
|
|
swm.sendPushEvent(originAttributes, registration.scope);
|
|
return { type: "pushed" };
|
|
},
|
|
|
|
onClose() {
|
|
if (this._attached) {
|
|
this._detach();
|
|
}
|
|
|
|
this.conn.sendActorEvent(this.actorID, "close");
|
|
},
|
|
|
|
onError(filename, lineno, message) {
|
|
reportError("ERROR:" + filename + ":" + lineno + ":" + message + "\n");
|
|
},
|
|
|
|
_getServiceWorkerRegistrationInfo() {
|
|
return swm.getRegistrationByPrincipal(this._dbg.principal, this._dbg.url);
|
|
},
|
|
|
|
_getServiceWorkerInfo() {
|
|
let registration = this._getServiceWorkerRegistrationInfo();
|
|
return registration.getWorkerByID(this._dbg.serviceWorkerID);
|
|
},
|
|
|
|
_detach() {
|
|
if (this._threadActor !== null) {
|
|
this._transport.close();
|
|
this._transport = null;
|
|
this._threadActor = null;
|
|
}
|
|
|
|
// If the worker is already destroyed, nsIWorkerDebugger.type throws
|
|
// (_dbg.closed appears to be false when it throws)
|
|
let type;
|
|
try {
|
|
type = this._dbg.type;
|
|
} catch (e) {}
|
|
|
|
if (type == Ci.nsIWorkerDebugger.TYPE_SERVICE) {
|
|
let worker = this._getServiceWorkerInfo();
|
|
if (worker) {
|
|
worker.detachDebugger();
|
|
}
|
|
}
|
|
|
|
this._dbg.removeListener(this);
|
|
this._attached = false;
|
|
}
|
|
});
|
|
|
|
exports.WorkerActor = WorkerActor;
|
|
|
|
function WorkerActorList(conn, options) {
|
|
this._conn = conn;
|
|
this._options = options;
|
|
this._actors = new Map();
|
|
this._onListChanged = null;
|
|
this._mustNotify = false;
|
|
this.onRegister = this.onRegister.bind(this);
|
|
this.onUnregister = this.onUnregister.bind(this);
|
|
}
|
|
|
|
WorkerActorList.prototype = {
|
|
getList() {
|
|
// Create a set of debuggers.
|
|
let dbgs = new Set();
|
|
let e = wdm.getWorkerDebuggerEnumerator();
|
|
while (e.hasMoreElements()) {
|
|
let dbg = e.getNext().QueryInterface(Ci.nsIWorkerDebugger);
|
|
if (matchWorkerDebugger(dbg, this._options)) {
|
|
dbgs.add(dbg);
|
|
}
|
|
}
|
|
|
|
// Delete each actor for which we don't have a debugger.
|
|
for (let [dbg, ] of this._actors) {
|
|
if (!dbgs.has(dbg)) {
|
|
this._actors.delete(dbg);
|
|
}
|
|
}
|
|
|
|
// Create an actor for each debugger for which we don't have one.
|
|
for (let dbg of dbgs) {
|
|
if (!this._actors.has(dbg)) {
|
|
this._actors.set(dbg, new WorkerActor(this._conn, dbg));
|
|
}
|
|
}
|
|
|
|
let actors = [];
|
|
for (let [, actor] of this._actors) {
|
|
actors.push(actor);
|
|
}
|
|
|
|
if (!this._mustNotify) {
|
|
if (this._onListChanged !== null) {
|
|
wdm.addListener(this);
|
|
}
|
|
this._mustNotify = true;
|
|
}
|
|
|
|
return Promise.resolve(actors);
|
|
},
|
|
|
|
get onListChanged() {
|
|
return this._onListChanged;
|
|
},
|
|
|
|
set onListChanged(onListChanged) {
|
|
if (typeof onListChanged !== "function" && onListChanged !== null) {
|
|
throw new Error("onListChanged must be either a function or null.");
|
|
}
|
|
if (onListChanged === this._onListChanged) {
|
|
return;
|
|
}
|
|
|
|
if (this._mustNotify) {
|
|
if (this._onListChanged === null && onListChanged !== null) {
|
|
wdm.addListener(this);
|
|
}
|
|
if (this._onListChanged !== null && onListChanged === null) {
|
|
wdm.removeListener(this);
|
|
}
|
|
}
|
|
this._onListChanged = onListChanged;
|
|
},
|
|
|
|
_notifyListChanged() {
|
|
this._onListChanged();
|
|
|
|
if (this._onListChanged !== null) {
|
|
wdm.removeListener(this);
|
|
}
|
|
this._mustNotify = false;
|
|
},
|
|
|
|
onRegister(dbg) {
|
|
if (matchWorkerDebugger(dbg, this._options)) {
|
|
this._notifyListChanged();
|
|
}
|
|
},
|
|
|
|
onUnregister(dbg) {
|
|
if (matchWorkerDebugger(dbg, this._options)) {
|
|
this._notifyListChanged();
|
|
}
|
|
}
|
|
};
|
|
|
|
exports.WorkerActorList = WorkerActorList;
|
|
|
|
let PushSubscriptionActor = protocol.ActorClassWithSpec(pushSubscriptionSpec, {
|
|
initialize(conn, subscription) {
|
|
protocol.Actor.prototype.initialize.call(this, conn);
|
|
this._subscription = subscription;
|
|
},
|
|
|
|
form(detail) {
|
|
if (detail === "actorid") {
|
|
return this.actorID;
|
|
}
|
|
let subscription = this._subscription;
|
|
return {
|
|
actor: this.actorID,
|
|
endpoint: subscription.endpoint,
|
|
pushCount: subscription.pushCount,
|
|
lastPush: subscription.lastPush,
|
|
quota: subscription.quota
|
|
};
|
|
},
|
|
|
|
destroy() {
|
|
protocol.Actor.prototype.destroy.call(this);
|
|
this._subscription = null;
|
|
},
|
|
});
|
|
|
|
// Lazily load the service-worker-child.js process script only once.
|
|
let _serviceWorkerProcessScriptLoaded = false;
|
|
|
|
let ServiceWorkerRegistrationActor =
|
|
protocol.ActorClassWithSpec(serviceWorkerRegistrationSpec, {
|
|
/**
|
|
* Create the ServiceWorkerRegistrationActor
|
|
* @param DebuggerServerConnection conn
|
|
* The server connection.
|
|
* @param ServiceWorkerRegistrationInfo registration
|
|
* The registration's information.
|
|
*/
|
|
initialize(conn, registration) {
|
|
protocol.Actor.prototype.initialize.call(this, conn);
|
|
this._conn = conn;
|
|
this._registration = registration;
|
|
this._pushSubscriptionActor = null;
|
|
Services.obs.addObserver(this, PushService.subscriptionModifiedTopic, false);
|
|
},
|
|
|
|
form(detail) {
|
|
if (detail === "actorid") {
|
|
return this.actorID;
|
|
}
|
|
let registration = this._registration;
|
|
return {
|
|
actor: this.actorID,
|
|
scope: registration.scope,
|
|
url: registration.scriptSpec
|
|
};
|
|
},
|
|
|
|
destroy() {
|
|
protocol.Actor.prototype.destroy.call(this);
|
|
Services.obs.removeObserver(this, PushService.subscriptionModifiedTopic, false);
|
|
this._registration = null;
|
|
if (this._pushSubscriptionActor) {
|
|
this._pushSubscriptionActor.destroy();
|
|
}
|
|
this._pushSubscriptionActor = null;
|
|
},
|
|
|
|
disconnect() {
|
|
this.destroy();
|
|
},
|
|
|
|
/**
|
|
* Standard observer interface to listen to push messages and changes.
|
|
*/
|
|
observe(subject, topic, data) {
|
|
let scope = this._registration.scope;
|
|
if (data !== scope) {
|
|
// This event doesn't concern us, pretend nothing happened.
|
|
return;
|
|
}
|
|
switch (topic) {
|
|
case PushService.subscriptionModifiedTopic:
|
|
if (this._pushSubscriptionActor) {
|
|
this._pushSubscriptionActor.destroy();
|
|
this._pushSubscriptionActor = null;
|
|
}
|
|
events.emit(this, "push-subscription-modified");
|
|
break;
|
|
}
|
|
},
|
|
|
|
start() {
|
|
if (!_serviceWorkerProcessScriptLoaded) {
|
|
Services.ppmm.loadProcessScript(
|
|
"resource://devtools/server/service-worker-child.js", true);
|
|
_serviceWorkerProcessScriptLoaded = true;
|
|
}
|
|
Services.ppmm.broadcastAsyncMessage("serviceWorkerRegistration:start", {
|
|
scope: this._registration.scope
|
|
});
|
|
return { type: "started" };
|
|
},
|
|
|
|
unregister() {
|
|
let { principal, scope } = this._registration;
|
|
let unregisterCallback = {
|
|
unregisterSucceeded: function () {},
|
|
unregisterFailed: function () {
|
|
console.error("Failed to unregister the service worker for " + scope);
|
|
},
|
|
QueryInterface: XPCOMUtils.generateQI(
|
|
[Ci.nsIServiceWorkerUnregisterCallback])
|
|
};
|
|
swm.propagateUnregister(principal, unregisterCallback, scope);
|
|
|
|
return { type: "unregistered" };
|
|
},
|
|
|
|
getPushSubscription() {
|
|
let registration = this._registration;
|
|
let pushSubscriptionActor = this._pushSubscriptionActor;
|
|
if (pushSubscriptionActor) {
|
|
return Promise.resolve(pushSubscriptionActor);
|
|
}
|
|
return new Promise((resolve, reject) => {
|
|
PushService.getSubscription(
|
|
registration.scope,
|
|
registration.principal,
|
|
(result, subscription) => {
|
|
if (!subscription) {
|
|
resolve(null);
|
|
return;
|
|
}
|
|
pushSubscriptionActor = new PushSubscriptionActor(this._conn, subscription);
|
|
this._pushSubscriptionActor = pushSubscriptionActor;
|
|
resolve(pushSubscriptionActor);
|
|
}
|
|
);
|
|
});
|
|
},
|
|
});
|
|
|
|
function ServiceWorkerRegistrationActorList(conn) {
|
|
this._conn = conn;
|
|
this._actors = new Map();
|
|
this._onListChanged = null;
|
|
this._mustNotify = false;
|
|
this.onRegister = this.onRegister.bind(this);
|
|
this.onUnregister = this.onUnregister.bind(this);
|
|
}
|
|
|
|
ServiceWorkerRegistrationActorList.prototype = {
|
|
getList() {
|
|
// Create a set of registrations.
|
|
let registrations = new Set();
|
|
let array = swm.getAllRegistrations();
|
|
for (let index = 0; index < array.length; ++index) {
|
|
registrations.add(
|
|
array.queryElementAt(index, Ci.nsIServiceWorkerRegistrationInfo));
|
|
}
|
|
|
|
// Delete each actor for which we don't have a registration.
|
|
for (let [registration, ] of this._actors) {
|
|
if (!registrations.has(registration)) {
|
|
this._actors.delete(registration);
|
|
}
|
|
}
|
|
|
|
// Create an actor for each registration for which we don't have one.
|
|
for (let registration of registrations) {
|
|
if (!this._actors.has(registration)) {
|
|
this._actors.set(registration,
|
|
new ServiceWorkerRegistrationActor(this._conn, registration));
|
|
}
|
|
}
|
|
|
|
if (!this._mustNotify) {
|
|
if (this._onListChanged !== null) {
|
|
swm.addListener(this);
|
|
}
|
|
this._mustNotify = true;
|
|
}
|
|
|
|
let actors = [];
|
|
for (let [, actor] of this._actors) {
|
|
actors.push(actor);
|
|
}
|
|
|
|
return Promise.resolve(actors);
|
|
},
|
|
|
|
get onListchanged() {
|
|
return this._onListchanged;
|
|
},
|
|
|
|
set onListChanged(onListChanged) {
|
|
if (typeof onListChanged !== "function" && onListChanged !== null) {
|
|
throw new Error("onListChanged must be either a function or null.");
|
|
}
|
|
|
|
if (this._mustNotify) {
|
|
if (this._onListChanged === null && onListChanged !== null) {
|
|
swm.addListener(this);
|
|
}
|
|
if (this._onListChanged !== null && onListChanged === null) {
|
|
swm.removeListener(this);
|
|
}
|
|
}
|
|
this._onListChanged = onListChanged;
|
|
},
|
|
|
|
_notifyListChanged() {
|
|
this._onListChanged();
|
|
|
|
if (this._onListChanged !== null) {
|
|
swm.removeListener(this);
|
|
}
|
|
this._mustNotify = false;
|
|
},
|
|
|
|
onRegister(registration) {
|
|
this._notifyListChanged();
|
|
},
|
|
|
|
onUnregister(registration) {
|
|
this._notifyListChanged();
|
|
}
|
|
};
|
|
|
|
exports.ServiceWorkerRegistrationActorList = ServiceWorkerRegistrationActorList;
|