node/test/parallel/test-internal-module-require.js
Joyee Cheung 92e95f17b6
src: simplify NativeModule caching and remove redundant data
- Remove `NativeModule._source` - the compilation is now entirely
  done in C++ and `process.binding('natives')` is implemented
  directly in the binding loader so there is no need to store
  additional source code strings.
- Instead of using an object as `NativeModule._cached` and insert
  into it after compilation of each native module, simply prebuild
  a JS map filled with all the native modules and infer the
  state of compilation through `mod.loading`/`mod.loaded`.
- Rename `NativeModule.nonInternalExists` to
  `NativeModule.canBeRequiredByUsers` and precompute that
  property for all the native modules during bootstrap instead
  of branching in every require call during runtime. This also fixes
  the bug where `worker_threads` can be made available with
  `--expose-internals`.
- Rename `NativeModule.requireForDeps` to
  `NativeModule.requireWithFallbackInDeps`.
- Add a test to make sure we do not accidentally leak any module
  to the global namespace.

PR-URL: https://github.com/nodejs/node/pull/25352
Reviewed-By: Anna Henningsen <anna@addaleax.net>
2019-01-12 22:56:02 +08:00

113 lines
2.5 KiB
JavaScript

'use strict';
// Flags: --expose-internals
// This verifies that
// 1. We do not leak internal modules unless the --require-internals option
// is on.
// 2. We do not accidentally leak any modules to the public global scope.
// 3. Deprecated modules are properly deprecated.
const common = require('../common');
if (!common.isMainThread) {
common.skip('Cannot test the existence of --expose-internals from worker');
}
const assert = require('assert');
const fork = require('child_process').fork;
const expectedPublicModules = new Set([
'_http_agent',
'_http_client',
'_http_common',
'_http_incoming',
'_http_outgoing',
'_http_server',
'_stream_duplex',
'_stream_passthrough',
'_stream_readable',
'_stream_transform',
'_stream_wrap',
'_stream_writable',
'_tls_common',
'_tls_wrap',
'assert',
'async_hooks',
'buffer',
'child_process',
'cluster',
'console',
'constants',
'crypto',
'dgram',
'dns',
'domain',
'events',
'fs',
'http',
'http2',
'https',
'inspector',
'module',
'net',
'os',
'path',
'perf_hooks',
'process',
'punycode',
'querystring',
'readline',
'repl',
'stream',
'string_decoder',
'sys',
'timers',
'tls',
'trace_events',
'tty',
'url',
'util',
'v8',
'vm',
'worker_threads',
'zlib'
]);
if (process.argv[2] === 'child') {
assert(!process.execArgv.includes('--expose-internals'));
process.once('message', ({ allBuiltins }) => {
const publicModules = new Set();
for (const id of allBuiltins) {
if (id.startsWith('internal/')) {
common.expectsError(() => {
require(id);
}, {
code: 'MODULE_NOT_FOUND',
message: `Cannot find module '${id}'`
});
} else {
require(id);
publicModules.add(id);
}
}
assert(allBuiltins.length > publicModules.size);
// Make sure all the public modules are available through
// require('module').builtinModules
assert.deepStrictEqual(
publicModules,
new Set(require('module').builtinModules)
);
assert.deepStrictEqual(publicModules, expectedPublicModules);
});
} else {
assert(process.execArgv.includes('--expose-internals'));
const child = fork(__filename, ['child'], {
execArgv: []
});
const { builtinModules } = require('module');
// When --expose-internals is on, require('module').builtinModules
// contains internal modules.
const message = { allBuiltins: builtinModules };
child.send(message);
}