mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-10-30 05:35:31 +00:00
281 lines
8.7 KiB
JavaScript
281 lines
8.7 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/. */
|
|
|
|
// TODO Bug 907060 Per off-line discussion, after the MessagePort is done
|
|
// at Bug 643325, we will start to refactorize the common logic of both
|
|
// Inter-App Communication and Shared Worker. For now, we hope to design an
|
|
// MozInterAppMessagePort to meet the timeline, which still follows exactly
|
|
// the same interface and semantic as the MessagePort is. In the future,
|
|
// we can then align it back to MessagePort with backward compatibility.
|
|
|
|
"use strict";
|
|
|
|
const { classes: Cc, interfaces: Ci, utils: Cu, results: Cr } = Components;
|
|
|
|
Cu.import("resource://gre/modules/Services.jsm");
|
|
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
|
|
Cu.import("resource://gre/modules/DOMRequestHelper.jsm");
|
|
|
|
const DEBUG = false;
|
|
function debug(aMsg) {
|
|
dump("-- InterAppMessagePort: " + Date.now() + ": " + aMsg + "\n");
|
|
}
|
|
|
|
XPCOMUtils.defineLazyServiceGetter(this, "cpmm",
|
|
"@mozilla.org/childprocessmessagemanager;1",
|
|
"nsIMessageSender");
|
|
|
|
XPCOMUtils.defineLazyServiceGetter(this, "appsService",
|
|
"@mozilla.org/AppsService;1",
|
|
"nsIAppsService");
|
|
|
|
const kMessages = ["InterAppMessagePort:OnClose",
|
|
"InterAppMessagePort:OnMessage",
|
|
"InterAppMessagePort:Shutdown"];
|
|
|
|
function InterAppMessagePort() {
|
|
if (DEBUG) debug("InterAppMessagePort()");
|
|
};
|
|
|
|
InterAppMessagePort.prototype = {
|
|
__proto__: DOMRequestIpcHelper.prototype,
|
|
|
|
classDescription: "MozInterAppMessagePort",
|
|
|
|
classID: Components.ID("{c66e0f8c-e3cb-11e2-9e85-43ef6244b884}"),
|
|
|
|
contractID: "@mozilla.org/dom/inter-app-message-port;1",
|
|
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIDOMGlobalPropertyInitializer,
|
|
Ci.nsISupportsWeakReference,
|
|
Ci.nsIObserver]),
|
|
|
|
// Ci.nsIDOMGlobalPropertyInitializer implementation.
|
|
init: function(aWindow) {
|
|
if (DEBUG) debug("Calling init().");
|
|
|
|
this.initDOMRequestHelper(aWindow, kMessages);
|
|
|
|
let principal = aWindow.document.nodePrincipal;
|
|
this._manifestURL = appsService.getManifestURLByLocalId(principal.appId);
|
|
this._pageURL = principal.URI.specIgnoringRef;
|
|
|
|
// Remove query string.
|
|
this._pageURL = this._pageURL.split("?")[0];
|
|
|
|
this._started = false;
|
|
this._closed = false;
|
|
this._messageQueue = [];
|
|
this._deferredClose = false;
|
|
},
|
|
|
|
// WebIDL implementation for constructor.
|
|
__init: function(aMessagePortID) {
|
|
if (DEBUG) {
|
|
debug("Calling __init(): aMessagePortID: " + aMessagePortID);
|
|
}
|
|
|
|
this._messagePortID = aMessagePortID;
|
|
|
|
cpmm.sendAsyncMessage("InterAppMessagePort:Register",
|
|
{ messagePortID: this._messagePortID,
|
|
manifestURL: this._manifestURL,
|
|
pageURL: this._pageURL });
|
|
},
|
|
|
|
// DOMRequestIpcHelper implementation.
|
|
uninit: function() {
|
|
if (DEBUG) debug("Calling uninit().");
|
|
|
|
// When the message port is uninitialized, we need to disentangle the
|
|
// coupling ports, as if the close() method had been called.
|
|
if (this._closed) {
|
|
if (DEBUG) debug("close() has been called. Don't need to close again.");
|
|
return;
|
|
}
|
|
|
|
this.close();
|
|
},
|
|
|
|
postMessage: function(aMessage) {
|
|
if (DEBUG) debug("Calling postMessage().");
|
|
|
|
if (this._closed) {
|
|
if (DEBUG) debug("close() has been called. Cannot post message.");
|
|
return;
|
|
}
|
|
|
|
cpmm.sendAsyncMessage("InterAppMessagePort:PostMessage",
|
|
{ messagePortID: this._messagePortID,
|
|
manifestURL: this._manifestURL,
|
|
message: aMessage });
|
|
},
|
|
|
|
start: function() {
|
|
// Begin dispatching messages received on the port.
|
|
if (DEBUG) debug("Calling start().");
|
|
|
|
if (this._closed) {
|
|
if (DEBUG) debug("close() has been called. Cannot call start().");
|
|
return;
|
|
}
|
|
|
|
if (this._started) {
|
|
if (DEBUG) debug("start() has been called. Don't need to start again.");
|
|
return;
|
|
}
|
|
|
|
// When a port's port message queue is enabled, the event loop must use it
|
|
// as one of its task sources.
|
|
this._started = true;
|
|
while (this._messageQueue.length) {
|
|
let message = this._messageQueue.shift();
|
|
this._dispatchMessage(message);
|
|
}
|
|
|
|
if (this._deferredClose) {
|
|
this._dispatchClose();
|
|
}
|
|
},
|
|
|
|
close: function() {
|
|
// Disconnecting the port, so that it is no longer active.
|
|
if (DEBUG) debug("Calling close().");
|
|
|
|
if (this._closed) {
|
|
if (DEBUG) debug("close() has been called. Don't need to close again.");
|
|
return;
|
|
}
|
|
|
|
this._closed = true;
|
|
this._deferredClose = false;
|
|
this._messageQueue.length = 0;
|
|
|
|
// When this method called on a local port that is entangled with another
|
|
// port, must cause the user agent to disentangle the coupling ports.
|
|
cpmm.sendAsyncMessage("InterAppMessagePort:Unregister",
|
|
{ messagePortID: this._messagePortID,
|
|
manifestURL: this._manifestURL });
|
|
|
|
this.removeMessageListeners(kMessages);
|
|
|
|
this._dispatchClose();
|
|
},
|
|
|
|
get onmessage() {
|
|
if (DEBUG) debug("Getting onmessage handler.");
|
|
|
|
return this.__DOM_IMPL__.getEventHandler("onmessage");
|
|
},
|
|
|
|
set onmessage(aHandler) {
|
|
if (DEBUG) debug("Setting onmessage handler.");
|
|
|
|
this.__DOM_IMPL__.setEventHandler("onmessage", aHandler);
|
|
|
|
// The first time a MessagePort object's onmessage IDL attribute is set,
|
|
// the port's message queue must be enabled, as if the start() method had
|
|
// been called.
|
|
if (this._started) {
|
|
if (DEBUG) debug("start() has been called. Don't need to start again.");
|
|
return;
|
|
}
|
|
|
|
this.start();
|
|
},
|
|
|
|
get onclose() {
|
|
if (DEBUG) debug("Getting onclose handler.");
|
|
return this.__DOM_IMPL__.getEventHandler("onclose");
|
|
},
|
|
|
|
set onclose(aHandler) {
|
|
if (DEBUG) debug("Setting onclose handler.");
|
|
this.__DOM_IMPL__.setEventHandler("onclose", aHandler);
|
|
},
|
|
|
|
_dispatchMessage: function _dispatchMessage(aMessage) {
|
|
let wrappedMessage = Cu.cloneInto(aMessage, this._window);
|
|
if (DEBUG) {
|
|
debug("_dispatchMessage: wrappedMessage: " +
|
|
JSON.stringify(wrappedMessage));
|
|
}
|
|
|
|
let event = new this._window
|
|
.MozInterAppMessageEvent("message",
|
|
{ data: wrappedMessage });
|
|
this.__DOM_IMPL__.dispatchEvent(event);
|
|
},
|
|
|
|
_dispatchClose() {
|
|
if (DEBUG) debug("_dispatchClose");
|
|
let event = new this._window.Event("close", {
|
|
bubbles: true,
|
|
cancelable: true
|
|
});
|
|
this.__DOM_IMPL__.dispatchEvent(event);
|
|
},
|
|
|
|
receiveMessage: function(aMessage) {
|
|
if (DEBUG) debug("receiveMessage: name: " + aMessage.name);
|
|
|
|
let message = aMessage.json;
|
|
if (message.manifestURL != this._manifestURL ||
|
|
message.pageURL != this._pageURL ||
|
|
message.messagePortID != this._messagePortID) {
|
|
if (DEBUG) debug("The message doesn't belong to this page. Returning. " +
|
|
uneval(message));
|
|
return;
|
|
}
|
|
|
|
switch (aMessage.name) {
|
|
case "InterAppMessagePort:OnMessage":
|
|
if (this._closed) {
|
|
if (DEBUG) debug("close() has been called. Drop the message.");
|
|
return;
|
|
}
|
|
|
|
if (!this._started) {
|
|
if (DEBUG) debug("Not yet called start(). Queue up the message.");
|
|
this._messageQueue.push(message.message);
|
|
return;
|
|
}
|
|
|
|
this._dispatchMessage(message.message);
|
|
break;
|
|
|
|
case "InterAppMessagePort:OnClose":
|
|
if (this._closed) {
|
|
if (DEBUG) debug("close() has been called. Drop the message.");
|
|
return;
|
|
}
|
|
|
|
// It is possible that one side of the port posts messages and calls
|
|
// close() before calling start() or setting the onmessage handler. In
|
|
// that case we need to queue the messages and defer the onclose event
|
|
// until the messages are delivered to the other side of the port.
|
|
if (!this._started) {
|
|
if (DEBUG) debug("Not yet called start(). Defer close notification.");
|
|
this._deferredClose = true;
|
|
return;
|
|
}
|
|
|
|
this._dispatchClose();
|
|
break;
|
|
|
|
case "InterAppMessagePort:Shutdown":
|
|
this.close();
|
|
break;
|
|
|
|
default:
|
|
dump("WARNING - Invalid InterAppMessagePort message type " +
|
|
aMessage.name + "\n");
|
|
break;
|
|
}
|
|
}
|
|
};
|
|
|
|
this.NSGetFactory = XPCOMUtils.generateNSGetFactory([InterAppMessagePort]);
|
|
|