Bug 1790761 - Add do_importESModule. r=mccr8
Differential Revision: https://phabricator.services.mozilla.com/D157545
This commit is contained in:
@@ -77,5 +77,37 @@ nsresult ImportModule(const char* aURI, const char* aExportName,
|
|||||||
return nsXPConnect::XPConnect()->WrapJS(cx, exports, aIID, aResult);
|
return nsXPConnect::XPConnect()->WrapJS(cx, exports, aIID, aResult);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
nsresult ImportESModule(const char* aURI, const char* aExportName,
|
||||||
|
const nsIID& aIID, void** aResult, bool aInfallible) {
|
||||||
|
AutoJSAPI jsapi;
|
||||||
|
MOZ_ALWAYS_TRUE(jsapi.Init(xpc::PrivilegedJunkScope()));
|
||||||
|
JSContext* cx = jsapi.cx();
|
||||||
|
|
||||||
|
JS::RootedObject moduleNamespace(cx);
|
||||||
|
nsresult rv = mozJSModuleLoader::Get()->ImportESModule(
|
||||||
|
cx, nsDependentCString(aURI), &moduleNamespace);
|
||||||
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||||
|
if (aInfallible) {
|
||||||
|
AnnotateCrashReportWithJSException(cx, aURI);
|
||||||
|
|
||||||
|
MOZ_CRASH_UNSAFE_PRINTF("Failed to load critical module \"%s\"", aURI);
|
||||||
|
}
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (aExportName) {
|
||||||
|
JS::RootedValue namedExport(cx);
|
||||||
|
if (!JS_GetProperty(cx, moduleNamespace, aExportName, &namedExport)) {
|
||||||
|
return NS_ERROR_FAILURE;
|
||||||
|
}
|
||||||
|
if (!namedExport.isObject()) {
|
||||||
|
return NS_ERROR_XPC_BAD_CONVERT_JS;
|
||||||
|
}
|
||||||
|
moduleNamespace.set(&namedExport.toObject());
|
||||||
|
}
|
||||||
|
|
||||||
|
return nsXPConnect::XPConnect()->WrapJS(cx, moduleNamespace, aIID, aResult);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace loader
|
} // namespace loader
|
||||||
} // namespace mozilla
|
} // namespace mozilla
|
||||||
|
|||||||
@@ -19,6 +19,9 @@ namespace loader {
|
|||||||
nsresult ImportModule(const char* aURI, const char* aExportName,
|
nsresult ImportModule(const char* aURI, const char* aExportName,
|
||||||
const nsIID& aIID, void** aResult, bool aInfallible);
|
const nsIID& aIID, void** aResult, bool aInfallible);
|
||||||
|
|
||||||
|
nsresult ImportESModule(const char* aURI, const char* aExportName,
|
||||||
|
const nsIID& aIID, void** aResult, bool aInfallible);
|
||||||
|
|
||||||
} // namespace loader
|
} // namespace loader
|
||||||
} // namespace mozilla
|
} // namespace mozilla
|
||||||
|
|
||||||
@@ -131,4 +134,107 @@ inline nsImportModule do_ImportModule(const char (&aURI)[N],
|
|||||||
return {aURI, aExportName, aRv, /* infallible */ false};
|
return {aURI, aExportName, aRv, /* infallible */ false};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class MOZ_STACK_CLASS nsImportESModule final : public nsCOMPtr_helper {
|
||||||
|
public:
|
||||||
|
nsImportESModule(const char* aURI, const char* aExportName,
|
||||||
|
nsresult* aErrorPtr, bool aInfallible)
|
||||||
|
: mURI(aURI),
|
||||||
|
mExportName(aExportName),
|
||||||
|
mErrorPtr(aErrorPtr),
|
||||||
|
mInfallible(aInfallible) {
|
||||||
|
MOZ_ASSERT_IF(mErrorPtr, !mInfallible);
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual nsresult NS_FASTCALL operator()(const nsIID& aIID,
|
||||||
|
void** aResult) const override {
|
||||||
|
nsresult rv = ::mozilla::loader::ImportESModule(mURI, mExportName, aIID,
|
||||||
|
aResult, mInfallible);
|
||||||
|
if (mErrorPtr) {
|
||||||
|
*mErrorPtr = rv;
|
||||||
|
}
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
const char* mURI;
|
||||||
|
const char* mExportName;
|
||||||
|
nsresult* mErrorPtr;
|
||||||
|
bool mInfallible;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Usage with exported name:
|
||||||
|
*
|
||||||
|
* Foo.sys.mjs:
|
||||||
|
*
|
||||||
|
* export function foo(bar) {
|
||||||
|
* return bar.toString();
|
||||||
|
* }
|
||||||
|
*
|
||||||
|
* mozIFoo.idl:
|
||||||
|
*
|
||||||
|
* interface mozIFoo : nsISupports {
|
||||||
|
* AString foo(double meh);
|
||||||
|
* }
|
||||||
|
*
|
||||||
|
* Thing.cpp:
|
||||||
|
*
|
||||||
|
* nsCOMPtr<mozIFoo> foo = do_ImportESModule(
|
||||||
|
* "resource://meh/Foo.sys.mjs");
|
||||||
|
*
|
||||||
|
* MOZ_TRY(foo->Foo(42));
|
||||||
|
*
|
||||||
|
* Usage with a single named object:
|
||||||
|
*
|
||||||
|
* Foo.sys.mjs:
|
||||||
|
*
|
||||||
|
* export var Foo = {
|
||||||
|
* function foo(bar) {
|
||||||
|
* return bar.toString();
|
||||||
|
* }
|
||||||
|
* };
|
||||||
|
*
|
||||||
|
* Thing.cpp:
|
||||||
|
*
|
||||||
|
* nsCOMPtr<mozIFoo> foo = do_ImportESModule(
|
||||||
|
* "resource:://meh/Foo.sys.mjs", "Foo");
|
||||||
|
*/
|
||||||
|
|
||||||
|
template <size_t N>
|
||||||
|
inline nsImportESModule do_ImportESModule(const char (&aURI)[N]) {
|
||||||
|
return {aURI, nullptr, nullptr, /* infallible */ true};
|
||||||
|
}
|
||||||
|
|
||||||
|
template <size_t N>
|
||||||
|
inline nsImportESModule do_ImportESModule(const char (&aURI)[N],
|
||||||
|
const mozilla::fallible_t&) {
|
||||||
|
return {aURI, nullptr, nullptr, /* infallible */ false};
|
||||||
|
}
|
||||||
|
|
||||||
|
template <size_t N>
|
||||||
|
inline nsImportESModule do_ImportESModule(const char (&aURI)[N],
|
||||||
|
nsresult* aRv) {
|
||||||
|
return {aURI, nullptr, aRv, /* infallible */ false};
|
||||||
|
}
|
||||||
|
|
||||||
|
template <size_t N, size_t N2>
|
||||||
|
inline nsImportESModule do_ImportESModule(const char (&aURI)[N],
|
||||||
|
const char (&aExportName)[N2]) {
|
||||||
|
return {aURI, aExportName, nullptr, /* infallible */ true};
|
||||||
|
}
|
||||||
|
|
||||||
|
template <size_t N, size_t N2>
|
||||||
|
inline nsImportESModule do_ImportESModule(const char (&aURI)[N],
|
||||||
|
const char (&aExportName)[N2],
|
||||||
|
const mozilla::fallible_t&) {
|
||||||
|
return {aURI, aExportName, nullptr, /* infallible */ false};
|
||||||
|
}
|
||||||
|
|
||||||
|
template <size_t N, size_t N2>
|
||||||
|
inline nsImportESModule do_ImportESModule(const char (&aURI)[N],
|
||||||
|
const char (&aExportName)[N2],
|
||||||
|
nsresult* aRv) {
|
||||||
|
return {aURI, aExportName, aRv, /* infallible */ false};
|
||||||
|
}
|
||||||
|
|
||||||
#endif // defined nsImportModule_h
|
#endif // defined nsImportModule_h
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ EXPORTS += [
|
|||||||
UNIFIED_SOURCES += [
|
UNIFIED_SOURCES += [
|
||||||
"xpctest_attributes.cpp",
|
"xpctest_attributes.cpp",
|
||||||
"xpctest_cenums.cpp",
|
"xpctest_cenums.cpp",
|
||||||
|
"xpctest_esmreturncode.cpp",
|
||||||
"xpctest_module.cpp",
|
"xpctest_module.cpp",
|
||||||
"xpctest_params.cpp",
|
"xpctest_params.cpp",
|
||||||
"xpctest_returncode.cpp",
|
"xpctest_returncode.cpp",
|
||||||
|
|||||||
@@ -0,0 +1,20 @@
|
|||||||
|
/* 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/. */
|
||||||
|
|
||||||
|
#include "xpctest_private.h"
|
||||||
|
#include "nsComponentManagerUtils.h"
|
||||||
|
#include "nsImportModule.h"
|
||||||
|
|
||||||
|
NS_IMPL_ISUPPORTS(nsXPCTestESMReturnCodeParent, nsIXPCTestReturnCodeParent)
|
||||||
|
|
||||||
|
NS_IMETHODIMP nsXPCTestESMReturnCodeParent::CallChild(int32_t childBehavior,
|
||||||
|
nsresult* _retval) {
|
||||||
|
nsresult rv;
|
||||||
|
nsCOMPtr<nsIXPCTestReturnCodeChild> child(do_ImportESModule(
|
||||||
|
"resource://test/ReturnCodeChild.sys.mjs", "ReturnCodeChild", &rv));
|
||||||
|
NS_ENSURE_SUCCESS(rv, rv);
|
||||||
|
rv = child->DoIt(childBehavior);
|
||||||
|
*_retval = rv;
|
||||||
|
return NS_OK;
|
||||||
|
}
|
||||||
@@ -36,6 +36,8 @@ nsresult xpcTestRegisterComponents() {
|
|||||||
"@mozilla.org/js/xpc/test/native/Params;1"));
|
"@mozilla.org/js/xpc/test/native/Params;1"));
|
||||||
MOZ_TRY(RegisterFactory<nsXPCTestReturnCodeParent>(
|
MOZ_TRY(RegisterFactory<nsXPCTestReturnCodeParent>(
|
||||||
"@mozilla.org/js/xpc/test/native/ReturnCodeParent;1"));
|
"@mozilla.org/js/xpc/test/native/ReturnCodeParent;1"));
|
||||||
|
MOZ_TRY(RegisterFactory<nsXPCTestESMReturnCodeParent>(
|
||||||
|
"@mozilla.org/js/xpc/test/native/ESMReturnCodeParent;1"));
|
||||||
MOZ_TRY(RegisterFactory<xpcTestCEnums>(
|
MOZ_TRY(RegisterFactory<xpcTestCEnums>(
|
||||||
"@mozilla.org/js/xpc/test/native/CEnums;1"));
|
"@mozilla.org/js/xpc/test/native/CEnums;1"));
|
||||||
|
|
||||||
|
|||||||
@@ -79,6 +79,17 @@ class nsXPCTestReturnCodeParent final : public nsIXPCTestReturnCodeParent {
|
|||||||
~nsXPCTestReturnCodeParent() = default;
|
~nsXPCTestReturnCodeParent() = default;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class nsXPCTestESMReturnCodeParent final : public nsIXPCTestReturnCodeParent {
|
||||||
|
public:
|
||||||
|
NS_DECL_ISUPPORTS
|
||||||
|
NS_DECL_NSIXPCTESTRETURNCODEPARENT
|
||||||
|
|
||||||
|
nsXPCTestESMReturnCodeParent() = default;
|
||||||
|
|
||||||
|
private:
|
||||||
|
~nsXPCTestESMReturnCodeParent() = default;
|
||||||
|
};
|
||||||
|
|
||||||
class xpcTestCEnums final : public nsIXPCTestCEnums {
|
class xpcTestCEnums final : public nsIXPCTestCEnums {
|
||||||
public:
|
public:
|
||||||
NS_DECL_ISUPPORTS
|
NS_DECL_ISUPPORTS
|
||||||
|
|||||||
45
js/xpconnect/tests/idl/xpctest_esmreturncode.idl
Normal file
45
js/xpconnect/tests/idl/xpctest_esmreturncode.idl
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
||||||
|
*
|
||||||
|
* 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/. */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test the use of Components.returnCode with system ESM
|
||||||
|
*
|
||||||
|
* This ("parent") interface defines a method that in-turn calls another
|
||||||
|
* ("child") interface implemented in JS, and returns the nsresult from that
|
||||||
|
* child interface. The child interface manages the return code by way of
|
||||||
|
* Components.returnCode.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "nsISupports.idl"
|
||||||
|
|
||||||
|
|
||||||
|
[scriptable, uuid(494f9336-ad06-46ad-bbb4-b0010e27e12d)]
|
||||||
|
interface nsIXPCTestESMReturnCodeParent : nsISupports {
|
||||||
|
// Calls the "child" interface with the specified behavior flag. Returns
|
||||||
|
// the NSRESULT from the child interface.
|
||||||
|
nsresult callChild(in long childBehavior);
|
||||||
|
};
|
||||||
|
|
||||||
|
[scriptable, uuid(dee07408-75d8-4968-a37c-fe0d48ccd1ac)]
|
||||||
|
interface nsIXPCTestESMReturnCodeChild : nsISupports {
|
||||||
|
void doIt(in long behavior);
|
||||||
|
|
||||||
|
// Flags to control that the child does.
|
||||||
|
// child will throw a JS exception
|
||||||
|
const long CHILD_SHOULD_THROW = 0;
|
||||||
|
|
||||||
|
// child will just return normally
|
||||||
|
const long CHILD_SHOULD_RETURN_SUCCESS = 1;
|
||||||
|
|
||||||
|
// child will return after setting Components.returnCode to NS_ERROR_FAILURE
|
||||||
|
const long CHILD_SHOULD_RETURN_RESULTCODE = 2;
|
||||||
|
|
||||||
|
// child will set Components.returnCode to NS_ERROR_UNEXPECTED, then create
|
||||||
|
// a new component that sets Components.returnCode to NS_ERROR_FAILURE.
|
||||||
|
// Our caller should see the NS_ERROR_UNEXPECTED we set rather than the
|
||||||
|
// value set later by the "inner" child.
|
||||||
|
const long CHILD_SHOULD_NEST_RESULTCODES = 3;
|
||||||
|
};
|
||||||
@@ -20,7 +20,7 @@
|
|||||||
interface nsIXPCTestReturnCodeParent : nsISupports {
|
interface nsIXPCTestReturnCodeParent : nsISupports {
|
||||||
// Calls the "child" interface with the specified behavior flag. Returns
|
// Calls the "child" interface with the specified behavior flag. Returns
|
||||||
// the NSRESULT from the child interface.
|
// the NSRESULT from the child interface.
|
||||||
nsresult callChild(in long childBehavior);
|
nsresult callChild(in long childBehavior);
|
||||||
};
|
};
|
||||||
|
|
||||||
[scriptable, uuid(672cfd34-1fd1-455d-9901-d879fa6fdb95)]
|
[scriptable, uuid(672cfd34-1fd1-455d-9901-d879fa6fdb95)]
|
||||||
|
|||||||
49
js/xpconnect/tests/unit/ReturnCodeChild.sys.mjs
Normal file
49
js/xpconnect/tests/unit/ReturnCodeChild.sys.mjs
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
/* 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/. */
|
||||||
|
|
||||||
|
function xpcWrap(obj, iface) {
|
||||||
|
let ifacePointer = Cc[
|
||||||
|
"@mozilla.org/supports-interface-pointer;1"
|
||||||
|
].createInstance(Ci.nsISupportsInterfacePointer);
|
||||||
|
|
||||||
|
ifacePointer.data = obj;
|
||||||
|
return ifacePointer.data.QueryInterface(iface);
|
||||||
|
}
|
||||||
|
|
||||||
|
export var ReturnCodeChild = {
|
||||||
|
QueryInterface: ChromeUtils.generateQI(["nsIXPCTestReturnCodeChild"]),
|
||||||
|
|
||||||
|
doIt(behaviour) {
|
||||||
|
switch (behaviour) {
|
||||||
|
case Ci.nsIXPCTestReturnCodeChild.CHILD_SHOULD_THROW:
|
||||||
|
throw(new Error("a requested error"));
|
||||||
|
case Ci.nsIXPCTestReturnCodeChild.CHILD_SHOULD_RETURN_SUCCESS:
|
||||||
|
return;
|
||||||
|
case Ci.nsIXPCTestReturnCodeChild.CHILD_SHOULD_RETURN_RESULTCODE:
|
||||||
|
Components.returnCode = Cr.NS_ERROR_FAILURE;
|
||||||
|
return;
|
||||||
|
case Ci.nsIXPCTestReturnCodeChild.CHILD_SHOULD_NEST_RESULTCODES:
|
||||||
|
// Use xpconnect to create another instance of *this* component and
|
||||||
|
// call that. This way we have crossed the xpconnect bridge twice.
|
||||||
|
|
||||||
|
// We set *our* return code early - this should be what is returned
|
||||||
|
// to our caller, even though our "inner" component will set it to
|
||||||
|
// a different value that we will see (but our caller should not)
|
||||||
|
Components.returnCode = Cr.NS_ERROR_UNEXPECTED;
|
||||||
|
// call the child asking it to do the .returnCode set.
|
||||||
|
let sub = xpcWrap(ReturnCodeChild, Ci.nsIXPCTestReturnCodeChild);
|
||||||
|
let childResult = Cr.NS_OK;
|
||||||
|
try {
|
||||||
|
sub.doIt(Ci.nsIXPCTestReturnCodeChild.CHILD_SHOULD_RETURN_RESULTCODE);
|
||||||
|
} catch (ex) {
|
||||||
|
childResult = ex.result;
|
||||||
|
}
|
||||||
|
// write it to the console so the test can check it.
|
||||||
|
let consoleService = Cc["@mozilla.org/consoleservice;1"]
|
||||||
|
.getService(Ci.nsIConsoleService);
|
||||||
|
consoleService.logStringMessage("nested child returned " + childResult);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -15,13 +15,15 @@ function run_test() {
|
|||||||
registerXPCTestComponents();
|
registerXPCTestComponents();
|
||||||
|
|
||||||
// and the tests.
|
// and the tests.
|
||||||
test_simple();
|
test_simple("@mozilla.org/js/xpc/test/native/ReturnCodeParent;1");
|
||||||
test_nested();
|
test_nested("@mozilla.org/js/xpc/test/native/ReturnCodeParent;1");
|
||||||
|
|
||||||
|
test_simple("@mozilla.org/js/xpc/test/native/ESMReturnCodeParent;1");
|
||||||
|
test_nested("@mozilla.org/js/xpc/test/native/ESMReturnCodeParent;1");
|
||||||
}
|
}
|
||||||
|
|
||||||
function test_simple() {
|
function test_simple(contractID) {
|
||||||
let parent = Cc["@mozilla.org/js/xpc/test/native/ReturnCodeParent;1"]
|
let parent = Cc[contractID].createInstance(Ci.nsIXPCTestReturnCodeParent);
|
||||||
.createInstance(Ci.nsIXPCTestReturnCodeParent);
|
|
||||||
let result;
|
let result;
|
||||||
|
|
||||||
// flush existing messages before we start testing.
|
// flush existing messages before we start testing.
|
||||||
@@ -51,9 +53,8 @@ function test_simple() {
|
|||||||
Assert.deepEqual(getConsoleMessages(), [], "no messages reported with .returnCode");
|
Assert.deepEqual(getConsoleMessages(), [], "no messages reported with .returnCode");
|
||||||
}
|
}
|
||||||
|
|
||||||
function test_nested() {
|
function test_nested(contractID) {
|
||||||
let parent = Cc["@mozilla.org/js/xpc/test/native/ReturnCodeParent;1"]
|
let parent = Cc[contractID].createInstance(Ci.nsIXPCTestReturnCodeParent);
|
||||||
.createInstance(Ci.nsIXPCTestReturnCodeParent);
|
|
||||||
let result;
|
let result;
|
||||||
|
|
||||||
// flush existing messages before we start testing.
|
// flush existing messages before we start testing.
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ support-files =
|
|||||||
recursive_importA.jsm
|
recursive_importA.jsm
|
||||||
recursive_importB.jsm
|
recursive_importB.jsm
|
||||||
ReturnCodeChild.jsm
|
ReturnCodeChild.jsm
|
||||||
|
ReturnCodeChild.sys.mjs
|
||||||
syntax_error.jsm
|
syntax_error.jsm
|
||||||
uninitialized_lexical.jsm
|
uninitialized_lexical.jsm
|
||||||
es6module.js
|
es6module.js
|
||||||
|
|||||||
Reference in New Issue
Block a user