json-web3 0.1.1 → 1.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/README.md CHANGED
@@ -20,16 +20,22 @@ import jsonWeb3 from 'json-web3'
20
20
  const payload = {
21
21
  balance: 1234567890123456789n,
22
22
  decimals: 18n,
23
+ u8Array: new Uint8Array([1, 2, 3, 255]),
24
+ u16Array: new Uint16Array([1, 2, 3, 255]),
25
+ bigIntArray: new BigInt64Array([18446744073709551615n, 2n, 3n, 255n]),
23
26
  }
24
27
 
25
28
  const text = jsonWeb3.stringify(payload)
26
- // => {"balance":{"__@json.bigint__":"1234567890123456789"},"decimals":{"__@json.bigint__":"18"}}
29
+ // => {"balance":{"__@json.bigint__":"1234567890123456789"},"decimals":{"__@json.bigint__":"18"},"u8Array":{"__@json.typedarray__":{"type":"Uint8Array","bytes":"0x010203ff"}},"u16Array":{"__@json.typedarray__":{"type":"Uint16Array","bytes":"0x010002000300ff00"}},"bigIntArray":{"__@json.typedarray__":{"type":"BigInt64Array","bytes":"0xffffffffffffffff02000000000000000300000000000000ff00000000000000"}}}
27
30
 
28
31
  const restored = jsonWeb3.parse(text)
29
32
  /* =>
30
33
  {
31
- "balance": 1234567890123456789n,
32
- "decimals": 18n
34
+ balance: 1234567890123456789n,
35
+ decimals: 18n,
36
+ u8Array: Uint8Array(4) [1, 2, 3, 255]...,
37
+ u16Array: Uint16Array(4)...,
38
+ bigIntArray: BigInt64Array(4)...
33
39
  }
34
40
  */
