| /** |
| * @fileoverview Serialization utils. |
| * @author Bryan Mishkin |
| */ |
| |
| "use strict"; |
| |
| /** |
| * Check if a value is a primitive or plain object created by the Object constructor. |
| * @param {any} val the value to check |
| * @returns {boolean} true if so |
| * @private |
| */ |
| function isSerializablePrimitiveOrPlainObject(val) { |
| return ( |
| val === null || |
| typeof val === "string" || |
| typeof val === "boolean" || |
| typeof val === "number" || |
| (typeof val === "object" && val.constructor === Object) || |
| Array.isArray(val) |
| ); |
| } |
| |
| /** |
| * Check if a value is serializable. |
| * Functions or objects like RegExp cannot be serialized by JSON.stringify(). |
| * Inspired by: https://stackoverflow.com/questions/30579940/reliable-way-to-check-if-objects-is-serializable-in-javascript |
| * @param {any} val The value |
| * @param {Set<Object>} seenObjects Objects already seen in this path from the root object. |
| * @returns {boolean} `true` if the value is serializable |
| */ |
| function isSerializable(val, seenObjects = new Set()) { |
| if (!isSerializablePrimitiveOrPlainObject(val)) { |
| return false; |
| } |
| if (typeof val === "object" && val !== null) { |
| if (seenObjects.has(val)) { |
| /* |
| * Since this is a depth-first traversal, encountering |
| * the same object again means there is a circular reference. |
| * Objects with circular references are not serializable. |
| */ |
| return false; |
| } |
| for (const property in val) { |
| if (Object.hasOwn(val, property)) { |
| if (!isSerializablePrimitiveOrPlainObject(val[property])) { |
| return false; |
| } |
| if ( |
| typeof val[property] === "object" && |
| val[property] !== null |
| ) { |
| if ( |
| /* |
| * We're creating a new Set of seen objects because we want to |
| * ensure that `val` doesn't appear again in this path, but it can appear |
| * in other paths. This allows for reusing objects in the graph, as long as |
| * there are no cycles. |
| */ |
| !isSerializable( |
| val[property], |
| new Set([...seenObjects, val]), |
| ) |
| ) { |
| return false; |
| } |
| } |
| } |
| } |
| } |
| return true; |
| } |
| |
| module.exports = { |
| isSerializable, |
| }; |