@colyseus/schema 3.0.70 → 3.0.72

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.
Files changed (73) hide show
  1. package/build/cjs/index.js +17 -6
  2. package/build/cjs/index.js.map +1 -1
  3. package/build/esm/index.mjs +17 -6
  4. package/build/esm/index.mjs.map +1 -1
  5. package/build/umd/index.js +17 -6
  6. package/lib/Decoder.d.ts +16 -0
  7. package/lib/Decoder.js +182 -0
  8. package/lib/Decoder.js.map +1 -0
  9. package/lib/Encoder.d.ts +13 -0
  10. package/lib/Encoder.js +79 -0
  11. package/lib/Encoder.js.map +1 -0
  12. package/lib/annotations.js +18 -3
  13. package/lib/annotations.js.map +1 -1
  14. package/lib/bench_encode.d.ts +1 -0
  15. package/lib/bench_encode.js +93 -0
  16. package/lib/bench_encode.js.map +1 -0
  17. package/lib/changes/ChangeSet.d.ts +12 -0
  18. package/lib/changes/ChangeSet.js +35 -0
  19. package/lib/changes/ChangeSet.js.map +1 -0
  20. package/lib/changes/ChangeTree.d.ts +53 -0
  21. package/lib/changes/ChangeTree.js +202 -0
  22. package/lib/changes/ChangeTree.js.map +1 -0
  23. package/lib/changes/DecodeOperation.d.ts +15 -0
  24. package/lib/changes/DecodeOperation.js +186 -0
  25. package/lib/changes/DecodeOperation.js.map +1 -0
  26. package/lib/changes/EncodeOperation.d.ts +18 -0
  27. package/lib/changes/EncodeOperation.js +130 -0
  28. package/lib/changes/EncodeOperation.js.map +1 -0
  29. package/lib/changes/ReferenceTracker.d.ts +14 -0
  30. package/lib/changes/ReferenceTracker.js +83 -0
  31. package/lib/changes/ReferenceTracker.js.map +1 -0
  32. package/lib/changes/consts.d.ts +14 -0
  33. package/lib/changes/consts.js +18 -0
  34. package/lib/changes/consts.js.map +1 -0
  35. package/lib/decoding/decode.d.ts +48 -0
  36. package/lib/decoding/decode.js +267 -0
  37. package/lib/decoding/decode.js.map +1 -0
  38. package/lib/ecs.d.ts +11 -0
  39. package/lib/ecs.js +160 -0
  40. package/lib/ecs.js.map +1 -0
  41. package/lib/filters/index.d.ts +8 -0
  42. package/lib/filters/index.js +24 -0
  43. package/lib/filters/index.js.map +1 -0
  44. package/lib/spec.d.ts +13 -0
  45. package/lib/spec.js +42 -0
  46. package/lib/spec.js.map +1 -0
  47. package/lib/types/ArraySchema.d.ts +238 -0
  48. package/lib/types/ArraySchema.js +555 -0
  49. package/lib/types/ArraySchema.js.map +1 -0
  50. package/lib/types/CollectionSchema.d.ts +35 -0
  51. package/lib/types/CollectionSchema.js +150 -0
  52. package/lib/types/CollectionSchema.js.map +1 -0
  53. package/lib/types/MapSchema.d.ts +38 -0
  54. package/lib/types/MapSchema.js +215 -0
  55. package/lib/types/MapSchema.js.map +1 -0
  56. package/lib/types/SetSchema.d.ts +32 -0
  57. package/lib/types/SetSchema.js +162 -0
  58. package/lib/types/SetSchema.js.map +1 -0
  59. package/lib/types/typeRegistry.d.ts +5 -0
  60. package/lib/types/typeRegistry.js +13 -0
  61. package/lib/types/typeRegistry.js.map +1 -0
  62. package/lib/usage.d.ts +1 -0
  63. package/lib/usage.js +22 -0
  64. package/lib/usage.js.map +1 -0
  65. package/lib/v3.d.ts +1 -0
  66. package/lib/v3.js +427 -0
  67. package/lib/v3.js.map +1 -0
  68. package/lib/v3_experiment.d.ts +1 -0
  69. package/lib/v3_experiment.js +407 -0
  70. package/lib/v3_experiment.js.map +1 -0
  71. package/package.json +1 -1
  72. package/src/annotations.ts +19 -4
  73. package/src/bench_encode.ts +64 -0
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,93 @@
1
+ "use strict";
2
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
3
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
4
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
5
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
6
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
7
+ };
8
+ Object.defineProperty(exports, "__esModule", { value: true });
9
+ const nanoid_1 = require("nanoid");
10
+ const _1 = require(".");
11
+ const benchmark = require("benchmark");
12
+ const suite = new benchmark.Suite();
13
+ class AttributeNew extends _1.Schema {
14
+ }
15
+ __decorate([
16
+ (0, _1.type)("string")
17
+ ], AttributeNew.prototype, "name", void 0);
18
+ __decorate([
19
+ (0, _1.type)("number")
20
+ ], AttributeNew.prototype, "value", void 0);
21
+ class ItemNew extends _1.Schema {
22
+ constructor() {
23
+ super(...arguments);
24
+ this.attributes = new _1.ArraySchema();
25
+ }
26
+ }
27
+ __decorate([
28
+ (0, _1.type)("number")
29
+ ], ItemNew.prototype, "price", void 0);
30
+ __decorate([
31
+ (0, _1.type)([AttributeNew])
32
+ ], ItemNew.prototype, "attributes", void 0);
33
+ class PositionNew extends _1.Schema {
34
+ }
35
+ __decorate([
36
+ (0, _1.type)("number")
37
+ ], PositionNew.prototype, "x", void 0);
38
+ __decorate([
39
+ (0, _1.type)("number")
40
+ ], PositionNew.prototype, "y", void 0);
41
+ class PlayerNew extends _1.Schema {
42
+ constructor() {
43
+ super(...arguments);
44
+ this.position = new PositionNew();
45
+ this.items = new _1.MapSchema();
46
+ }
47
+ }
48
+ __decorate([
49
+ (0, _1.type)(PositionNew)
50
+ ], PlayerNew.prototype, "position", void 0);
51
+ __decorate([
52
+ (0, _1.type)({ map: ItemNew })
53
+ ], PlayerNew.prototype, "items", void 0);
54
+ class StateNew extends _1.Schema {
55
+ constructor() {
56
+ super(...arguments);
57
+ this.players = new _1.MapSchema();
58
+ }
59
+ }
60
+ __decorate([
61
+ (0, _1.type)({ map: PlayerNew })
62
+ ], StateNew.prototype, "players", void 0);
63
+ __decorate([
64
+ (0, _1.type)("string")
65
+ ], StateNew.prototype, "currentTurn", void 0);
66
+ const state = new StateNew();
67
+ for (let i = 0; i < 50; i++) {
68
+ const player = new PlayerNew();
69
+ state.players.set(`p-${(0, nanoid_1.nanoid)()}`, player);
70
+ player.position.x = (i + 1) * 100;
71
+ player.position.y = (i + 1) * 100;
72
+ for (let j = 0; j < 10; j++) {
73
+ const item = new ItemNew();
74
+ item.price = (i + 1) * 50;
75
+ for (let k = 0; k < 5; k++) {
76
+ const attr = new AttributeNew();
77
+ attr.name = `Attribute ${k}`;
78
+ attr.value = k;
79
+ item.attributes.push(attr);
80
+ }
81
+ player.items.set(`item-${j}`, item);
82
+ }
83
+ }
84
+ _1.Encoder.BUFFER_SIZE = 1024 * 128;
85
+ const encoder = new _1.Encoder(state);
86
+ // measure time to .encodeAll()
87
+ const now = Date.now();
88
+ for (let i = 0; i < 1000; i++) {
89
+ encoder.encodeAll();
90
+ }
91
+ console.log(Date.now() - now);
92
+ console.log(Array.from(encoder.encodeAll()).join(","));
93
+ //# sourceMappingURL=bench_encode.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"bench_encode.js","sourceRoot":"","sources":["../src/bench_encode.ts"],"names":[],"mappings":"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","sourcesContent":["import { nanoid } from \"nanoid\";\nimport { Schema, type, MapSchema, ArraySchema, Encoder } from \".\";\nimport * as benchmark from \"benchmark\";\n\nconst suite = new benchmark.Suite();\n\nclass AttributeNew extends Schema {\n @type(\"string\") name: string;\n @type(\"number\") value: number;\n}\n\nclass ItemNew extends Schema {\n @type(\"number\") price: number;\n @type([ AttributeNew ]) attributes = new ArraySchema<AttributeNew>();\n}\n\nclass PositionNew extends Schema {\n @type(\"number\") x: number;\n @type(\"number\") y: number;\n}\n\nclass PlayerNew extends Schema {\n @type(PositionNew) position = new PositionNew();\n @type({ map: ItemNew }) items = new MapSchema<ItemNew>();\n}\n\nclass StateNew extends Schema {\n @type({ map: PlayerNew }) players = new MapSchema<PlayerNew>();\n @type(\"string\") currentTurn: string;\n}\n\nconst state = new StateNew();\n\nfor (let i = 0; i < 50; i++) {\n const player = new PlayerNew();\n state.players.set(`p-${nanoid()}`, player);\n\n player.position.x = (i + 1) * 100;\n player.position.y = (i + 1) * 100;\n for (let j = 0; j < 10; j++) {\n const item = new ItemNew();\n item.price = (i + 1) * 50;\n for (let k = 0; k < 5; k++) {\n const attr = new AttributeNew();\n attr.name = `Attribute ${k}`;\n attr.value = k;\n item.attributes.push(attr);\n\n }\n player.items.set(`item-${j}`, item);\n }\n}\n\nEncoder.BUFFER_SIZE = 1024 * 128;\nconst encoder = new Encoder(state);\n\n// measure time to .encodeAll()\n\nconst now = Date.now();\nfor (let i = 0; i < 1000; i++) {\n encoder.encodeAll();\n}\nconsole.log(Date.now() - now);\nconsole.log(Array.from(encoder.encodeAll()).join(\",\"));\n"]}
@@ -0,0 +1,12 @@
1
+ import { OPERATION } from "../spec";
2
+ import { ChangeOperation } from "./ChangeTree";
3
+ export declare class ChangeSet {
4
+ changes: {
5
+ [index: number]: ChangeOperation;
6
+ };
7
+ changed: boolean;
8
+ allChanges: Set<number>;
9
+ change(index: number, operation?: OPERATION): void;
10
+ discardAll(): void;
11
+ setParent(...args: any[]): void;
12
+ }
@@ -0,0 +1,35 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ChangeSet = void 0;
4
+ const spec_1 = require("../spec");
5
+ class ChangeSet {
6
+ changes = {};
7
+ changed = false;
8
+ allChanges = new Set();
9
+ // definition: SchemaDefinition;
10
+ change(index, operation = spec_1.OPERATION.ADD) {
11
+ const previousChange = this.changes[index];
12
+ if (!previousChange ||
13
+ previousChange.op === spec_1.OPERATION.DELETE ||
14
+ previousChange.op === spec_1.OPERATION.TOUCH // (mazmorra.io's BattleAction issue)
15
+ ) {
16
+ this.changes[index] = {
17
+ op: (!previousChange)
18
+ ? operation
19
+ : (previousChange.op === spec_1.OPERATION.DELETE)
20
+ ? spec_1.OPERATION.DELETE_AND_ADD
21
+ : operation,
22
+ index
23
+ };
24
+ }
25
+ this.allChanges.add(index);
26
+ this.changed = true;
27
+ }
28
+ discardAll() {
29
+ this.changes = {};
30
+ }
31
+ setParent(...args) {
32
+ }
33
+ }
34
+ exports.ChangeSet = ChangeSet;
35
+ //# sourceMappingURL=ChangeSet.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ChangeSet.js","sourceRoot":"","sources":["../../src/changes/ChangeSet.ts"],"names":[],"mappings":";;;AAAA,kCAAoC;AAIpC,MAAa,SAAS;IAClB,OAAO,GAAyC,EAAE,CAAC;IACnD,OAAO,GAAY,KAAK,CAAC;IACzB,UAAU,GAAgB,IAAI,GAAG,EAAE,CAAC;IACpC,gCAAgC;IAEhC,MAAM,CAAC,KAAa,EAAE,YAAuB,gBAAS,CAAC,GAAG;QACtD,MAAM,cAAc,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QAE3C,IACI,CAAC,cAAc;YACf,cAAc,CAAC,EAAE,KAAK,gBAAS,CAAC,MAAM;YACtC,cAAc,CAAC,EAAE,KAAK,gBAAS,CAAC,KAAK,CAAC,qCAAqC;UAC7E,CAAC;YACC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,GAAG;gBAClB,EAAE,EAAE,CAAC,CAAC,cAAc,CAAC;oBACjB,CAAC,CAAC,SAAS;oBACX,CAAC,CAAC,CAAC,cAAc,CAAC,EAAE,KAAK,gBAAS,CAAC,MAAM,CAAC;wBACtC,CAAC,CAAC,gBAAS,CAAC,cAAc;wBAC1B,CAAC,CAAC,SAAS;gBACnB,KAAK;aACR,CAAC;QACN,CAAC;QAED,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAE3B,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;IACxB,CAAC;IAED,UAAU;QACN,IAAI,CAAC,OAAO,GAAG,EAAE,CAAC;IACtB,CAAC;IAED,SAAS,CAAC,GAAG,IAAW;IACxB,CAAC;CAEJ;AApCD,8BAoCC","sourcesContent":["import { OPERATION } from \"../spec\";\nimport { SchemaDefinition } from \"../annotations\";\nimport { ChangeOperation, Ref } from \"./ChangeTree\";\n\nexport class ChangeSet {\n changes: { [index: number]: ChangeOperation } = {};\n changed: boolean = false;\n allChanges: Set<number> = new Set();\n // definition: SchemaDefinition;\n\n change(index: number, operation: OPERATION = OPERATION.ADD) {\n const previousChange = this.changes[index];\n\n if (\n !previousChange ||\n previousChange.op === OPERATION.DELETE ||\n previousChange.op === OPERATION.TOUCH // (mazmorra.io's BattleAction issue)\n ) {\n this.changes[index] = {\n op: (!previousChange)\n ? operation\n : (previousChange.op === OPERATION.DELETE)\n ? OPERATION.DELETE_AND_ADD\n : operation,\n index\n };\n }\n\n this.allChanges.add(index);\n\n this.changed = true;\n }\n\n discardAll() {\n this.changes = {};\n }\n\n setParent(...args: any[]) {\n }\n\n}"]}
@@ -0,0 +1,53 @@
1
+ import { OPERATION } from "../spec";
2
+ import { Schema } from "../Schema";
3
+ import { FilterChildrenCallback } from "../annotations";
4
+ import { MapSchema } from "../types/MapSchema";
5
+ import { ArraySchema } from "../types/ArraySchema";
6
+ import { CollectionSchema } from "../types/CollectionSchema";
7
+ import { SetSchema } from "../types/SetSchema";
8
+ import { ReferenceTracker } from "./ReferenceTracker";
9
+ export type Ref = Schema | ArraySchema | MapSchema | CollectionSchema | SetSchema;
10
+ export interface ChangeOperation {
11
+ op: OPERATION;
12
+ index: number;
13
+ }
14
+ export interface FieldCache {
15
+ beginIndex: number;
16
+ endIndex: number;
17
+ }
18
+ export declare class ChangeTree {
19
+ ref: Ref;
20
+ refId: number;
21
+ root?: ReferenceTracker;
22
+ parent?: Ref;
23
+ parentIndex?: number;
24
+ indexes: {
25
+ [index: string]: any;
26
+ };
27
+ changed: boolean;
28
+ changes: Map<number, ChangeOperation>;
29
+ allChanges: Set<number>;
30
+ caches: {
31
+ [field: number]: number[];
32
+ };
33
+ currentCustomOperation: number;
34
+ constructor(ref: Ref, parent?: Ref, root?: ReferenceTracker);
35
+ setParent(parent: Ref, root?: ReferenceTracker, parentIndex?: number): void;
36
+ operation(op: ChangeOperation): void;
37
+ change(fieldName: string | number, operation?: OPERATION): void;
38
+ touch(fieldName: string | number): void;
39
+ touchParents(): void;
40
+ getType(index?: number): any;
41
+ getChildrenFilter(): FilterChildrenCallback;
42
+ getValue(index: number): any;
43
+ delete(fieldName: string | number): void;
44
+ discard(changed?: boolean, discardAll?: boolean): void;
45
+ /**
46
+ * Recursively discard all changes from this, and child structures.
47
+ */
48
+ discardAll(): void;
49
+ cache(field: number, cachedBytes: number[]): void;
50
+ clone(): ChangeTree;
51
+ ensureRefId(): void;
52
+ protected assertValidIndex(index: number, fieldName: string | number): void;
53
+ }
@@ -0,0 +1,202 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ChangeTree = void 0;
4
+ const spec_1 = require("../spec");
5
+ const Schema_1 = require("../Schema");
6
+ class ChangeTree {
7
+ constructor(ref, parent, root) {
8
+ this.changed = false;
9
+ this.changes = new Map();
10
+ this.allChanges = new Set();
11
+ // cached indexes for filtering
12
+ this.caches = {};
13
+ this.currentCustomOperation = 0;
14
+ this.ref = ref;
15
+ this.setParent(parent, root);
16
+ }
17
+ setParent(parent, root, parentIndex) {
18
+ if (!this.indexes) {
19
+ this.indexes = (this.ref instanceof Schema_1.Schema)
20
+ ? this.ref['_definition'].indexes
21
+ : {};
22
+ }
23
+ this.parent = parent;
24
+ this.parentIndex = parentIndex;
25
+ // avoid setting parents with empty `root`
26
+ if (!root) {
27
+ return;
28
+ }
29
+ this.root = root;
30
+ //
31
+ // assign same parent on child structures
32
+ //
33
+ if (this.ref instanceof Schema_1.Schema) {
34
+ const definition = this.ref['_definition'];
35
+ for (let field in definition.schema) {
36
+ const value = this.ref[field];
37
+ if (value && value['$changes']) {
38
+ const parentIndex = definition.indexes[field];
39
+ value['$changes'].setParent(this.ref, root, parentIndex);
40
+ }
41
+ }
42
+ }
43
+ else if (typeof (this.ref) === "object") {
44
+ this.ref.forEach((value, key) => {
45
+ if (value instanceof Schema_1.Schema) {
46
+ const changeTreee = value['$changes'];
47
+ const parentIndex = this.ref['$changes'].indexes[key];
48
+ changeTreee.setParent(this.ref, this.root, parentIndex);
49
+ }
50
+ });
51
+ }
52
+ }
53
+ operation(op) {
54
+ this.changes.set(--this.currentCustomOperation, op);
55
+ }
56
+ change(fieldName, operation = spec_1.OPERATION.ADD) {
57
+ const index = (typeof (fieldName) === "number")
58
+ ? fieldName
59
+ : this.indexes[fieldName];
60
+ this.assertValidIndex(index, fieldName);
61
+ const previousChange = this.changes.get(index);
62
+ if (!previousChange ||
63
+ previousChange.op === spec_1.OPERATION.DELETE ||
64
+ previousChange.op === spec_1.OPERATION.TOUCH // (mazmorra.io's BattleAction issue)
65
+ ) {
66
+ this.changes.set(index, {
67
+ op: (!previousChange)
68
+ ? operation
69
+ : (previousChange.op === spec_1.OPERATION.DELETE)
70
+ ? spec_1.OPERATION.DELETE_AND_ADD
71
+ : operation,
72
+ // : OPERATION.REPLACE,
73
+ index
74
+ });
75
+ }
76
+ this.allChanges.add(index);
77
+ this.changed = true;
78
+ this.touchParents();
79
+ }
80
+ touch(fieldName) {
81
+ const index = (typeof (fieldName) === "number")
82
+ ? fieldName
83
+ : this.indexes[fieldName];
84
+ this.assertValidIndex(index, fieldName);
85
+ if (!this.changes.has(index)) {
86
+ this.changes.set(index, { op: spec_1.OPERATION.TOUCH, index });
87
+ }
88
+ this.allChanges.add(index);
89
+ // ensure touch is placed until the $root is found.
90
+ this.touchParents();
91
+ }
92
+ touchParents() {
93
+ if (this.parent) {
94
+ this.parent['$changes'].touch(this.parentIndex);
95
+ }
96
+ }
97
+ getType(index) {
98
+ if (this.ref['_definition']) {
99
+ const definition = this.ref['_definition'];
100
+ return definition.schema[definition.fieldsByIndex[index]];
101
+ }
102
+ else {
103
+ const definition = this.parent['_definition'];
104
+ const parentType = definition.schema[definition.fieldsByIndex[this.parentIndex]];
105
+ //
106
+ // Get the child type from parent structure.
107
+ // - ["string"] => "string"
108
+ // - { map: "string" } => "string"
109
+ // - { set: "string" } => "string"
110
+ //
111
+ return Object.values(parentType)[0];
112
+ }
113
+ }
114
+ getChildrenFilter() {
115
+ const childFilters = this.parent['_definition'].childFilters;
116
+ return childFilters && childFilters[this.parentIndex];
117
+ }
118
+ //
119
+ // used during `.encode()`
120
+ //
121
+ getValue(index) {
122
+ return this.ref['getByIndex'](index);
123
+ }
124
+ delete(fieldName) {
125
+ const index = (typeof (fieldName) === "number")
126
+ ? fieldName
127
+ : this.indexes[fieldName];
128
+ if (index === undefined) {
129
+ console.warn(`@colyseus/schema ${this.ref.constructor.name}: trying to delete non-existing index: ${fieldName} (${index})`);
130
+ return;
131
+ }
132
+ const previousValue = this.getValue(index);
133
+ // console.log("$changes.delete =>", { fieldName, index, previousValue });
134
+ this.changes.set(index, { op: spec_1.OPERATION.DELETE, index });
135
+ this.allChanges.delete(index);
136
+ // delete cache
137
+ delete this.caches[index];
138
+ // remove `root` reference
139
+ if (previousValue && previousValue['$changes']) {
140
+ previousValue['$changes'].parent = undefined;
141
+ }
142
+ this.changed = true;
143
+ this.touchParents();
144
+ }
145
+ discard(changed = false, discardAll = false) {
146
+ //
147
+ // Map, Array, etc:
148
+ // Remove cached key to ensure ADD operations is unsed instead of
149
+ // REPLACE in case same key is used on next patches.
150
+ //
151
+ // TODO: refactor this. this is not relevant for Collection and Set.
152
+ //
153
+ if (!(this.ref instanceof Schema_1.Schema)) {
154
+ this.changes.forEach((change) => {
155
+ if (change.op === spec_1.OPERATION.DELETE) {
156
+ const index = this.ref['getIndex'](change.index);
157
+ delete this.indexes[index];
158
+ }
159
+ });
160
+ }
161
+ this.changes.clear();
162
+ this.changed = changed;
163
+ if (discardAll) {
164
+ this.allChanges.clear();
165
+ }
166
+ // re-set `currentCustomOperation`
167
+ this.currentCustomOperation = 0;
168
+ }
169
+ /**
170
+ * Recursively discard all changes from this, and child structures.
171
+ */
172
+ discardAll() {
173
+ this.changes.forEach((change) => {
174
+ const value = this.getValue(change.index);
175
+ if (value && value['$changes']) {
176
+ value['$changes'].discardAll();
177
+ }
178
+ });
179
+ this.discard();
180
+ }
181
+ // cache(field: number, beginIndex: number, endIndex: number) {
182
+ cache(field, cachedBytes) {
183
+ this.caches[field] = cachedBytes;
184
+ }
185
+ clone() {
186
+ return new ChangeTree(this.ref, this.parent, this.root);
187
+ }
188
+ ensureRefId() {
189
+ // skip if refId is already set.
190
+ if (this.refId !== undefined) {
191
+ return;
192
+ }
193
+ this.refId = this.root.getNextUniqueId();
194
+ }
195
+ assertValidIndex(index, fieldName) {
196
+ if (index === undefined) {
197
+ throw new Error(`ChangeTree: missing index for field "${fieldName}"`);
198
+ }
199
+ }
200
+ }
201
+ exports.ChangeTree = ChangeTree;
202
+ //# sourceMappingURL=ChangeTree.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ChangeTree.js","sourceRoot":"","sources":["../../src/changes/ChangeTree.ts"],"names":[],"mappings":"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","sourcesContent":["import { OPERATION } from \"../spec\";\nimport { Schema } from \"../Schema\";\nimport { SchemaDefinition, FilterChildrenCallback } from \"../annotations\";\n\nimport { MapSchema } from \"../types/MapSchema\";\nimport { ArraySchema } from \"../types/ArraySchema\";\nimport { CollectionSchema } from \"../types/CollectionSchema\";\nimport { SetSchema } from \"../types/SetSchema\";\nimport { ReferenceTracker } from \"./ReferenceTracker\";\n\nexport type Ref = Schema\n | ArraySchema\n | MapSchema\n | CollectionSchema\n | SetSchema;\n\nexport interface ChangeOperation {\n op: OPERATION,\n index: number,\n}\n\n//\n// FieldCache is used for @filter()\n//\nexport interface FieldCache {\n beginIndex: number;\n endIndex: number;\n}\n\n\nexport class ChangeTree {\n ref: Ref;\n refId: number;\n\n root?: ReferenceTracker;\n\n parent?: Ref;\n parentIndex?: number;\n\n indexes: {[index: string]: any};\n\n changed: boolean = false;\n changes = new Map<number, ChangeOperation>();\n allChanges = new Set<number>();\n\n // cached indexes for filtering\n caches: {[field: number]: number[]} = {};\n\n currentCustomOperation: number = 0;\n\n constructor(ref: Ref, parent?: Ref, root?: ReferenceTracker) {\n this.ref = ref;\n this.setParent(parent, root);\n }\n\n setParent(\n parent: Ref,\n root?: ReferenceTracker,\n parentIndex?: number,\n ) {\n if (!this.indexes) {\n this.indexes = (this.ref instanceof Schema)\n ? this.ref['_definition'].indexes\n : {};\n }\n\n this.parent = parent;\n this.parentIndex = parentIndex;\n\n // avoid setting parents with empty `root`\n if (!root) { return; }\n this.root = root;\n\n //\n // assign same parent on child structures\n //\n if (this.ref instanceof Schema) {\n const definition: SchemaDefinition = this.ref['_definition'];\n\n for (let field in definition.schema) {\n const value = this.ref[field];\n\n if (value && value['$changes']) {\n const parentIndex = definition.indexes[field];\n\n (value['$changes'] as ChangeTree).setParent(\n this.ref,\n root,\n parentIndex,\n );\n }\n }\n\n } else if (typeof (this.ref) === \"object\") {\n this.ref.forEach((value, key) => {\n if (value instanceof Schema) {\n const changeTreee = value['$changes'];\n const parentIndex = this.ref['$changes'].indexes[key];\n\n changeTreee.setParent(\n this.ref,\n this.root,\n parentIndex,\n );\n }\n });\n }\n }\n\n operation(op: ChangeOperation) {\n this.changes.set(--this.currentCustomOperation, op);\n }\n\n change(fieldName: string | number, operation: OPERATION = OPERATION.ADD) {\n const index = (typeof (fieldName) === \"number\")\n ? fieldName\n : this.indexes[fieldName];\n\n this.assertValidIndex(index, fieldName);\n\n const previousChange = this.changes.get(index);\n\n if (\n !previousChange ||\n previousChange.op === OPERATION.DELETE ||\n previousChange.op === OPERATION.TOUCH // (mazmorra.io's BattleAction issue)\n ) {\n this.changes.set(index, {\n op: (!previousChange)\n ? operation\n : (previousChange.op === OPERATION.DELETE)\n ? OPERATION.DELETE_AND_ADD\n : operation,\n // : OPERATION.REPLACE,\n index\n });\n }\n\n this.allChanges.add(index);\n\n this.changed = true;\n this.touchParents();\n }\n\n touch(fieldName: string | number) {\n const index = (typeof (fieldName) === \"number\")\n ? fieldName\n : this.indexes[fieldName];\n\n this.assertValidIndex(index, fieldName);\n\n if (!this.changes.has(index)) {\n this.changes.set(index, { op: OPERATION.TOUCH, index });\n }\n\n this.allChanges.add(index);\n\n // ensure touch is placed until the $root is found.\n this.touchParents();\n }\n\n touchParents() {\n if (this.parent) {\n (this.parent['$changes'] as ChangeTree).touch(this.parentIndex);\n }\n }\n\n getType(index?: number) {\n if (this.ref['_definition']) {\n const definition = (this.ref as Schema)['_definition'];\n return definition.schema[ definition.fieldsByIndex[index] ];\n\n } else {\n const definition = (this.parent as Schema)['_definition'];\n const parentType = definition.schema[ definition.fieldsByIndex[this.parentIndex] ];\n\n //\n // Get the child type from parent structure.\n // - [\"string\"] => \"string\"\n // - { map: \"string\" } => \"string\"\n // - { set: \"string\" } => \"string\"\n //\n return Object.values(parentType)[0];\n }\n }\n\n getChildrenFilter(): FilterChildrenCallback {\n const childFilters = (this.parent as Schema)['_definition'].childFilters;\n return childFilters && childFilters[this.parentIndex];\n }\n\n //\n // used during `.encode()`\n //\n getValue(index: number) {\n return this.ref['getByIndex'](index);\n }\n\n delete(fieldName: string | number) {\n const index = (typeof (fieldName) === \"number\")\n ? fieldName\n : this.indexes[fieldName];\n\n if (index === undefined) {\n console.warn(`@colyseus/schema ${this.ref.constructor.name}: trying to delete non-existing index: ${fieldName} (${index})`);\n return;\n }\n\n const previousValue = this.getValue(index);\n // console.log(\"$changes.delete =>\", { fieldName, index, previousValue });\n\n this.changes.set(index, { op: OPERATION.DELETE, index });\n\n this.allChanges.delete(index);\n\n // delete cache\n delete this.caches[index];\n\n // remove `root` reference\n if (previousValue && previousValue['$changes']) {\n previousValue['$changes'].parent = undefined;\n }\n\n this.changed = true;\n this.touchParents();\n }\n\n discard(changed: boolean = false, discardAll: boolean = false) {\n //\n // Map, Array, etc:\n // Remove cached key to ensure ADD operations is unsed instead of\n // REPLACE in case same key is used on next patches.\n //\n // TODO: refactor this. this is not relevant for Collection and Set.\n //\n if (!(this.ref instanceof Schema)) {\n this.changes.forEach((change) => {\n if (change.op === OPERATION.DELETE) {\n const index = this.ref['getIndex'](change.index)\n delete this.indexes[index];\n }\n });\n }\n\n this.changes.clear();\n this.changed = changed;\n\n if (discardAll) {\n this.allChanges.clear();\n }\n\n // re-set `currentCustomOperation`\n this.currentCustomOperation = 0;\n }\n\n /**\n * Recursively discard all changes from this, and child structures.\n */\n discardAll() {\n this.changes.forEach((change) => {\n const value = this.getValue(change.index);\n\n if (value && value['$changes']) {\n value['$changes'].discardAll();\n }\n });\n\n this.discard();\n }\n\n // cache(field: number, beginIndex: number, endIndex: number) {\n cache(field: number, cachedBytes: number[]) {\n this.caches[field] = cachedBytes;\n }\n\n clone() {\n return new ChangeTree(this.ref, this.parent, this.root);\n }\n\n ensureRefId() {\n // skip if refId is already set.\n if (this.refId !== undefined) {\n return;\n }\n\n this.refId = this.root.getNextUniqueId();\n }\n\n protected assertValidIndex(index: number, fieldName: string | number) {\n if (index === undefined) {\n throw new Error(`ChangeTree: missing index for field \"${fieldName}\"`);\n }\n }\n\n}\n"]}
@@ -0,0 +1,15 @@
1
+ import { OPERATION } from "../spec";
2
+ import { DataChange, Schema } from "../Schema";
3
+ import type { Ref } from "./ChangeTree";
4
+ import type { Decoder } from "../Decoder";
5
+ import * as decode from "../encoding/decode";
6
+ export declare enum DecodeState {
7
+ DEFINITION_MISMATCH = 0
8
+ }
9
+ export type DecodeOperation<T extends Schema = any> = (decoder: Decoder<T>, bytes: number[], it: decode.Iterator, ref: Ref, allChanges: DataChange[]) => void;
10
+ export declare function decodeValue(decoder: Decoder, operation: OPERATION, ref: Ref, index: number, type: any, bytes: number[], it: decode.Iterator, allChanges: DataChange[]): {
11
+ value: any;
12
+ previousValue: any;
13
+ };
14
+ export declare const decodeSchemaOperation: DecodeOperation;
15
+ export declare const decodeKeyValueOperation: DecodeOperation;
@@ -0,0 +1,186 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.decodeKeyValueOperation = exports.decodeSchemaOperation = exports.decodeValue = exports.DecodeState = void 0;
4
+ const spec_1 = require("../spec");
5
+ const Schema_1 = require("../Schema");
6
+ const decode = require("../encoding/decode");
7
+ const typeRegistry_1 = require("../types/typeRegistry");
8
+ const consts_1 = require("./consts");
9
+ const __1 = require("..");
10
+ var DecodeState;
11
+ (function (DecodeState) {
12
+ DecodeState[DecodeState["DEFINITION_MISMATCH"] = 0] = "DEFINITION_MISMATCH";
13
+ })(DecodeState || (exports.DecodeState = DecodeState = {}));
14
+ function decodeValue(decoder, operation, ref, index, type, bytes, it, allChanges) {
15
+ const $root = decoder.refs;
16
+ const previousValue = ref[consts_1.$getByIndex](index);
17
+ let value;
18
+ if ((operation & spec_1.OPERATION.DELETE) === spec_1.OPERATION.DELETE) {
19
+ //
20
+ // Delete operations
21
+ //
22
+ if (operation !== spec_1.OPERATION.DELETE_AND_ADD) {
23
+ ref[consts_1.$deleteByIndex](index);
24
+ }
25
+ // Flag `refId` for garbage collection.
26
+ const previousRefId = $root.refIds.get(previousValue);
27
+ if (previousRefId) {
28
+ $root.removeRef(previousRefId);
29
+ }
30
+ value = null;
31
+ }
32
+ else if (Schema_1.Schema.is(type)) {
33
+ const refId = decode.number(bytes, it);
34
+ value = $root.refs.get(refId);
35
+ if (operation !== spec_1.OPERATION.REPLACE) {
36
+ const childType = decoder.getInstanceType(bytes, it, type);
37
+ if (!value) {
38
+ value = decoder.createInstanceOfType(childType);
39
+ if (previousValue) {
40
+ // value.$callbacks = previousValue.$callbacks;
41
+ // value.$listeners = previousValue.$listeners;
42
+ const previousRefId = $root.refIds.get(previousValue);
43
+ if (previousRefId && refId !== previousRefId) {
44
+ $root.removeRef(previousRefId);
45
+ }
46
+ }
47
+ }
48
+ // console.log("ADD REF!", refId, value, ", TYPE =>", Metadata.getFor(childType));
49
+ $root.addRef(refId, value, (value !== previousValue));
50
+ }
51
+ }
52
+ else if (typeof (type) === "string") {
53
+ //
54
+ // primitive value (number, string, boolean, etc)
55
+ //
56
+ value = decode[type](bytes, it);
57
+ }
58
+ else {
59
+ const typeDef = (0, typeRegistry_1.getType)(Object.keys(type)[0]);
60
+ const refId = decode.number(bytes, it);
61
+ const valueRef = ($root.refs.has(refId))
62
+ ? previousValue || $root.refs.get(refId)
63
+ : new typeDef.constructor();
64
+ value = valueRef.clone(true);
65
+ value[consts_1.$childType] = Object.values(type)[0]; // cache childType for ArraySchema and MapSchema
66
+ // preserve schema callbacks
67
+ if (previousValue) {
68
+ // value['$callbacks'] = previousValue['$callbacks'];
69
+ const previousRefId = $root.refIds.get(previousValue);
70
+ if (previousRefId && refId !== previousRefId) {
71
+ $root.removeRef(previousRefId);
72
+ //
73
+ // Trigger onRemove if structure has been replaced.
74
+ //
75
+ const entries = previousValue.entries();
76
+ let iter;
77
+ while ((iter = entries.next()) && !iter.done) {
78
+ const [key, value] = iter.value;
79
+ allChanges.push({
80
+ refId,
81
+ op: spec_1.OPERATION.DELETE,
82
+ field: key,
83
+ value: undefined,
84
+ previousValue: value,
85
+ });
86
+ }
87
+ }
88
+ }
89
+ // console.log("ADD REF!", { refId, value });
90
+ $root.addRef(refId, value, (valueRef !== previousValue));
91
+ }
92
+ return { value, previousValue };
93
+ }
94
+ exports.decodeValue = decodeValue;
95
+ const decodeSchemaOperation = function (decoder, bytes, it, ref, allChanges) {
96
+ const first_byte = bytes[it.offset++];
97
+ const metadata = ref['constructor'][Symbol.metadata];
98
+ // "compressed" index + operation
99
+ const operation = (first_byte >> 6) << 6;
100
+ const index = first_byte % (operation || 255);
101
+ // skip early if field is not defined
102
+ const field = metadata[index];
103
+ if (field === undefined) {
104
+ return DecodeState.DEFINITION_MISMATCH;
105
+ }
106
+ const { value, previousValue } = decodeValue(decoder, operation, ref, index, metadata[field].type, bytes, it, allChanges);
107
+ if (value !== null && value !== undefined) {
108
+ ref[field] = value;
109
+ }
110
+ // add change
111
+ if (previousValue !== value) {
112
+ allChanges.push({
113
+ refId: decoder.currentRefId,
114
+ op: operation,
115
+ field: field,
116
+ value,
117
+ previousValue,
118
+ });
119
+ }
120
+ };
121
+ exports.decodeSchemaOperation = decodeSchemaOperation;
122
+ const decodeKeyValueOperation = function (decoder, bytes, it, ref, allChanges) {
123
+ const first_byte = bytes[it.offset++];
124
+ // "uncompressed" index + operation (array/map items)
125
+ const operation = first_byte;
126
+ if (operation === spec_1.OPERATION.CLEAR) {
127
+ //
128
+ // TODO: refactor me!
129
+ // The `.clear()` method is calling `$root.removeRef(refId)` for
130
+ // each item inside this collection
131
+ //
132
+ ref.clear(allChanges);
133
+ return;
134
+ }
135
+ const index = decode.number(bytes, it);
136
+ const type = ref[consts_1.$childType];
137
+ let dynamicIndex;
138
+ if ((operation & spec_1.OPERATION.ADD) === spec_1.OPERATION.ADD) { // ADD or DELETE_AND_ADD
139
+ dynamicIndex = (ref instanceof __1.MapSchema)
140
+ ? decode.string(bytes, it)
141
+ : index;
142
+ ref['setIndex'](index, dynamicIndex);
143
+ }
144
+ else {
145
+ // here
146
+ dynamicIndex = ref['getIndex'](index);
147
+ }
148
+ const { value, previousValue } = decodeValue(decoder, operation, ref, dynamicIndex, type, bytes, it, allChanges);
149
+ if (value !== null && value !== undefined) {
150
+ if (ref instanceof __1.MapSchema) {
151
+ // const key = ref['$indexes'].get(field);
152
+ const key = dynamicIndex;
153
+ // ref.set(key, value);
154
+ ref['$items'].set(key, value);
155
+ }
156
+ else if (ref instanceof __1.ArraySchema) {
157
+ // const key = ref['$indexes'][field];
158
+ // console.log("SETTING FOR ArraySchema =>", { field, key, value });
159
+ // ref[key] = value;
160
+ ref.setAt(index, value);
161
+ }
162
+ else if (ref instanceof __1.CollectionSchema) {
163
+ const index = ref.add(value);
164
+ ref['setIndex'](index, index);
165
+ }
166
+ else if (ref instanceof __1.SetSchema) {
167
+ const index = ref.add(value);
168
+ if (index !== false) {
169
+ ref['setIndex'](index, index);
170
+ }
171
+ }
172
+ }
173
+ // add change
174
+ if (previousValue !== value) {
175
+ allChanges.push({
176
+ refId: decoder.currentRefId,
177
+ op: operation,
178
+ field: "", // FIXME: remove this
179
+ dynamicIndex,
180
+ value,
181
+ previousValue,
182
+ });
183
+ }
184
+ };
185
+ exports.decodeKeyValueOperation = decodeKeyValueOperation;
186
+ //# sourceMappingURL=DecodeOperation.js.map