// Copyright 2008 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. (function() { "use strict"; // A more universal stringify that supports more types than JSON. // Used by the d8 shell to output results. var stringifyDepthLimit = 4; // To avoid crashing on cyclic objects // Hacky solution to circumvent forcing --allow-natives-syntax for d8 function isProxy(o) { return false }; function JSProxyGetTarget(proxy) { }; function JSProxyGetHandler(proxy) { }; try { isProxy = Function(['object'], 'return %_IsJSProxy(object)'); JSProxyGetTarget = Function(['proxy'], 'return %JSProxyGetTarget(proxy)'); JSProxyGetHandler = Function(['proxy'], 'return %JSProxyGetHandler(proxy)'); } catch(e) {}; function Stringify(x, depth) { if (depth === undefined) depth = stringifyDepthLimit; else if (depth === 0) return "..."; if (isProxy(x)) { return StringifyProxy(x, depth); } switch (typeof x) { case "undefined": return "undefined"; case "boolean": case "number": case "function": case "symbol": return x.toString(); case "string": return "\"" + x.toString() + "\""; case "bigint": // TODO(neis): Use x.toString() once we have it. return String(x) + "n"; case "object": if (IS_NULL(x)) return "null"; if (x.constructor && x.constructor.name === "Array") { var elems = []; for (var i = 0; i < x.length; ++i) { elems.push( {}.hasOwnProperty.call(x, i) ? Stringify(x[i], depth - 1) : ""); } return "[" + elems.join(", ") + "]"; } try { var string = String(x); if (string && string !== "[object Object]") return string; } catch(e) {} var props = []; var names = Object.getOwnPropertyNames(x); names = names.concat(Object.getOwnPropertySymbols(x)); for (var i in names) { var name = names[i]; var desc = Object.getOwnPropertyDescriptor(x, name); if (IS_UNDEFINED(desc)) continue; if (IS_SYMBOL(name)) name = "[" + Stringify(name) + "]"; if ("value" in desc) { props.push(name + ": " + Stringify(desc.value, depth - 1)); } if (desc.get) { var getter = Stringify(desc.get); props.push("get " + name + getter.slice(getter.indexOf('('))); } if (desc.set) { var setter = Stringify(desc.set); props.push("set " + name + setter.slice(setter.indexOf('('))); } } return "{" + props.join(", ") + "}"; default: return "[crazy non-standard value]"; } } function StringifyProxy(proxy, depth) { var proxy_type = typeof proxy; var info_object = { target: JSProxyGetTarget(proxy), handler: JSProxyGetHandler(proxy) } return '[' + proxy_type + ' Proxy ' + Stringify(info_object, depth-1) + ']'; } return Stringify; })();