Skip to content

Commit f5ff88b

Browse files
BridgeARdanielleadams
authored andcommitted
util: pass through the inspect function to custom inspect functions
This allows to use more portable custom inspect functions. Fixes: #35956 Signed-off-by: Ruben Bridgewater <[email protected]> PR-URL: #41019 Reviewed-By: Michaël Zasso <[email protected]>
1 parent 1fc6fd6 commit f5ff88b

File tree

3 files changed

+35
-16
lines changed

3 files changed

+35
-16
lines changed

doc/api/util.md

+24-11
Original file line numberDiff line numberDiff line change
@@ -579,7 +579,7 @@ changes:
579579
codes. Colors are customizable. See [Customizing `util.inspect` colors][].
580580
**Default:** `false`.
581581
* `customInspect` {boolean} If `false`,
582-
`[util.inspect.custom](depth, opts)` functions are not invoked.
582+
`[util.inspect.custom](depth, opts, inspect)` functions are not invoked.
583583
**Default:** `true`.
584584
* `showProxy` {boolean} If `true`, `Proxy` inspection includes
585585
the [`target` and `handler`][] objects. **Default:** `false`.
@@ -874,10 +874,18 @@ ignored, if not supported.
874874

875875
<!-- type=misc -->
876876

877+
<!-- YAML
878+
added: v0.1.97
879+
changes:
880+
- version: REPLACEME
881+
pr-url: https://github.com/nodejs/node/pull/41019
882+
description: The inspect argument is added for more interoperability.
883+
-->
884+
877885
Objects may also define their own
878-
[`[util.inspect.custom](depth, opts)`][util.inspect.custom] function,
886+
[`[util.inspect.custom](depth, opts, inspect)`][util.inspect.custom] function,
879887
which `util.inspect()` will invoke and use the result of when inspecting
880-
the object:
888+
the object.
881889

882890
```js
883891
const util = require('util');
@@ -887,7 +895,7 @@ class Box {
887895
this.value = value;
888896
}
889897

890-
[util.inspect.custom](depth, options) {
898+
[util.inspect.custom](depth, options, inspect) {
891899
if (depth < 0) {
892900
return options.stylize('[Box]', 'special');
893901
}
@@ -898,8 +906,8 @@ class Box {
898906

899907
// Five space padding because that's the size of "Box< ".
900908
const padding = ' '.repeat(5);
901-
const inner = util.inspect(this.value, newOptions)
902-
.replace(/\n/g, `\n${padding}`);
909+
const inner = inspect(this.value, newOptions)
910+
.replace(/\n/g, `\n${padding}`);
903911
return `${options.stylize('Box', 'special')}< ${inner} >`;
904912
}
905913
}
@@ -910,9 +918,9 @@ util.inspect(box);
910918
// Returns: "Box< true >"
911919
```
912920

913-
Custom `[util.inspect.custom](depth, opts)` functions typically return a string
914-
but may return a value of any type that will be formatted accordingly by
915-
`util.inspect()`.
921+
Custom `[util.inspect.custom](depth, opts, inspect)` functions typically return
922+
a string but may return a value of any type that will be formatted accordingly
923+
by `util.inspect()`.
916924

917925
```js
918926
const util = require('util');
@@ -942,8 +950,13 @@ In addition to being accessible through `util.inspect.custom`, this
942950
symbol is [registered globally][global symbol registry] and can be
943951
accessed in any environment as `Symbol.for('nodejs.util.inspect.custom')`.
944952

953+
Using this allows code to be written in a portable fashion, so that the custom
954+
inspect function is used in an Node.js environment and ignored in the browser.
955+
The `util.inspect()` function itself is passed as third argument to the custom
956+
inspect function to allow further portability.
957+
945958
```js
946-
const inspect = Symbol.for('nodejs.util.inspect.custom');
959+
const customInspectSymbol = Symbol.for('nodejs.util.inspect.custom');
947960

948961
class Password {
949962
constructor(value) {
@@ -954,7 +967,7 @@ class Password {
954967
return 'xxxxxxxx';
955968
}
956969

957-
[inspect]() {
970+
[customInspectSymbol](depth, inspectOptions, inspect) {
958971
return `Password <${this.toString()}>`;
959972
}
960973
}

lib/internal/util/inspect.js

+7-2
Original file line numberDiff line numberDiff line change
@@ -778,7 +778,12 @@ function formatValue(ctx, value, recurseTimes, typedArray) {
778778
const isCrossContext =
779779
proxy !== undefined || !(context instanceof Object);
780780
const ret = FunctionPrototypeCall(
781-
maybeCustom, context, depth, getUserOptions(ctx, isCrossContext));
781+
maybeCustom,
782+
context,
783+
depth,
784+
getUserOptions(ctx, isCrossContext),
785+
inspect
786+
);
782787
// If the custom inspection method returned `this`, don't go into
783788
// infinite recursion.
784789
if (ret !== context) {
@@ -1143,7 +1148,7 @@ function getClassBase(value, constructor, tag) {
11431148

11441149
function getFunctionBase(value, constructor, tag) {
11451150
const stringified = FunctionPrototypeToString(value);
1146-
if (stringified.slice(0, 5) === 'class' && stringified.endsWith('}')) {
1151+
if (stringified.startsWith('class') && stringified.endsWith('}')) {
11471152
const slice = stringified.slice(5, -1);
11481153
const bracketIndex = slice.indexOf('{');
11491154
if (bracketIndex !== -1 &&

test/parallel/test-util-inspect.js

+4-3
Original file line numberDiff line numberDiff line change
@@ -974,7 +974,7 @@ util.inspect({ hasOwnProperty: null });
974974

975975
assert.strictEqual(util.inspect(subject), "{ foo: 'bar' }");
976976

977-
subject[util.inspect.custom] = common.mustCall((depth, opts) => {
977+
subject[util.inspect.custom] = common.mustCall((depth, opts, inspect) => {
978978
const clone = { ...opts };
979979
// This might change at some point but for now we keep the stylize function.
980980
// The function should either be documented or an alternative should be
@@ -984,12 +984,13 @@ util.inspect({ hasOwnProperty: null });
984984
assert.strictEqual(opts.budget, undefined);
985985
assert.strictEqual(opts.indentationLvl, undefined);
986986
assert.strictEqual(opts.showHidden, false);
987+
assert.strictEqual(inspect, util.inspect);
987988
assert.deepStrictEqual(
988-
new Set(Object.keys(util.inspect.defaultOptions).concat(['stylize'])),
989+
new Set(Object.keys(inspect.defaultOptions).concat(['stylize'])),
989990
new Set(Object.keys(opts))
990991
);
991992
opts.showHidden = true;
992-
return { [util.inspect.custom]: common.mustCall((depth, opts2) => {
993+
return { [inspect.custom]: common.mustCall((depth, opts2) => {
993994
assert.deepStrictEqual(clone, opts2);
994995
}) };
995996
});

0 commit comments

Comments
 (0)