Skip to content

Commit bf8c687

Browse files
boneskullnaugtur
authored andcommitted
fix(compartment-mapper/node): support anonymous entrypoints
An "anonymous" entrypoint is one where the closest `package.json` does not contain a `name` field. It is not unusual for CJS packages to create a subdirectory containing `package.json` like this: ```json {"type": "module"} ``` The effect is that any `.js` file in this subdir will be treated as an ECMAScript module by Node.js. A problem arises when a module in this subdir is provided as an entrypoint to `mapNodeModules()`. In this case, the "root" package descriptor _has no name_. Ultimately, the resulting shape of the `CompartmentMapDescriptor` fails `assertCompartmentMap()`, presumably due to an `undefined` value (or `"undefined"` string!) where one was not expected. To avoid the failure, we can assign a string value to the root package descriptor's `name` field. This is very much a corner case and was discovered by running arbitrary packages through `mapNodeModules()` (specifically, `@babel/runtime`). - Also added a test for this - Also added a broad/shallow test case for `mapNodeModules()` since one did not exist * * * Questions at time of commit: - Does the name need to be more unique to avoid collisions with other packages in the dependency tree? - Severe enough to pull in `node:crypto`? - Can there ever be more than one "anonymous" package?
1 parent c589ed0 commit bf8c687

File tree

5 files changed

+86
-0
lines changed

5 files changed

+86
-0
lines changed

packages/compartment-mapper/src/node-modules.js

+14
Original file line numberDiff line numberDiff line change
@@ -547,6 +547,11 @@ const gatherDependency = async (
547547
);
548548
};
549549

550+
/**
551+
* The name used by a root entry compartment that has no `name` in its package descriptor.
552+
*/
553+
export const ANONYMOUS_COMPARTMENT = '<ANONYMOUS>';
554+
550555
/**
551556
* graphPackages returns a graph whose keys are nominally URLs, one per
552557
* package, with values that are label: (an informative Compartment name, built
@@ -621,6 +626,15 @@ const graphPackages = async (
621626
};
622627
}
623628

629+
/**
630+
* If the entry package descriptor has no `name`, we need to put _something_
631+
* in there to pass compartment map validation (go find
632+
* `assertCompartmentMap()`).
633+
*/
634+
if (!packageDescriptor.name) {
635+
packageDescriptor.name = ANONYMOUS_COMPARTMENT;
636+
}
637+
624638
const graph = create(null);
625639
await graphPackage(
626640
packageDescriptor.name,

packages/compartment-mapper/test/fixtures-anonymous/node_modules/unnamed/incognito/index.js

+1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

packages/compartment-mapper/test/fixtures-anonymous/node_modules/unnamed/incognito/package.json

+3
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

packages/compartment-mapper/test/fixtures-anonymous/node_modules/unnamed/package.json

+11
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
import 'ses';
2+
import fs from 'node:fs';
3+
import url from 'node:url';
4+
import test from 'ava';
5+
import { ANONYMOUS_COMPARTMENT, mapNodeModules } from '../src/node-modules.js';
6+
import { makeReadPowers } from '../src/node-powers.js';
7+
8+
const { keys, values } = Object;
9+
10+
test('mapNodeModules() should fulfill with a denormalized CompartmentMapDescriptor', async t => {
11+
t.plan(4);
12+
13+
const readPowers = makeReadPowers({ fs, url });
14+
const moduleLocation = `${new URL(
15+
'fixtures-0/node_modules/bundle/main.js',
16+
import.meta.url,
17+
)}`;
18+
19+
const { compartments, entry } = await mapNodeModules(
20+
readPowers,
21+
moduleLocation,
22+
);
23+
24+
t.deepEqual(
25+
values(compartments)
26+
.map(({ name }) => name)
27+
.sort(),
28+
['bundle', 'bundle-dep'],
29+
);
30+
31+
t.true(keys(compartments).every(name => name.startsWith('file://')));
32+
33+
t.is(compartments[entry.compartment].name, 'bundle');
34+
35+
t.deepEqual(keys(compartments[entry.compartment].modules).sort(), [
36+
'.',
37+
'bundle',
38+
'bundle-dep',
39+
]);
40+
});
41+
42+
test(`mapNodeModules() should assign a package name when the entry point's package descriptor lacks a "name" field`, async t => {
43+
t.plan(1);
44+
45+
const readPowers = makeReadPowers({ fs, url });
46+
const moduleLocation = `${new URL(
47+
'fixtures-anonymous/node_modules/unnamed/incognito/index.js',
48+
import.meta.url,
49+
)}`;
50+
51+
const { compartments } = await mapNodeModules(readPowers, moduleLocation);
52+
53+
t.deepEqual(
54+
values(compartments).map(({ name }) => name),
55+
[ANONYMOUS_COMPARTMENT],
56+
);
57+
});

0 commit comments

Comments
 (0)