summaryrefslogtreecommitdiffstats
path: root/dom/manifest/ManifestObtainer.jsm
diff options
context:
space:
mode:
Diffstat (limited to 'dom/manifest/ManifestObtainer.jsm')
-rw-r--r--dom/manifest/ManifestObtainer.jsm160
1 files changed, 160 insertions, 0 deletions
diff --git a/dom/manifest/ManifestObtainer.jsm b/dom/manifest/ManifestObtainer.jsm
new file mode 100644
index 000000000..88cba0d91
--- /dev/null
+++ b/dom/manifest/ManifestObtainer.jsm
@@ -0,0 +1,160 @@
+/* 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/.
+ */
+ /*
+ * ManifestObtainer is an implementation of:
+ * http://w3c.github.io/manifest/#obtaining
+ *
+ * Exposes 2 public method:
+ *
+ * .contentObtainManifest(aContent) - used in content process
+ * .browserObtainManifest(aBrowser) - used in browser/parent process
+ *
+ * both return a promise. If successful, you get back a manifest object.
+ *
+ * Import it with URL:
+ * 'chrome://global/content/manifestMessages.js'
+ *
+ * e10s IPC message from this components are handled by:
+ * dom/ipc/manifestMessages.js
+ *
+ * Which is injected into every browser instance via browser.js.
+ *
+ * exported ManifestObtainer
+ */
+/*globals Components, Task, PromiseMessage, XPCOMUtils, ManifestProcessor, BrowserUtils*/
+"use strict";
+const {
+ utils: Cu,
+ classes: Cc,
+ interfaces: Ci
+} = Components;
+Cu.import("resource://gre/modules/Task.jsm");
+Cu.import("resource://gre/modules/PromiseMessage.jsm");
+Cu.import("resource://gre/modules/XPCOMUtils.jsm");
+Cu.import("resource://gre/modules/ManifestProcessor.jsm");
+XPCOMUtils.defineLazyModuleGetter(this, "BrowserUtils", // jshint ignore:line
+ "resource://gre/modules/BrowserUtils.jsm");
+
+this.ManifestObtainer = { // jshint ignore:line
+ /**
+ * Public interface for obtaining a web manifest from a XUL browser, to use
+ * on the parent process.
+ * @param {XULBrowser} The browser to check for the manifest.
+ * @return {Promise<Object>} The processed manifest.
+ */
+ browserObtainManifest: Task.async(function* (aBrowser) {
+ const msgKey = "DOM:ManifestObtainer:Obtain";
+ if (!isXULBrowser(aBrowser)) {
+ throw new TypeError("Invalid input. Expected XUL browser.");
+ }
+ const mm = aBrowser.messageManager;
+ const {data: {success, result}} = yield PromiseMessage.send(mm, msgKey);
+ if (!success) {
+ const error = toError(result);
+ throw error;
+ }
+ return result;
+ }),
+ /**
+ * Public interface for obtaining a web manifest from a XUL browser.
+ * @param {Window} The content Window from which to extract the manifest.
+ * @return {Promise<Object>} The processed manifest.
+ */
+ contentObtainManifest: Task.async(function* (aContent) {
+ if (!aContent || isXULBrowser(aContent)) {
+ throw new TypeError("Invalid input. Expected a DOM Window.");
+ }
+ let manifest;
+ try {
+ manifest = yield fetchManifest(aContent);
+ } catch (err) {
+ throw err;
+ }
+ return manifest;
+ }
+)};
+
+function toError(aErrorClone) {
+ let error;
+ switch (aErrorClone.name) {
+ case "TypeError":
+ error = new TypeError();
+ break;
+ default:
+ error = new Error();
+ }
+ Object.getOwnPropertyNames(aErrorClone)
+ .forEach(name => error[name] = aErrorClone[name]);
+ return error;
+}
+
+function isXULBrowser(aBrowser) {
+ if (!aBrowser || !aBrowser.namespaceURI || !aBrowser.localName) {
+ return false;
+ }
+ const XUL = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul";
+ return (aBrowser.namespaceURI === XUL && aBrowser.localName === "browser");
+}
+
+/**
+ * Asynchronously processes the result of response after having fetched
+ * a manifest.
+ * @param {Response} aResp Response from fetch().
+ * @param {Window} aContentWindow The content window.
+ * @return {Promise<Object>} The processed manifest.
+ */
+const processResponse = Task.async(function* (aResp, aContentWindow) {
+ const badStatus = aResp.status < 200 || aResp.status >= 300;
+ if (aResp.type === "error" || badStatus) {
+ const msg =
+ `Fetch error: ${aResp.status} - ${aResp.statusText} at ${aResp.url}`;
+ throw new Error(msg);
+ }
+ const text = yield aResp.text();
+ const args = {
+ jsonText: text,
+ manifestURL: aResp.url,
+ docURL: aContentWindow.location.href
+ };
+ const manifest = ManifestProcessor.process(args);
+ return manifest;
+});
+
+/**
+ * Asynchronously fetches a web manifest.
+ * @param {Window} a The content Window from where to extract the manifest.
+ * @return {Promise<Object>}
+ */
+const fetchManifest = Task.async(function* (aWindow) {
+ if (!aWindow || aWindow.top !== aWindow) {
+ let msg = "Window must be a top-level browsing context.";
+ throw new Error(msg);
+ }
+ const elem = aWindow.document.querySelector("link[rel~='manifest']");
+ if (!elem || !elem.getAttribute("href")) {
+ let msg = `No manifest to fetch at ${aWindow.location}`;
+ throw new Error(msg);
+ }
+ // Throws on malformed URLs
+ const manifestURL = new aWindow.URL(elem.href, elem.baseURI);
+ const reqInit = {
+ mode: "cors"
+ };
+ if (elem.crossOrigin === "use-credentials") {
+ reqInit.credentials = "include";
+ }
+ const request = new aWindow.Request(manifestURL, reqInit);
+ request.overrideContentPolicyType(Ci.nsIContentPolicy.TYPE_WEB_MANIFEST);
+ let response;
+ try {
+ response = yield aWindow.fetch(request);
+ } catch (err) {
+ throw err;
+ }
+ const manifest = yield processResponse(response, aWindow);
+ return manifest;
+});
+
+this.EXPORTED_SYMBOLS = ["ManifestObtainer"]; // jshint ignore:line