Skip to content

Commit 5498236

Browse files
MarshallOfSoundLukáš Budínský
authored and
Lukáš Budínský
committed
fix: expose the built-in electron module via the ESM loader (#35930)
1 parent f3956f4 commit 5498236

File tree

3 files changed

+100
-0
lines changed

3 files changed

+100
-0
lines changed

patches/node/.patches

+1
Original file line numberDiff line numberDiff line change
@@ -51,3 +51,4 @@ fixup_for_error_declaration_shadows_a_local_variable.patch
5151
fixup_for_wc_98-compat-extra-semi.patch
5252
drop_deserializerequest_move_constructor_for_c_20_compat.patch
5353
fix_parallel_test-v8-stats.patch
54+
fix_expose_the_built-in_electron_module_via_the_esm_loader.patch
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,87 @@
1+
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
2+
From: Samuel Attard <[email protected]>
3+
Date: Thu, 6 Oct 2022 04:09:16 -0700
4+
Subject: fix: expose the built-in electron module via the ESM loader
5+
6+
This allows usage of `import { app } from 'electron'` and `import('electron')` natively in the browser + non-sandboxed renderer
7+
8+
diff --git a/lib/internal/modules/esm/get_format.js b/lib/internal/modules/esm/get_format.js
9+
index 5ae0e17dcfb5e24a1a117c33c4d42891686e693f..619fe6cef3b02eb575410225f41d3e7d51f37b93 100644
10+
--- a/lib/internal/modules/esm/get_format.js
11+
+++ b/lib/internal/modules/esm/get_format.js
12+
@@ -31,6 +31,7 @@ const protocolHandlers = ObjectAssign(ObjectCreate(null), {
13+
'http:': getHttpProtocolModuleFormat,
14+
'https:': getHttpProtocolModuleFormat,
15+
'node:'() { return 'builtin'; },
16+
+ 'electron:'() { return 'commonjs'; },
17+
});
18+
19+
function getDataProtocolModuleFormat(parsed) {
20+
diff --git a/lib/internal/modules/esm/resolve.js b/lib/internal/modules/esm/resolve.js
21+
index fc5dcd6863dc358102a74bd2cc723d54436fae64..97eea17d815967671a2a0fc2a9c95a9bb85947ac 100644
22+
--- a/lib/internal/modules/esm/resolve.js
23+
+++ b/lib/internal/modules/esm/resolve.js
24+
@@ -883,6 +883,8 @@ function parsePackageName(specifier, base) {
25+
return { packageName, packageSubpath, isScoped };
26+
}
27+
28+
+const electronSpecifiers = new SafeSet(['electron', 'electron/main', 'electron/common', 'electron/renderer']);
29+
+
30+
/**
31+
* @param {string} specifier
32+
* @param {string | URL | undefined} base
33+
@@ -895,6 +897,10 @@ function packageResolve(specifier, base, conditions) {
34+
return new URL('node:' + specifier);
35+
}
36+
37+
+ if (electronSpecifiers.has(specifier)) {
38+
+ return new URL('electron:electron');
39+
+ }
40+
+
41+
const { packageName, packageSubpath, isScoped } =
42+
parsePackageName(specifier, base);
43+
44+
@@ -1095,7 +1101,7 @@ function checkIfDisallowedImport(specifier, parsed, parsedParentURL) {
45+
46+
function throwIfUnsupportedURLProtocol(url) {
47+
if (url.protocol !== 'file:' && url.protocol !== 'data:' &&
48+
- url.protocol !== 'node:') {
49+
+ url.protocol !== 'node:' && url.protocol !== 'electron:') {
50+
throw new ERR_UNSUPPORTED_ESM_URL_SCHEME(url);
51+
}
52+
}
53+
diff --git a/lib/internal/modules/esm/translators.js b/lib/internal/modules/esm/translators.js
54+
index 8fb3c96f8dc4c535c3898755f7846ae24d9867c4..bda3ca0797e8f1b1d69295c2276c0f841cae99f2 100644
55+
--- a/lib/internal/modules/esm/translators.js
56+
+++ b/lib/internal/modules/esm/translators.js
57+
@@ -155,7 +155,7 @@ translators.set('commonjs', async function commonjsStrategy(url, source,
58+
59+
if (!cjsParse) await initCJSParse();
60+
const { module, exportNames } = cjsPreparseModuleExports(filename);
61+
- const namesWithDefault = exportNames.has('default') ?
62+
+ const namesWithDefault = filename === 'electron' ? ['default', ...Object.keys(module.exports)] : exportNames.has('default') ?
63+
[...exportNames] : ['default', ...exportNames];
64+
65+
return new ModuleWrap(url, undefined, namesWithDefault, function() {
66+
@@ -174,7 +174,7 @@ translators.set('commonjs', async function commonjsStrategy(url, source,
67+
}
68+
}
69+
70+
- for (const exportName of exportNames) {
71+
+ for (const exportName of namesWithDefault) {
72+
if (!ObjectPrototypeHasOwnProperty(exports, exportName) ||
73+
exportName === 'default')
74+
continue;
75+
diff --git a/lib/internal/url.js b/lib/internal/url.js
76+
index 22bff28595f6f5b109ae47c79aa1f5ac463ec6c3..d4eb2e044cc1152f48c92d0503f9216e3aad5f4b 100644
77+
--- a/lib/internal/url.js
78+
+++ b/lib/internal/url.js
79+
@@ -1432,6 +1432,8 @@ function fileURLToPath(path) {
80+
path = new URL(path);
81+
else if (!isURLInstance(path))
82+
throw new ERR_INVALID_ARG_TYPE('path', ['string', 'URL'], path);
83+
+ if (path.protocol === 'electron:')
84+
+ return 'electron';
85+
if (path.protocol !== 'file:')
86+
throw new ERR_INVALID_URL_SCHEME('file');
87+
return isWindows ? getPathFromURLWin32(path) : getPathFromURLPosix(path);

spec/modules-spec.ts

+12
Original file line numberDiff line numberDiff line change
@@ -174,4 +174,16 @@ describe('modules support', () => {
174174
});
175175
});
176176
});
177+
178+
describe('esm', () => {
179+
it('can load the built-in "electron" module via ESM import', async () => {
180+
await expect(import('electron')).to.eventually.be.ok();
181+
});
182+
183+
it('the built-in "electron" module loaded via ESM import has the same exports as the CJS module', async () => {
184+
const esmElectron = await import('electron');
185+
const cjsElectron = require('electron');
186+
expect(Object.keys(esmElectron)).to.deep.equal(Object.keys(cjsElectron));
187+
});
188+
});
177189
});

0 commit comments

Comments
 (0)