Skip to content

Commit 3c7873b

Browse files
isaacsry
authored andcommitted
Use duck-typing as well as instanceof in sys.inspect
This makes it so that inspecting objects from other contexts works as expected.
1 parent 2fa4de0 commit 3c7873b

File tree

1 file changed

+30
-7
lines changed

1 file changed

+30
-7
lines changed

lib/sys.js

+30-7
Original file line numberDiff line numberDiff line change
@@ -65,21 +65,21 @@ exports.inspect = function (obj, showHidden, depth) {
6565

6666
// Functions without properties can be shortcutted.
6767
if (typeof value === 'function' && keys.length === 0) {
68-
if (value instanceof RegExp) {
68+
if (isRegExp(value)) {
6969
return '' + value;
7070
} else {
7171
return '[Function]';
7272
}
7373
}
7474

7575
// Dates without properties can be shortcutted
76-
if (value instanceof Date && keys.length === 0) {
76+
if (isDate(value) && keys.length === 0) {
7777
return value.toUTCString();
7878
}
7979

8080
var base, type, braces;
8181
// Determine the object type
82-
if (value instanceof Array) {
82+
if (isArray(value)) {
8383
type = 'Array';
8484
braces = ["[", "]"];
8585
} else {
@@ -89,13 +89,13 @@ exports.inspect = function (obj, showHidden, depth) {
8989

9090
// Make functions say that they are functions
9191
if (typeof value === 'function') {
92-
base = (value instanceof RegExp) ? ' ' + value : ' [Function]';
92+
base = (isRegExp(value)) ? ' ' + value : ' [Function]';
9393
} else {
9494
base = "";
9595
}
9696

9797
// Make dates with properties first say the date
98-
if (value instanceof Date) {
98+
if (isDate(value)) {
9999
base = ' ' + value.toUTCString();
100100
}
101101

@@ -106,7 +106,7 @@ exports.inspect = function (obj, showHidden, depth) {
106106
}
107107

108108
if (recurseTimes < 0) {
109-
if (value instanceof RegExp) {
109+
if (isRegExp(value)) {
110110
return '' + value;
111111
} else {
112112
return "[Object]";
@@ -140,7 +140,7 @@ exports.inspect = function (obj, showHidden, depth) {
140140
str = format(value[key], recurseTimes - 1);
141141
}
142142
if (str.indexOf('\n') > -1) {
143-
if (value instanceof Array) {
143+
if (isArray(value)) {
144144
str = str.split('\n').map(function(line) {
145145
return ' ' + line;
146146
}).join('\n').substr(2);
@@ -191,6 +191,29 @@ exports.inspect = function (obj, showHidden, depth) {
191191
}
192192
return format(obj, (typeof depth === 'undefined' ? 2 : depth));
193193
};
194+
function isArray (ar) {
195+
return ar instanceof Array
196+
|| Array.isArray(ar)
197+
|| (ar && ar !== Object.prototype && isArray(ar.__proto__));
198+
}
199+
function isRegExp (re) {
200+
var s = ""+re;
201+
return re instanceof RegExp // easy case
202+
|| typeof(re) === "function" // duck-type for context-switching evalcx case
203+
&& re.constructor.name === "RegExp"
204+
&& re.compile
205+
&& re.test
206+
&& re.exec
207+
&& s.charAt(0) === "/"
208+
&& s.substr(-1) === "/";
209+
}
210+
function isDate (d) {
211+
if (d instanceof Date) return true;
212+
if (typeof d !== "object") return false;
213+
var properties = Date.prototype && Object.getOwnPropertyNames(Date.prototype);
214+
var proto = d.__proto__ && Object.getOwnPropertyNames(d.__proto__);
215+
return JSON.stringify(proto) === JSON.stringify(properties);
216+
}
194217

195218
var pWarning;
196219

0 commit comments

Comments
 (0)