1
0
Fork 0
firefox/remote/shared/messagehandler/transports/BrowsingContextUtils.sys.mjs
Daniel Baumann 5e9a113729
Adding upstream version 140.0.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
2025-06-25 09:37:52 +02:00

177 lines
5.8 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/. */
const lazy = {};
ChromeUtils.defineESModuleGetters(lazy, {
error: "chrome://remote/content/shared/messagehandler/Errors.sys.mjs",
PollPromise: "chrome://remote/content/shared/Sync.sys.mjs",
});
/**
* A browsing context might be replaced before reaching the parent process,
* instead we serialize enough information to retrieve the navigable in the
* parent process.
*
* If the browsing context is top level, then the browserId can be used to
* find the browser element and the new browsing context.
* Otherwise (frames) the browsing context should not be replaced and the
* browsing context id should be enough to find the browsing context.
*
* Should be used when preparing an event payload from the content to the
* parent process.
*
* @param {BrowsingContext} browsingContext
* The browsing context for which we want to get details.
* @returns {object}
* An object that returns the following properties:
* - browserId: browser id for this browsing context
* - browsingContextId: browsing context id
* - isTopBrowsingContext: flag that indicates if the browsing context is
* top level
*/
export function getBrowsingContextDetails(browsingContext) {
return {
browserId: browsingContext.browserId,
browsingContextId: browsingContext.id,
isTopBrowsingContext: browsingContext.parent === null,
};
}
function isExtensionContext(browsingContext) {
let principal;
try {
if (CanonicalBrowsingContext.isInstance(browsingContext)) {
principal = browsingContext.currentWindowGlobal.documentPrincipal;
} else {
principal = browsingContext.window.document.nodePrincipal;
}
} catch (e) {
throw new Error(
`Could not retrieve principal for browsingContext (${e.message})`
);
}
// In practice, note that the principal will never be an expanded principal.
// The are only used for content scripts executed in a Sandbox, and do not
// have a browsing context on their own.
// But we still use this flag because there is no isAddonPrincipal flag.
return principal.isAddonOrExpandedAddonPrincipal;
}
function isParentProcess(browsingContext) {
if (CanonicalBrowsingContext.isInstance(browsingContext)) {
return browsingContext.currentWindowGlobal.osPid === -1;
}
// If `browsingContext` is not a `CanonicalBrowsingContext`, then we are
// necessarily in a content process page.
return false;
}
/**
* Check if the provided browsing context is currently displaying its initial
* document. For top level browsing contexts, this is usually the initial
* about:blank which will be replaced soon.
*
* @param {BrowsingContext} browsingContext
* The browsing context to check.
*
* @returns {boolean}
* True if the browsing context is on the initial document, false otherwise.
*/
export function isInitialDocument(browsingContext) {
if (!browsingContext.currentWindowGlobal) {
// Right after a browsing context has been attached it could happen that
// no window global has been set yet. Consider this as nothing has been
// loaded yet.
return true;
}
return browsingContext.currentWindowGlobal.isInitialDocument;
}
/**
* Check if the given browsing context is valid for the message handler
* to use.
*
* @param {BrowsingContext} browsingContext
* The browsing context to check.
* @param {object=} options
* @param {string=} options.browserId
* The id of the browser to filter the browsing contexts by (optional).
* @param {string=} options.userContext
* The id of the user context to filter the browsing contexts by (optional).
*
* @returns {boolean}
* True if the browsing context is valid, false otherwise.
*/
export function isBrowsingContextCompatible(browsingContext, options = {}) {
const { browserId, userContext } = options;
// If a browserId was provided, skip browsing contexts which are not
// associated with this browserId.
if (browserId !== undefined && browsingContext.browserId !== browserId) {
return false;
}
// If a userContext was provided, skip browsing contexts which are not
// associated with this userContext.
if (
userContext !== undefined &&
browsingContext.originAttributes.userContextId !== userContext
) {
return false;
}
// If this is a CanonicalBrowsingContext but the currentWindowGlobal is not
// attached yet, skip it.
if (
CanonicalBrowsingContext.isInstance(browsingContext) &&
!browsingContext.currentWindowGlobal
) {
return false;
}
// Skip:
// - extension contexts until we support debugging webextensions, see Bug 1755014.
// - privileged contexts until we support debugging Chrome context, see Bug 1713440.
return (
!isExtensionContext(browsingContext) && !isParentProcess(browsingContext)
);
}
/**
* Wait until `currentWindowGlobal` is available on a browsing context. When a
* browsing context has just been created, the `currentWindowGlobal` might not
* be attached yet.
*
* @param {CanonicalBrowsingContext} browsingContext
* The browsing context to wait for.
*
* @returns {Promise}
* Promise which resolves when `currentWindowGlobal` is set on the browsing
* context or throws a `DiscardedBrowsingContextError` error if it is still
* not available after 100ms.
*/
export async function waitForCurrentWindowGlobal(browsingContext) {
await lazy.PollPromise(
(resolve, reject) => {
if (browsingContext.currentWindowGlobal) {
resolve();
} else {
reject();
}
},
{
timeout: 100,
}
);
if (!browsingContext.currentWindowGlobal) {
throw new lazy.error.DiscardedBrowsingContextError(
`BrowsingContext does no longer exist`
);
}
}