nytra 0.0.4 → 0.0.6

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/Nytra.d.ts CHANGED
@@ -1,8 +1,11 @@
1
1
  import { Reader } from "./Reader.ts";
2
2
  import { Writer } from "./Writer.ts";
3
+ type TypedArrayRegisterFieldOptions = {
4
+ typedArrayTargetTypeId?: number | Function;
5
+ };
3
6
  export type RegisterFieldOptions = {
4
7
  nullable?: boolean;
5
- };
8
+ } & TypedArrayRegisterFieldOptions;
6
9
  export declare class Nytra {
7
10
  #private;
8
11
  static registerField(position: number, targetTypeId?: number | Function, options?: RegisterFieldOptions): (v: undefined, ctx: ClassFieldDecoratorContext) => void;
@@ -12,3 +15,4 @@ export declare class Nytra {
12
15
  static encode(data: unknown, type?: number | null, withType?: boolean, writer?: Writer | null): Uint8Array;
13
16
  static decode(data: Uint8Array | Reader, type?: number | null): unknown;
14
17
  }
18
+ export {};
package/dist/Nytra.js CHANGED
@@ -1,7 +1,7 @@
1
1
  import { Reader } from "./Reader.js";
2
2
  import { Registry } from "./Registry.js";
3
3
  import { Writer } from "./Writer.js";
4
- import { TYPE_ARRAY, TYPE_BIGINT, TYPE_BOOLEAN, TYPE_EXTENSION, TYPE_FLOAT32, TYPE_FLOAT64, TYPE_INT16, TYPE_INT32, TYPE_INT64, TYPE_INT8, TYPE_JSON, TYPE_NULL, TYPE_OBJECT, TYPE_STRING, TYPE_STRING_16_INTERNAL, TYPE_STRING_32_INTERNAL, TYPE_UINT16, TYPE_UINT32, TYPE_UINT64, TYPE_UINT8 } from "./Types.js";
4
+ import { TYPE_ARRAY, TYPE_BIGINT, TYPE_BOOLEAN, TYPE_EXTENSION, TYPE_FLOAT32, TYPE_FLOAT64, TYPE_INT16, TYPE_INT32, TYPE_INT64, TYPE_INT8, TYPE_JSON, TYPE_NULL, TYPE_OBJECT, TYPE_STRING, TYPE_STRING_16_INTERNAL, TYPE_STRING_32_INTERNAL, TYPE_TYPED_ARRAY, TYPE_UINT16, TYPE_UINT32, TYPE_UINT64, TYPE_UINT8 } from "./Types.js";
5
5
  const MAX_UINT_32 = 2 ** 32;
6
6
  const MAX_UINT_16 = 2 ** 16;
7
7
  const MAX_UINT_8 = 2 ** 8;
@@ -34,6 +34,17 @@ function createDecoder(cls) {
34
34
  meta.targetTypeId = found ? found.typeId : TYPE_JSON;
35
35
  }
36
36
  let type = meta.targetTypeId;
37
+ if (type === TYPE_TYPED_ARRAY) {
38
+ const targetType = typeof meta.options.typedArrayTargetTypeId === 'function' ? classMetaStore.get(meta.options.typedArrayTargetTypeId)?.typeId : meta.options.typedArrayTargetTypeId;
39
+ const size = reader.readUINT32();
40
+ const end = reader.offset + size;
41
+ let data = [];
42
+ while (reader.offset < end) {
43
+ data.push(Nytra.decode(reader, targetType === TYPE_STRING ? null : targetType));
44
+ }
45
+ obj[name] = data;
46
+ continue;
47
+ }
37
48
  if (type === TYPE_STRING) {
38
49
  type = reader.readType();
39
50
  }
@@ -43,6 +54,35 @@ function createDecoder(cls) {
43
54
  return obj;
44
55
  };
45
56
  }
