summaryrefslogtreecommitdiffstats
path: root/dom/push/Push.sys.mjs
diff options
context:
space:
mode:
Diffstat (limited to 'dom/push/Push.sys.mjs')
-rw-r--r--dom/push/Push.sys.mjs335
1 files changed, 335 insertions, 0 deletions
diff --git a/dom/push/Push.sys.mjs b/dom/push/Push.sys.mjs
new file mode 100644
index 0000000000..07f763040f
--- /dev/null
+++ b/dom/push/Push.sys.mjs
@@ -0,0 +1,335 @@
+/* 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/. */
+
+import { XPCOMUtils } from "resource://gre/modules/XPCOMUtils.sys.mjs";
+
+import { DOMRequestIpcHelper } from "resource://gre/modules/DOMRequestHelper.sys.mjs";
+
+const lazy = {};
+
+ChromeUtils.defineLazyGetter(lazy, "console", () => {
+ let { ConsoleAPI } = ChromeUtils.importESModule(
+ "resource://gre/modules/Console.sys.mjs"
+ );
+ return new ConsoleAPI({
+ maxLogLevelPref: "dom.push.loglevel",
+ prefix: "Push",
+ });
+});
+
+XPCOMUtils.defineLazyServiceGetter(
+ lazy,
+ "PushService",
+ "@mozilla.org/push/Service;1",
+ "nsIPushService"
+);
+
+const PUSH_CID = Components.ID("{cde1d019-fad8-4044-b141-65fb4fb7a245}");
+
+/**
+ * The Push component runs in the child process and exposes the Push API
+ * to the web application. The PushService running in the parent process is the
+ * one actually performing all operations.
+ */
+export function Push() {
+ lazy.console.debug("Push()");
+}
+
+Push.prototype = {
+ __proto__: DOMRequestIpcHelper.prototype,
+
+ contractID: "@mozilla.org/push/PushManager;1",
+
+ classID: PUSH_CID,
+
+ QueryInterface: ChromeUtils.generateQI([
+ "nsIDOMGlobalPropertyInitializer",
+ "nsISupportsWeakReference",
+ "nsIObserver",
+ ]),
+
+ init(win) {
+ lazy.console.debug("init()");
+
+ this._window = win;
+
+ this.initDOMRequestHelper(win);
+
+ // Get the client principal from the window. This won't be null because the
+ // service worker should be available when accessing the push manager.
+ this._principal = win.clientPrincipal;
+
+ if (!this._principal) {
+ throw new Error(" The client principal of the window is not available");
+ }
+
+ try {
+ this._topLevelPrincipal = win.top.document.nodePrincipal;
+ } catch (error) {
+ // Accessing the top-level document might fails if cross-origin
+ this._topLevelPrincipal = undefined;
+ }
+ },
+
+ __init(scope) {
+ this._scope = scope;
+ },
+
+ askPermission() {
+ lazy.console.debug("askPermission()");
+
+ let hasValidTransientUserGestureActivation =
+ this._window.document.hasValidTransientUserGestureActivation;
+
+ return this.createPromise((resolve, reject) => {
+ // Test permission before requesting to support GeckoView:
+ // * GeckoViewPermissionChild wants to return early when requested without user activation
+ // before doing actual permission check:
+ // https://searchfox.org/mozilla-central/rev/0ba4632ee85679a1ccaf652df79c971fa7e9b9f7/mobile/android/actors/GeckoViewPermissionChild.sys.mjs#46-56
+ // which is partly because:
+ // * GeckoView test runner has no real permission check but just returns VALUE_ALLOW.
+ // https://searchfox.org/mozilla-central/rev/6e5b9a5a1edab13a1b2e2e90944b6e06b4d8149c/mobile/android/test_runner/src/main/java/org/mozilla/geckoview/test_runner/TestRunnerActivity.java#108-123
+ if (this._testPermission() === Ci.nsIPermissionManager.ALLOW_ACTION) {
+ resolve();
+ return;
+ }
+
+ let permissionDenied = () => {
+ reject(
+ new this._window.DOMException(
+ "User denied permission to use the Push API.",
+ "NotAllowedError"
+ )
+ );
+ };
+
+ if (
+ Services.prefs.getBoolPref("dom.push.testing.ignorePermission", false)
+ ) {
+ resolve();
+ return;
+ }
+
+ this._requestPermission(
+ hasValidTransientUserGestureActivation,
+ resolve,
+ permissionDenied
+ );
+ });
+ },
+
+ subscribe(options) {
+ lazy.console.debug("subscribe()", this._scope);
+
+ return this.askPermission().then(() =>
+ this.createPromise((resolve, reject) => {
+ let callback = new PushSubscriptionCallback(this, resolve, reject);
+
+ if (!options || options.applicationServerKey === null) {
+ lazy.PushService.subscribe(this._scope, this._principal, callback);
+ return;
+ }
+
+ let keyView = this._normalizeAppServerKey(options.applicationServerKey);
+ if (keyView.byteLength === 0) {
+ callback._rejectWithError(Cr.NS_ERROR_DOM_PUSH_INVALID_KEY_ERR);
+ return;
+ }
+ lazy.PushService.subscribeWithKey(
+ this._scope,
+ this._principal,
+ keyView,
+ callback
+ );
+ })
+ );
+ },
+
+ _normalizeAppServerKey(appServerKey) {
+ let key;
+ if (typeof appServerKey == "string") {
+ try {
+ key = Cu.cloneInto(
+ ChromeUtils.base64URLDecode(appServerKey, {
+ padding: "reject",
+ }),
+ this._window
+ );
+ } catch (e) {
+ throw new this._window.DOMException(
+ "String contains an invalid character",
+ "InvalidCharacterError"
+ );
+ }
+ } else if (this._window.ArrayBuffer.isView(appServerKey)) {
+ key = appServerKey.buffer;
+ } else {
+ // `appServerKey` is an array buffer.
+ key = appServerKey;
+ }
+ return new this._window.Uint8Array(key);
+ },
+
+ getSubscription() {
+ lazy.console.debug("getSubscription()", this._scope);
+
+ return this.createPromise((resolve, reject) => {
+ let callback = new PushSubscriptionCallback(this, resolve, reject);
+ lazy.PushService.getSubscription(this._scope, this._principal, callback);
+ });
+ },
+
+ permissionState() {
+ lazy.console.debug("permissionState()", this._scope);
+
+ return this.createPromise((resolve, reject) => {
+ let permission = Ci.nsIPermissionManager.UNKNOWN_ACTION;
+
+ try {
+ permission = this._testPermission();
+ } catch (e) {
+ reject();
+ return;
+ }
+
+ let pushPermissionStatus = "prompt";
+ if (permission == Ci.nsIPermissionManager.ALLOW_ACTION) {
+ pushPermissionStatus = "granted";
+ } else if (permission == Ci.nsIPermissionManager.DENY_ACTION) {
+ pushPermissionStatus = "denied";
+ }
+ resolve(pushPermissionStatus);
+ });
+ },
+
+ _testPermission() {
+ let permission = Services.perms.testExactPermissionFromPrincipal(
+ this._principal,
+ "desktop-notification"
+ );
+ if (permission == Ci.nsIPermissionManager.ALLOW_ACTION) {
+ return permission;
+ }
+ try {
+ if (Services.prefs.getBoolPref("dom.push.testing.ignorePermission")) {
+ permission = Ci.nsIPermissionManager.ALLOW_ACTION;
+ }
+ } catch (e) {}
+ return permission;
+ },
+
+ _requestPermission(
+ hasValidTransientUserGestureActivation,
+ allowCallback,
+ cancelCallback
+ ) {
+ // Create an array with a single nsIContentPermissionType element.
+ let type = {
+ type: "desktop-notification",
+ options: [],
+ QueryInterface: ChromeUtils.generateQI(["nsIContentPermissionType"]),
+ };
+ let typeArray = Cc["@mozilla.org/array;1"].createInstance(
+ Ci.nsIMutableArray
+ );
+ typeArray.appendElement(type);
+
+ // create a nsIContentPermissionRequest
+ let request = {
+ QueryInterface: ChromeUtils.generateQI(["nsIContentPermissionRequest"]),
+ types: typeArray,
+ principal: this._principal,
+ hasValidTransientUserGestureActivation,
+ topLevelPrincipal: this._topLevelPrincipal,
+ allow: allowCallback,
+ cancel: cancelCallback,
+ window: this._window,
+ };
+
+ // Using askPermission from nsIDOMWindowUtils that takes care of the
+ // remoting if needed.
+ let windowUtils = this._window.windowUtils;
+ windowUtils.askPermission(request);
+ },
+};
+
+function PushSubscriptionCallback(pushManager, resolve, reject) {
+ this.pushManager = pushManager;
+ this.resolve = resolve;
+ this.reject = reject;
+}
+
+PushSubscriptionCallback.prototype = {
+ QueryInterface: ChromeUtils.generateQI(["nsIPushSubscriptionCallback"]),
+
+ onPushSubscription(ok, subscription) {
+ let { pushManager } = this;
+ if (!Components.isSuccessCode(ok)) {
+ this._rejectWithError(ok);
+ return;
+ }
+
+ if (!subscription) {
+ this.resolve(null);
+ return;
+ }
+
+ let p256dhKey = this._getKey(subscription, "p256dh");
+ let authSecret = this._getKey(subscription, "auth");
+ let options = {
+ endpoint: subscription.endpoint,
+ scope: pushManager._scope,
+ p256dhKey,
+ authSecret,
+ };
+ let appServerKey = this._getKey(subscription, "appServer");
+ if (appServerKey) {
+ // Avoid passing null keys to work around bug 1256449.
+ options.appServerKey = appServerKey;
+ }
+ let sub = new pushManager._window.PushSubscription(options);
+ this.resolve(sub);
+ },
+
+ _getKey(subscription, name) {
+ let rawKey = Cu.cloneInto(
+ subscription.getKey(name),
+ this.pushManager._window
+ );
+ if (!rawKey.length) {
+ return null;
+ }
+
+ let key = new this.pushManager._window.ArrayBuffer(rawKey.length);
+ let keyView = new this.pushManager._window.Uint8Array(key);
+ keyView.set(rawKey);
+ return key;
+ },
+
+ _rejectWithError(result) {
+ let error;
+ switch (result) {
+ case Cr.NS_ERROR_DOM_PUSH_INVALID_KEY_ERR:
+ error = new this.pushManager._window.DOMException(
+ "Invalid raw ECDSA P-256 public key.",
+ "InvalidAccessError"
+ );
+ break;
+
+ case Cr.NS_ERROR_DOM_PUSH_MISMATCHED_KEY_ERR:
+ error = new this.pushManager._window.DOMException(
+ "A subscription with a different application server key already exists.",
+ "InvalidStateError"
+ );
+ break;
+
+ default:
+ error = new this.pushManager._window.DOMException(
+ "Error retrieving push subscription.",
+ "AbortError"
+ );
+ }
+ this.reject(error);
+ },
+};