Skip to content

Commit 957fc0f

Browse files
committed
lib: promote process.binding/_tickCallback to runtime deprecation
These have been pendingDeprecation for long enough. This also adds the `--deprecated-process-binding` (and matching `--no-deprecated-process-binding`) CLI option as a future transition path for when we do ultimately remove `process.binding()`. `--deprecated-process-binding` ensures that `process.binding(...)` is available and is currently the default. In the future, when we are ready to begin moving `process.binding(...)` to EOF, we can flip the default so that `process.binding('...')` is unavailable by default *but still able to be switched back on if necessary*, giving users an escape hatch if they really need it. This will give us a more nuanced path than abruptly removing it completely.
1 parent 242cbd7 commit 957fc0f

File tree

8 files changed

+66
-14
lines changed

8 files changed

+66
-14
lines changed

doc/api/cli.md

+11
Original file line numberDiff line numberDiff line change
@@ -432,6 +432,12 @@ added: v12.0.0
432432
433433
Specify the file name of the CPU profile generated by `--cpu-prof`.
434434

435+
### `--deprecated-process-binding`
436+
437+
Ensure that the deprecated `process.binding(...)` API is available. This
438+
is currently the default. A runtime deprecation warning, however, will be
439+
emitted when `process.binding(...)` is accessed.
440+
435441
### `--diagnostic-dir=directory`
436442

437443
Set the directory to which all diagnostic output files are written.
@@ -1172,6 +1178,10 @@ Disable the `node-addons` exports condition as well as disable loading
11721178
native addons. When `--no-addons` is specified, calling `process.dlopen` or
11731179
requiring a native C++ addon will fail and throw an exception.
11741180

1181+
### `--no-deprecated-process-binding`
1182+
1183+
Makes the deprecated `process.binding(...)` API unavailable in the process.
1184+
11751185
### `--no-deprecation`
11761186

11771187
<!-- YAML
@@ -2325,6 +2335,7 @@ Node.js options that are allowed are:
23252335
* `--allow-fs-write`
23262336
* `--allow-worker`
23272337
* `--conditions`, `-C`
2338+
* `--deprecated-process-binding`
23282339
* `--diagnostic-dir`
23292340
* `--disable-proto`
23302341
* `--dns-result-order`

doc/api/deprecations.md

+8-2
Original file line numberDiff line numberDiff line change
@@ -2203,6 +2203,9 @@ The `produceCachedData` option is deprecated. Use
22032203

22042204
<!-- YAML
22052205
changes:
2206+
- version: REPLACEME
2207+
pr-url: https://github.com/nodejs/node/pull/50687
2208+
description: Upgraded to a runtime deprecation.
22062209
- version: v11.12.0
22072210
pr-url: https://github.com/nodejs/node/pull/26500
22082211
description: Added support for `--pending-deprecation`.
@@ -2211,7 +2214,7 @@ changes:
22112214
description: Documentation-only deprecation.
22122215
-->
22132216

2214-
Type: Documentation-only (supports [`--pending-deprecation`][])
2217+
Type: Runtime
22152218

22162219
`process.binding()` is for use by Node.js internal code only.
22172220

@@ -2604,12 +2607,15 @@ Prefer [`response.socket`][] over [`response.connection`][] and
26042607

26052608
<!-- YAML
26062609
changes:
2610+
- version: REPLACEME
2611+
pr-url: https://github.com/nodejs/node/pull/50687
2612+
description: Upgraded to a runtime deprecation.
26072613
- version: v12.12.0
26082614
pr-url: https://github.com/nodejs/node/pull/29781
26092615
description: Documentation-only deprecation.
26102616
-->
26112617

2612-
Type: Documentation-only (supports [`--pending-deprecation`][])
2618+
Type: Runtime
26132619

26142620
The `process._tickCallback` property was never documented as
26152621
an officially supported API.

lib/internal/bootstrap/realm.js

+3-1
Original file line numberDiff line numberDiff line change
@@ -77,7 +77,6 @@ ObjectDefineProperty(process, 'moduleLoadList', {
7777
writable: false,
7878
});
7979