57
+ function createTypedArrayEncoder(targetType) {
58
+ if (typeof targetType === 'function') {
59
+ const meta = classMetaStore.get(targetType);
60
+ if (!meta) {
61
+ throw new Error(`Cannot convert object type ${targetType}`);
62
+ }
63
+ targetType = meta ? meta.typeId : TYPE_JSON;
64
+ meta.typeId;
65
+ }
66
+ return function (data, writer) {
67
+ if (!Array.isArray(data)) {
68
+ throw new Error('Data must be an array');
69
+ }
70
+ if (writer === null) {
71
+ writer = new Writer();
72
+ }
73
+ const arr = data;
74
+ const startIndex = writer.offset;
75
+ writer.setOffset(startIndex + 4); // reserve space for length
76
+ for (let value of arr) {
77
+ Nytra.encode(value, targetType, false, writer);
78
+ }
79
+ const endIndex = writer.offset;
80
+ writer.setOffset(startIndex);
81
+ writer.writeUint32(endIndex - startIndex - 4);
82
+ writer.setOffset(endIndex);
83
+ return writer.toUint8Array();
84
+ };
85
+ }
46
86
  function createEncoder(cls) {
47
87
  const meta = classMetaStore.get(cls);
48
88
  const cachedEncoders = [];
@@ -69,7 +109,14 @@ function createEncoder(cls) {
69
109
  }
70
110
  // Encoder-Funktion vorgebunden
71
111
  let encodeFn;
72
- if (typeId < 255) {
112
+ if (typeId === TYPE_TYPED_ARRAY) {
113
+ if (typeof fieldMeta.options.typedArrayTargetTypeId === 'function') {
114
+ const typeId = classMetaStore.get(fieldMeta.options.typedArrayTargetTypeId)?.typeId;
115
+ fieldMeta.options.typedArrayTargetTypeId = typeId;
116
+ }
117
+ encodeFn = createTypedArrayEncoder(fieldMeta.options.typedArrayTargetTypeId);
118
+ }
119
+ else if (typeId < 255) {
73
120
  encodeFn = (value, writer) => Nytra.encode(value, typeId, false, writer);
74
121
  }
75
122
  else {
@@ -107,11 +154,15 @@ const classMetaStore = new WeakMap();
107
154
  const SYMBOL_FIELDS = Symbol('Nytra:fields');
108
155
  const CustomRegistry = new Registry();
109
156
  export class Nytra {
110
- static registerField(position, targetTypeId, options = {}) {
157
+ static registerField(position, targetTypeId, options) {
111
158
  const defaultOpts = {
112
- nullable: false
159
+ nullable: false,
160
+ typedArrayTargetTypeId: -1,
113
161
  };
114
162
  Object.assign(defaultOpts, options);
163
+ if (targetTypeId === TYPE_TYPED_ARRAY && defaultOpts.typedArrayTargetTypeId === -1) {
164
+ throw new Error('You must provide options.typedArrayTargetTypeId if you want to register a typed array field');
165
+ }
115
166
  return function (v, ctx) {
116
167
  if (ctx.private) {
117
168
  throw new Error('Only public fields can be registered');
@@ -430,6 +481,16 @@ export class Nytra {
430
481
  case TYPE_FLOAT32: {
431
482
  return reader.readFloat32();
432
483
  }
484
+ case TYPE_TYPED_ARRAY: {
485
+ const type = reader.readType();
486
+ const len = reader.readUINT32();
487
+ const end = reader.offset + len;
488
+ const targetArray = [];
489
+ while (reader.offset < end) {
490
+ targetArray.push(this.decode(reader, type));
491
+ }
492
+ return targetArray;
493
+ }
433
494
  case TYPE_ARRAY: {
434
495
  const len = reader.readUINT32();
435
496
  const end = reader.offset + len;
package/dist/Registry.js CHANGED
@@ -1,8 +1,4 @@
1
1
  export class Registry {
2
- /**
3
- *
4
- * @type {Map<number, TypeHandler>}
5
- */
6
2
  #registry = new Map();
7
3
  register(type, handler) {
8
4
  if (this.#registry.has(type)) {
package/dist/Types.d.ts CHANGED
@@ -1 +1 @@
1
- export declare const TYPE_NULL = 0, TYPE_BOOLEAN = 1, TYPE_ARRAY = 2, TYPE_OBJECT = 3, TYPE_STRING_16_INTERNAL = 4, TYPE_STRING_32_INTERNAL = 5, TYPE_JSON = 9, TYPE_UINT8 = 11, TYPE_UINT16 = 12, TYPE_UINT32 = 13, TYPE_UINT64 = 14, TYPE_INT8 = 21, TYPE_INT16 = 22, TYPE_INT32 = 23, TYPE_INT64 = 24, TYPE_FLOAT32 = 30, TYPE_FLOAT64 = 31, TYPE_BIGINT = 40, TYPE_EXTENSION = 127, TYPE_STRING = 128;
1
+ export declare const TYPE_NULL = 0, TYPE_BOOLEAN = 1, TYPE_ARRAY = 2, TYPE_OBJECT = 3, TYPE_STRING_16_INTERNAL = 4, TYPE_STRING_32_INTERNAL = 5, TYPE_JSON = 9, TYPE_UINT8 = 11, TYPE_UINT16 = 12, TYPE_UINT32 = 13, TYPE_UINT64 = 14, TYPE_INT8 = 21, TYPE_INT16 = 22, TYPE_INT32 = 23, TYPE_INT64 = 24, TYPE_FLOAT32 = 30, TYPE_FLOAT64 = 31, TYPE_BIGINT = 40, TYPE_TYPED_ARRAY = 50, TYPE_EXTENSION = 127, TYPE_STRING = 128;
package/dist/Types.js CHANGED
@@ -1,3 +1,3 @@
1
- export const TYPE_NULL = 0, TYPE_BOOLEAN = 1, TYPE_ARRAY = 2, TYPE_OBJECT = 3, TYPE_STRING_16_INTERNAL = 4, TYPE_STRING_32_INTERNAL = 5, TYPE_JSON = 9, TYPE_UINT8 = 11, TYPE_UINT16 = 12, TYPE_UINT32 = 13, TYPE_UINT64 = 14, TYPE_INT8 = 21, TYPE_INT16 = 22, TYPE_INT32 = 23, TYPE_INT64 = 24, TYPE_FLOAT32 = 30, TYPE_FLOAT64 = 31, TYPE_BIGINT = 40, TYPE_EXTENSION = 127,
1
+ export const TYPE_NULL = 0, TYPE_BOOLEAN = 1, TYPE_ARRAY = 2, TYPE_OBJECT = 3, TYPE_STRING_16_INTERNAL = 4, TYPE_STRING_32_INTERNAL = 5, TYPE_JSON = 9, TYPE_UINT8 = 11, TYPE_UINT16 = 12, TYPE_UINT32 = 13, TYPE_UINT64 = 14, TYPE_INT8 = 21, TYPE_INT16 = 22, TYPE_INT32 = 23, TYPE_INT64 = 24, TYPE_FLOAT32 = 30, TYPE_FLOAT64 = 31, TYPE_BIGINT = 40, TYPE_TYPED_ARRAY = 50, TYPE_EXTENSION = 127,
2
2
  //128-255 -> String use 7 bits for length (max 127)
3
3
  TYPE_STRING = 128;
package/dist/Writer.d.ts CHANGED
@@ -5,6 +5,7 @@ export declare class Writer {
5
5
  get offset(): number;
6
6
  constructor(initialSize?: number);
7
7
  private ensureCapacity;
8
+ writeType(type: number): this;
8
9
  writeInt8(value: number): this;
9
10
  writeInt16(value: number): this;
10
11
  writeInt32(value: number): this;
package/dist/Writer.js CHANGED
@@ -1,3 +1,4 @@
1
+ import { TYPE_EXTENSION } from "./Types.js";
1
2
  const LITTLE_ENDIAN = true;
2
3
  export class Writer {
3
4
  static DEFAULT_SIZE = 1024;
@@ -38,6 +39,13 @@ export class Writer {
38
39
  this.#buffer = newBuffer;
39
40
  this.#view = new DataView(newBuffer);
40
41
  }
42
+ writeType(type) {
43
+ if (type <= 255) {
44
+ return this.writeUint8(type);
45
+ }
46
+ this.writeUint8(TYPE_EXTENSION);
47
+ return this.writeUint16(type);
48
+ }
41
49
  writeInt8(value) {
42
50
  this.ensureCapacity(1);
43
51
  this.#view.setInt8(this.#offset, value);
package/package.json CHANGED
@@ -1,10 +1,21 @@
1
1
  {
2
2
  "private": false,
3
3
  "name": "nytra",
4
- "version": "0.0.4",
4
+ "version": "0.0.6",
5
5
  "description": "A javascript ES6 Decorator driven library to encode/decode javascript objects to/from binary optimized buffers to reduce payload on network",
6
6
  "license": "MIT",
7
- "author": "",
7
+ "author": {
8
+ "name": "Kapsonfire",
9
+ "email": "kapsonfire@googlemail.com"
10
+ },
11
+ "repository": {
12
+ "type": "git",
13
+ "url": "git+https://github.com/nytra-lib/nytra-js.git"
14
+ },
15
+ "homepage": "https://github.com/nytra-lib/nytra-js#readme",
16
+ "bugs": {
17
+ "url": "https://github.com/nytra-lib/nytra-js/issues"
18
+ },
8
19
  "type": "module",
9
20
  "main": "index.js",
10
21
  "scripts": {
package/dist/Binson.d.ts DELETED
@@ -1,14 +0,0 @@
1
- import { Reader } from "./Reader.ts";
2
- import { Writer } from "./Writer.ts";
3
- export type RegisterFieldOptions = {
4
- nullable?: boolean;
5
- };
6
- export declare class Binson {
7
- #private;
8
- static registerField(position: number, targetTypeId?: number | Function, options?: RegisterFieldOptions): (v: undefined, ctx: ClassFieldDecoratorContext) => void;
9
- static registerClass(typeId: number): <T extends Function>(decoratedClass: T, ctx: ClassDecoratorContext) => void;
10
- static autoguessType(data: unknown): number;
11
- static getTypeIdForClass(ctor: Function): number;
12
- static encode(data: unknown, type?: number | null, withType?: boolean, writer?: Writer | null): Uint8Array;
13
- static decode(data: Uint8Array | Reader, type?: number | null): unknown;
14
- }
package/dist/Binson.js DELETED
@@ -1,468 +0,0 @@
1
- import { Reader } from "./Reader.js";
2
- import { Registry } from "./Registry.js";
3
- import { Writer } from "./Writer.js";
4
- import { TYPE_ARRAY, TYPE_BIGINT, TYPE_BOOLEAN, TYPE_EXTENSION, TYPE_FLOAT32, TYPE_FLOAT64, TYPE_INT16, TYPE_INT32, TYPE_INT64, TYPE_INT8, TYPE_JSON, TYPE_NULL, TYPE_OBJECT, TYPE_STRING, TYPE_STRING_16_INTERNAL, TYPE_STRING_32_INTERNAL, TYPE_UINT16, TYPE_UINT32, TYPE_UINT64, TYPE_UINT8 } from "./Types.js";
5
- const MAX_UINT_32 = 2 ** 32;
6
- const MAX_UINT_16 = 2 ** 16;
7
- const MAX_UINT_8 = 2 ** 8;
8
- const MIN_INT_32 = -(2 ** 31);
9
- const MIN_INT_16 = -(2 ** 15);
10
- const MIN_INT_8 = -(2 ** 8);
11
- function createDecoder(cls) {
12
- const meta = classMetaStore.get(cls);
13
- const fields = meta?.fields;
14
- return function (reader) {
15
- let obj = {};
16
- const nullableFields = fields.filter(([_name, meta]) => meta.options.nullable).map(([name, _meta]) => name);
17
- const nullableBytes = Math.ceil(nullableFields.length / 8);
18
- const nullableBitfield = nullableBytes > 0 ? reader.readBytes(nullableBytes) : [];
19
- const isFieldNull = (index) => {
20
- const byteIndex = Math.floor(index / 8);
21
- const bitOffset = index % 8;
22
- return (nullableBitfield[byteIndex] & (1 << bitOffset)) !== 0;
23
- };
24
- for (let [name, meta] of fields) {
25
- if (nullableBytes > 0 && meta.options.nullable) {
26
- const nullIndex = nullableFields.indexOf(name);
27
- if (nullIndex !== -1 && isFieldNull(nullIndex)) {
28
- obj[name] = null;
29
- continue;
30
- }
31
- }
32
- if (typeof meta.targetTypeId === "function") {
33
- const found = classMetaStore.get(meta.targetTypeId);
34
- meta.targetTypeId = found ? found.typeId : TYPE_JSON;
35
- }
36
- let type = meta.targetTypeId;
37
- if (type === TYPE_STRING) {
38
- type = reader.readType();
39
- }
40
- obj[name] = Binson.decode(reader, type);
41
- }
42
- Object.setPrototypeOf(obj, cls.prototype);
43
- return obj;
44
- };
45
- }
46
- function createEncoder(cls) {
47
- const meta = classMetaStore.get(cls);
48
- const cachedEncoders = [];
49
- const nullableFields = [];
50
- let nullableBytes = 0;
51
- return function (data, writer) {
52
- if (writer === null) {
53
- writer = new Writer();
54
- }
55
- if (cachedEncoders.length == 0) {
56
- meta.fields.forEach(([name, fieldMeta]) => {
57
- if (fieldMeta.options.nullable) {
58
- nullableFields.push(name);
59
- }
60
- let typeId = fieldMeta.targetTypeId;
61
- if (!typeId) {
62
- cachedEncoders.push({ name: name, encode: null, options: fieldMeta.options });
63
- return;
64
- }
65
- // Resolve Function → typeId hier, nicht im Loop!
66
- if (typeof typeId === 'function') {
67
- const found = classMetaStore.get(typeId);
68
- typeId = found ? found.typeId : TYPE_JSON;
69
- }
70
- // Encoder-Funktion vorgebunden
71
- let encodeFn;
72
- if (typeId < 255) {
73
- encodeFn = (value, writer) => Binson.encode(value, typeId, false, writer);
74
- }
75
- else {
76
- encodeFn = CustomRegistry.getEncoder(typeId);
77
- }
78
- cachedEncoders.push({ name: name, encode: encodeFn, options: fieldMeta.options });
79
- });
80
- nullableBytes = Math.ceil(nullableFields.length / 8);
81
- }
82
- if (nullableBytes > 0) {
83
- const nullableBitfield = new Uint8Array(nullableBytes);
84
- for (let i = 0; i < nullableFields.length; i++) {
85
- const field = nullableFields[i];
86
- if (data[field] === null) {
87
- const byteIndex = Math.floor(i / 8);
88
- const bitOffset = i % 8;
89
- nullableBitfield[byteIndex] |= (1 << bitOffset);
90
- }
91
- }
92
- writer.writeBytes(nullableBitfield);
93
- }
94
- for (const encodeInfo of cachedEncoders) {
95
- const value = data[encodeInfo.name];
96
- if (value === null && encodeInfo.options.nullable) {
97
- continue;
98
- }
99
- encodeInfo.encode ?
100
- encodeInfo.encode(data[encodeInfo.name], writer) :
101
- Binson.encode(data[encodeInfo.name], null, true, writer);
102
- }
103
- return writer.toUint8Array();
104
- };
105
- }
106
- const classMetaStore = new WeakMap();
107
- const SYMBOL_FIELDS = Symbol('binson:fields');
108
- const CustomRegistry = new Registry();
109
- export class Binson {
110
- static registerField(position, targetTypeId, options = {}) {
111
- const defaultOpts = {
112
- nullable: false
113
- };
114
- Object.assign(defaultOpts, options);
115
- return function (v, ctx) {
116
- if (ctx.private) {
117
- throw new Error('Only public fields can be registered');
118
- }
119
- const metadata = ctx.metadata;
120
- if (typeof metadata[SYMBOL_FIELDS] === 'undefined') {
121
- metadata[SYMBOL_FIELDS] = new Map();
122
- }
123
- const fieldMeta = {
124
- position,
125
- targetTypeId,
126
- options: defaultOpts
127
- };
128
- metadata[SYMBOL_FIELDS].set(ctx.name, fieldMeta);
129
- };
130
- }
131
- static registerClass(typeId) {
132
- if (typeId < 256 || typeId > 65535) {
133
- throw new Error('typeId must be in range of 256 and 65535');
134
- }
135
- return function (decoratedClass, ctx) {
136
- const metadata = ctx.metadata;
137
- const fields = metadata[SYMBOL_FIELDS] ?
138
- [...metadata[SYMBOL_FIELDS].entries()].sort(([_aName, aMeta], [_bName, bMeta]) => aMeta.position - bMeta.position)
139
- : [];
140
- classMetaStore.set(decoratedClass, { typeId, fields });
141
- CustomRegistry.register(typeId, {
142
- encoder: createEncoder(decoratedClass),
143
- decoder: createDecoder(decoratedClass)
144
- });
145
- };
146
- }
147
- static autoguessType(data) {
148
- if (data === null) {
149
- return TYPE_NULL;
150
- }
151
- if (typeof data === 'object') {
152
- const ctor = data.constructor;
153
- const found = classMetaStore.get(ctor);
154
- if (found) {
155
- return found.typeId;
156
- }
157
- }
158
- if (Array.isArray(data)) {
159
- return TYPE_ARRAY;
160
- }
161
- if (typeof data === 'object') {
162
- return TYPE_OBJECT;
163
- }
164
- if (typeof data === 'string') {
165
- return TYPE_STRING;
166
- }
167
- if (typeof data === 'boolean') {
168
- return TYPE_BOOLEAN;
169
- }
170
- if (typeof data === 'number') {
171
- if (Number.isInteger(data)) {
172
- if (data >= 0) {
173
- //unsigned possible
174
- if (data <= MAX_UINT_8) {
175
- return TYPE_UINT8;
176
- }
177
- if (data <= MAX_UINT_16) {
178
- return TYPE_UINT16;
179
- }
180
- if (data <= MAX_UINT_32) {
181
- return TYPE_UINT32;
182
- }
183
- return TYPE_UINT64;
184
- }
185
- else {
186
- if (data >= MIN_INT_8) {
187
- return TYPE_INT8;
188
- }
189
- if (data >= MIN_INT_16) {
190
- return TYPE_INT16;
191
- }
192
- if (data >= MIN_INT_32) {
193
- return TYPE_INT32;
194
- }
195
- return TYPE_INT64;
196
- }
197
- }
198
- if (Math.fround(data) === data) {
199
- return TYPE_FLOAT32;
200
- }
201
- return TYPE_FLOAT64;
202
- }
203
- if (typeof data === 'bigint') {
204
- return TYPE_BIGINT;
205
- }
206
- return TYPE_JSON;
207
- }
208
- static getTypeIdForClass(ctor) {
209
- const found = classMetaStore.get(ctor);
210
- return found ? found.typeId : TYPE_JSON;
211
- }
212
- static #TEXT_ENCODER = new TextEncoder();
213
- static encode(data, type = null, withType = true, writer = null) {
214
- if (writer === null) {
215
- writer = new Writer();
216
- }
217
- if (type === null) {
218
- type = this.autoguessType(data);
219
- }
220
- if (type >= 256) {
221
- const encodeFunction = CustomRegistry.getEncoder(type);
222
- if (typeof encodeFunction !== 'function') {
223
- throw new Error('Unknown type:' + type);
224
- }
225
- if (withType) {
226
- writer.writeUint8(TYPE_EXTENSION);
227
- writer.writeUint16(type);
228
- }
229
- encodeFunction(data, writer);
230
- return writer.toUint8Array();
231
- }
232
- if (type >= TYPE_STRING) { //automatically
233
- const str = data;
234
- const bytes = this.#TEXT_ENCODER.encode(str);
235
- const len = bytes.length;
236
- if (len <= 127) {
237
- writer.writeUint8(128 + len);
238
- writer.writeBytes(bytes);
239
- return writer.toUint8Array();
240
- }
241
- if (len <= 65535) {
242
- writer.writeUint8(TYPE_STRING_16_INTERNAL);
243
- writer.writeUint16(len);
244
- writer.writeBytes(bytes);
245
- return writer.toUint8Array();
246
- }
247
- writer.writeUint8(TYPE_STRING_32_INTERNAL);
248
- writer.writeUint32(len);
249
- writer.writeBytes(bytes);
250
- return writer.toUint8Array();
251
- }
252
- switch (type) {
253
- case TYPE_NULL:
254
- writer.writeUint8(TYPE_NULL);
255
- return writer.toUint8Array();
256
- case TYPE_ARRAY: {
257
- if (!Array.isArray(data)) {
258
- throw new Error('Data must be an array');
259
- }
260
- const arr = data;
261
- if (withType)
262
- writer.writeUint8(TYPE_ARRAY);
263
- const startIndex = writer.offset;
264
- writer.setOffset(startIndex + 4); // reserve space for length
265
- for (let value of arr) {
266
- this.encode(value, null, true, writer);
267
- }
268
- const endIndex = writer.offset;
269
- writer.setOffset(startIndex);
270
- writer.writeUint32(endIndex - startIndex - 4);
271
- writer.setOffset(endIndex);
272
- return writer.toUint8Array();
273
- }
274
- case TYPE_OBJECT: {
275
- if (typeof data !== 'object') {
276
- throw new Error('Data must be an array');
277
- }
278
- const obj = data;
279
- if (withType)
280
- writer.writeUint8(TYPE_OBJECT);
281
- const startIndex = writer.offset;
282
- writer.setOffset(startIndex + 4);
283
- let keys = Object.keys(obj);
284
- for (let key of keys) {
285
- this.encode(key, TYPE_STRING, true, writer);
286
- this.encode(obj[key], null, true, writer);
287
- }
288
- const endIndex = writer.offset;
289
- writer.setOffset(startIndex);
290
- writer.writeUint32(endIndex - startIndex - 4);
291
- writer.setOffset(endIndex);
292
- return writer.toUint8Array();
293
- }
294
- case TYPE_UINT8: {
295
- if (withType)
296
- writer.writeUint8(TYPE_UINT8);
297
- writer.writeUint8(data);
298
- return writer.toUint8Array();
299
- }
300
- case TYPE_UINT16: {
301
- if (withType)
302
- writer.writeUint8(TYPE_UINT16);
303
- writer.writeUint16(data);
304
- return writer.toUint8Array();
305
- }
306
- case TYPE_UINT32: {
307
- if (withType)
308
- writer.writeUint8(TYPE_UINT32);
309
- writer.writeUint32(data);
310
- return writer.toUint8Array();
311
- }
312
- case TYPE_UINT64: {
313
- if (withType)
314
- writer.writeUint8(TYPE_UINT64);
315
- writer.writeUint64(BigInt(data));
316
- return writer.toUint8Array();
317
- }
318
- case TYPE_INT8: {
319
- if (withType)
320
- writer.writeUint8(TYPE_INT8);
321
- writer.writeInt8(data);
322
- return writer.toUint8Array();
323
- }
324
- case TYPE_INT16: {
325
- if (withType)
326
- writer.writeUint8(TYPE_INT16);
327
- writer.writeInt16(data);
328
- return writer.toUint8Array();
329
- }
330
- case TYPE_INT32: {
331
- if (withType)
332
- writer.writeUint8(TYPE_INT32);
333
- writer.writeInt32(data);
334
- return writer.toUint8Array();
335
- }
336
- case TYPE_INT64: {
337
- if (withType)
338
- writer.writeUint8(TYPE_INT64);
339
- writer.writeInt64(BigInt(data));
340
- return writer.toUint8Array();
341
- }
342
- case TYPE_BOOLEAN: {
343
- if (withType)
344
- writer.writeUint8(TYPE_BOOLEAN);
345
- writer.writeUint8(data ? 1 : 0);
346
- return writer.toUint8Array();
347
- }
348
- case TYPE_JSON: {
349
- return this.encode(JSON.stringify(data), TYPE_STRING, withType, writer);
350
- }
351
- case TYPE_FLOAT32: {
352
- if (withType)
353
- writer.writeUint8(TYPE_FLOAT32);
354
- writer.writeFloat32(data);
355
- return writer.toUint8Array();
356
- }
357
- case TYPE_FLOAT64: {
358
- if (withType)
359
- writer.writeUint8(TYPE_FLOAT64);
360
- writer.writeFloat64(data);
361
- return writer.toUint8Array();
362
- }
363
- case TYPE_BIGINT: {
364
- if (withType)
365
- writer.writeUint8(TYPE_BIGINT);
366
- writer.writeBigInt(BigInt(data));
367
- return writer.toUint8Array();
368
- }
369
- }
370
- throw new Error('Type unknown: ' + type);
371
- }
372
- static decode(data, type = null) {
373
- let reader = data instanceof Reader ? data : new Reader(data);
374
- if (type === null)
375
- type = reader.readType();
376
- if (type > 255) {
377
- const decodeFunction = CustomRegistry.getDecoder(type);
378
- if (typeof decodeFunction !== 'function') {
379
- throw new Error('Unknown type:' + type);
380
- }
381
- return decodeFunction(reader);
382
- }
383
- if (type >= TYPE_STRING) {
384
- let length = type & 0b01111111;
385
- return reader.readString(length);
386
- }
387
- switch (type) {
388
- case TYPE_STRING_16_INTERNAL: {
389
- const length = reader.readUINT16();
390
- return reader.readString(length);
391
- }
392
- case TYPE_STRING_32_INTERNAL: {
393
- const length = reader.readUINT32();
394
- return reader.readString(length);
395
- }
396
- case TYPE_BOOLEAN: {
397
- return reader.readUINT8() !== 0;
398
- }
399
- case TYPE_JSON: {
400
- const json = this.decode(reader);
401
- return JSON.parse(json);
402
- }
403
- case TYPE_UINT8: {
404
- return reader.readUINT8();
405
- }
406
- case TYPE_UINT16: {
407
- return reader.readUINT16();
408
- }
409
- case TYPE_UINT32: {
410
- return reader.readUINT32();
411
- }
412
- case TYPE_UINT64: {
413
- return bigintToSafeNumber(reader.readUINT64());
414
- }
415
- case TYPE_INT8: {
416
- return reader.readINT8();
417
- }
418
- case TYPE_INT16: {
419
- return reader.readINT16();
420
- }
421
- case TYPE_INT32: {
422
- return reader.readINT32();
423
- }
424
- case TYPE_INT64: {
425
- return bigintToSafeNumber(reader.readINT64());
426
- }
427
- case TYPE_FLOAT64: {
428
- return reader.readFloat64();
429
- }
430
- case TYPE_FLOAT32: {
431
- return reader.readFloat32();
432
- }
433
- case TYPE_ARRAY: {
434
- const len = reader.readUINT32();
435
- const end = reader.offset + len;
436
- const targetArray = [];
437
- while (reader.offset < end) {
438
- targetArray.push(this.decode(reader));
439
- }
440
- return targetArray;
441
- }
442
- case TYPE_OBJECT: {
443
- const len = reader.readUINT32();
444
- const end = reader.offset + len;
445
- const targetObj = {};
446
- while (reader.offset < end) {
447
- const key = this.decode(reader);
448
- targetObj[key] = this.decode(reader);
449
- }
450
- return targetObj;
451
- }
452
- case TYPE_BIGINT: {
453
- return reader.readBigInt();
454
- }
455
- case TYPE_NULL: {
456
- return null;
457
- }
458
- }
459
- throw new Error('Unknown type:' + type);
460
- }
461
- }
462
- function bigintToSafeNumber(value) {
463
- if (value > BigInt(Number.MAX_SAFE_INTEGER) ||
464
- value < BigInt(Number.MIN_SAFE_INTEGER)) {
465
- return value;
466
- }
467
- return Number(value);
468
- }
@@ -1,25 +0,0 @@
1
- import { Reader } from "./Reader.ts";
2
- export declare const NativeHandlers: {
3
- readonly [x: number]: {
4
- encoder: (data: null) => Uint8Array<ArrayBuffer>;
5
- decoder: (reader: Reader) => null;
6
- } | {
7
- encoder: (data: number) => Uint8Array<ArrayBufferLike>;
8
- decoder: (reader: Reader) => number;
9
- } | {
10
- encoder: (data: bigint) => Uint8Array<ArrayBufferLike>;
11
- decoder: (reader: Reader) => bigint;
12
- } | {
13
- encoder: (data: boolean) => Uint8Array<ArrayBuffer>;
14
- decoder: (reader: Reader) => boolean;
15
- } | {
16
- decoder(reader: Reader): object;
17
- encoder(data: object): Uint8Array;
18
- } | {
19
- encoder: (json: unknown) => Uint8Array<ArrayBufferLike>;
20
- decoder: (reader: Reader) => any;
21
- } | {
22
- encoder: (data: string) => Uint8Array<ArrayBufferLike>;
23
- decoder: (reader: Reader) => string;
24
- };
25
- };
@@ -1,137 +0,0 @@
1
- import { Writer } from "./Writer.js";
2
- import { Binson } from "./Binson.js";
3
- import { Reader } from "./Reader.js";
4
- const TRUE = new Uint8Array([1]);
5
- const FALSE = new Uint8Array([0]);
6
- export const NativeHandlers = {
7
- [Types.NULL]: {
8
- encoder: (data) => new Uint8Array(0),
9
- decoder: (reader) => null
10
- },
11
- [Types.UINT8]: {
12
- encoder: (data) => new Writer(1).writeUint8(data).toUint8Array(),
13
- decoder: (reader) => reader.readUINT8()
14
- },
15
- [Types.UINT16]: {
16
- encoder: (data) => new Writer(2).writeUint16(data).toUint8Array(),
17
- decoder: (reader) => reader.readUINT16()
18
- },
19
- [Types.UINT32]: {
20
- encoder: (data) => new Writer(4).writeUint32(data).toUint8Array(),
21
- decoder: (reader) => reader.readUINT32()
22
- },
23
- [Types.UINT64]: {
24
- encoder: (data) => new Writer(8).writeUint64(BigInt(data)).toUint8Array(),
25
- decoder: (reader) => Number(reader.readUINT64())
26
- },
27
- [Types.INT8]: {
28
- encoder: (data) => new Writer(1).writeUint8(data).toUint8Array(),
29
- decoder: (reader) => reader.readINT8()
30
- },
31
- [Types.INT16]: {
32
- encoder: (data) => new Writer(2).writeInt16(data).toUint8Array(),
33
- decoder: (reader) => reader.readINT16()
34
- },
35
- [Types.INT32]: {
36
- encoder: (data) => new Writer(4).writeInt32(data).toUint8Array(),
37
- decoder: (reader) => reader.readINT32()
38
- },
39
- [Types.INT64]: {
40
- encoder: (data) => new Writer(8).writeInt64(BigInt(data)).toUint8Array(),
41
- decoder: (reader) => Number(reader.readINT64())
42
- },
43
- [Types.FLOAT32]: {
44
- encoder: (data) => new Writer(4).writeFloat32(data).toUint8Array(),
45
- decoder: (reader) => reader.readFloat32()
46
- },
47
- [Types.FLOAT64]: {
48
- encoder: (data) => new Writer(4).writeFloat32(data).toUint8Array(),
49
- decoder: (reader) => reader.readFloat32()
50
- },
51
- [Types.BIGINT]: {
52
- encoder: (data) => new Writer(10).writeBigInt(data).toUint8Array(),
53
- decoder: (reader) => reader.readBigInt()
54
- },
55
- [Types.BOOLEAN]: {
56
- encoder: (data) => data ? TRUE : FALSE,
57
- decoder: (reader) => reader.readUINT8() > 0
58
- },
59
- [Types.ARRAY]: {
60
- decoder(reader) {
61
- let fullData = [];
62
- let length = reader.readUINT32();
63
- let buffer = reader.readBytes(length);
64
- let bufferReader = new Reader(buffer);
65
- while (bufferReader.remaining > 0) {
66
- fullData.push(Binson.decode(bufferReader));
67
- }
68
- return fullData;
69
- }, encoder(data) {
70
- const buffer = flattenUint8Array(data.map(d => Binson.encode(d, null)));
71
- const length = new Uint8Array(4);
72
- new DataView(length.buffer).setUint32(0, buffer.length, true);
73
- const out = new Uint8Array(4 + buffer.length);
74
- out.set(length, 0);
75
- out.set(buffer, 4);
76
- return out;
77
- }
78
- },
79
- [Types.OBJECT]: {
80
- decoder(reader) {
81
- let fullData = {};
82
- let length = reader.readUINT32();
83
- let buffer = reader.readBytes(length);
84
- let bufferReader = new Reader(buffer);
85
- while (bufferReader.remaining > 0) {
86
- let key = bufferReader.readString();
87
- fullData[key] = Binson.decode(bufferReader);
88
- }
89
- return fullData;
90
- }, encoder(data) {
91
- let keys = Object.keys(data);
92
- let dataArray = new Array(keys.length * 2);
93
- for (let i = 0; i < keys.length; i++) {
94
- let key = keys[i];
95
- dataArray[i * 2] = Binson.encodeValue(key, Types.STRING);
96
- dataArray[i * 2 + 1] = Binson.encode(data[key], null);
97
- }
98
- let buffer = flattenUint8Array(dataArray);
99
- const length = new Uint8Array(4);
100
- new DataView(length.buffer).setUint32(0, buffer.length, true);
101
- const out = new Uint8Array(4 + buffer.length);
102
- out.set(length, 0);
103
- out.set(buffer, 4);
104
- return out;
105
- }
106
- },
107
- [Types.JSON]: {
108
- encoder: (json) => {
109
- let data = JSON.stringify(json);
110
- const bytes = new TextEncoder().encode(data);
111
- return new Writer(2 + bytes.length)
112
- .writeString(data)
113
- .toUint8Array();
114
- },
115
- decoder: (reader) => JSON.parse(reader.readString())
116
- },
117
- [Types.STRING]: {
118
- encoder: (data) => {
119
- const bytes = new TextEncoder().encode(data);
120
- return new Writer(2 + bytes.length)
121
- .writeString(data)
122
- .toUint8Array();
123
- },
124
- decoder: (reader) => reader.readString()
125
- },
126
- };
127
- function flattenUint8Array(chunks) {
128
- const totalLength = chunks.reduce((sum, c) => sum + c.length, 0);
129
- const result = new Uint8Array(totalLength);
130
- let offset = 0;
131
- for (let i = 0; i < chunks.length; i++) {
132
- const chunk = chunks[i];
133
- result.set(chunk, offset);
134
- offset += chunk.length;
135
- }
136
- return result;
137
- }