@tomer/uneval 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.d.mts +54 -0
- package/dist/index.js +2 -0
- package/license +18 -0
- package/package.json +79 -0
- package/readme.md +320 -0
package/dist/index.d.mts
ADDED
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
//#region src/index.d.ts
|
|
2
|
+
/** Options for {@link uneval}. */
|
|
3
|
+
type UnevalOptions = {
|
|
4
|
+
/**
|
|
5
|
+
* A custom function for unevaling values.
|
|
6
|
+
*
|
|
7
|
+
* Return the following types depending on the desired behavior:
|
|
8
|
+
* - `string` to provide a custom uneval result for {@link value}
|
|
9
|
+
* - `null` to omit {@link value} from the output (e.g. in arrays, objects,
|
|
10
|
+
* Sets, Maps). Omitting the root value throws an `Error`.
|
|
11
|
+
* - `undefined` (or don't return anything, which is equivalent) to use the
|
|
12
|
+
* default behavior for {@link value}
|
|
13
|
+
*
|
|
14
|
+
* It can be used to uneval already supported values differently or to uneval
|
|
15
|
+
* unsupported values such as functions.
|
|
16
|
+
*
|
|
17
|
+
* The {@link uneval} param is the same uneval function these options were
|
|
18
|
+
* passed to. You may use it to delegate back to uneval for sub-objects.
|
|
19
|
+
*/
|
|
20
|
+
custom?: (value: unknown, uneval: (value: unknown) => string) => string | null | undefined;
|
|
21
|
+
};
|
|
22
|
+
/**
|
|
23
|
+
* Converts the given {@link value} to JavaScript source code.
|
|
24
|
+
*
|
|
25
|
+
* @example
|
|
26
|
+
* ```js
|
|
27
|
+
* import assert from 'node:assert'
|
|
28
|
+
* import uneval from '@tomer/uneval'
|
|
29
|
+
*
|
|
30
|
+
* const object = { message: `hello world` }
|
|
31
|
+
*
|
|
32
|
+
* const source = uneval(object)
|
|
33
|
+
* console.log(source)
|
|
34
|
+
* //=> {message:"hello world"}
|
|
35
|
+
*
|
|
36
|
+
* const roundtrippedObject = (0, eval)(`(${source})`)
|
|
37
|
+
* assert.deepEqual(roundtrippedObject, object)
|
|
38
|
+
*
|
|
39
|
+
* const circularObject = {}
|
|
40
|
+
* circularObject.self = circularObject
|
|
41
|
+
*
|
|
42
|
+
* const circularSource = uneval(circularObject)
|
|
43
|
+
* console.log(circularSource)
|
|
44
|
+
* //=> (a=>a.self=a)({})
|
|
45
|
+
*
|
|
46
|
+
* const roundtrippedCircularObject = (0, eval)(`(${circularSource})`)
|
|
47
|
+
* assert.deepEqual(roundtrippedCircularObject, circularObject)
|
|
48
|
+
* ```
|
|
49
|
+
*/
|
|
50
|
+
declare const uneval: (value: unknown, {
|
|
51
|
+
custom
|
|
52
|
+
}?: UnevalOptions) => string;
|
|
53
|
+
//#endregion
|
|
54
|
+
export { UnevalOptions, uneval as default };
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
let e=e=>{let r="";do{r=t[e%t.length]+r,e=~~(e/t.length)-1}while(e>=0);return r.length>1&&(r=`$${r}`),r},t="abcdefghijklmnopqrstuvwxyz";t+=t.toUpperCase();let r=(e,t="")=>`new ${e}${""===t?"":`(${t})`}`,l=(e,t)=>t.l.get(e).t,n=/^[$_\p{ID_Start}][$_\p{ID_Continue}]*$/u,u=e=>!!isNaN(e)&&(o.setFloat64(0,e),o.getBigUint64(0)!=i),o=new DataView(new ArrayBuffer(8));o.setFloat64(0,NaN);let i=o.getBigUint64(0),f=({byteOffset:e,byteLength:t,buffer:r,BYTES_PER_ELEMENT:l},n)=>`${D(r,n)}${e+t==r.byteLength?e>0?`,${e}`:"":`,${+e},${t/l}`}`,$=e=>`Object.assign(${e})`,a=(e,t)=>{let r=t.u;t.u=void 0;let l=D(e,t);return t.u=r,l},s=(e,t)=>{let r="",l=0;for(let n=0;n<e.length;n+=1){let u=e[n],o=t[u];if(o){r+=e.slice(l,n)+o,l=n+1;continue}let i,f=u.charCodeAt(0);if(f>=55296&&f<=56319){let t=e.charCodeAt(n+1);t>=56320&&t<=57343?n++:i=!0}else i=f>=56320&&f<=57343;i?(r+=`${e.slice(l,n)}\\u${f.toString(16)}`,l=n+1):"/"==u&&n>0&&"<"==e[n-1]&&"script"==e.slice(n+1,n+7).toLowerCase()&&(r+=`${e.slice(l,n)}\\u002f`,l=n+1)}return r+=e.slice(l),r},c={"\0":"\\0","\n":"\\n","\r":"\\r","\t":"\\t","\b":"\\b","\f":"\\f","\v":"\\v","\u2028":"\\u2028","\u2029":"\\u2029"},b={'"':'\\"',"\\":"\\\\",...c},y=new Map(Reflect.ownKeys(Symbol).flatMap(e=>{if("string"!=typeof e)return[];if(!n.test(e))return[];let t=Symbol[e];return"symbol"==typeof t?[[t,e]]:[]})),_=e=>{let t=e.constructor?.name;return[m[t],t]},m={Boolean:0,Number:0,RegExp:1,String:0,Array:2,Set:3,Map:4,ArrayBuffer:5,Int8Array:6,Uint8Array:6,Uint8ClampedArray:6,Int16Array:6,Uint16Array:6,Int32Array:6,Uint32Array:6,Float16Array:6,Float32Array:6,Float64Array:6,BigInt64Array:6,BigUint64Array:6,Buffer:7,Date:8,Duration:9,Instant:9,PlainDate:9,PlainDateTime:9,PlainMonthDay:9,PlainTime:9,PlainYearMonth:9,ZonedDateTime:9,URL:10,URLSearchParams:10},p=(e,t)=>{let r=t.o.get(e);if("string"==typeof r)return r;if("function"==typeof e)throw new TypeError("Unsupported function");let[l,n]=_(e);return d[l]?.(e,t,n)??v(e,t)},d=[(e,t)=>`Object(${D(e.valueOf(),t)})`,({source:e,flags:t},l,n)=>e==s(e,c)&&/^[a-z]*$/u.test(t)?`/${e}/${t}`:r(n,`${D(e,l)}${t&&`,${D(t,l)}`}`),(e,t)=>{let r,n=Object.keys(e).flatMap(e=>isNaN(+e)?[]:+e),u=e.length-1 in e?"[]":`Array(${e.length})`,o=e.length-n.length,i=(`${e.length}`.length+2)*n.length;if($(u).length+i<o){let r=n.flatMap(r=>{let n=e[r],u=D(n,t);if(u)return`${r}:${u}`;if(null===u){let u=l(n,t);t.i.push({$:`${l(e,t)}[${r}]=${u}`,_:u})}return[]}).join();return r?$(`${u},{${r}}`):u}let f=Array.from(e,(n,u)=>{if(!(u in e))return r=!0,"";r=!1;let o=D(n,t);if(o)return o;if(null===o){let r=l(n,t);t.i.push({$:`${l(e,t)}[${u}]=${r}`,_:r})}else r=!0;return""});return r&&f.push(""),`[${f.join()}]`},(e,t,n)=>{let u,o=[];for(let r of e){let n=(u?a:D)(r,t);if(void 0===n)continue;if(null===n&&(u=!0,n=l(r,t)),!u){o.push(n);continue}let i=l(e,t);t.i.push({$:`${i}.add(${n})`,_:i})}let i=o.join();return r(n,i&&`[${i}]`)},(e,t,n)=>{let u,o=[];for(let[r,n]of e){let i=(u?a:D)(r,t);null===i&&(u=!0);let f=(u?a:D)(n,t);if(void 0===i||void 0===f)continue;let $=i??l(r,t),s=f??l(n,t);if(!u&&null!=f){o.push(`[${$},${s}]`);continue}let c=l(e,t);t.i.push({$:`${c}.set(${$},${s})`,_:c}),u||o.push(`[${$}]`)}let i=o.join();return r(n,i&&`[${i}]`)},(e,t,n)=>{let u,o,{detached:i,resizable:f,byteLength:$,maxByteLength:a}=e;if(i&&t.i.push({$:`${l(e,t)}.transfer()`}),0==$||-1==(o=(u=new Uint8Array(e.slice())).findIndex(e=>0!=e)))return r(n,f?`${+$},{maxByteLength:${+a}}`:$>0?$:"");if(!f)return`${D(u,t)}.buffer`;let s=u.findLastIndex(e=>0!=e);return t.i.push({$:`new Uint8Array(${l(e,t)}).set([${u.slice(o,s+1)}]${o>0?`,${o}`:""})`}),r(n,`${+$},{maxByteLength:${+a}}`)},(e,t,l)=>{let n="F"==l[0]&&[...e].some(u),o=e.buffer;if(t.o.has(o)||t.l.has(o)||e.byteLength!=o.byteLength||o.resizable||n)return r(l,f(e,t));let i="B"==l[0]?0n:0;return e.some(e=>!Object.is(e,i))?`${l}.of(${Array.from(e,e=>D(e,t)).join()})`:r(l,e.length||"")},(e,t,r)=>{let l=e.buffer;return e.byteOffset||e.byteLength||l.resizable||l.detached?`${r}.from(${f(e,t)})`:`${r}.alloc(0)`},(e,t,l)=>r(l,D(+e,t)),(e,t,r)=>{if("Z"!=r[0])return`Temporal.${r}.from(${D(`${e}`,t)})`;let l=e;return`new Temporal.${r}(${D(l.epochNanoseconds,t)},${D(l.timeZoneId,t)})`},(e,t,l)=>{let n=`${e}`;return r(l,n&&D(n,t))}],v=(e,t)=>{let r,u=[],o=Reflect.ownKeys(e);for(r=0;r<o.length;r++){let i=o[r],f=Object.getOwnPropertyDescriptor(e,i);if(!g(f))break;let $=f.value;if("symbol"==typeof i&&null===t.o.get(i))continue;let a=D($,t);if(void 0===a)continue;if(null!==a){let{$:e,m:r}=j(i,t);u.push({$:r&&e==a?e:`${e}:${a}`});continue}let s=l(e,t),c=l($,t),b="symbol"==typeof i?{$:`${s}[${D(i,t)}]=${c}`,_:c}:i==h?{$:`Object.defineProperty(${s},"${h}",{value:${c},configurable:!0,enumerable:!0,writable:!0})`,_:s}:{$:`${s}${n.test(i)?`.${i}`:`[${D(i,t)}]`}=${c}`,_:c};u.push({p:!0,$:`${j(i,t).$}:null`,v:()=>b})}let i=r;if(i<o.length)for(;r<o.length;r++){let l=o[r],n=Object.getOwnPropertyDescriptor(e,l);u.push(w(l,n,e,t))}let f=u.length;for(;f>0&&u[f-1].p;)f--;for(let e=0;e<u.length;e++){let r=u[e];r.p&&t.i.push(r.v(e<f))}let $=u.slice(0,f),a=`{${$.slice(0,i).map(e=>e.$).join()}}`,s=$.slice(i).map(e=>e.$);s.length&&(a=`Object.defineProperties(${a},{${s.join()}})`);let c=Object.getPrototypeOf(e);return A(c)||(a=`Object.setPrototypeOf(${a},${D(c,t)})`),a},g=e=>"value"in e&&e.enumerable&&e.configurable&&e.writable,w=(e,t,r,n)=>{let u,o,i=[];for(let e of B){if(!(e in t))continue;let r=t[e];if(U.includes(e)){if(null==r&&o)continue;o=!0}let f=D(r,n);void 0!==f&&(null===f&&(u=!0,f=l(r,n)),i.push(`${e}:${f}`))}for(let e of S)t[e]&&i.push(`${e}:!0`);let f=`{${i.join()}}`;return u?{p:!0,$:`${j(e,n).$}:{configurable:!0}`,v:u=>{let o=l(r,n),$=u&&!t.configurable?`{${[...i,"configurable:!1"].join()}}`:f;return{$:`Object.defineProperty(${o},${D(e,n)},${$})`,_:o}}}:{$:`${j(e,n).$}:${f}`}},j=(e,t)=>{if("symbol"==typeof e||e==h)return{$:`[${D(e,t)}]`};let r=e[0];if(r>="0"&&r<="9"){let r=+e;return{$:r>=0&&Number.isSafeInteger(r)&&e==`${r}`?e:D(e,t)}}return n.test(e)?{$:e,m:!0}:{$:D(e,t)}},A=e=>e==Object.prototype||O(e)&&T(e)==P,O=e=>{let t=typeof e;return"object"==t&&!!e||"function"==t},T=e=>Object.getOwnPropertyNames(e).sort().join("\0"),h="__proto__",P=T(Object.prototype),S=["configurable","enumerable","writable"],U=["get","set"],B=["value",...U],D=(e,t)=>{if(!O(e)){let r=t.o.get(e);if(null===r)return;if(void 0!==r)return r}let r=typeof e;return"undefined"==r?"void 0":"boolean"==r?(e=>"!"+ +!e)(e):"number"==r?(e=>{if(e==1/0)return"1/0";if(e==-1/0)return"-1/0";let t=`${e}`;return Object.is(e,-0)?`-${t}`:t.startsWith("0.")?t.slice(1):t.startsWith("-0.")?`-${t.slice(2)}`:t})(e):"bigint"==r?(e=>`${e}n`)(e):"string"==r?`"${s(e,b)}"`:"symbol"==r?((e,t)=>{let r=y.get(e);if(r)return`Symbol.${r}`;if(r=Symbol.keyFor(e),r)return`Symbol.for(${D(r,t)})`;throw new TypeError("Unsupported symbol")})(e,t):null==e?"null":((e,t)=>{if(null===t.o.get(e))return;let r=t.l.get(e);if(!r)return p(e,t);if(t.j.has(e))return null;if(t.u&&(t.u.A=!0),!r.$){let l=t.u;t.u=r,t.j.add(e),r.$=p(e,t),t.j.delete(e),t.u=l,t.O.push(r)}return r.t})(e,t)},E=(e,{custom:t}={})=>{let r=I(e,t),l=D(e,r);if(void 0===l)throw new Error("Root omitted");if(!r.l.size)return l;let n=r.i.map(e=>e.$),u=r.l.get(e);u?r.i.at(-1)?._!=u.t&&n.push(u.t):n.push(l);let o=n.join();(n.length>1||"{"==o[0])&&(o=`(${o})`);let i,f=[],$=[];for(let e of r.O)i||=e.A,i?f.push(`${e.t}=${e.$}`):(f.push(e.t),$.push(e.$));let a=f.join();return`(${f.length>1?`(${a})`:a}=>${o})(${$.join()})`},I=(t,r)=>{let l=new Map,n=new Map,u=t=>{l.has(t)||l.set(t,{t:e(l.size)})},o=new Map,i=(e,t)=>{if(r){let t=n.get(e);if(void 0===t&&(t=r(e,e=>E(e,{custom:r})),void 0!==t&&n.set(e,t)),null===t)return}if(!e||"object"!=typeof e)return;let l=o.get(e);if(l)return u(e),void(2==l&&u(t));o.set(e,2),f(e),o.set(e,1)},f=e=>{let[t]=_(e);if(null==t){for(let t of Reflect.ownKeys(e)){"symbol"==typeof t&&i(t);let r=Object.getOwnPropertyDescriptor(e,t);i(r.value,e),i(r.get,e),i(r.set,e)}let t=Object.getPrototypeOf(e);A(t)||i(t,e)}else if(0==t){let t=e.valueOf();i(t),null===n.get(t)&&n.set(e,null)}else if(2==t)for(let t of Object.keys(e))i(e[t],e);else if(3==t)for(let t of e)i(t,e);else if(4==t)for(let[t,r]of e)i(t,e),i(r,e),2==o.get(r)&&O(t)&&u(t);else if(5==t)(e.detached||e.resizable&&new Uint8Array(e).some(e=>0!=e))&&u(e);else if(7==t||6==t){let{buffer:t}=e;i(t,e),null===n.get(t)&&n.set(e,null)}};return i(t),{l,o:n,j:new Set,O:[],i:[]}};export{E as default};
|
|
2
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.js","names":[],"sources":["../src/internal/identifier.ts","../src/internal/common.ts","../src/internal/buffer.ts","../src/internal/collection.ts","../src/internal/primitive.ts","../src/internal/temporal.ts","../src/internal/type.ts","../src/internal/url.ts","../src/internal/object.ts","../src/internal/index.ts","../src/index.ts"],"sourcesContent":["/** Returns a unique valid JS identifier for the given {@link index}. */\nexport const generateIdentifier = (index: number): string => {\n  let identifier = ``\n  do {\n    identifier = IDENTIFIER_CHARS[index % IDENTIFIER_CHARS.length]! + identifier\n    // For smaller bundle size.\n    // eslint-disable-next-line unicorn/prefer-math-trunc\n    index = ~~(index / IDENTIFIER_CHARS.length) - 1\n  } while (index >= 0)\n\n  if (identifier.length > 1) {\n    // Any identifiers with 2 or more characters could clash with reserved\n    // keywords or global identifiers we reference. We could list all those out\n    // and check them, but ending up with identifiers are more than\n    // `IDENTIFIER_CHARS.length` in length is so rare in `uneval` (requires a\n    // ton of shared or circular references) that it's not worth increasing the\n    // bundle size for.\n    identifier = `$${identifier}`\n  }\n\n  return identifier\n}\n\nlet IDENTIFIER_CHARS = `abcdefghijklmnopqrstuvwxyz`\nIDENTIFIER_CHARS += IDENTIFIER_CHARS.toUpperCase()\n","import type { State } from './types.ts'\n\nexport const newInstance = (name: string, args: string | number = ``): string =>\n  `new ${name}${args === `` ? `` : `(${args})`}`\n\nexport const bindingName = (value: object, state: State): string =>\n  state._bindings.get(value)!._name\n\nexport const PROPERTY_REG_EXP = /^[$_\\p{ID_Start}][$_\\p{ID_Continue}]*$/u\n","// For smaller bundle size.\n/* eslint-disable unicorn/prefer-number-properties */\n/* eslint-disable eqeqeq */\n\nimport { bindingName, newInstance } from './common.ts'\nimport { unevalInternal } from './index.ts'\nimport type { State, Uneval } from './types.ts'\n\nexport type TypedArray =\n  | Int8Array\n  | Uint8Array\n  | Uint8ClampedArray\n  | Int16Array\n  | Uint16Array\n  | Int32Array\n  | Uint32Array\n  | Float16Array\n  | Float32Array\n  | Float64Array\n  | BigInt64Array\n  | BigUint64Array\n\nexport const unevalTypedArray = (\n  typedArray: TypedArray,\n  state: State,\n  name: string,\n): string => {\n  // eslint-disable-next-line @typescript-eslint/prefer-string-starts-ends-with\n  const isFloatingPoint = name[0] == `F`\n  const hasNonCanonicalNaN =\n    isFloatingPoint &&\n    [...(typedArray as Float16Array | Float32Array | Float64Array)].some(\n      isNonCanonicalNaN,\n    )\n\n  const arrayBuffer = typedArray.buffer\n  if (\n    // We have to construct from a buffer if the end-user provided custom source\n    // for it.\n    state._customSources.has(arrayBuffer) ||\n    // We have to construct from a buffer if it has a binding, meaning that it's\n    // shared between multiple values.\n    state._bindings.has(arrayBuffer) ||\n    // Also, if the byte lengths differ between the typed array and the buffer,\n    // then that means this typed array is a view of a slice of a buffer. The\n    // only way to achieve that is by constructing from a buffer.\n    typedArray.byteLength != arrayBuffer.byteLength ||\n    // Lastly, if the underlying buffer is resizable, then we must also\n    // construct from a buffer. The default buffer created from\n    // `new TypedArray(...)` is not resizable.\n    (arrayBuffer as ArrayBuffer).resizable ||\n    // For floating-point arrays with non-canonical NaN values, we must\n    // construct from the buffer to preserve the exact NaN bit pattern.\n    hasNonCanonicalNaN\n  ) {\n    return newInstance(name, unevalBufferWrapperArgs(typedArray, state))\n  }\n\n  // eslint-disable-next-line @typescript-eslint/prefer-string-starts-ends-with\n  const zero = name[0] == `B` ? 0n : 0\n  if (typedArray.some(value => !Object.is(value, zero))) {\n    return `${name}.of(${Array.from<number | bigint, string>(\n      typedArray,\n      value => unevalInternal(value, state)!,\n    ).join()})`\n  }\n\n  return newInstance(name, typedArray.length || ``)\n}\n\nconst isNonCanonicalNaN = (value: number): boolean => {\n  if (!isNaN(value)) {\n    return false\n  }\n\n  float64ScratchView.setFloat64(0, value)\n  return float64ScratchView.getBigUint64(0) != CANONICAL_NAN_BITS\n}\n\nconst float64ScratchView = new DataView(new ArrayBuffer(8))\nfloat64ScratchView.setFloat64(0, NaN)\nconst CANONICAL_NAN_BITS = float64ScratchView.getBigUint64(0)\n\nexport const unevalBuffer: Uneval<Buffer> = (buffer, state, name) => {\n  const arrayBuffer = buffer.buffer as ArrayBuffer\n  if (\n    !buffer.byteOffset &&\n    !buffer.byteLength &&\n    !arrayBuffer.resizable &&\n    !arrayBuffer.detached\n  ) {\n    return `${name}.alloc(0)`\n  }\n\n  return `${name}.from(${unevalBufferWrapperArgs(buffer, state)})`\n}\n\nconst unevalBufferWrapperArgs = (\n  {\n    byteOffset,\n    byteLength,\n    buffer,\n    BYTES_PER_ELEMENT,\n  }: {\n    byteOffset: number\n    byteLength: number\n    buffer: ArrayBufferLike\n    BYTES_PER_ELEMENT: number\n  },\n  state: State,\n): string =>\n  `${unevalInternal(buffer, state)!}${\n    byteOffset + byteLength == buffer.byteLength\n      ? byteOffset > 0\n        ? `,${byteOffset}`\n        : ``\n      : `,${+byteOffset},${byteLength / BYTES_PER_ELEMENT}`\n  }`\n\nexport const unevalArrayBuffer: Uneval<ArrayBuffer> = (\n  arrayBuffer,\n  state,\n  name,\n) => {\n  const { detached, resizable, byteLength, maxByteLength } = arrayBuffer\n\n  if (detached) {\n    state._mutations.push({\n      _source: `${bindingName(arrayBuffer, state)}.transfer()`,\n    })\n  }\n\n  let uint8Array: Uint8Array\n  let firstNonZeroIndex: number\n  if (\n    byteLength == 0 ||\n    (firstNonZeroIndex = (uint8Array = new Uint8Array(\n      arrayBuffer.slice(),\n    )).findIndex(value => value != 0)) == -1\n  ) {\n    return newInstance(\n      name,\n      resizable\n        ? `${+byteLength},{maxByteLength:${+maxByteLength}}`\n        : byteLength > 0\n          ? byteLength\n          : ``,\n    )\n  }\n\n  if (!resizable) {\n    return `${unevalInternal(uint8Array, state)!}.buffer`\n  }\n\n  const lastNonZeroIndex = uint8Array.findLastIndex(value => value != 0)\n  state._mutations.push({\n    _source: `new Uint8Array(${bindingName(arrayBuffer, state)}).set([${\n      // eslint-disable-next-line @typescript-eslint/restrict-template-expressions\n      uint8Array.slice(firstNonZeroIndex, lastNonZeroIndex + 1)\n    }]${firstNonZeroIndex > 0 ? `,${firstNonZeroIndex}` : ``})`,\n  })\n  return newInstance(name, `${+byteLength},{maxByteLength:${+maxByteLength}}`)\n}\n","// For smaller bundle size.\n/* eslint-disable unicorn/prefer-number-properties */\n\nimport { bindingName, newInstance } from './common.ts'\nimport { unevalInternal } from './index.ts'\nimport type { State, Uneval } from './types.ts'\n\nexport const unevalArray: Uneval<unknown[]> = (array, state) => {\n  // Get own numeric index keys to check sparsity in `O(keys)` time instead of\n  // `O(length)` time, which could result in DoS.\n  const indices = Object.keys(array).flatMap(key => (isNaN(+key) ? [] : +key))\n\n  const hasTrailingEmptySlots = !(array.length - 1 in array)\n  const emptyArraySource = hasTrailingEmptySlots\n    ? // We'll have to set the length explicitly if there's no entry that\n      // implicitly grows the array to its length.\n      `Array(${array.length})`\n    : `[]`\n\n  // A comma for each sparse slot in a dense array.\n  const denseOverhead = array.length - indices.length\n  // A `${index}: ,` for each non-sparse slot in a sparse array.\n  const maxIndexLength = `${array.length}`.length\n  const entryOverhead = (maxIndexLength + 2) * indices.length\n  const sparseOverhead =\n    unevalObjectAssign(emptyArraySource).length + entryOverhead\n  if (sparseOverhead < denseOverhead) {\n    // The array is sparse enough that the `Object.assign` representation is\n    // likely more compact.\n    const entriesSource = indices\n      .flatMap(index => {\n        const item = array[index]\n        const result = unevalInternal(item, state)\n        if (result) {\n          return `${index}:${result}`\n        }\n\n        if (result === null) {\n          const itemName = bindingName(item as object, state)\n          state._mutations.push({\n            _source: `${bindingName(array, state)}[${index}]=${itemName}`,\n            _evaluatesTo: itemName,\n          })\n        }\n        return []\n      })\n      .join()\n    return entriesSource\n      ? unevalObjectAssign(`${emptyArraySource},{${entriesSource}}`)\n      : emptyArraySource\n  }\n\n  let trailingEmptySlot: boolean | undefined\n  const itemSources = Array.from(array, (item, index) => {\n    if (!(index in array)) {\n      trailingEmptySlot = true\n      return ``\n    }\n\n    trailingEmptySlot = false\n    const result = unevalInternal(item, state)\n    if (result) {\n      return result\n    }\n\n    if (result === null) {\n      const itemName = bindingName(item as object, state)\n      state._mutations.push({\n        _source: `${bindingName(array, state)}[${index}]=${itemName}`,\n        // An assignment evaluates to the right-hand side.\n        _evaluatesTo: itemName,\n      })\n    } else {\n      // Omitted value. Render it as an empty slot.\n      trailingEmptySlot = true\n    }\n    return ``\n  })\n  if (trailingEmptySlot) {\n    // The array has a trailing empty slot (either sparse input or custom\n    // omitted). This requires an extra comma because otherwise the last\n    // comma is interpreted as a no-op trailing comma.\n    itemSources.push(``)\n  }\n  return `[${itemSources.join()}]`\n}\n\nconst unevalObjectAssign = (args: string) => `Object.assign(${args})`\n\nexport const unevalSet: Uneval<Set<unknown>> = (set, state, name) => {\n  let foundCircular: true | undefined\n  const argSources: string[] = []\n  for (const value of set) {\n    // If we've seen a circular value, then all remaining values will end\n    // up in mutations so we uneval them without the current binding to not\n    // add unnecessary dependencies to the current binding.\n    let result = (foundCircular ? unevalWithoutCurrentBinding : unevalInternal)(\n      value,\n      state,\n    )\n    if (result === undefined) {\n      // Skip if omitted.\n      continue\n    }\n    if (result === null) {\n      foundCircular = true\n      result = bindingName(value as object, state)\n    }\n\n    if (!foundCircular) {\n      argSources.push(result)\n      continue\n    }\n\n    // After and including the first circular value, we add set values\n    // via mutation to preserve iteration order.\n    const valueName = bindingName(set, state)\n    state._mutations.push({\n      _source: `${valueName}.add(${result})`,\n      // `Set.add` always returns `this`.\n      _evaluatesTo: valueName,\n    })\n  }\n\n  const argsSource = argSources.join()\n  return newInstance(name, argsSource && `[${argsSource}]`)\n}\n\nexport const unevalMap: Uneval<Map<unknown, unknown>> = (map, state, name) => {\n  let foundCircularKey: true | undefined\n  const argSources: string[] = []\n  for (const [key, value] of map) {\n    // If we've seen a circular key, then all remaining entries will end\n    // up in mutations so we uneval them without the current binding to\n    // not add unnecessary dependencies to the current binding.\n    const keyResult = (\n      foundCircularKey ? unevalWithoutCurrentBinding : unevalInternal\n    )(key, state)\n    if (keyResult === null) {\n      foundCircularKey = true\n    }\n    const valueResult = (\n      foundCircularKey ? unevalWithoutCurrentBinding : unevalInternal\n    )(value, state)\n    if (keyResult === undefined || valueResult === undefined) {\n      // Skip the entire entry if key or value is omitted.\n      continue\n    }\n\n    const keySource = keyResult ?? bindingName(key as object, state)\n    const valueSource = valueResult ?? bindingName(value as object, state)\n\n    if (\n      // After and including the first circular key, we add entries via\n      // mutation below to preserve iteration order.\n      !foundCircularKey &&\n      valueResult != null\n    ) {\n      argSources.push(`[${keySource},${valueSource}]`)\n      continue\n    }\n\n    const valueName = bindingName(map, state)\n    state._mutations.push({\n      _source: `${valueName}.set(${keySource},${valueSource})`,\n      // `Map.set` always returns `this`.\n      _evaluatesTo: valueName,\n    })\n    if (!foundCircularKey) {\n      // If the value is circular, but we've not seen a circular key, then\n      // include the key in the constructor, but set the value later via\n      // mutation.\n      argSources.push(`[${keySource}]`)\n    }\n  }\n\n  const argsSource = argSources.join()\n  return newInstance(name, argsSource && `[${argsSource}]`)\n}\n\nconst unevalWithoutCurrentBinding = (\n  value: unknown,\n  state: State,\n): string | null | undefined => {\n  const previousBinding = state._currentBinding\n  state._currentBinding = undefined\n  const result = unevalInternal(value, state)\n  state._currentBinding = previousBinding\n  return result\n}\n","// For smaller bundle size.\n/* eslint-disable eqeqeq */\n\nimport { newInstance, PROPERTY_REG_EXP } from './common.ts'\nimport { unevalInternal } from './index.ts'\nimport type { State, Uneval } from './types.ts'\n\nexport const unevalBoolean = (value: boolean): string =>\n  // Convert `false` to `!1` and `true` to `!0`\n  `!${+!value}`\n\nexport const unevalNumber = (value: number): string => {\n  if (value == Infinity) {\n    return `1/0`\n  } else if (value == -Infinity) {\n    return `-1/0`\n  }\n\n  const source = `${value}`\n  if (Object.is(value, -0)) {\n    // Converting -0 to a string becomes `0` so we have to special-case it.\n    return `-${source}`\n  }\n\n  // Convert `0.123` to `.123` and  `-0.123` to `-.123`.\n  if (source.startsWith(`0.`)) {\n    return source.slice(1)\n  } else if (source.startsWith(`-0.`)) {\n    return `-${source.slice(2)}`\n  }\n\n  return source\n}\n\nexport const unevalBigint = (value: bigint): string => `${value}n`\n\n// `charCodeAt` is more performant for our use-case because we're dealing with\n// strings known to be single code units.\n/* eslint-disable unicorn/prefer-code-point */\nexport const unevalLiteral = (\n  value: string,\n  codeUnitEscapes: Readonly<Record<string, string>>,\n): string => {\n  let source = ``\n\n  let lastIndex = 0\n  for (let i = 0; i < value.length; i += 1) {\n    const codeUnit = value[i]!\n    const escaped = codeUnitEscapes[codeUnit]\n    if (escaped) {\n      source += value.slice(lastIndex, i) + escaped\n      lastIndex = i + 1\n      continue\n    }\n\n    const code = codeUnit.charCodeAt(0)\n\n    // https://en.wikipedia.org/wiki/UTF-16#U+D800_to_U+DFFF_(surrogates)\n    let isUnpairedSurrogate: boolean | undefined\n    const isLowSurrogate = code >= 0xd800 && code <= 0xdbff\n    if (isLowSurrogate) {\n      const next = value.charCodeAt(i + 1)\n      const isHighSurrogate = next >= 0xdc00 && next <= 0xdfff\n      if (isHighSurrogate) {\n        i++\n      } else {\n        isUnpairedSurrogate = true\n      }\n    } else {\n      const isHighSurrogate = code >= 0xdc00 && code <= 0xdfff\n      // If this is a high surrogate, then it's unpaired. If it were paired,\n      // then we would have skipped it in the previous iteration of the loop.\n      isUnpairedSurrogate = isHighSurrogate\n    }\n\n    if (isUnpairedSurrogate) {\n      // Escape unpaired surrogates in the source.\n      source += `${value.slice(lastIndex, i)}\\\\u${code.toString(16)}`\n      lastIndex = i + 1\n      continue\n    }\n\n    // Prevent XSS attack via closing an inline script tag.\n    if (\n      codeUnit == `/` &&\n      i > 0 &&\n      value[i - 1] == `<` &&\n      value.slice(i + 1, i + 7).toLowerCase() == `script`\n    ) {\n      source += `${value.slice(lastIndex, i)}\\\\u002f`\n      lastIndex = i + 1\n    }\n  }\n\n  source += value.slice(lastIndex)\n  return source\n}\n/* eslint-enable unicorn/prefer-code-point */\n\n/**\n * Code unit escapes for code units that are not safe to include in JS source\n * code for a literal (like a `string` or `RegExp`).\n */\nexport const UNSAFE_CODE_UNIT_ESCAPES: Readonly<Record<string, string>> = {\n  '\\0': `\\\\0`,\n  '\\n': `\\\\n`,\n  '\\r': `\\\\r`,\n  '\\t': `\\\\t`,\n  '\\b': `\\\\b`,\n  '\\f': `\\\\f`,\n  '\\v': `\\\\v`,\n  // https://stackoverflow.com/a/9168133\n  '\\u2028': `\\\\u2028`,\n  '\\u2029': `\\\\u2029`,\n}\n\nexport const STRING_CODE_UNIT_ESCAPES: Readonly<Record<string, string>> = {\n  '\"': `\\\\\"`,\n  '\\\\': `\\\\\\\\`,\n  ...UNSAFE_CODE_UNIT_ESCAPES,\n}\n\nexport const unevalSymbol = (value: symbol, state: State): string => {\n  let key = WELL_KNOWN_SYMBOL_TO_KEY.get(value)\n  if (key) {\n    return `Symbol.${key}`\n  }\n\n  key = Symbol.keyFor(value)\n  if (key) {\n    return `Symbol.for(${unevalInternal(key, state)})`\n  }\n\n  throw new TypeError(`Unsupported symbol`)\n}\n\nconst WELL_KNOWN_SYMBOL_TO_KEY: ReadonlyMap<symbol, string> = new Map(\n  Reflect.ownKeys(Symbol).flatMap(key => {\n    // This doesn't happen in practice, but best to be safe if one day a\n    // non-string key as added to `Symbol`.\n    if (typeof key != `string`) {\n      return []\n    }\n    if (!PROPERTY_REG_EXP.test(key)) {\n      // Defend against pollution attacks.\n      return []\n    }\n    const value = Symbol[key as keyof typeof Symbol]\n    return typeof value == `symbol` ? [[value, key]] : []\n  }),\n)\n\n// eslint-disable-next-line @typescript-eslint/no-wrapper-object-types\nexport const unevalPrimitiveWrapper: Uneval<Boolean | Number | String> = (\n  value,\n  state,\n) => `Object(${unevalInternal(value.valueOf(), state)})`\n\nexport const unevalRegExp: Uneval<RegExp> = (\n  { source, flags },\n  state,\n  name,\n) => {\n  const escapedSource = unevalLiteral(source, UNSAFE_CODE_UNIT_ESCAPES)\n  return (\n    // `RegExp.prototype.source` will return the escaped version of the\n    // source between the forward slashes for a literal. We can use it\n    // directly in the `RegExp` literal, but only if the `source` is safe\n    // for JS (e.g. no unescaped `\\0` inline). We can't simply use\n    // `escapedSource` here because that doesn't roundtrip.\n    // i.e. `/\\0/.source` is does not equal `new RegExp('\\0').source`.\n    // The former is `'\\\\0'` while the later is `'\\0'`.\n    source == escapedSource &&\n      // This protects against RCE from monkey-patched `RegExp` objects.\n      /^[a-z]*$/u.test(flags)\n      ? `/${source}/${flags}`\n      : newInstance(\n          name,\n          `${unevalInternal(source, state)}${\n            flags && `,${unevalInternal(flags, state)}`\n          }`,\n        )\n  )\n}\n","// For smaller bundle size.\n/* eslint-disable eqeqeq */\n\nimport { newInstance } from './common.ts'\nimport { unevalInternal } from './index.ts'\nimport type { Uneval } from './types.ts'\n\nexport const unevalDate: Uneval<Date> = (date, state, name) =>\n  newInstance(name, unevalInternal(+date, state))\n\nexport const unevalTemporal: Uneval<\n  | Temporal.Duration\n  | Temporal.Instant\n  | Temporal.PlainDate\n  | Temporal.PlainDateTime\n  | Temporal.PlainMonthDay\n  | Temporal.PlainTime\n  | Temporal.PlainYearMonth\n  | Temporal.ZonedDateTime\n> = (temporal, state, name) => {\n  // eslint-disable-next-line @typescript-eslint/prefer-string-starts-ends-with\n  if (name[0] != `Z`) {\n    return `Temporal.${name}.from(${unevalInternal(\n      // eslint-disable-next-line @typescript-eslint/restrict-template-expressions\n      `${temporal}`,\n      state,\n    )})`\n  }\n\n  // We handle `ZonedDateTime` differently from that other `Temporal`\n  // objects because `Temporal.ZonedDateTime.from(zonedDateTime.toString())`\n  // does not always roundtrip:\n  // https://github.com/tc39/proposal-temporal/pull/3014#issuecomment-3856086253\n  const zonedDateTime = temporal as Temporal.ZonedDateTime\n  return `new Temporal.${name}(${unevalInternal(\n    zonedDateTime.epochNanoseconds,\n    state,\n  )},${unevalInternal(zonedDateTime.timeZoneId, state)})`\n}\n","// For smaller bundle size.\n/* eslint-disable eqeqeq */\n\nexport const getType = (\n  value: object,\n): [] | [number, string] | [undefined, string] => {\n  // `.constructor` returns `undefined` for objects with null prototype.\n  // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n  const name = value.constructor?.name\n  return [(TYPES as Record<string, number>)[name], name]\n}\n\nexport const T_PRIMITIVE_WRAPPER = 0\nexport const T_REG_EXP = 1\nexport const T_ARRAY = 2\nexport const T_SET = 3\nexport const T_MAP = 4\nexport const T_ARRAY_BUFFER = 5\nexport const T_TYPED_ARRAY = 6\nexport const T_BUFFER = 7\nexport const T_DATE = 8\nexport const T_TEMPORAL = 9\nexport const T_URL = 10\n\nconst TYPES = {\n  Boolean: T_PRIMITIVE_WRAPPER,\n  Number: T_PRIMITIVE_WRAPPER,\n  RegExp: T_REG_EXP,\n  String: T_PRIMITIVE_WRAPPER,\n\n  Array: T_ARRAY,\n  Set: T_SET,\n  Map: T_MAP,\n\n  ArrayBuffer: T_ARRAY_BUFFER,\n  Int8Array: T_TYPED_ARRAY,\n  Uint8Array: T_TYPED_ARRAY,\n  Uint8ClampedArray: T_TYPED_ARRAY,\n  Int16Array: T_TYPED_ARRAY,\n  Uint16Array: T_TYPED_ARRAY,\n  Int32Array: T_TYPED_ARRAY,\n  Uint32Array: T_TYPED_ARRAY,\n  Float16Array: T_TYPED_ARRAY,\n  Float32Array: T_TYPED_ARRAY,\n  Float64Array: T_TYPED_ARRAY,\n  BigInt64Array: T_TYPED_ARRAY,\n  BigUint64Array: T_TYPED_ARRAY,\n  Buffer: T_BUFFER,\n\n  Date: T_DATE,\n  Duration: T_TEMPORAL,\n  Instant: T_TEMPORAL,\n  PlainDate: T_TEMPORAL,\n  PlainDateTime: T_TEMPORAL,\n  PlainMonthDay: T_TEMPORAL,\n  PlainTime: T_TEMPORAL,\n  PlainYearMonth: T_TEMPORAL,\n  ZonedDateTime: T_TEMPORAL,\n\n  URL: T_URL,\n  URLSearchParams: T_URL,\n} as const\n","import { newInstance } from './common.ts'\nimport { unevalInternal } from './index.ts'\nimport type { Uneval } from './types.ts'\n\nexport const unevalURL: Uneval<URL | URLSearchParams> = (url, state, name) => {\n  // eslint-disable-next-line @typescript-eslint/restrict-template-expressions\n  const source = `${url}`\n  return newInstance(name, source && unevalInternal(source, state))\n}\n","// For smaller bundle size.\n/* eslint-disable eqeqeq */\n\nimport { unevalArrayBuffer, unevalBuffer, unevalTypedArray } from './buffer.ts'\nimport { unevalArray, unevalMap, unevalSet } from './collection.ts'\nimport { bindingName, PROPERTY_REG_EXP } from './common.ts'\nimport { unevalInternal } from './index.ts'\nimport { unevalPrimitiveWrapper, unevalRegExp } from './primitive.ts'\nimport { unevalDate, unevalTemporal } from './temporal.ts'\nimport { getType } from './type.ts'\nimport type { Mutation, State, Uneval } from './types.ts'\nimport { unevalURL } from './url.ts'\n\nexport const unevalObject = (\n  value: object,\n  state: State,\n): string | null | undefined => {\n  if (state._customSources.get(value) === null) {\n    // The user decided to omit this value.\n    return undefined\n  }\n\n  const binding = state._bindings.get(value)\n  if (!binding) {\n    // This value has no binding so we can render its source inline.\n    return unevalObjectInternal(value, state)\n  }\n\n  if (state._currentParents.has(value)) {\n    // Return null to indicate that this value is circularly referenced and\n    // should not be rendered here. Instead, we'll add a mutation to attach the\n    // circular reference later (in the caller).\n    return null\n  }\n\n  // Note the current binding as having a dependency (on the binding we're\n  // rendering below).\n  if (state._currentBinding) {\n    state._currentBinding._hasDependency = true\n  }\n\n  if (!binding._source) {\n    const previousBinding = state._currentBinding\n\n    // If this is the first time we're encountering this binding, then we must\n    // compute and set its source.\n    state._currentBinding = binding\n    state._currentParents.add(value)\n    binding._source = unevalObjectInternal(value, state)\n    state._currentParents.delete(value)\n    state._currentBinding = previousBinding\n\n    // It's important to add the binding to the order after its source is\n    // rendered so that it's rendered after its dependencies.\n    state._bindingOrder.push(binding)\n  }\n\n  return binding._name\n}\n\nconst unevalObjectInternal = (value: object, state: State): string => {\n  const customSource = state._customSources.get(value)\n  if (typeof customSource == `string`) {\n    return customSource\n  }\n\n  if (typeof value == `function`) {\n    throw new TypeError(`Unsupported function`)\n  }\n\n  const [type, name] = getType(value)\n  return unevals[type!]?.(value, state, name!) ?? unevalObjectLike(value, state)\n}\n\n// The order of this array must match the numeric values of `T_*` variables.\nconst unevals: Uneval<any>[] = [\n  unevalPrimitiveWrapper,\n  unevalRegExp,\n  unevalArray,\n  unevalSet,\n  unevalMap,\n  unevalArrayBuffer,\n  unevalTypedArray,\n  unevalBuffer,\n  unevalDate,\n  unevalTemporal,\n  unevalURL,\n]\n\nconst unevalObjectLike = (object: object, state: State): string => {\n  const entries: ObjectEntry[] = []\n\n  const keys = Reflect.ownKeys(object)\n  let keyIndex: number\n  for (keyIndex = 0; keyIndex < keys.length; keyIndex++) {\n    const key = keys[keyIndex]!\n    const descriptor = Object.getOwnPropertyDescriptor(object, key)!\n    if (!isRegularDataDescriptor(descriptor)) {\n      // All properties from this index onward will be rendered via\n      // `Object.defineProperties` to preserve property order.\n      break\n    }\n\n    const value = descriptor.value as unknown\n\n    if (typeof key == `symbol` && state._customSources.get(key) === null) {\n      // Skip properties with omitted symbol keys.\n      continue\n    }\n\n    const valueResult = unevalInternal(value, state)\n    if (valueResult === undefined) {\n      // Skip properties with omitted values.\n      continue\n    }\n    if (valueResult !== null) {\n      const { _source: keySource, _isIdentifier: isIdentifier } =\n        unevalObjectLiteralKey(key, state)\n      entries.push({\n        _source:\n          isIdentifier && keySource == valueResult\n            ? keySource\n            : `${keySource}:${valueResult}`,\n      })\n      continue\n    }\n\n    const objectName = bindingName(object, state)\n    const valueName = bindingName(value as object, state)\n    const mutation: Mutation =\n      typeof key == `symbol`\n        ? {\n            _source: `${objectName}[${unevalInternal(key, state)}]=${valueName}`,\n            // An assignment evaluates to the right-hand side.\n            _evaluatesTo: valueName,\n          }\n        : key == __PROTO__\n          ? {\n              _source: `Object.defineProperty(${objectName},\"${__PROTO__}\",{value:${\n                valueName\n              },configurable:!0,enumerable:!0,writable:!0})`,\n              // `Object.defineProperty` always returns the input object.\n              _evaluatesTo: objectName,\n            }\n          : {\n              _source: `${objectName}${\n                PROPERTY_REG_EXP.test(key)\n                  ? `.${key}`\n                  : `[${unevalInternal(key, state)}]`\n              }=${valueName}`,\n              // An assignment evaluates to the right-hand side.\n              _evaluatesTo: valueName,\n            }\n    entries.push({\n      _isCircular: true,\n      // This is a placeholder property for preserving property order. We'll set\n      // the property's actual value later with the mutation below.\n      _source: `${unevalObjectLiteralKey(key, state)._source}:null`,\n      _mutation: () => mutation,\n    })\n  }\n\n  const firstDescriptorIndex = keyIndex\n  if (firstDescriptorIndex < keys.length) {\n    for (; keyIndex < keys.length; keyIndex++) {\n      const key = keys[keyIndex]!\n      const descriptor = Object.getOwnPropertyDescriptor(object, key)!\n      entries.push(unevalDescriptorEntry(key, descriptor, object, state))\n    }\n  }\n\n  // We use this to trim trailing circular placeholders because they're not\n  // necessary for preserving property order. The mutations of trailing circular\n  // properties will end up placing them in the right order.\n  let trailingCircularEntriesStartIndex = entries.length\n  while (\n    trailingCircularEntriesStartIndex > 0 &&\n    entries[trailingCircularEntriesStartIndex - 1]!._isCircular\n  ) {\n    trailingCircularEntriesStartIndex--\n  }\n\n  // Push all circular mutations now that we know which entries have\n  // placeholders (non-trailing) vs which don't (trailing).\n  for (let i = 0; i < entries.length; i++) {\n    const entry = entries[i]!\n    if (entry._isCircular) {\n      const hasPlaceholder = i < trailingCircularEntriesStartIndex\n      state._mutations.push(entry._mutation(hasPlaceholder))\n    }\n  }\n\n  // Create the initial object literal source.\n  const leadingEntries = entries.slice(0, trailingCircularEntriesStartIndex)\n  let source = `{${leadingEntries\n    .slice(0, firstDescriptorIndex)\n    .map(entry => entry._source)\n    .join()}}`\n  // Then set the trailing object descriptors if they are any.\n  const descriptorEntrySources = leadingEntries\n    .slice(firstDescriptorIndex)\n    .map(entry => entry._source)\n  if (descriptorEntrySources.length) {\n    source = `Object.defineProperties(${source},{${descriptorEntrySources.join()}})`\n  }\n\n  const prototype = Object.getPrototypeOf(object) as unknown\n  if (!isDefaultObjectPrototype(prototype)) {\n    // We only render the prototype if it's not equal the default one in this\n    // realm.\n    source = `Object.setPrototypeOf(${source},${\n      // This must be non-null because prototypes cannot be circular. Trying\n      // to make them circular results in an error.\n      unevalInternal(prototype, state)!\n    })`\n  }\n\n  return source\n}\n\n/**\n * Whether the {@link descriptor} is for a regular property, meaning its\n * descriptor attributes are the ones you'd get from a regular object literal\n * property (not through `Object.defineProperty`).\n */\nconst isRegularDataDescriptor = (descriptor: PropertyDescriptor): boolean =>\n  `value` in descriptor &&\n  descriptor.enumerable! &&\n  descriptor.configurable! &&\n  descriptor.writable!\n\nconst unevalDescriptorEntry = (\n  key: string | symbol,\n  descriptor: PropertyDescriptor,\n  object: object,\n  state: State,\n): ObjectEntry => {\n  const descriptorEntrySources: string[] = []\n\n  let isCircular: true | undefined\n  let isGetSet: true | undefined\n  for (const key of UNKNOWN_DESCRIPTOR_KEYS) {\n    if (!(key in descriptor)) {\n      continue\n    }\n\n    const value = descriptor[key] as unknown\n    if (\n      GET_SET_DESCRIPTOR_KEYS.includes(\n        key as (typeof GET_SET_DESCRIPTOR_KEYS)[number],\n      )\n    ) {\n      if (value == null && isGetSet) {\n        // We only need one of `get` and `set` to be set in order for the\n        // descriptor to not be a data descriptor. If `get` or `set` is already\n        // set, and this other one is `undefined`, then setting it wouldn't do\n        // anything.\n        continue\n      }\n      isGetSet = true\n    }\n\n    let result = unevalInternal(value, state)\n    if (result === undefined) {\n      // Skip properties with omitted values.\n      continue\n    }\n    if (result === null) {\n      isCircular = true\n      result = bindingName(value as object, state)\n    }\n\n    descriptorEntrySources.push(`${key}:${result}`)\n  }\n\n  // Set any non-`false` (the default) descriptor values.\n  for (const key of BOOLEAN_DESCRIPTOR_KEYS) {\n    if (descriptor[key]) {\n      descriptorEntrySources.push(`${key}:!0`)\n    }\n  }\n  const descriptorSource = `{${descriptorEntrySources.join()}}`\n\n  if (!isCircular) {\n    return {\n      _source: `${unevalObjectLiteralKey(key, state)._source}:${descriptorSource}`,\n    }\n  }\n\n  return {\n    _isCircular: true,\n    // This is a placeholder property for preserving property order. We'll set\n    // the property's actual value later with the mutation.\n    //\n    // We leave all descriptor properties defaulting to `false` and `undefined`\n    // so we only need to explicitly list the `true` and non-`undefined`\n    // descriptor properties in the mutation. However, we must set\n    // `configurable: true` because otherwise we can't modify the property\n    // later at all.\n    _source: `${unevalObjectLiteralKey(key, state)._source}:{configurable:!0}`,\n    _mutation: hasPlaceholder => {\n      const objectName = bindingName(object, state)\n\n      // When a placeholder with `configurable: true` was emitted, the\n      // mutation must explicitly set `configurable: false` to overwrite it.\n      const mutationDescriptorSource =\n        hasPlaceholder && !descriptor.configurable\n          ? `{${[...descriptorEntrySources, `configurable:!1`].join()}}`\n          : descriptorSource\n\n      return {\n        _source: `Object.defineProperty(${\n          objectName\n        },${unevalInternal(key, state)},${mutationDescriptorSource})`,\n        // `Object.defineProperty` always returns the input object.\n        _evaluatesTo: objectName,\n      }\n    },\n  }\n}\n\ntype ObjectEntry = { _source: string } & (\n  | { _isCircular?: never }\n  | { _isCircular: true; _mutation: (hasPlaceholder: boolean) => Mutation }\n)\n\nconst unevalObjectLiteralKey = (\n  key: string | symbol,\n  state: State,\n): { _source: string; _isIdentifier?: boolean } => {\n  if (\n    typeof key == `symbol` ||\n    // `{ ['__proto__']: ...}` is a hack for setting `__proto__` as an own\n    // property rather than setting `Object.prototype`.\n    key == __PROTO__\n  ) {\n    return { _source: `[${unevalInternal(key, state)!}]` }\n  }\n\n  // The vast majority of keys are non-numeric so don't bother with the\n  // expensive numeric key check below if the key is definitely not numeric\n  // based on the first character.\n  const firstChar = key[0]!\n  if (firstChar >= `0` && firstChar <= `9`) {\n    const number = +key\n    const isNumericKey =\n      // Negative numbers must be quoted.\n      number >= 0 &&\n      // Numbers that can't be safely represented as integers (e.g. because\n      // they are too large) must be quoted.\n      Number.isSafeInteger(number) &&\n      // If the key doesn't roundtrip through numeric conversion, then it's\n      // padded (e.g. `01`) and must be quoted to retain that.\n      key == `${number}`\n    return { _source: isNumericKey ? key : unevalInternal(key, state)! }\n  }\n\n  if (PROPERTY_REG_EXP.test(key)) {\n    return { _source: key, _isIdentifier: true }\n  }\n\n  return { _source: unevalInternal(key, state)! }\n}\n\nexport const isDefaultObjectPrototype = (value: unknown): boolean =>\n  value == Object.prototype ||\n  (isObject(value) &&\n    ownKeysString(value) == DEFAULT_OBJECT_PROTOTYPE_KEYS_STRING)\n\nexport const isObject = (value: unknown): value is object => {\n  const type = typeof value\n  return (type == `object` && !!value) || type == `function`\n}\n\nconst ownKeysString = (value: object) =>\n  Object.getOwnPropertyNames(value).sort().join(`\\0`)\n\nconst __PROTO__ = `__proto__`\nconst DEFAULT_OBJECT_PROTOTYPE_KEYS_STRING = ownKeysString(Object.prototype)\n\nconst BOOLEAN_DESCRIPTOR_KEYS = [\n  `configurable`,\n  `enumerable`,\n  `writable`,\n] as const\nconst GET_SET_DESCRIPTOR_KEYS = [`get`, `set`] as const\nconst UNKNOWN_DESCRIPTOR_KEYS = [`value`, ...GET_SET_DESCRIPTOR_KEYS] as const\n","// For smaller bundle size.\n/* eslint-disable eqeqeq */\n\nimport { isObject, unevalObject } from './object.ts'\nimport {\n  STRING_CODE_UNIT_ESCAPES,\n  unevalBigint,\n  unevalBoolean,\n  unevalLiteral,\n  unevalNumber,\n  unevalSymbol,\n} from './primitive.ts'\nimport type { State } from './types.ts'\n\nexport const unevalInternal = ((\n  value: unknown,\n  state: State,\n): string | null | undefined => {\n  // Don't check the custom source for objects now because we may need to create\n  // a binding in `unevalObject`.\n  if (!isObject(value)) {\n    const customSource = state._customSources.get(value)\n    if (customSource === null) {\n      // The user decided to omit this value.\n      return undefined\n    }\n    if (customSource !== undefined) {\n      // The user provided a custom source for this value.\n      return customSource\n    }\n  }\n\n  const type = typeof value\n  if (type == `undefined`) {\n    return `void 0`\n  } else if (type == `boolean`) {\n    return unevalBoolean(value as boolean)\n  } else if (type == `number`) {\n    return unevalNumber(value as number)\n  } else if (type == `bigint`) {\n    return unevalBigint(value as bigint)\n  } else if (type == `string`) {\n    return `\"${unevalLiteral(value as string, STRING_CODE_UNIT_ESCAPES)}\"`\n  } else if (type == `symbol`) {\n    return unevalSymbol(value as symbol, state)\n  } else {\n    return value == null ? `null` : unevalObject(value, state)\n  }\n}) as {\n  (\n    value: string | number | boolean | bigint | symbol | null | undefined,\n    state: State,\n  ): string | undefined\n  (value: unknown, state: State): string | null | undefined\n}\n","// For smaller bundle size.\n/* eslint-disable eqeqeq */\n\nimport type { TypedArray } from './internal/buffer.ts'\nimport { generateIdentifier } from './internal/identifier.ts'\nimport { unevalInternal } from './internal/index.ts'\nimport { isDefaultObjectPrototype, isObject } from './internal/object.ts'\nimport {\n  getType,\n  T_ARRAY,\n  T_ARRAY_BUFFER,\n  T_BUFFER,\n  T_MAP,\n  T_PRIMITIVE_WRAPPER,\n  T_SET,\n  T_TYPED_ARRAY,\n} from './internal/type.ts'\nimport type { Binding, State } from './internal/types.ts'\n\n/** Options for {@link uneval}. */\nexport type UnevalOptions = {\n  /**\n   * A custom function for unevaling values.\n   *\n   * Return the following types depending on the desired behavior:\n   * - `string` to provide a custom uneval result for {@link value}\n   * - `null` to omit {@link value} from the output (e.g. in arrays, objects,\n   *   Sets, Maps). Omitting the root value throws an `Error`.\n   * - `undefined` (or don't return anything, which is equivalent) to use the\n   *   default behavior for {@link value}\n   *\n   * It can be used to uneval already supported values differently or to uneval\n   * unsupported values such as functions.\n   *\n   * The {@link uneval} param is the same uneval function these options were\n   * passed to. You may use it to delegate back to uneval for sub-objects.\n   */\n  custom?: (\n    value: unknown,\n    uneval: (value: unknown) => string,\n  ) => string | null | undefined\n}\n\n// TODO(#51): Support arbitrary properties on random types (e.g. Arrays, etc.)\n/**\n * Converts the given {@link value} to JavaScript source code.\n *\n * @example\n * ```js\n * import assert from 'node:assert'\n * import uneval from '@tomer/uneval'\n *\n * const object = { message: `hello world` }\n *\n * const source = uneval(object)\n * console.log(source)\n * //=> {message:\"hello world\"}\n *\n * const roundtrippedObject = (0, eval)(`(${source})`)\n * assert.deepEqual(roundtrippedObject, object)\n *\n * const circularObject = {}\n * circularObject.self = circularObject\n *\n * const circularSource = uneval(circularObject)\n * console.log(circularSource)\n * //=> (a=>a.self=a)({})\n *\n * const roundtrippedCircularObject = (0, eval)(`(${circularSource})`)\n * assert.deepEqual(roundtrippedCircularObject, circularObject)\n * ```\n */\nconst uneval = (value: unknown, { custom }: UnevalOptions = {}): string => {\n  const state = createState(value, custom)\n  const result = unevalInternal(value, state)\n  if (result === undefined) {\n    throw new Error(`Root omitted`)\n  }\n\n  if (!state._bindings.size) {\n    // If no bindings were created, then it's impossible for the returned source\n    // to reference one, so it must be a string.\n    return result!\n  }\n\n  const bodySources = state._mutations.map(result => result._source)\n  const returnBinding = state._bindings.get(\n    // The value must be an object if we ended up needing to create bindings for\n    // it. A primitive value would always result in fully inline source above.\n    value as object,\n  )\n  if (!returnBinding) {\n    // The value the returned source evaluates to doesn't have a binding, so we\n    // add its source inline at the end of the body expression to return it.\n    bodySources.push(\n      // Guaranteed to be non-null because that only happens when it's circular,\n      // which it can't be if it has no binding.\n      result!,\n    )\n  } else if (state._mutations.at(-1)?._evaluatesTo != returnBinding._name) {\n    // The value the returned source evaluates to does have a binding, but the\n    // last mutation in the body does not evaluate to it, so we have to add a\n    // reference to the value's binding at the end of the body expression to\n    // return it.\n    bodySources.push(returnBinding._name)\n  }\n\n  let bodySource = bodySources.join()\n  // eslint-disable-next-line @typescript-eslint/prefer-string-starts-ends-with\n  if (bodySources.length > 1 || bodySource[0] == `{`) {\n    // If the body is a comma expression, then it requires a parentheses to be a\n    // syntactically correct expression return. If the body is an object, then\n    // it also needs parentheses so that it isn't interpreted as a block.\n    bodySource = `(${bodySource})`\n  }\n\n  // We use an IIFE's lambda parameters to declare bindings because it's fewer\n  // bytes than using `var` or `let`. We place the binding values in the call\n  // arguments for the leading bindings that don't have dependencies and place\n  // in the default parameter values for the rest. This is because default\n  // parameter values require more bytes than call arguments (due to the `=`).\n  const parameterSources: string[] = []\n  const argSources: string[] = []\n  let hasDependency: boolean | undefined\n  for (const binding of state._bindingOrder) {\n    // eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing\n    hasDependency ||= binding._hasDependency\n    if (hasDependency) {\n      parameterSources.push(`${binding._name}=${binding._source}`)\n    } else {\n      parameterSources.push(binding._name)\n      argSources.push(binding._source!)\n    }\n  }\n\n  const parametersSource = parameterSources.join()\n  return `(${\n    parameterSources.length > 1 ? `(${parametersSource})` : parametersSource\n  }=>${bodySource})(${argSources.join()})`\n}\n\nconst createState = (\n  value: unknown,\n  custom: UnevalOptions[`custom`],\n): State => {\n  const bindings = new Map<object, Binding>()\n  const customSources = new Map<unknown, string | null>()\n\n  const ensureBinding = (value: object) => {\n    if (!bindings.has(value)) {\n      bindings.set(value, { _name: generateIdentifier(bindings.size) })\n    }\n  }\n\n  // Values seen at any point during traversal. Used to detect circular and\n  // shared references.\n  const SOMEWHERE = 1\n  const PARENT = 2\n  const seenLocation = new Map<object, typeof SOMEWHERE | typeof PARENT>()\n\n  const traverse = (value: unknown, parent?: object) => {\n    if (custom) {\n      let source = customSources.get(value)\n      if (source === undefined) {\n        source = custom(value, value => uneval(value, { custom }))\n        if (source !== undefined) {\n          customSources.set(value, source)\n        }\n      }\n      if (source === null) {\n        return\n      }\n    }\n\n    if (!value || typeof value != `object`) {\n      // Note that we purposefully do not traverse functions here because we\n      // don't support them natively.\n      return\n    }\n\n    const location = seenLocation.get(value)\n    if (location) {\n      // If a object value is used more than once, then it needs a binding for\n      // the shared reference.\n      ensureBinding(value)\n\n      if (location == PARENT) {\n        // If this value is referenced circularly, then we'll need a binding for\n        // its parent so that we can mutate it later to attach the circular\n        // reference.\n        ensureBinding(parent!)\n      }\n      return\n    }\n\n    seenLocation.set(value, PARENT)\n    traverseObject(value)\n    seenLocation.set(value, SOMEWHERE)\n  }\n\n  const traverseObject = (value: object) => {\n    const [type] = getType(value)\n    if (type == undefined) {\n      for (const key of Reflect.ownKeys(value)) {\n        if (typeof key == `symbol`) {\n          traverse(key)\n        }\n        const descriptor = Object.getOwnPropertyDescriptor(value, key)!\n        traverse(descriptor.value as unknown, value)\n        // eslint-disable-next-line @typescript-eslint/unbound-method\n        traverse(descriptor.get, value)\n        // eslint-disable-next-line @typescript-eslint/unbound-method\n        traverse(descriptor.set, value)\n      }\n\n      const prototype = Object.getPrototypeOf(value) as unknown\n      if (!isDefaultObjectPrototype(prototype)) {\n        // We only render the prototype if it's not equal the default one in\n        // this realm.\n        traverse(prototype, value)\n      }\n    } else if (type == T_PRIMITIVE_WRAPPER) {\n      // Traverse the underlying unboxed value to apply `custom` to it.\n      const underlyingValue = value.valueOf()\n      traverse(underlyingValue)\n      if (customSources.get(underlyingValue) === null) {\n        customSources.set(value, null)\n      }\n    } else if (type == T_ARRAY) {\n      // Use `Object.keys` to avoid iterating empty slots, which are no-ops for\n      // traversal, and to safely handle huge sparse arrays without DoS.\n      for (const key of Object.keys(value)) {\n        traverse((value as Record<string, unknown>)[key], value)\n      }\n    } else if (type == T_SET) {\n      for (const item of value as Set<unknown>) {\n        traverse(item, value)\n      }\n    } else if (type == T_MAP) {\n      for (const [key, item] of value as Map<unknown, unknown>) {\n        traverse(key, value)\n        traverse(item, value)\n\n        if (\n          seenLocation.get(\n            // Not guaranteed to be an object, but that doesn't matter because\n            // it's just a lookup, and this saves bytes.\n            item as object,\n          ) == PARENT &&\n          isObject(key)\n        ) {\n          // If the item is circular and the key is an object, then the key also\n          // needs a binding so we can reference it in mutations.\n          ensureBinding(key)\n        }\n      }\n    } else if (type == T_ARRAY_BUFFER) {\n      // TODO(#42): Support TypedArrays containing detached ArrayBuffers.\n      const arrayBuffer = value as ArrayBuffer\n      if (\n        // If the `ArrayBuffer` is detached, then we need a binding to call\n        // `.transfer()` on.\n        arrayBuffer.detached ||\n        // If the `ArrayBuffer` is resizable and has non-zero values, then\n        // we'll have to construct it, put it in a binding, then `.set(...)` it.\n        (arrayBuffer.resizable &&\n          new Uint8Array(arrayBuffer).some(value => value != 0))\n      ) {\n        ensureBinding(value)\n      }\n    } else if (type == T_BUFFER || type == T_TYPED_ARRAY) {\n      const { buffer } = value as Buffer | TypedArray\n      traverse(buffer, value)\n      if (customSources.get(buffer) === null) {\n        customSources.set(value, null)\n      }\n    }\n  }\n\n  traverse(value)\n\n  return {\n    _bindings: bindings,\n    _customSources: customSources,\n    _currentParents: new Set(),\n    _bindingOrder: [],\n    _mutations: [],\n  }\n}\n\nexport default uneval\n"],"mappings":"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"}
|
package/license
ADDED
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2025-2026 Tomer Aberbach
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and
|
|
6
|
+
associated documentation files (the "Software"), to deal in the Software without restriction, including
|
|
7
|
+
without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
8
|
+
copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the
|
|
9
|
+
following conditions:
|
|
10
|
+
|
|
11
|
+
The above copyright notice and this permission notice shall be included in all copies or substantial
|
|
12
|
+
portions of the Software.
|
|
13
|
+
|
|
14
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT
|
|
15
|
+
LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO
|
|
16
|
+
EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
|
17
|
+
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
|
18
|
+
USE OR OTHER DEALINGS IN THE SOFTWARE.
|
package/package.json
ADDED
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@tomer/uneval",
|
|
3
|
+
"version": "0.0.1",
|
|
4
|
+
"author": {
|
|
5
|
+
"name": "Tomer Aberbach",
|
|
6
|
+
"email": "tomer@aberba.ch",
|
|
7
|
+
"url": "https://tomeraberba.ch"
|
|
8
|
+
},
|
|
9
|
+
"description": "Convert a JS value to JS source code, like eval in reverse.",
|
|
10
|
+
"keywords": [
|
|
11
|
+
"eval",
|
|
12
|
+
"source",
|
|
13
|
+
"stringify",
|
|
14
|
+
"serialize",
|
|
15
|
+
"roundtrip",
|
|
16
|
+
"uneval"
|
|
17
|
+
],
|
|
18
|
+
"homepage": "https://github.com/TomerAberbach/uneval",
|
|
19
|
+
"repository": "TomerAberbach/uneval",
|
|
20
|
+
"bugs": {
|
|
21
|
+
"url": "https://github.com/TomerAberbach/uneval/issues"
|
|
22
|
+
},
|
|
23
|
+
"funding": {
|
|
24
|
+
"url": "https://github.com/sponsors/TomerAberbach"
|
|
25
|
+
},
|
|
26
|
+
"license": "MIT",
|
|
27
|
+
"files": [
|
|
28
|
+
"dist"
|
|
29
|
+
],
|
|
30
|
+
"type": "module",
|
|
31
|
+
"sideEffects": false,
|
|
32
|
+
"engines": {
|
|
33
|
+
"node": ">= 22"
|
|
34
|
+
},
|
|
35
|
+
"exports": {
|
|
36
|
+
".": {
|
|
37
|
+
"types": "./dist/index.d.mts",
|
|
38
|
+
"default": "./dist/index.js"
|
|
39
|
+
},
|
|
40
|
+
"./package.json": "./package.json"
|
|
41
|
+
},
|
|
42
|
+
"prettier": "@tomer/prettier-config",
|
|
43
|
+
"devDependencies": {
|
|
44
|
+
"@fast-check/poisoning": "^0.2.3",
|
|
45
|
+
"@fast-check/vitest": "^0.2.4",
|
|
46
|
+
"@js-temporal/polyfill": "^0.5.1",
|
|
47
|
+
"@rollup/plugin-terser": "^1.0.0",
|
|
48
|
+
"@tomer/eslint-config": "^4.6.1",
|
|
49
|
+
"@tomer/prettier-config": "^4.0.0",
|
|
50
|
+
"@types/jsesc": "^3.0.3",
|
|
51
|
+
"@types/serialize-javascript": "^5.0.4",
|
|
52
|
+
"@vitest/coverage-v8": "^4.0.6",
|
|
53
|
+
"@vitest/expect": "^4.0.6",
|
|
54
|
+
"devalue": "^5.6.3",
|
|
55
|
+
"eslint": "^9.39.2",
|
|
56
|
+
"jest-extended": "^7.0.0",
|
|
57
|
+
"jsdom": "^28.1.0",
|
|
58
|
+
"jsesc": "^3.1.0",
|
|
59
|
+
"magic-string": "^0.30.21",
|
|
60
|
+
"prettier": "^3.8.1",
|
|
61
|
+
"publint": "^0.3.18",
|
|
62
|
+
"serialize-javascript": "^7.0.4",
|
|
63
|
+
"tosource": "2.0.0-alpha.3",
|
|
64
|
+
"tsdown": "^0.20.3",
|
|
65
|
+
"typescript": "^5.9.3",
|
|
66
|
+
"vitest": "^4.0.6"
|
|
67
|
+
},
|
|
68
|
+
"scripts": {
|
|
69
|
+
"format": "prettier --cache --write .",
|
|
70
|
+
"lint": "eslint --cache --cache-location node_modules/.cache/eslint/ --fix .",
|
|
71
|
+
"typecheck": "tsc --noEmit",
|
|
72
|
+
"test": "vitest",
|
|
73
|
+
"coverage": "vitest --coverage",
|
|
74
|
+
"bench": "vitest bench",
|
|
75
|
+
"build": "tsdown",
|
|
76
|
+
"generate-comparison-table": "node scripts/generate-comparison-table.ts",
|
|
77
|
+
"check-comparison-table": "node scripts/generate-comparison-table.ts --check"
|
|
78
|
+
}
|
|
79
|
+
}
|