80-
8180
// processBindingAllowList contains the name of bindings that are allowed
8281
// for access via process.binding(). This is used to provide a transition
8382
// path for modules that are being moved over to internalBinding.
@@ -144,6 +143,9 @@ const experimentalModuleList = new SafeSet();
144143
{
145144
const bindingObj = { __proto__: null };
146145

146+
// TODO(@jasnell): If the --deprecated-process-binding option is false we really
147+
// ought to skip setting this here. However, we cannot require internal/options or
148+
// use internalBinding here so we'll have to pull it back out later.
147149
process.binding = function binding(module) {
148150
module = String(module);
149151
// Deprecated specific process.binding() modules, but not all, allow

lib/internal/process/policy.js

+9-3
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,10 @@ const {
1111
ERR_MANIFEST_TDZ,
1212
} = require('internal/errors').codes;
1313
const { Manifest } = require('internal/policy/manifest');
14+
15+
const { getOptionValue } = require('internal/options');
16+
const deprecatedProcessBinding = getOptionValue('--deprecated-process-binding');
17+
1418
let manifest;
1519
let manifestSrc;
1620
let manifestURL;
@@ -36,9 +40,11 @@ module.exports = ObjectFreeze({
3640

3741
// process.binding() is deprecated (DEP0111) and trivially allows bypassing
3842
// policies, so if policies are enabled, make this API unavailable.
39-
process.binding = function binding(_module) {
40-
throw new ERR_ACCESS_DENIED('process.binding');
41-
};
43+
if (deprecatedProcessBinding) {
44+
process.binding = function binding(_module) {
45+
throw new ERR_ACCESS_DENIED('process.binding');
46+
};
47+
}
4248
process._linkedBinding = function _linkedBinding(_module) {
4349
throw new ERR_ACCESS_DENIED('process._linkedBinding');
4450
};

lib/internal/process/pre_execution.js

+15-8
Original file line numberDiff line numberDiff line change
@@ -545,15 +545,20 @@ function initializeDeprecations() {
545545
});
546546
}
547547

548-
if (pendingDeprecation) {
548+
if (getOptionValue('--deprecated-process-binding')) {
549549
process.binding = deprecate(process.binding,
550550
'process.binding() is deprecated. ' +
551551
'Please use public APIs instead.', 'DEP0111');
552-
553-
process._tickCallback = deprecate(process._tickCallback,
554-
'process._tickCallback() is deprecated',
555-
'DEP0134');
552+
} else {
553+
// Ideally it wouldn't be defined at all already but realm.js does not
554+
// provide a means of checking to see if the option is enabled or not,
555+
// so we do it here.
556+
process.binding = undefined;
556557
}
558+
559+
process._tickCallback = deprecate(process._tickCallback,
560+
'process._tickCallback() is deprecated',
561+
'DEP0134');
557562
}
558563

559564
function setupChildProcessIpcChannel() {
@@ -587,9 +592,11 @@ function initializeClusterIPC() {
587592
function initializePermission() {
588593
const experimentalPermission = getOptionValue('--experimental-permission');
589594
if (experimentalPermission) {
590-
process.binding = function binding(_module) {
591-
throw new ERR_ACCESS_DENIED('process.binding');
592-
};
595+
if (getOptionValue('--deprecated-process-binding') && process.binding !== undefined) {
596+
process.binding = function binding(_module) {
597+
throw new ERR_ACCESS_DENIED('process.binding');
598+
};
599+
}
593600
process.emitWarning('Permission is an experimental feature',
594601
'ExperimentalWarning');
595602
const { has, deny } = require('internal/process/permission');

src/node_options.cc

+5
Original file line numberDiff line numberDiff line change
@@ -535,6 +535,11 @@ EnvironmentOptionsParser::EnvironmentOptionsParser() {
535535
"emit pending deprecation warnings",
536536
&EnvironmentOptions::pending_deprecation,
537537
kAllowedInEnvvar);
538+
AddOption("--deprecated-process-binding",
539+
"enable the deprecated process.binding(...) api",
540+
&EnvironmentOptions::deprecated_process_binding,
541+
kAllowedInEnvvar,
542+
true);
538543
AddOption("--preserve-symlinks",
539544
"preserve symbolic links when resolving",
540545
&EnvironmentOptions::preserve_symlinks,

src/node_options.h

+10
Original file line numberDiff line numberDiff line change
@@ -188,6 +188,16 @@ class EnvironmentOptions : public Options {
188188
false;
189189
#endif // DEBUG
190190

191+
// TODO(@jasnell): In preparation for the eventual complete removal of
192+
// the deprecated process.binding, this option controls whether or not
193+
// process.binding is available. Currently the default is true, making
194+
// the deprecated API available, albeit with a runtime deprecation message.
195+
// Then, later, we can change the default to false, giving folks the option
196+
// to re-enable if they really need it. Then, as the final step, we would
197+
// remove the deprecated api entirely, in which case this option becomes
198+
// a non-op.
199+
bool deprecated_process_binding = true;
200+
191201
bool watch_mode = false;
192202
bool watch_mode_report_to_parent = false;
193203
bool watch_mode_preserve_output = false;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
'use strict';
2+
// Flags: --no-deprecated-process-binding
3+
require('../common');
4+
const { strictEqual } = require('node:assert');
5+
strictEqual(process.binding, undefined);

0 commit comments

Comments
 (0)