35
41
  ```
@@ -37,8 +43,8 @@ const restored = jsonWeb3.parse(text)
37
43
  ## API (Fully compatible with native globalThis.JSON)
38
44
 
39
45
  - `stringify(value, replacer?, space?)`
40
- - `parse(text)`
46
+ - `parse(text, reviver?)`
41
47
 
42
48
  ## Notes
43
49
 
44
- `bigint` values are encoded as objects: `{"__@json.bigint__":"<value>"}`. More types can be added later.
50
+ `bigint` values are encoded as objects: `{"__@json.bigint__":"<value>"}`. `ArrayBuffer`, Node `Buffer` JSON shapes, and typed arrays are encoded as `{"__@json.typedarray__":{"type":"<Name>","bytes":"0x..."}}` and decoded back to the original typed array (`Uint8Array`, `Uint8ClampedArray`, `Uint16Array`, `Uint32Array`, `Int8Array`, `Int16Array`, `Int32Array`, `Float32Array`, `Float64Array`, `BigInt64Array`, `BigUint64Array`).
package/dist/index.cjs CHANGED
@@ -25,32 +25,142 @@ __export(index_exports, {
25
25
  stringify: () => stringify
26
26
  });
27
27
  module.exports = __toCommonJS(index_exports);
28
+
29
+ // src/utils.ts
28
30
  var BIGINT_TAG = "__@json.bigint__";
31
+ var TYPEDARRAY_TAG = "__@json.typedarray__";
32
+ var undefined = void 0;
33
+ var isUndefined = (value) => value === undefined;
34
+ var isString = (value) => typeof value === "string";
35
+ var hasOwnProperty = (obj, prop) => Object.prototype.hasOwnProperty.call(obj, prop);
36
+ var isArray = (value) => Array.isArray(value);
37
+ var isObject = (value) => value !== null && typeof value === "object" && !isArray(value);
38
+ var isFunction = (value) => typeof value === "function";
39
+ var isBigInt = (value) => typeof value === "bigint";
40
+ var hasBuffer = () => typeof Buffer !== "undefined";
41
+ var isArrayBuffer = (value) => !isUndefined(ArrayBuffer) && value instanceof ArrayBuffer;
42
+ var isTypedArray = (value) => {
43
+ if (isUndefined(value)) return false;
44
+ if (!isUndefined(ArrayBuffer) && isFunction(ArrayBuffer.isView) && ArrayBuffer.isView(value)) {
45
+ return Object.prototype.toString.call(value) !== "[object DataView]";
46
+ }
47
+ const tag = Object.prototype.toString.call(value);
48
+ return tag.endsWith("Array]") && tag !== "[object Array]";
49
+ };
50
+ var getTypedArrayName = (value) => {
51
+ if (isUndefined(value)) return null;
52
+ const ctor = value.constructor;
53
+ if (ctor && isString(ctor.name)) return ctor.name;
54
+ const tag = Object.prototype.toString.call(value);
55
+ const match = /^\[object (.+)\]$/.exec(tag);
56
+ return match ? match[1] : null;
57
+ };
58
+ var TYPEDARRAY_CTORS = {
59
+ Uint8Array: !isUndefined(Uint8Array) ? Uint8Array : undefined,
60
+ Uint8ClampedArray: !isUndefined(Uint8ClampedArray) ? Uint8ClampedArray : undefined,
61
+ Uint16Array: !isUndefined(Uint16Array) ? Uint16Array : undefined,
62
+ Uint32Array: !isUndefined(Uint32Array) ? Uint32Array : undefined,
63
+ Int8Array: !isUndefined(Int8Array) ? Int8Array : undefined,
64
+ Int16Array: !isUndefined(Int16Array) ? Int16Array : undefined,
65
+ Int32Array: !isUndefined(Int32Array) ? Int32Array : undefined,
66
+ Float32Array: !isUndefined(Float32Array) ? Float32Array : undefined,
67
+ Float64Array: !isUndefined(Float64Array) ? Float64Array : undefined,
68
+ BigInt64Array: !isUndefined(BigInt64Array) ? BigInt64Array : undefined,
69
+ BigUint64Array: !isUndefined(BigUint64Array) ? BigUint64Array : undefined
70
+ };
71
+ var toHex = (value) => {
72
+ let out = "0x";
73
+ for (const byte of value) {
74
+ out += byte.toString(16).padStart(2, "0");
75
+ }
76
+ return out;
77
+ };
78
+ var fromHex = (hex) => {
79
+ const normalized = hex.startsWith("0x") ? hex.slice(2) : hex;
80
+ if (normalized.length % 2 !== 0) {
81
+ throw new Error("Invalid hex string for bytes");
82
+ }
83
+ if (hasBuffer()) {
84
+ return Uint8Array.from(Buffer.from(normalized, "hex"));
85
+ }
86
+ const out = new Uint8Array(normalized.length / 2);
87
+ for (let i = 0; i < normalized.length; i += 2) {
88
+ out[i / 2] = Number.parseInt(normalized.slice(i, i + 2), 16);
89
+ }
90
+ return out;
91
+ };
92
+ var toSerializable = (value) => {
93
+ if (isBigInt(value)) {
94
+ return { [BIGINT_TAG]: value.toString() };
95
+ }
96
+ if (isTypedArray(value)) {
97
+ const type = getTypedArrayName(value);
98
+ if (type && TYPEDARRAY_CTORS[type]) {
99
+ const bytes = new Uint8Array(value.buffer, value.byteOffset, value.byteLength);
100
+ return { [TYPEDARRAY_TAG]: { type, bytes: toHex(bytes) } };
101
+ }
102
+ }
103
+ if (isArrayBuffer(value)) {
104
+ return { [TYPEDARRAY_TAG]: { type: "Uint8Array", bytes: toHex(new Uint8Array(value)) } };
105
+ }
106
+ if (value && isObject(value) && value.type === "Buffer" && isArray(value.data) && Object.keys(value).length === 2 && value.data.every((entry) => Number.isInteger(entry) && entry >= 0 && entry <= 255)) {
107
+ return { [TYPEDARRAY_TAG]: { type: "Uint8Array", bytes: toHex(Uint8Array.from(value.data)) } };
108
+ }
109
+ return value;
110
+ };
111
+ var fromSerializable = (value) => {
112
+ if (value && isObject(value)) {
113
+ if (hasOwnProperty(value, BIGINT_TAG)) {
114
+ return BigInt(value[BIGINT_TAG]);
115
+ }
116
+ if (hasOwnProperty(value, TYPEDARRAY_TAG)) {
117
+ const payload = value[TYPEDARRAY_TAG];
118
+ if (!payload || !isObject(payload) || !isString(payload.type) || !isString(payload.bytes)) {
119
+ throw new Error("Invalid typed array payload");
120
+ }
121
+ const bytes = fromHex(payload.bytes);
122
+ const ctor = TYPEDARRAY_CTORS[payload.type];
123
+ if (!ctor) return bytes;
124
+ const buffer = new ArrayBuffer(bytes.byteLength);
125
+ new Uint8Array(buffer).set(bytes);
126
+ return new ctor(buffer);
127
+ }
128
+ if (value.type === "Buffer" && isArray(value.data) && Object.keys(value).length === 2 && value.data.every((entry) => Number.isInteger(entry) && entry >= 0 && entry <= 255)) {
129
+ return Uint8Array.from(value.data);
130
+ }
131
+ }
132
+ return value;
133
+ };
134
+ var isTypedArrayPayload = (value) => isObject(value) && isString(value.type) && isString(value.bytes) && Object.keys(value).length === 2;
135
+
136
+ // src/index.ts
29
137
  var RAW_JSON = JSON;
30
- var applyReplacer = (key, value, replacer) => {
31
- if (typeof replacer === "function") {
138
+ var applyReplacer = (holder, key, value, replacer) => {
139
+ if (isFunction(replacer)) {
32
140
  return replacer(key, value);
33
141
  }
34
- if (Array.isArray(replacer)) {
142
+ if (isArray(replacer)) {
35
143
  if (key === "") return value;
36
144
  if (key === BIGINT_TAG) return value;
37
- return replacer.includes(key) ? value : void 0;
145
+ if (key === TYPEDARRAY_TAG) return value;
146
+ if (isTypedArrayPayload(holder)) return value;
147
+ if (isArray(holder)) return value;
148
+ return replacer.includes(key) ? value : undefined;
38
149
  }
39
150
  return value;
40
151
  };
41
- var toSerializable = (value) => typeof value === "bigint" ? { [BIGINT_TAG]: value.toString() } : value;
42
152
  var stringify = (value, replacer = null, space) => RAW_JSON.stringify(
43
153
  value,
44
- (key, v) => {
45
- const replaced = applyReplacer(key, v, replacer);
154
+ function replacerFn(key, v) {
155
+ const replaced = applyReplacer(this, key, v, replacer);
46
156
  return toSerializable(replaced);
47
157
  },
48
158
  space
49
159
  );
50
- var parse = (text) => RAW_JSON.parse(
51
- text,
52
- (_, v) => v && typeof v === "object" && BIGINT_TAG in v ? BigInt(v[BIGINT_TAG]) : v
53
- );
160
+ var parse = (text, reviver = null) => RAW_JSON.parse(text, (key, v) => {
161
+ const decoded = fromSerializable(v);
162
+ return isFunction(reviver) ? reviver(key, decoded) : decoded;
163
+ });
54
164
  var index_default = {
55
165
  stringify,
56
166
  parse
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts"],"sourcesContent":["const BIGINT_TAG = '__@json.bigint__'\nconst RAW_JSON = JSON\ntype Replacer = ((this: any, key: string, value: any) => any) | Array<string | number> | null\n\nconst applyReplacer = (key: string, value: any, replacer: Replacer): any => {\n if (typeof replacer === 'function') {\n return replacer(key, value)\n }\n if (Array.isArray(replacer)) {\n if (key === '') return value\n if (key === BIGINT_TAG) return value\n return replacer.includes(key) ? value : undefined\n }\n return value\n}\n\nconst toSerializable = (value: any): any =>\n typeof value === 'bigint' ? { [BIGINT_TAG]: value.toString() } : value\n\nexport const stringify = (value: any, replacer: Replacer = null, space?: string | number): string =>\n RAW_JSON.stringify(\n value,\n (key, v) => {\n const replaced = applyReplacer(key, v, replacer)\n return toSerializable(replaced)\n },\n space,\n )\n\nexport const parse = (text: string): any =>\n RAW_JSON.parse(text, (_, v) =>\n v && typeof v === 'object' && BIGINT_TAG in v ? BigInt(v[BIGINT_TAG]) : v,\n )\n\nexport default {\n stringify,\n parse,\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAAM,aAAa;AACnB,IAAM,WAAW;AAGjB,IAAM,gBAAgB,CAAC,KAAa,OAAY,aAA4B;AAC1E,MAAI,OAAO,aAAa,YAAY;AAClC,WAAO,SAAS,KAAK,KAAK;AAAA,EAC5B;AACA,MAAI,MAAM,QAAQ,QAAQ,GAAG;AAC3B,QAAI,QAAQ,GAAI,QAAO;AACvB,QAAI,QAAQ,WAAY,QAAO;AAC/B,WAAO,SAAS,SAAS,GAAG,IAAI,QAAQ;AAAA,EAC1C;AACA,SAAO;AACT;AAEA,IAAM,iBAAiB,CAAC,UACtB,OAAO,UAAU,WAAW,EAAE,CAAC,UAAU,GAAG,MAAM,SAAS,EAAE,IAAI;AAE5D,IAAM,YAAY,CAAC,OAAY,WAAqB,MAAM,UAC/D,SAAS;AAAA,EACP;AAAA,EACA,CAAC,KAAK,MAAM;AACV,UAAM,WAAW,cAAc,KAAK,GAAG,QAAQ;AAC/C,WAAO,eAAe,QAAQ;AAAA,EAChC;AAAA,EACA;AACF;AAEK,IAAM,QAAQ,CAAC,SACpB,SAAS;AAAA,EAAM;AAAA,EAAM,CAAC,GAAG,MACvB,KAAK,OAAO,MAAM,YAAY,cAAc,IAAI,OAAO,EAAE,UAAU,CAAC,IAAI;AAC1E;AAEF,IAAO,gBAAQ;AAAA,EACb;AAAA,EACA;AACF;","names":[]}
1
+ {"version":3,"sources":["../src/index.ts","../src/utils.ts"],"sourcesContent":["import {\n BIGINT_TAG,\n fromSerializable,\n isArray,\n isFunction,\n isTypedArrayPayload,\n toSerializable,\n TYPEDARRAY_TAG,\n undefined,\n} from './utils'\n\nconst RAW_JSON = JSON\ntype Replacer = ((this: any, key: string, value: any) => any) | Array<string | number> | null\ntype Reviver = ((this: any, key: string, value: any) => any) | null\n\nconst applyReplacer = (holder: any, key: string, value: any, replacer: Replacer): any => {\n if (isFunction(replacer)) {\n return replacer(key, value)\n }\n if (isArray(replacer)) {\n if (key === '') return value\n if (key === BIGINT_TAG) return value\n if (key === TYPEDARRAY_TAG) return value\n if (isTypedArrayPayload(holder)) return value\n if (isArray(holder)) return value\n return replacer.includes(key) ? value : undefined\n }\n return value\n}\n\nexport const stringify = (value: any, replacer: Replacer = null, space?: string | number): string =>\n RAW_JSON.stringify(\n value,\n function replacerFn(key, v) {\n const replaced = applyReplacer(this, key, v, replacer)\n return toSerializable(replaced)\n },\n space,\n )\n\nexport const parse = (text: string, reviver: Reviver = null): any =>\n RAW_JSON.parse(text, (key, v) => {\n const decoded = fromSerializable(v)\n return isFunction(reviver) ? reviver(key, decoded) : decoded\n })\n\nexport default {\n stringify,\n parse,\n}\n","export const BIGINT_TAG = '__@json.bigint__'\nexport const TYPEDARRAY_TAG = '__@json.typedarray__'\n\nexport const undefined = void 0\nexport const isUndefined = (value: any): value is undefined => value === undefined\nexport const isString = (value: any): value is string => typeof value === 'string'\nexport const hasOwnProperty = (obj: any, prop: string): boolean =>\n Object.prototype.hasOwnProperty.call(obj, prop)\nexport const isArray = (value: any): value is Array<any> => Array.isArray(value)\nexport const isObject = (value: any): value is Record<string, any> =>\n value !== null && typeof value === 'object' && !isArray(value)\nexport const isFunction = (value: any): value is Function => typeof value === 'function'\nexport const isBigInt = (value: any): value is bigint => typeof value === 'bigint'\nconst hasBuffer = (): boolean => typeof Buffer !== 'undefined'\n\nexport const isBuffer = (value: any): value is Buffer =>\n hasBuffer() && isFunction(Buffer.isBuffer) && Buffer.isBuffer(value)\nexport const isArrayBuffer = (value: any): value is ArrayBuffer =>\n !isUndefined(ArrayBuffer) && value instanceof ArrayBuffer\n\nexport const isTypedArray = (value: any): value is ArrayBufferView => {\n if (isUndefined(value)) return false\n if (!isUndefined(ArrayBuffer) && isFunction(ArrayBuffer.isView) && ArrayBuffer.isView(value)) {\n return Object.prototype.toString.call(value) !== '[object DataView]'\n }\n const tag = Object.prototype.toString.call(value)\n return tag.endsWith('Array]') && tag !== '[object Array]'\n}\n\nexport const getTypedArrayName = (value: any): string | null => {\n if (isUndefined(value)) return null\n const ctor = (value as any).constructor\n if (ctor && isString(ctor.name)) return ctor.name\n const tag = Object.prototype.toString.call(value)\n const match = /^\\[object (.+)\\]$/.exec(tag)\n return match ? match[1] : null\n}\n\nconst TYPEDARRAY_CTORS: Record<string, any> = {\n Uint8Array: !isUndefined(Uint8Array) ? Uint8Array : undefined,\n Uint8ClampedArray: !isUndefined(Uint8ClampedArray) ? Uint8ClampedArray : undefined,\n Uint16Array: !isUndefined(Uint16Array) ? Uint16Array : undefined,\n Uint32Array: !isUndefined(Uint32Array) ? Uint32Array : undefined,\n Int8Array: !isUndefined(Int8Array) ? Int8Array : undefined,\n Int16Array: !isUndefined(Int16Array) ? Int16Array : undefined,\n Int32Array: !isUndefined(Int32Array) ? Int32Array : undefined,\n Float32Array: !isUndefined(Float32Array) ? Float32Array : undefined,\n Float64Array: !isUndefined(Float64Array) ? Float64Array : undefined,\n BigInt64Array: !isUndefined(BigInt64Array) ? BigInt64Array : undefined,\n BigUint64Array: !isUndefined(BigUint64Array) ? BigUint64Array : undefined,\n}\n\nexport const toHex = (value: Uint8Array): string => {\n let out = '0x'\n for (const byte of value) {\n out += byte.toString(16).padStart(2, '0')\n }\n return out\n}\n\nexport const fromHex = (hex: string): Uint8Array => {\n const normalized = hex.startsWith('0x') ? hex.slice(2) : hex\n if (normalized.length % 2 !== 0) {\n throw new Error('Invalid hex string for bytes')\n }\n\n if (hasBuffer()) {\n return Uint8Array.from(Buffer.from(normalized, 'hex'))\n }\n\n const out = new Uint8Array(normalized.length / 2)\n for (let i = 0; i < normalized.length; i += 2) {\n out[i / 2] = Number.parseInt(normalized.slice(i, i + 2), 16)\n }\n return out\n}\n\nexport const toSerializable = (value: any): any => {\n if (isBigInt(value)) {\n return { [BIGINT_TAG]: value.toString() }\n }\n\n if (isTypedArray(value)) {\n const type = getTypedArrayName(value)\n if (type && TYPEDARRAY_CTORS[type]) {\n const bytes = new Uint8Array(value.buffer, value.byteOffset, value.byteLength)\n return { [TYPEDARRAY_TAG]: { type, bytes: toHex(bytes) } }\n }\n }\n\n if (isArrayBuffer(value)) {\n return { [TYPEDARRAY_TAG]: { type: 'Uint8Array', bytes: toHex(new Uint8Array(value)) } }\n }\n\n if (\n value &&\n isObject(value) &&\n value.type === 'Buffer' &&\n isArray(value.data) &&\n Object.keys(value).length === 2 &&\n value.data.every((entry: any) => Number.isInteger(entry) && entry >= 0 && entry <= 255)\n ) {\n return { [TYPEDARRAY_TAG]: { type: 'Uint8Array', bytes: toHex(Uint8Array.from(value.data)) } }\n }\n\n return value\n}\n\nexport const fromSerializable = (value: any): any => {\n if (value && isObject(value)) {\n if (hasOwnProperty(value, BIGINT_TAG)) {\n return BigInt(value[BIGINT_TAG])\n }\n if (hasOwnProperty(value, TYPEDARRAY_TAG)) {\n const payload = value[TYPEDARRAY_TAG]\n if (!payload || !isObject(payload) || !isString(payload.type) || !isString(payload.bytes)) {\n throw new Error('Invalid typed array payload')\n }\n const bytes = fromHex(payload.bytes)\n const ctor = TYPEDARRAY_CTORS[payload.type]\n if (!ctor) return bytes\n const buffer = new ArrayBuffer(bytes.byteLength)\n new Uint8Array(buffer).set(bytes)\n return new ctor(buffer)\n }\n if (\n value.type === 'Buffer' &&\n isArray(value.data) &&\n Object.keys(value).length === 2 &&\n value.data.every((entry: any) => Number.isInteger(entry) && entry >= 0 && entry <= 255)\n ) {\n return Uint8Array.from(value.data)\n }\n }\n return value\n}\n\nexport const isTypedArrayPayload = (value: any): boolean =>\n isObject(value) &&\n isString(value.type) &&\n isString(value.bytes) &&\n Object.keys(value).length === 2\n"],"mappings":"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","names":[]}
package/dist/index.d.cts CHANGED
@@ -1,9 +1,10 @@
1
1
  type Replacer = ((this: any, key: string, value: any) => any) | Array<string | number> | null;
2
+ type Reviver = ((this: any, key: string, value: any) => any) | null;
2
3
  declare const stringify: (value: any, replacer?: Replacer, space?: string | number) => string;
3
- declare const parse: (text: string) => any;
4
+ declare const parse: (text: string, reviver?: Reviver) => any;
4
5
  declare const _default: {
5
6
  stringify: (value: any, replacer?: Replacer, space?: string | number) => string;
6
- parse: (text: string) => any;
7
+ parse: (text: string, reviver?: Reviver) => any;
7
8
  };
8
9
 
9
10
  export { _default as default, parse, stringify };
package/dist/index.d.ts CHANGED
@@ -1,9 +1,10 @@
1
1
  type Replacer = ((this: any, key: string, value: any) => any) | Array<string | number> | null;
2
+ type Reviver = ((this: any, key: string, value: any) => any) | null;
2
3
  declare const stringify: (value: any, replacer?: Replacer, space?: string | number) => string;
3
- declare const parse: (text: string) => any;
4
+ declare const parse: (text: string, reviver?: Reviver) => any;
4
5
  declare const _default: {
5
6
  stringify: (value: any, replacer?: Replacer, space?: string | number) => string;
6
- parse: (text: string) => any;
7
+ parse: (text: string, reviver?: Reviver) => any;
7
8
  };
8
9
 
9
10
  export { _default as default, parse, stringify };
package/dist/index.js CHANGED
@@ -1,30 +1,138 @@
1
- // src/index.ts
1
+ // src/utils.ts
2
2
  var BIGINT_TAG = "__@json.bigint__";
3
+ var TYPEDARRAY_TAG = "__@json.typedarray__";
4
+ var undefined = void 0;
5
+ var isUndefined = (value) => value === undefined;
6
+ var isString = (value) => typeof value === "string";
7
+ var hasOwnProperty = (obj, prop) => Object.prototype.hasOwnProperty.call(obj, prop);
8
+ var isArray = (value) => Array.isArray(value);
9
+ var isObject = (value) => value !== null && typeof value === "object" && !isArray(value);
10
+ var isFunction = (value) => typeof value === "function";
11
+ var isBigInt = (value) => typeof value === "bigint";
12
+ var hasBuffer = () => typeof Buffer !== "undefined";
13
+ var isArrayBuffer = (value) => !isUndefined(ArrayBuffer) && value instanceof ArrayBuffer;
14
+ var isTypedArray = (value) => {
15
+ if (isUndefined(value)) return false;
16
+ if (!isUndefined(ArrayBuffer) && isFunction(ArrayBuffer.isView) && ArrayBuffer.isView(value)) {
17
+ return Object.prototype.toString.call(value) !== "[object DataView]";
18
+ }
19
+ const tag = Object.prototype.toString.call(value);
20
+ return tag.endsWith("Array]") && tag !== "[object Array]";
21
+ };
22
+ var getTypedArrayName = (value) => {
23
+ if (isUndefined(value)) return null;
24
+ const ctor = value.constructor;
25
+ if (ctor && isString(ctor.name)) return ctor.name;
26
+ const tag = Object.prototype.toString.call(value);
27
+ const match = /^\[object (.+)\]$/.exec(tag);
28
+ return match ? match[1] : null;
29
+ };
30
+ var TYPEDARRAY_CTORS = {
31
+ Uint8Array: !isUndefined(Uint8Array) ? Uint8Array : undefined,
32
+ Uint8ClampedArray: !isUndefined(Uint8ClampedArray) ? Uint8ClampedArray : undefined,
33
+ Uint16Array: !isUndefined(Uint16Array) ? Uint16Array : undefined,
34
+ Uint32Array: !isUndefined(Uint32Array) ? Uint32Array : undefined,
35
+ Int8Array: !isUndefined(Int8Array) ? Int8Array : undefined,
36
+ Int16Array: !isUndefined(Int16Array) ? Int16Array : undefined,
37
+ Int32Array: !isUndefined(Int32Array) ? Int32Array : undefined,
38
+ Float32Array: !isUndefined(Float32Array) ? Float32Array : undefined,
39
+ Float64Array: !isUndefined(Float64Array) ? Float64Array : undefined,
40
+ BigInt64Array: !isUndefined(BigInt64Array) ? BigInt64Array : undefined,
41
+ BigUint64Array: !isUndefined(BigUint64Array) ? BigUint64Array : undefined
42
+ };
43
+ var toHex = (value) => {
44
+ let out = "0x";
45
+ for (const byte of value) {
46
+ out += byte.toString(16).padStart(2, "0");
47
+ }
48
+ return out;
49
+ };
50
+ var fromHex = (hex) => {
51
+ const normalized = hex.startsWith("0x") ? hex.slice(2) : hex;
52
+ if (normalized.length % 2 !== 0) {
53
+ throw new Error("Invalid hex string for bytes");
54
+ }
55
+ if (hasBuffer()) {
56
+ return Uint8Array.from(Buffer.from(normalized, "hex"));
57
+ }
58
+ const out = new Uint8Array(normalized.length / 2);
59
+ for (let i = 0; i < normalized.length; i += 2) {
60
+ out[i / 2] = Number.parseInt(normalized.slice(i, i + 2), 16);
61
+ }
62
+ return out;
63
+ };
64
+ var toSerializable = (value) => {
65
+ if (isBigInt(value)) {
66
+ return { [BIGINT_TAG]: value.toString() };
67
+ }
68
+ if (isTypedArray(value)) {
69
+ const type = getTypedArrayName(value);
70
+ if (type && TYPEDARRAY_CTORS[type]) {
71
+ const bytes = new Uint8Array(value.buffer, value.byteOffset, value.byteLength);
72
+ return { [TYPEDARRAY_TAG]: { type, bytes: toHex(bytes) } };
73
+ }
74
+ }
75
+ if (isArrayBuffer(value)) {
76
+ return { [TYPEDARRAY_TAG]: { type: "Uint8Array", bytes: toHex(new Uint8Array(value)) } };
77
+ }
78
+ if (value && isObject(value) && value.type === "Buffer" && isArray(value.data) && Object.keys(value).length === 2 && value.data.every((entry) => Number.isInteger(entry) && entry >= 0 && entry <= 255)) {
79
+ return { [TYPEDARRAY_TAG]: { type: "Uint8Array", bytes: toHex(Uint8Array.from(value.data)) } };
80
+ }
81
+ return value;
82
+ };
83
+ var fromSerializable = (value) => {
84
+ if (value && isObject(value)) {
85
+ if (hasOwnProperty(value, BIGINT_TAG)) {
86
+ return BigInt(value[BIGINT_TAG]);
87
+ }
88
+ if (hasOwnProperty(value, TYPEDARRAY_TAG)) {
89
+ const payload = value[TYPEDARRAY_TAG];
90
+ if (!payload || !isObject(payload) || !isString(payload.type) || !isString(payload.bytes)) {
91
+ throw new Error("Invalid typed array payload");
92
+ }
93
+ const bytes = fromHex(payload.bytes);
94
+ const ctor = TYPEDARRAY_CTORS[payload.type];
95
+ if (!ctor) return bytes;
96
+ const buffer = new ArrayBuffer(bytes.byteLength);
97
+ new Uint8Array(buffer).set(bytes);
98
+ return new ctor(buffer);
99
+ }
100
+ if (value.type === "Buffer" && isArray(value.data) && Object.keys(value).length === 2 && value.data.every((entry) => Number.isInteger(entry) && entry >= 0 && entry <= 255)) {
101
+ return Uint8Array.from(value.data);
102
+ }
103
+ }
104
+ return value;
105
+ };
106
+ var isTypedArrayPayload = (value) => isObject(value) && isString(value.type) && isString(value.bytes) && Object.keys(value).length === 2;
107
+
108
+ // src/index.ts
3
109
  var RAW_JSON = JSON;
4
- var applyReplacer = (key, value, replacer) => {
5
- if (typeof replacer === "function") {
110
+ var applyReplacer = (holder, key, value, replacer) => {
111
+ if (isFunction(replacer)) {
6
112
  return replacer(key, value);
7
113
  }
8
- if (Array.isArray(replacer)) {
114
+ if (isArray(replacer)) {
9
115
  if (key === "") return value;
10
116
  if (key === BIGINT_TAG) return value;
11
- return replacer.includes(key) ? value : void 0;
117
+ if (key === TYPEDARRAY_TAG) return value;
118
+ if (isTypedArrayPayload(holder)) return value;
119
+ if (isArray(holder)) return value;
120
+ return replacer.includes(key) ? value : undefined;
12
121
  }
13
122
  return value;
14
123
  };
15
- var toSerializable = (value) => typeof value === "bigint" ? { [BIGINT_TAG]: value.toString() } : value;
16
124
  var stringify = (value, replacer = null, space) => RAW_JSON.stringify(
17
125
  value,
18
- (key, v) => {
19
- const replaced = applyReplacer(key, v, replacer);
126
+ function replacerFn(key, v) {
127
+ const replaced = applyReplacer(this, key, v, replacer);
20
128
  return toSerializable(replaced);
21
129
  },
22
130
  space
23
131
  );
24
- var parse = (text) => RAW_JSON.parse(
25
- text,
26
- (_, v) => v && typeof v === "object" && BIGINT_TAG in v ? BigInt(v[BIGINT_TAG]) : v
27
- );
132
+ var parse = (text, reviver = null) => RAW_JSON.parse(text, (key, v) => {
133
+ const decoded = fromSerializable(v);
134
+ return isFunction(reviver) ? reviver(key, decoded) : decoded;
135
+ });
28
136
  var index_default = {
29
137
  stringify,
30
138
  parse
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts"],"sourcesContent":["const BIGINT_TAG = '__@json.bigint__'\nconst RAW_JSON = JSON\ntype Replacer = ((this: any, key: string, value: any) => any) | Array<string | number> | null\n\nconst applyReplacer = (key: string, value: any, replacer: Replacer): any => {\n if (typeof replacer === 'function') {\n return replacer(key, value)\n }\n if (Array.isArray(replacer)) {\n if (key === '') return value\n if (key === BIGINT_TAG) return value\n return replacer.includes(key) ? value : undefined\n }\n return value\n}\n\nconst toSerializable = (value: any): any =>\n typeof value === 'bigint' ? { [BIGINT_TAG]: value.toString() } : value\n\nexport const stringify = (value: any, replacer: Replacer = null, space?: string | number): string =>\n RAW_JSON.stringify(\n value,\n (key, v) => {\n const replaced = applyReplacer(key, v, replacer)\n return toSerializable(replaced)\n },\n space,\n )\n\nexport const parse = (text: string): any =>\n RAW_JSON.parse(text, (_, v) =>\n v && typeof v === 'object' && BIGINT_TAG in v ? BigInt(v[BIGINT_TAG]) : v,\n )\n\nexport default {\n stringify,\n parse,\n}\n"],"mappings":";AAAA,IAAM,aAAa;AACnB,IAAM,WAAW;AAGjB,IAAM,gBAAgB,CAAC,KAAa,OAAY,aAA4B;AAC1E,MAAI,OAAO,aAAa,YAAY;AAClC,WAAO,SAAS,KAAK,KAAK;AAAA,EAC5B;AACA,MAAI,MAAM,QAAQ,QAAQ,GAAG;AAC3B,QAAI,QAAQ,GAAI,QAAO;AACvB,QAAI,QAAQ,WAAY,QAAO;AAC/B,WAAO,SAAS,SAAS,GAAG,IAAI,QAAQ;AAAA,EAC1C;AACA,SAAO;AACT;AAEA,IAAM,iBAAiB,CAAC,UACtB,OAAO,UAAU,WAAW,EAAE,CAAC,UAAU,GAAG,MAAM,SAAS,EAAE,IAAI;AAE5D,IAAM,YAAY,CAAC,OAAY,WAAqB,MAAM,UAC/D,SAAS;AAAA,EACP;AAAA,EACA,CAAC,KAAK,MAAM;AACV,UAAM,WAAW,cAAc,KAAK,GAAG,QAAQ;AAC/C,WAAO,eAAe,QAAQ;AAAA,EAChC;AAAA,EACA;AACF;AAEK,IAAM,QAAQ,CAAC,SACpB,SAAS;AAAA,EAAM;AAAA,EAAM,CAAC,GAAG,MACvB,KAAK,OAAO,MAAM,YAAY,cAAc,IAAI,OAAO,EAAE,UAAU,CAAC,IAAI;AAC1E;AAEF,IAAO,gBAAQ;AAAA,EACb;AAAA,EACA;AACF;","names":[]}
1
+ {"version":3,"sources":["../src/utils.ts","../src/index.ts"],"sourcesContent":["export const BIGINT_TAG = '__@json.bigint__'\nexport const TYPEDARRAY_TAG = '__@json.typedarray__'\n\nexport const undefined = void 0\nexport const isUndefined = (value: any): value is undefined => value === undefined\nexport const isString = (value: any): value is string => typeof value === 'string'\nexport const hasOwnProperty = (obj: any, prop: string): boolean =>\n Object.prototype.hasOwnProperty.call(obj, prop)\nexport const isArray = (value: any): value is Array<any> => Array.isArray(value)\nexport const isObject = (value: any): value is Record<string, any> =>\n value !== null && typeof value === 'object' && !isArray(value)\nexport const isFunction = (value: any): value is Function => typeof value === 'function'\nexport const isBigInt = (value: any): value is bigint => typeof value === 'bigint'\nconst hasBuffer = (): boolean => typeof Buffer !== 'undefined'\n\nexport const isBuffer = (value: any): value is Buffer =>\n hasBuffer() && isFunction(Buffer.isBuffer) && Buffer.isBuffer(value)\nexport const isArrayBuffer = (value: any): value is ArrayBuffer =>\n !isUndefined(ArrayBuffer) && value instanceof ArrayBuffer\n\nexport const isTypedArray = (value: any): value is ArrayBufferView => {\n if (isUndefined(value)) return false\n if (!isUndefined(ArrayBuffer) && isFunction(ArrayBuffer.isView) && ArrayBuffer.isView(value)) {\n return Object.prototype.toString.call(value) !== '[object DataView]'\n }\n const tag = Object.prototype.toString.call(value)\n return tag.endsWith('Array]') && tag !== '[object Array]'\n}\n\nexport const getTypedArrayName = (value: any): string | null => {\n if (isUndefined(value)) return null\n const ctor = (value as any).constructor\n if (ctor && isString(ctor.name)) return ctor.name\n const tag = Object.prototype.toString.call(value)\n const match = /^\\[object (.+)\\]$/.exec(tag)\n return match ? match[1] : null\n}\n\nconst TYPEDARRAY_CTORS: Record<string, any> = {\n Uint8Array: !isUndefined(Uint8Array) ? Uint8Array : undefined,\n Uint8ClampedArray: !isUndefined(Uint8ClampedArray) ? Uint8ClampedArray : undefined,\n Uint16Array: !isUndefined(Uint16Array) ? Uint16Array : undefined,\n Uint32Array: !isUndefined(Uint32Array) ? Uint32Array : undefined,\n Int8Array: !isUndefined(Int8Array) ? Int8Array : undefined,\n Int16Array: !isUndefined(Int16Array) ? Int16Array : undefined,\n Int32Array: !isUndefined(Int32Array) ? Int32Array : undefined,\n Float32Array: !isUndefined(Float32Array) ? Float32Array : undefined,\n Float64Array: !isUndefined(Float64Array) ? Float64Array : undefined,\n BigInt64Array: !isUndefined(BigInt64Array) ? BigInt64Array : undefined,\n BigUint64Array: !isUndefined(BigUint64Array) ? BigUint64Array : undefined,\n}\n\nexport const toHex = (value: Uint8Array): string => {\n let out = '0x'\n for (const byte of value) {\n out += byte.toString(16).padStart(2, '0')\n }\n return out\n}\n\nexport const fromHex = (hex: string): Uint8Array => {\n const normalized = hex.startsWith('0x') ? hex.slice(2) : hex\n if (normalized.length % 2 !== 0) {\n throw new Error('Invalid hex string for bytes')\n }\n\n if (hasBuffer()) {\n return Uint8Array.from(Buffer.from(normalized, 'hex'))\n }\n\n const out = new Uint8Array(normalized.length / 2)\n for (let i = 0; i < normalized.length; i += 2) {\n out[i / 2] = Number.parseInt(normalized.slice(i, i + 2), 16)\n }\n return out\n}\n\nexport const toSerializable = (value: any): any => {\n if (isBigInt(value)) {\n return { [BIGINT_TAG]: value.toString() }\n }\n\n if (isTypedArray(value)) {\n const type = getTypedArrayName(value)\n if (type && TYPEDARRAY_CTORS[type]) {\n const bytes = new Uint8Array(value.buffer, value.byteOffset, value.byteLength)\n return { [TYPEDARRAY_TAG]: { type, bytes: toHex(bytes) } }\n }\n }\n\n if (isArrayBuffer(value)) {\n return { [TYPEDARRAY_TAG]: { type: 'Uint8Array', bytes: toHex(new Uint8Array(value)) } }\n }\n\n if (\n value &&\n isObject(value) &&\n value.type === 'Buffer' &&\n isArray(value.data) &&\n Object.keys(value).length === 2 &&\n value.data.every((entry: any) => Number.isInteger(entry) && entry >= 0 && entry <= 255)\n ) {\n return { [TYPEDARRAY_TAG]: { type: 'Uint8Array', bytes: toHex(Uint8Array.from(value.data)) } }\n }\n\n return value\n}\n\nexport const fromSerializable = (value: any): any => {\n if (value && isObject(value)) {\n if (hasOwnProperty(value, BIGINT_TAG)) {\n return BigInt(value[BIGINT_TAG])\n }\n if (hasOwnProperty(value, TYPEDARRAY_TAG)) {\n const payload = value[TYPEDARRAY_TAG]\n if (!payload || !isObject(payload) || !isString(payload.type) || !isString(payload.bytes)) {\n throw new Error('Invalid typed array payload')\n }\n const bytes = fromHex(payload.bytes)\n const ctor = TYPEDARRAY_CTORS[payload.type]\n if (!ctor) return bytes\n const buffer = new ArrayBuffer(bytes.byteLength)\n new Uint8Array(buffer).set(bytes)\n return new ctor(buffer)\n }\n if (\n value.type === 'Buffer' &&\n isArray(value.data) &&\n Object.keys(value).length === 2 &&\n value.data.every((entry: any) => Number.isInteger(entry) && entry >= 0 && entry <= 255)\n ) {\n return Uint8Array.from(value.data)\n }\n }\n return value\n}\n\nexport const isTypedArrayPayload = (value: any): boolean =>\n isObject(value) &&\n isString(value.type) &&\n isString(value.bytes) &&\n Object.keys(value).length === 2\n","import {\n BIGINT_TAG,\n fromSerializable,\n isArray,\n isFunction,\n isTypedArrayPayload,\n toSerializable,\n TYPEDARRAY_TAG,\n undefined,\n} from './utils'\n\nconst RAW_JSON = JSON\ntype Replacer = ((this: any, key: string, value: any) => any) | Array<string | number> | null\ntype Reviver = ((this: any, key: string, value: any) => any) | null\n\nconst applyReplacer = (holder: any, key: string, value: any, replacer: Replacer): any => {\n if (isFunction(replacer)) {\n return replacer(key, value)\n }\n if (isArray(replacer)) {\n if (key === '') return value\n if (key === BIGINT_TAG) return value\n if (key === TYPEDARRAY_TAG) return value\n if (isTypedArrayPayload(holder)) return value\n if (isArray(holder)) return value\n return replacer.includes(key) ? value : undefined\n }\n return value\n}\n\nexport const stringify = (value: any, replacer: Replacer = null, space?: string | number): string =>\n RAW_JSON.stringify(\n value,\n function replacerFn(key, v) {\n const replaced = applyReplacer(this, key, v, replacer)\n return toSerializable(replaced)\n },\n space,\n )\n\nexport const parse = (text: string, reviver: Reviver = null): any =>\n RAW_JSON.parse(text, (key, v) => {\n const decoded = fromSerializable(v)\n return isFunction(reviver) ? reviver(key, decoded) : decoded\n })\n\nexport default {\n stringify,\n parse,\n}\n"],"mappings":"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","names":[]}
package/package.json CHANGED
@@ -1,10 +1,11 @@
1
1
  {
2
2
  "name": "json-web3",
3
- "version": "0.1.1",
3
+ "version": "1.0.1",
4
4
  "description": "BigInt-safe JSON serialization and deserialization for Web3 use cases.",
5
5
  "keywords": [
6
6
  "json",
7
7
  "bigint",
8
+ "arraybuffer",
8
9
  "serialization",
9
10
  "deserialization",
10
11
  "web3"
@@ -35,7 +36,12 @@
35
36
  ],
36
37
  "devDependencies": {
37
38
  "@types/node": "^25.0.9",
39
+ "@vitest/coverage-v8": "^4.0.17",
38
40
  "prettier": "^3.8.0",
41
+ "prettier-plugin-lint-md": "^1.0.1",
42
+ "prettier-plugin-organize-imports": "^4.3.0",
43
+ "prettier-plugin-sort-json": "^4.2.0",
44
+ "prettier-plugin-tailwindcss": "^0.7.2",
39
45
  "tsup": "^8.5.1",
40
46
  "typescript": "^5.9.3",
41
47
  "vitest": "^4.0.17"
@@ -45,8 +51,8 @@
45
51
  },
46
52
  "scripts": {
47
53
  "build": "tsup",
54
+ "test": "vitest run --coverage",
48
55
  "lint": "prettier . --write",
49
- "test": "vitest run",
50
56
  "test:watch": "vitest"
51
57
  }
52
58
  }