diff --git a/README.md b/README.md index c75cd226..37fe9c26 100644 --- a/README.md +++ b/README.md @@ -15,20 +15,33 @@ interface SomeInterface { combined with the JavaScript implementation class file ```js -exports.implementation = class SomeInterfaceImpl { +module.exports = class SomeInterfaceImpl { add(x, y) { return x + y; } }; ``` +> Note: It's also possible to use ES2015 module default export syntax: +> +> ```js +> export default class SomeInterfaceImpl { +> add(x, y) { +> return x + y; +> } +> }; +> ``` + will generate a JavaScript wrapper class file roughly like this: ```js const conversions = require("webidl-conversions"); -const impl = require("./utils.js").implSymbol; +const utils = require("./utils.js"); +const impl = utils.implSymbol; -const Impl = require("./SomeInterface-impl.js").implementation; +// utils.importStar is roughly equivalent to Babel's _interopRequireWildcard() +// and TypeScript's __importStar() functions. +const Impl = utils.importStar(require("./SomeInterface-impl.js")).default; class SomeInterface { constructor() { diff --git a/lib/constructs/attribute.js b/lib/constructs/attribute.js index 8cce5411..7f4641e3 100644 --- a/lib/constructs/attribute.js +++ b/lib/constructs/attribute.js @@ -44,8 +44,8 @@ class Attribute { if (this.static) { brandCheck = ""; - getterBody = `return Impl.implementation["${this.idl.name}"];`; - setterBody = `Impl.implementation["${this.idl.name}"] = V;`; + getterBody = `return Impl.default["${this.idl.name}"];`; + setterBody = `Impl.default["${this.idl.name}"] = V;`; } else if (shouldReflect) { if (!reflector[this.idl.idlType.idlType]) { throw new Error("Unknown reflector type: " + this.idl.idlType.idlType); diff --git a/lib/constructs/interface.js b/lib/constructs/interface.js index 8a72c69b..f3c4f173 100644 --- a/lib/constructs/interface.js +++ b/lib/constructs/interface.js @@ -528,7 +528,7 @@ class Interface { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -541,7 +541,7 @@ class Interface { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -1162,7 +1162,7 @@ class Interface { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); `; diff --git a/lib/constructs/operation.js b/lib/constructs/operation.js index 3530add2..b3c44840 100644 --- a/lib/constructs/operation.js +++ b/lib/constructs/operation.js @@ -67,7 +67,7 @@ class Operation { `; } - const callOn = this.static ? "Impl.implementation" : "this[impl]"; + const callOn = this.static ? "Impl.default" : "this[impl]"; // In case of stringifiers, use the named implementation function rather than hardcoded "toString". // All overloads will have the same name, so pick the first one. const implFunc = this.idls[0].name || this.name; diff --git a/lib/output/utils.js b/lib/output/utils.js index 0b0b0d49..c1a4cba3 100644 --- a/lib/output/utils.js +++ b/lib/output/utils.js @@ -74,6 +74,52 @@ const namedSetNew = Symbol("named property set new"); const namedSetExisting = Symbol("named property set existing"); const namedDelete = Symbol("named property delete"); +/** + * Used to cache importStar results. + */ +const importStarCache = new WeakMap(); + +/** + * Mostly copied from TypeScript and Babel. + * + * @template T + * @param {T} obj + * @return {T extends { default: any } + * ? T : T extends string | number | bigint | boolean | symbol | null | undefined + * ? { default: T } : { default: T } & { [K in keyof T]: T[K] }} + */ +function importStar(obj) { + if (obj && obj.__esModule) { + return obj; + } + + if (obj === null || (typeof obj !== "object" && typeof obj !== "function")) { + return { default: obj }; + } + + if (importStarCache.has(obj)) { + return importStarCache.get(obj); + } + + const newObj = {}; + + for (const key in obj) { + if (hasOwn(obj, key) && key !== "default") { + Object.defineProperty( + newObj, + key, + Object.getOwnPropertyDescriptor(obj, key) + ); + } + } + + newObj.default = obj; + + importStarCache.set(obj, newObj); + + return newObj; +} + module.exports = exports = { isObject, hasOwn, @@ -97,5 +143,6 @@ module.exports = exports = { namedGet, namedSetNew, namedSetExisting, - namedDelete + namedDelete, + importStar }; diff --git a/lib/transformer.js b/lib/transformer.js index 4ce29fa4..8fec6992 100644 --- a/lib/transformer.js +++ b/lib/transformer.js @@ -212,7 +212,7 @@ class Transformer { const conversions = require("webidl-conversions"); const utils = require("${relativeUtils}"); ${source} - const Impl = require("${implFile}.js"); + const Impl = utils.importStar(require("${implFile}.js")); `; source = this._prettify(source); diff --git a/test/__snapshots__/test.js.snap b/test/__snapshots__/test.js.snap index 8f0bf689..850fd9c2 100644 --- a/test/__snapshots__/test.js.snap +++ b/test/__snapshots__/test.js.snap @@ -137,7 +137,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -150,7 +150,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -188,7 +188,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -205,7 +205,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/DOMImplementation.js\\"); +const Impl = utils.importStar(require(\\"../implementations/DOMImplementation.js\\")); " `; @@ -336,7 +336,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -349,7 +349,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -387,7 +387,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -404,7 +404,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/DictionaryConvert.js\\"); +const Impl = utils.importStar(require(\\"../implementations/DictionaryConvert.js\\")); " `; @@ -475,7 +475,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -488,7 +488,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -526,7 +526,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -543,7 +543,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/Enum.js\\"); +const Impl = utils.importStar(require(\\"../implementations/Enum.js\\")); " `; @@ -622,7 +622,7 @@ module.exports = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -646,7 +646,7 @@ module.exports = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -659,7 +659,7 @@ module.exports = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -680,7 +680,7 @@ module.exports = { } }; // module.exports -const Impl = require(\\"../implementations/Factory.js\\"); +const Impl = utils.importStar(require(\\"../implementations/Factory.js\\")); " `; @@ -705,7 +705,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -718,7 +718,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -841,7 +841,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -858,7 +858,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/Global.js\\"); +const Impl = utils.importStar(require(\\"../implementations/Global.js\\")); " `; @@ -895,7 +895,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -908,7 +908,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -946,7 +946,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -963,7 +963,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/LegacyArrayClass.js\\"); +const Impl = utils.importStar(require(\\"../implementations/LegacyArrayClass.js\\")); " `; @@ -1056,7 +1056,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -1069,7 +1069,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -1107,7 +1107,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -1124,7 +1124,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/MixedIn.js\\"); +const Impl = utils.importStar(require(\\"../implementations/MixedIn.js\\")); " `; @@ -1460,7 +1460,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -1473,7 +1473,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -1511,7 +1511,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -1528,7 +1528,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/Overloads.js\\"); +const Impl = utils.importStar(require(\\"../implementations/Overloads.js\\")); " `; @@ -1608,7 +1608,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -1621,7 +1621,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -1659,7 +1659,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -1676,7 +1676,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/PromiseTypes.js\\"); +const Impl = utils.importStar(require(\\"../implementations/PromiseTypes.js\\")); " `; @@ -1838,7 +1838,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -1851,7 +1851,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -1889,7 +1889,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -1906,7 +1906,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/Reflect.js\\"); +const Impl = utils.importStar(require(\\"../implementations/Reflect.js\\")); " `; @@ -2168,7 +2168,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -2181,7 +2181,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -2219,7 +2219,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -2236,7 +2236,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/SeqAndRec.js\\"); +const Impl = utils.importStar(require(\\"../implementations/SeqAndRec.js\\")); " `; @@ -2280,15 +2280,15 @@ class Static { } static def() { - return Impl.implementation.def(); + return Impl.default.def(); } static get abc() { - return Impl.implementation[\\"abc\\"]; + return Impl.default[\\"abc\\"]; } static set abc(V) { - return Impl.implementation[\\"abc\\"]; + return Impl.default[\\"abc\\"]; } } Object.defineProperties(Static.prototype, { @@ -2304,7 +2304,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -2317,7 +2317,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -2355,7 +2355,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -2372,7 +2372,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/Static.js\\"); +const Impl = utils.importStar(require(\\"../implementations/Static.js\\")); " `; @@ -2504,7 +2504,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -2517,7 +2517,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -2555,7 +2555,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -2736,7 +2736,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/Storage.js\\"); +const Impl = utils.importStar(require(\\"../implementations/Storage.js\\")); " `; @@ -2780,7 +2780,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -2793,7 +2793,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -2831,7 +2831,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -2848,7 +2848,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/StringifierAttribute.js\\"); +const Impl = utils.importStar(require(\\"../implementations/StringifierAttribute.js\\")); " `; @@ -2884,7 +2884,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -2897,7 +2897,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -2935,7 +2935,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -2952,7 +2952,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/StringifierDefaultOperation.js\\"); +const Impl = utils.importStar(require(\\"../implementations/StringifierDefaultOperation.js\\")); " `; @@ -2997,7 +2997,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -3010,7 +3010,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -3048,7 +3048,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -3065,7 +3065,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/StringifierNamedOperation.js\\"); +const Impl = utils.importStar(require(\\"../implementations/StringifierNamedOperation.js\\")); " `; @@ -3101,7 +3101,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -3114,7 +3114,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -3152,7 +3152,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -3169,7 +3169,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/StringifierOperation.js\\"); +const Impl = utils.importStar(require(\\"../implementations/StringifierOperation.js\\")); " `; @@ -3609,7 +3609,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -3622,7 +3622,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -3660,7 +3660,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -3677,7 +3677,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/TypedefsAndUnions.js\\"); +const Impl = utils.importStar(require(\\"../implementations/TypedefsAndUnions.js\\")); " `; @@ -3947,7 +3947,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -3960,7 +3960,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -3998,7 +3998,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -4016,7 +4016,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/URL.js\\"); +const Impl = utils.importStar(require(\\"../implementations/URL.js\\")); " `; @@ -4077,7 +4077,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -4090,7 +4090,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -4128,7 +4128,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -4305,7 +4305,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/URLList.js\\"); +const Impl = utils.importStar(require(\\"../implementations/URLList.js\\")); " `; @@ -4664,7 +4664,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -4677,7 +4677,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -4724,7 +4724,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -4742,7 +4742,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/URLSearchParams.js\\"); +const Impl = utils.importStar(require(\\"../implementations/URLSearchParams.js\\")); " `; @@ -4827,7 +4827,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -4840,7 +4840,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -4878,7 +4878,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -5081,7 +5081,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/URLSearchParamsCollection.js\\"); +const Impl = utils.importStar(require(\\"../implementations/URLSearchParamsCollection.js\\")); " `; @@ -5111,7 +5111,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -5124,7 +5124,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -5164,7 +5164,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -5396,7 +5396,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/URLSearchParamsCollection2.js\\"); +const Impl = utils.importStar(require(\\"../implementations/URLSearchParamsCollection2.js\\")); " `; @@ -5485,7 +5485,7 @@ class UnderscoredProperties { }); args.push(curArg); } - return Impl.implementation.static(...args); + return Impl.default.static(...args); } } Object.defineProperties(UnderscoredProperties.prototype, { @@ -5505,7 +5505,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -5518,7 +5518,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -5556,7 +5556,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -5573,7 +5573,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/UnderscoredProperties.js\\"); +const Impl = utils.importStar(require(\\"../implementations/UnderscoredProperties.js\\")); " `; @@ -5598,7 +5598,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -5611,7 +5611,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -5734,7 +5734,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -5751,7 +5751,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/Unforgeable.js\\"); +const Impl = utils.importStar(require(\\"../implementations/Unforgeable.js\\")); " `; @@ -5778,7 +5778,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -5791,7 +5791,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -5844,7 +5844,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -6036,7 +6036,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/UnforgeableMap.js\\"); +const Impl = utils.importStar(require(\\"../implementations/UnforgeableMap.js\\")); " `; @@ -6106,7 +6106,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -6119,7 +6119,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -6157,7 +6157,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -6174,7 +6174,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/Unscopable.js\\"); +const Impl = utils.importStar(require(\\"../implementations/Unscopable.js\\")); " `; @@ -6349,7 +6349,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -6362,7 +6362,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -6400,7 +6400,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -6417,7 +6417,7 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/Variadic.js\\"); +const Impl = utils.importStar(require(\\"../implementations/Variadic.js\\")); " `; @@ -6444,7 +6444,7 @@ const iface = { _mixedIntoPredicates: [], is(obj) { if (obj) { - if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.implementation) { + if (utils.hasOwn(obj, impl) && obj[impl] instanceof Impl.default) { return true; } for (const isMixedInto of module.exports._mixedIntoPredicates) { @@ -6457,7 +6457,7 @@ const iface = { }, isImpl(obj) { if (obj) { - if (obj instanceof Impl.implementation) { + if (obj instanceof Impl.default) { return true; } @@ -6495,7 +6495,7 @@ const iface = { this._internalSetup(obj); Object.defineProperty(obj, impl, { - value: new Impl.implementation(constructorArgs, privateData), + value: new Impl.default(constructorArgs, privateData), configurable: true }); @@ -6512,6 +6512,6 @@ const iface = { }; // iface module.exports = iface; -const Impl = require(\\"../implementations/ZeroArgConstructor.js\\"); +const Impl = utils.importStar(require(\\"../implementations/ZeroArgConstructor.js\\")); " `; diff --git a/test/utils.test.js b/test/utils.test.js new file mode 100644 index 00000000..fd927f7d --- /dev/null +++ b/test/utils.test.js @@ -0,0 +1,127 @@ +// eslint-disable-next-line spaced-comment +/* eslint-env node, jest */ +/* global BigInt */ + +"use strict"; + +const utils = require("../lib/output/utils"); + +describe("utils.js", () => { + describe("importStar", () => { + test("importStar works on non-ES2015 modules", () => { + const exports = class Foo { + static doStuff() { + return "success"; + } + }; + + exports.init = function (instance, privateData) { + instance._privateData = privateData; + }; + + const Impl = utils.importStar(exports); + + expect(Impl).not.toBe(exports); + expect(Impl).not.toHaveProperty("prototype"); + expect(Impl).not.toHaveProperty("doStuff"); + expect(Impl).toHaveProperty("default"); + expect(Impl).toHaveProperty("init"); + expect(Impl).not.toHaveProperty("__esModule"); + + expect(Impl.default).toBe(exports); + expect(Impl.init).toBe(exports.init); + }); + + test("importStar works on ES2015 modules", () => { + const module = {}; + Object.defineProperty(module, "__esModule", { value: true }); + + module.default = class Foo { + static doStuff() { + return "success"; + } + }; + + module.init = function (instance, privateData) { + instance._privateData = privateData; + }; + + const Impl = utils.importStar(module); + + expect(Impl).toBe(module); + expect(Impl).toHaveProperty("default"); + expect(Impl).toHaveProperty("init"); + expect(Impl).toHaveProperty("__esModule"); + }); + + test("importStar caches the result for Object types", () => { + const exports = class Foo { + static doStuff() { + return "success"; + } + }; + + const Impl1 = utils.importStar(exports); + const Impl2 = utils.importStar(exports); + + expect(Impl1).toBe(Impl2); + }); + + test("importStar doesn't cache the result for primitive values", () => { + const exports = 12345; + + const Impl1 = utils.importStar(exports); + const Impl2 = utils.importStar(exports); + + expect(Impl1).not.toBe(Impl2); + expect(Impl1.default).toBe(Impl2.default); + }); + + describe("importStar works on primitive values:", () => { + const BIG_INT_PLACEHOLDER = "BigInt placeholder"; + + /** @type {Array} */ + const primitives = [ + 123, + "string", + Symbol.iterator, + true, + BIG_INT_PLACEHOLDER, + null, + undefined + ]; + + // eslint-disable-next-line new-cap, valid-typeof + if (typeof BigInt === "function" && typeof BigInt("123") === "bigint") { + primitives.splice( + primitives.indexOf(BIG_INT_PLACEHOLDER), + 1, + // eslint-disable-next-line new-cap + BigInt("123") + ); + } + + for (const primitive of primitives) { + /** @type {"string" | "number" | "bigint" | "boolean" | "symbol" | "null" | "undefined"} */ + let type = primitive === null ? "null" : typeof primitive; + + let testPrimitive = test; + + if (primitive === BIG_INT_PLACEHOLDER) { + // BigInt doesn't exist on Node 8, so we have to assign this manually: + type = "bigint"; + testPrimitive = test.skip; + } + + testPrimitive(type, () => { + const Impl = utils.importStar(primitive); + + expect(Impl).toEqual(expect.anything()); + expect(Impl).toHaveProperty("default"); + expect(typeof Impl.default).toEqual(typeof primitive); + expect(Impl.default).toEqual(primitive); + }); + } + }); + }); +});