@push.rocks/smartmongo 2.2.0 → 4.0.0

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 (126) hide show
  1. package/dist_ts/00_commitinfo_data.js +1 -1
  2. package/dist_ts/index.d.ts +1 -1
  3. package/dist_ts/index.js +3 -3
  4. package/dist_ts/tsmdb/engine/AggregationEngine.js +189 -0
  5. package/dist_ts/{congodb → tsmdb}/engine/IndexEngine.d.ts +23 -3
  6. package/dist_ts/tsmdb/engine/IndexEngine.js +678 -0
  7. package/dist_ts/tsmdb/engine/QueryEngine.js +271 -0
  8. package/dist_ts/tsmdb/engine/QueryPlanner.d.ts +64 -0
  9. package/dist_ts/tsmdb/engine/QueryPlanner.js +308 -0
  10. package/dist_ts/tsmdb/engine/SessionEngine.d.ts +117 -0
  11. package/dist_ts/tsmdb/engine/SessionEngine.js +232 -0
  12. package/dist_ts/{congodb → tsmdb}/engine/TransactionEngine.d.ts +1 -1
  13. package/dist_ts/tsmdb/engine/TransactionEngine.js +287 -0
  14. package/dist_ts/tsmdb/engine/UpdateEngine.js +461 -0
  15. package/dist_ts/{congodb/errors/CongoErrors.d.ts → tsmdb/errors/TsmdbErrors.d.ts} +16 -16
  16. package/dist_ts/tsmdb/errors/TsmdbErrors.js +155 -0
  17. package/dist_ts/{congodb → tsmdb}/index.d.ts +11 -4
  18. package/dist_ts/tsmdb/index.js +31 -0
  19. package/dist_ts/tsmdb/server/CommandRouter.d.ts +87 -0
  20. package/dist_ts/tsmdb/server/CommandRouter.js +222 -0
  21. package/dist_ts/{congodb/server/CongoServer.d.ts → tsmdb/server/TsmdbServer.d.ts} +6 -6
  22. package/dist_ts/tsmdb/server/TsmdbServer.js +229 -0
  23. package/dist_ts/{congodb → tsmdb}/server/WireProtocol.d.ts +1 -1
  24. package/dist_ts/tsmdb/server/WireProtocol.js +298 -0
  25. package/dist_ts/{congodb → tsmdb}/server/handlers/AdminHandler.d.ts +1 -1
  26. package/dist_ts/tsmdb/server/handlers/AdminHandler.js +668 -0
  27. package/dist_ts/{congodb → tsmdb}/server/handlers/AggregateHandler.d.ts +1 -1
  28. package/dist_ts/tsmdb/server/handlers/AggregateHandler.js +277 -0
  29. package/dist_ts/{congodb → tsmdb}/server/handlers/DeleteHandler.d.ts +1 -1
  30. package/dist_ts/tsmdb/server/handlers/DeleteHandler.js +95 -0
  31. package/dist_ts/{congodb → tsmdb}/server/handlers/FindHandler.d.ts +1 -1
  32. package/dist_ts/tsmdb/server/handlers/FindHandler.js +291 -0
  33. package/dist_ts/{congodb → tsmdb}/server/handlers/HelloHandler.d.ts +1 -1
  34. package/dist_ts/{congodb → tsmdb}/server/handlers/HelloHandler.js +2 -2
  35. package/dist_ts/{congodb → tsmdb}/server/handlers/IndexHandler.d.ts +1 -1
  36. package/dist_ts/tsmdb/server/handlers/IndexHandler.js +183 -0
  37. package/dist_ts/{congodb → tsmdb}/server/handlers/InsertHandler.d.ts +1 -1
  38. package/dist_ts/tsmdb/server/handlers/InsertHandler.js +79 -0
  39. package/dist_ts/{congodb → tsmdb}/server/handlers/UpdateHandler.d.ts +1 -1
  40. package/dist_ts/tsmdb/server/handlers/UpdateHandler.js +296 -0
  41. package/dist_ts/tsmdb/server/handlers/index.js +10 -0
  42. package/dist_ts/{congodb → tsmdb}/server/index.d.ts +2 -2
  43. package/dist_ts/tsmdb/server/index.js +7 -0
  44. package/dist_ts/{congodb → tsmdb}/storage/FileStorageAdapter.d.ts +27 -3
  45. package/dist_ts/tsmdb/storage/FileStorageAdapter.js +465 -0
  46. package/dist_ts/{congodb → tsmdb}/storage/IStorageAdapter.d.ts +7 -2
  47. package/dist_ts/{congodb → tsmdb}/storage/IStorageAdapter.js +1 -1
  48. package/dist_ts/{congodb → tsmdb}/storage/MemoryStorageAdapter.d.ts +3 -2
  49. package/dist_ts/tsmdb/storage/MemoryStorageAdapter.js +378 -0
  50. package/dist_ts/{congodb → tsmdb}/storage/OpLog.d.ts +1 -1
  51. package/dist_ts/tsmdb/storage/OpLog.js +221 -0
  52. package/dist_ts/tsmdb/storage/WAL.d.ts +117 -0
  53. package/dist_ts/tsmdb/storage/WAL.js +286 -0
  54. package/dist_ts/tsmdb/tsmdb.plugins.js +14 -0
  55. package/dist_ts/{congodb → tsmdb}/types/interfaces.d.ts +3 -3
  56. package/dist_ts/{congodb → tsmdb}/types/interfaces.js +1 -1
  57. package/dist_ts/tsmdb/utils/checksum.d.ts +30 -0
  58. package/dist_ts/tsmdb/utils/checksum.js +77 -0
  59. package/dist_ts/tsmdb/utils/index.d.ts +1 -0
  60. package/dist_ts/tsmdb/utils/index.js +2 -0
  61. package/package.json +1 -1
  62. package/readme.hints.md +7 -12
  63. package/readme.md +25 -25
  64. package/ts/00_commitinfo_data.ts +1 -1
  65. package/ts/index.ts +2 -2
  66. package/ts/{congodb → tsmdb}/engine/AggregationEngine.ts +1 -1
  67. package/ts/tsmdb/engine/IndexEngine.ts +798 -0
  68. package/ts/{congodb → tsmdb}/engine/QueryEngine.ts +1 -1
  69. package/ts/tsmdb/engine/QueryPlanner.ts +393 -0
  70. package/ts/tsmdb/engine/SessionEngine.ts +292 -0
  71. package/ts/{congodb → tsmdb}/engine/TransactionEngine.ts +12 -12
  72. package/ts/{congodb → tsmdb}/engine/UpdateEngine.ts +1 -1
  73. package/ts/{congodb/errors/CongoErrors.ts → tsmdb/errors/TsmdbErrors.ts} +34 -34
  74. package/ts/{congodb → tsmdb}/index.ts +16 -7
  75. package/ts/{congodb → tsmdb}/server/CommandRouter.ts +114 -5
  76. package/ts/{congodb/server/CongoServer.ts → tsmdb/server/TsmdbServer.ts} +11 -8
  77. package/ts/{congodb → tsmdb}/server/WireProtocol.ts +1 -1
  78. package/ts/{congodb → tsmdb}/server/handlers/AdminHandler.ts +116 -11
  79. package/ts/{congodb → tsmdb}/server/handlers/AggregateHandler.ts +1 -1
  80. package/ts/{congodb → tsmdb}/server/handlers/DeleteHandler.ts +18 -3
  81. package/ts/{congodb → tsmdb}/server/handlers/FindHandler.ts +43 -14
  82. package/ts/{congodb → tsmdb}/server/handlers/HelloHandler.ts +1 -1
  83. package/ts/{congodb → tsmdb}/server/handlers/IndexHandler.ts +1 -1
  84. package/ts/{congodb → tsmdb}/server/handlers/InsertHandler.ts +7 -1
  85. package/ts/{congodb → tsmdb}/server/handlers/UpdateHandler.ts +34 -5
  86. package/ts/{congodb → tsmdb}/server/index.ts +2 -2
  87. package/ts/{congodb → tsmdb}/storage/FileStorageAdapter.ts +90 -7
  88. package/ts/{congodb → tsmdb}/storage/IStorageAdapter.ts +8 -2
  89. package/ts/{congodb → tsmdb}/storage/MemoryStorageAdapter.ts +14 -2
  90. package/ts/{congodb → tsmdb}/storage/OpLog.ts +1 -1
  91. package/ts/tsmdb/storage/WAL.ts +375 -0
  92. package/ts/{congodb → tsmdb}/types/interfaces.ts +3 -3
  93. package/ts/tsmdb/utils/checksum.ts +88 -0
  94. package/ts/tsmdb/utils/index.ts +1 -0
  95. package/dist_ts/congodb/congodb.plugins.js +0 -14
  96. package/dist_ts/congodb/engine/AggregationEngine.js +0 -189
  97. package/dist_ts/congodb/engine/IndexEngine.js +0 -376
  98. package/dist_ts/congodb/engine/QueryEngine.js +0 -271
  99. package/dist_ts/congodb/engine/TransactionEngine.js +0 -287
  100. package/dist_ts/congodb/engine/UpdateEngine.js +0 -461
  101. package/dist_ts/congodb/errors/CongoErrors.js +0 -155
  102. package/dist_ts/congodb/index.js +0 -26
  103. package/dist_ts/congodb/server/CommandRouter.d.ts +0 -51
  104. package/dist_ts/congodb/server/CommandRouter.js +0 -132
  105. package/dist_ts/congodb/server/CongoServer.js +0 -227
  106. package/dist_ts/congodb/server/WireProtocol.js +0 -298
  107. package/dist_ts/congodb/server/handlers/AdminHandler.js +0 -568
  108. package/dist_ts/congodb/server/handlers/AggregateHandler.js +0 -277
  109. package/dist_ts/congodb/server/handlers/DeleteHandler.js +0 -83
  110. package/dist_ts/congodb/server/handlers/FindHandler.js +0 -261
  111. package/dist_ts/congodb/server/handlers/IndexHandler.js +0 -183
  112. package/dist_ts/congodb/server/handlers/InsertHandler.js +0 -76
  113. package/dist_ts/congodb/server/handlers/UpdateHandler.js +0 -270
  114. package/dist_ts/congodb/server/handlers/index.js +0 -10
  115. package/dist_ts/congodb/server/index.js +0 -7
  116. package/dist_ts/congodb/storage/FileStorageAdapter.js +0 -396
  117. package/dist_ts/congodb/storage/MemoryStorageAdapter.js +0 -367
  118. package/dist_ts/congodb/storage/OpLog.js +0 -221
  119. package/ts/congodb/engine/IndexEngine.ts +0 -479
  120. /package/dist_ts/{congodb → tsmdb}/engine/AggregationEngine.d.ts +0 -0
  121. /package/dist_ts/{congodb → tsmdb}/engine/QueryEngine.d.ts +0 -0
  122. /package/dist_ts/{congodb → tsmdb}/engine/UpdateEngine.d.ts +0 -0
  123. /package/dist_ts/{congodb → tsmdb}/server/handlers/index.d.ts +0 -0
  124. /package/dist_ts/{congodb/congodb.plugins.d.ts → tsmdb/tsmdb.plugins.d.ts} +0 -0
  125. /package/ts/{congodb → tsmdb}/server/handlers/index.ts +0 -0
  126. /package/ts/{congodb/congodb.plugins.ts → tsmdb/tsmdb.plugins.ts} +0 -0
@@ -1,298 +0,0 @@
1
- import * as plugins from '../congodb.plugins.js';
2
- /**
3
- * MongoDB Wire Protocol Implementation
4
- * Handles parsing and encoding of MongoDB wire protocol messages (OP_MSG primarily)
5
- *
6
- * Wire Protocol Message Format:
7
- * - Header (16 bytes): messageLength (4), requestID (4), responseTo (4), opCode (4)
8
- * - OP_MSG: flagBits (4), sections[], optional checksum (4)
9
- *
10
- * References:
11
- * - https://www.mongodb.com/docs/manual/reference/mongodb-wire-protocol/
12
- */
13
- // OpCodes
14
- export const OP_REPLY = 1; // Legacy reply
15
- export const OP_UPDATE = 2001; // Legacy update
16
- export const OP_INSERT = 2002; // Legacy insert
17
- export const OP_QUERY = 2004; // Legacy query (still used for initial handshake)
18
- export const OP_GET_MORE = 2005; // Legacy getMore
19
- export const OP_DELETE = 2006; // Legacy delete
20
- export const OP_KILL_CURSORS = 2007; // Legacy kill cursors
21
- export const OP_COMPRESSED = 2012; // Compressed message
22
- export const OP_MSG = 2013; // Modern protocol (MongoDB 3.6+)
23
- // OP_MSG Section Types
24
- export const SECTION_BODY = 0; // Single BSON document
25
- export const SECTION_DOCUMENT_SEQUENCE = 1; // Document sequence for bulk operations
26
- // OP_MSG Flag Bits
27
- export const MSG_FLAG_CHECKSUM_PRESENT = 1 << 0;
28
- export const MSG_FLAG_MORE_TO_COME = 1 << 1;
29
- export const MSG_FLAG_EXHAUST_ALLOWED = 1 << 16;
30
- /**
31
- * Wire Protocol parser and encoder
32
- */
33
- export class WireProtocol {
34
- /**
35
- * Parse a complete message from a buffer
36
- * Returns the parsed command and the number of bytes consumed
37
- */
38
- static parseMessage(buffer) {
39
- if (buffer.length < 16) {
40
- return null; // Not enough data for header
41
- }
42
- const header = this.parseHeader(buffer);
43
- if (buffer.length < header.messageLength) {
44
- return null; // Not enough data for complete message
45
- }
46
- const messageBuffer = buffer.subarray(0, header.messageLength);
47
- switch (header.opCode) {
48
- case OP_MSG:
49
- return this.parseOpMsg(messageBuffer, header);
50
- case OP_QUERY:
51
- return this.parseOpQuery(messageBuffer, header);
52
- default:
53
- throw new Error(`Unsupported opCode: ${header.opCode}`);
54
- }
55
- }
56
- /**
57
- * Parse message header (16 bytes)
58
- */
59
- static parseHeader(buffer) {
60
- return {
61
- messageLength: buffer.readInt32LE(0),
62
- requestID: buffer.readInt32LE(4),
63
- responseTo: buffer.readInt32LE(8),
64
- opCode: buffer.readInt32LE(12),
65
- };
66
- }
67
- /**
68
- * Parse OP_MSG message
69
- */
70
- static parseOpMsg(buffer, header) {
71
- let offset = 16; // Skip header
72
- const flagBits = buffer.readUInt32LE(offset);
73
- offset += 4;
74
- const sections = [];
75
- const documentSequences = new Map();
76
- // Parse sections until we reach the end (or checksum)
77
- const messageEnd = (flagBits & MSG_FLAG_CHECKSUM_PRESENT)
78
- ? header.messageLength - 4
79
- : header.messageLength;
80
- while (offset < messageEnd) {
81
- const sectionType = buffer.readUInt8(offset);
82
- offset += 1;
83
- if (sectionType === SECTION_BODY) {
84
- // Single BSON document
85
- const docSize = buffer.readInt32LE(offset);
86
- const docBuffer = buffer.subarray(offset, offset + docSize);
87
- const doc = plugins.bson.deserialize(docBuffer);
88
- sections.push({ type: SECTION_BODY, payload: doc });
89
- offset += docSize;
90
- }
91
- else if (sectionType === SECTION_DOCUMENT_SEQUENCE) {
92
- // Document sequence
93
- const sectionSize = buffer.readInt32LE(offset);
94
- const sectionEnd = offset + sectionSize;
95
- offset += 4;
96
- // Read sequence identifier (C string)
97
- let identifierEnd = offset;
98
- while (buffer[identifierEnd] !== 0 && identifierEnd < sectionEnd) {
99
- identifierEnd++;
100
- }
101
- const identifier = buffer.subarray(offset, identifierEnd).toString('utf8');
102
- offset = identifierEnd + 1; // Skip null terminator
103
- // Read documents
104
- const documents = [];
105
- while (offset < sectionEnd) {
106
- const docSize = buffer.readInt32LE(offset);
107
- const docBuffer = buffer.subarray(offset, offset + docSize);
108
- documents.push(plugins.bson.deserialize(docBuffer));
109
- offset += docSize;
110
- }
111
- sections.push({
112
- type: SECTION_DOCUMENT_SEQUENCE,
113
- payload: {},
114
- sequenceIdentifier: identifier,
115
- documents
116
- });
117
- documentSequences.set(identifier, documents);
118
- }
119
- else {
120
- throw new Error(`Unknown section type: ${sectionType}`);
121
- }
122
- }
123
- // The first section body contains the command
124
- const commandSection = sections.find(s => s.type === SECTION_BODY);
125
- if (!commandSection) {
126
- throw new Error('OP_MSG missing command body section');
127
- }
128
- const command = commandSection.payload;
129
- const commandName = Object.keys(command)[0];
130
- const database = command.$db || 'admin';
131
- return {
132
- command: {
133
- commandName,
134
- command,
135
- database,
136
- requestID: header.requestID,
137
- opCode: header.opCode,
138
- documentSequences: documentSequences.size > 0 ? documentSequences : undefined,
139
- },
140
- bytesConsumed: header.messageLength,
141
- };
142
- }
143
- /**
144
- * Parse OP_QUERY message (legacy, used for initial handshake)
145
- */
146
- static parseOpQuery(buffer, header) {
147
- let offset = 16; // Skip header
148
- const flags = buffer.readInt32LE(offset);
149
- offset += 4;
150
- // Read full collection name (C string)
151
- let nameEnd = offset;
152
- while (buffer[nameEnd] !== 0 && nameEnd < buffer.length) {
153
- nameEnd++;
154
- }
155
- const fullCollectionName = buffer.subarray(offset, nameEnd).toString('utf8');
156
- offset = nameEnd + 1;
157
- const numberToSkip = buffer.readInt32LE(offset);
158
- offset += 4;
159
- const numberToReturn = buffer.readInt32LE(offset);
160
- offset += 4;
161
- // Read query document
162
- const querySize = buffer.readInt32LE(offset);
163
- const queryBuffer = buffer.subarray(offset, offset + querySize);
164
- const query = plugins.bson.deserialize(queryBuffer);
165
- offset += querySize;
166
- // Extract database from collection name (format: "dbname.$cmd" or "dbname.collection")
167
- const parts = fullCollectionName.split('.');
168
- const database = parts[0];
169
- // For OP_QUERY to .$cmd, the query IS the command
170
- let commandName = 'find';
171
- let command = query;
172
- if (parts[1] === '$cmd') {
173
- // This is a command
174
- commandName = Object.keys(query)[0];
175
- // Handle special commands like isMaster, hello
176
- if (commandName === 'isMaster' || commandName === 'ismaster') {
177
- commandName = 'hello';
178
- }
179
- }
180
- return {
181
- command: {
182
- commandName,
183
- command,
184
- database,
185
- requestID: header.requestID,
186
- opCode: header.opCode,
187
- },
188
- bytesConsumed: header.messageLength,
189
- };
190
- }
191
- /**
192
- * Encode a response as OP_MSG
193
- */
194
- static encodeOpMsgResponse(responseTo, response, requestID = Math.floor(Math.random() * 0x7FFFFFFF)) {
195
- // Add $db if not present (optional in response)
196
- const responseDoc = { ...response };
197
- // Serialize the response document
198
- const bodyBson = plugins.bson.serialize(responseDoc);
199
- // Calculate message length
200
- // Header (16) + flagBits (4) + section type (1) + body BSON
201
- const messageLength = 16 + 4 + 1 + bodyBson.length;
202
- const buffer = Buffer.alloc(messageLength);
203
- let offset = 0;
204
- // Write header
205
- buffer.writeInt32LE(messageLength, offset); // messageLength
206
- offset += 4;
207
- buffer.writeInt32LE(requestID, offset); // requestID
208
- offset += 4;
209
- buffer.writeInt32LE(responseTo, offset); // responseTo
210
- offset += 4;
211
- buffer.writeInt32LE(OP_MSG, offset); // opCode
212
- offset += 4;
213
- // Write flagBits (0 = no flags)
214
- buffer.writeUInt32LE(0, offset);
215
- offset += 4;
216
- // Write section type 0 (body)
217
- buffer.writeUInt8(SECTION_BODY, offset);
218
- offset += 1;
219
- // Write body BSON
220
- Buffer.from(bodyBson).copy(buffer, offset);
221
- return buffer;
222
- }
223
- /**
224
- * Encode a response as OP_REPLY (legacy, for OP_QUERY responses)
225
- */
226
- static encodeOpReplyResponse(responseTo, documents, requestID = Math.floor(Math.random() * 0x7FFFFFFF), cursorId = BigInt(0)) {
227
- // Serialize all documents
228
- const docBuffers = documents.map(doc => plugins.bson.serialize(doc));
229
- const totalDocsSize = docBuffers.reduce((sum, buf) => sum + buf.length, 0);
230
- // Message format:
231
- // Header (16) + responseFlags (4) + cursorID (8) + startingFrom (4) + numberReturned (4) + documents
232
- const messageLength = 16 + 4 + 8 + 4 + 4 + totalDocsSize;
233
- const buffer = Buffer.alloc(messageLength);
234
- let offset = 0;
235
- // Write header
236
- buffer.writeInt32LE(messageLength, offset); // messageLength
237
- offset += 4;
238
- buffer.writeInt32LE(requestID, offset); // requestID
239
- offset += 4;
240
- buffer.writeInt32LE(responseTo, offset); // responseTo
241
- offset += 4;
242
- buffer.writeInt32LE(OP_REPLY, offset); // opCode
243
- offset += 4;
244
- // Write OP_REPLY fields
245
- buffer.writeInt32LE(0, offset); // responseFlags (0 = no errors)
246
- offset += 4;
247
- buffer.writeBigInt64LE(cursorId, offset); // cursorID
248
- offset += 8;
249
- buffer.writeInt32LE(0, offset); // startingFrom
250
- offset += 4;
251
- buffer.writeInt32LE(documents.length, offset); // numberReturned
252
- offset += 4;
253
- // Write documents
254
- for (const docBuffer of docBuffers) {
255
- Buffer.from(docBuffer).copy(buffer, offset);
256
- offset += docBuffer.length;
257
- }
258
- return buffer;
259
- }
260
- /**
261
- * Encode an error response
262
- */
263
- static encodeErrorResponse(responseTo, errorCode, errorMessage, commandName) {
264
- const response = {
265
- ok: 0,
266
- errmsg: errorMessage,
267
- code: errorCode,
268
- codeName: this.getErrorCodeName(errorCode),
269
- };
270
- return this.encodeOpMsgResponse(responseTo, response);
271
- }
272
- /**
273
- * Get error code name from error code
274
- */
275
- static getErrorCodeName(code) {
276
- const errorNames = {
277
- 0: 'OK',
278
- 1: 'InternalError',
279
- 2: 'BadValue',
280
- 11000: 'DuplicateKey',
281
- 11001: 'DuplicateKeyValue',
282
- 13: 'Unauthorized',
283
- 26: 'NamespaceNotFound',
284
- 27: 'IndexNotFound',
285
- 48: 'NamespaceExists',
286
- 59: 'CommandNotFound',
287
- 66: 'ImmutableField',
288
- 73: 'InvalidNamespace',
289
- 85: 'IndexOptionsConflict',
290
- 112: 'WriteConflict',
291
- 121: 'DocumentValidationFailure',
292
- 211: 'KeyNotFound',
293
- 251: 'NoSuchTransaction',
294
- };
295
- return errorNames[code] || 'UnknownError';
296
- }
297
- }
298
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiV2lyZVByb3RvY29sLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vdHMvY29uZ29kYi9zZXJ2ZXIvV2lyZVByb3RvY29sLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxPQUFPLE1BQU0sdUJBQXVCLENBQUM7QUFFakQ7Ozs7Ozs7Ozs7R0FVRztBQUVILFVBQVU7QUFDVixNQUFNLENBQUMsTUFBTSxRQUFRLEdBQUcsQ0FBQyxDQUFDLENBQVEsZUFBZTtBQUNqRCxNQUFNLENBQUMsTUFBTSxTQUFTLEdBQUcsSUFBSSxDQUFDLENBQUksZ0JBQWdCO0FBQ2xELE1BQU0sQ0FBQyxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsQ0FBSSxnQkFBZ0I7QUFDbEQsTUFBTSxDQUFDLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxDQUFLLGtEQUFrRDtBQUNwRixNQUFNLENBQUMsTUFBTSxXQUFXLEdBQUcsSUFBSSxDQUFDLENBQUUsaUJBQWlCO0FBQ25ELE1BQU0sQ0FBQyxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsQ0FBSSxnQkFBZ0I7QUFDbEQsTUFBTSxDQUFDLE1BQU0sZUFBZSxHQUFHLElBQUksQ0FBQyxDQUFDLHNCQUFzQjtBQUMzRCxNQUFNLENBQUMsTUFBTSxhQUFhLEdBQUcsSUFBSSxDQUFDLENBQUcscUJBQXFCO0FBQzFELE1BQU0sQ0FBQyxNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsQ0FBTyxpQ0FBaUM7QUFFbkUsdUJBQXVCO0FBQ3ZCLE1BQU0sQ0FBQyxNQUFNLFlBQVksR0FBRyxDQUFDLENBQUMsQ0FBVyx1QkFBdUI7QUFDaEUsTUFBTSxDQUFDLE1BQU0seUJBQXlCLEdBQUcsQ0FBQyxDQUFDLENBQUMsd0NBQXdDO0FBRXBGLG1CQUFtQjtBQUNuQixNQUFNLENBQUMsTUFBTSx5QkFBeUIsR0FBRyxDQUFDLElBQUksQ0FBQyxDQUFDO0FBQ2hELE1BQU0sQ0FBQyxNQUFNLHFCQUFxQixHQUFHLENBQUMsSUFBSSxDQUFDLENBQUM7QUFDNUMsTUFBTSxDQUFDLE1BQU0sd0JBQXdCLEdBQUcsQ0FBQyxJQUFJLEVBQUUsQ0FBQztBQXlEaEQ7O0dBRUc7QUFDSCxNQUFNLE9BQU8sWUFBWTtJQUN2Qjs7O09BR0c7SUFDSCxNQUFNLENBQUMsWUFBWSxDQUFDLE1BQWM7UUFDaEMsSUFBSSxNQUFNLENBQUMsTUFBTSxHQUFHLEVBQUUsRUFBRSxDQUFDO1lBQ3ZCLE9BQU8sSUFBSSxDQUFDLENBQUMsNkJBQTZCO1FBQzVDLENBQUM7UUFFRCxNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsV0FBVyxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBRXhDLElBQUksTUFBTSxDQUFDLE1BQU0sR0FBRyxNQUFNLENBQUMsYUFBYSxFQUFFLENBQUM7WUFDekMsT0FBTyxJQUFJLENBQUMsQ0FBQyx1Q0FBdUM7UUFDdEQsQ0FBQztRQUVELE1BQU0sYUFBYSxHQUFHLE1BQU0sQ0FBQyxRQUFRLENBQUMsQ0FBQyxFQUFFLE1BQU0sQ0FBQyxhQUFhLENBQUMsQ0FBQztRQUUvRCxRQUFRLE1BQU0sQ0FBQyxNQUFNLEVBQUUsQ0FBQztZQUN0QixLQUFLLE1BQU07Z0JBQ1QsT0FBTyxJQUFJLENBQUMsVUFBVSxDQUFDLGFBQWEsRUFBRSxNQUFNLENBQUMsQ0FBQztZQUNoRCxLQUFLLFFBQVE7Z0JBQ1gsT0FBTyxJQUFJLENBQUMsWUFBWSxDQUFDLGFBQWEsRUFBRSxNQUFNLENBQUMsQ0FBQztZQUNsRDtnQkFDRSxNQUFNLElBQUksS0FBSyxDQUFDLHVCQUF1QixNQUFNLENBQUMsTUFBTSxFQUFFLENBQUMsQ0FBQztRQUM1RCxDQUFDO0lBQ0gsQ0FBQztJQUVEOztPQUVHO0lBQ0ssTUFBTSxDQUFDLFdBQVcsQ0FBQyxNQUFjO1FBQ3ZDLE9BQU87WUFDTCxhQUFhLEVBQUUsTUFBTSxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUM7WUFDcEMsU0FBUyxFQUFFLE1BQU0sQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDO1lBQ2hDLFVBQVUsRUFBRSxNQUFNLENBQUMsV0FBVyxDQUFDLENBQUMsQ0FBQztZQUNqQyxNQUFNLEVBQUUsTUFBTSxDQUFDLFdBQVcsQ0FBQyxFQUFFLENBQUM7U0FDL0IsQ0FBQztJQUNKLENBQUM7SUFFRDs7T0FFRztJQUNLLE1BQU0sQ0FBQyxVQUFVLENBQUMsTUFBYyxFQUFFLE1BQXNCO1FBQzlELElBQUksTUFBTSxHQUFHLEVBQUUsQ0FBQyxDQUFDLGNBQWM7UUFFL0IsTUFBTSxRQUFRLEdBQUcsTUFBTSxDQUFDLFlBQVksQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUM3QyxNQUFNLElBQUksQ0FBQyxDQUFDO1FBRVosTUFBTSxRQUFRLEdBQW9CLEVBQUUsQ0FBQztRQUNyQyxNQUFNLGlCQUFpQixHQUFHLElBQUksR0FBRyxFQUFtQyxDQUFDO1FBRXJFLHNEQUFzRDtRQUN0RCxNQUFNLFVBQVUsR0FBRyxDQUFDLFFBQVEsR0FBRyx5QkFBeUIsQ0FBQztZQUN2RCxDQUFDLENBQUMsTUFBTSxDQUFDLGFBQWEsR0FBRyxDQUFDO1lBQzFCLENBQUMsQ0FBQyxNQUFNLENBQUMsYUFBYSxDQUFDO1FBRXpCLE9BQU8sTUFBTSxHQUFHLFVBQVUsRUFBRSxDQUFDO1lBQzNCLE1BQU0sV0FBVyxHQUFHLE1BQU0sQ0FBQyxTQUFTLENBQUMsTUFBTSxDQUFDLENBQUM7WUFDN0MsTUFBTSxJQUFJLENBQUMsQ0FBQztZQUVaLElBQUksV0FBVyxLQUFLLFlBQVksRUFBRSxDQUFDO2dCQUNqQyx1QkFBdUI7Z0JBQ3ZCLE1BQU0sT0FBTyxHQUFHLE1BQU0sQ0FBQyxXQUFXLENBQUMsTUFBTSxDQUFDLENBQUM7Z0JBQzNDLE1BQU0sU0FBUyxHQUFHLE1BQU0sQ0FBQyxRQUFRLENBQUMsTUFBTSxFQUFFLE1BQU0sR0FBRyxPQUFPLENBQUMsQ0FBQztnQkFDNUQsTUFBTSxHQUFHLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsU0FBUyxDQUFDLENBQUM7Z0JBQ2hELFFBQVEsQ0FBQyxJQUFJLENBQUMsRUFBRSxJQUFJLEVBQUUsWUFBWSxFQUFFLE9BQU8sRUFBRSxHQUFHLEVBQUUsQ0FBQyxDQUFDO2dCQUNwRCxNQUFNLElBQUksT0FBTyxDQUFDO1lBQ3BCLENBQUM7aUJBQU0sSUFBSSxXQUFXLEtBQUsseUJBQXlCLEVBQUUsQ0FBQztnQkFDckQsb0JBQW9CO2dCQUNwQixNQUFNLFdBQVcsR0FBRyxNQUFNLENBQUMsV0FBVyxDQUFDLE1BQU0sQ0FBQyxDQUFDO2dCQUMvQyxNQUFNLFVBQVUsR0FBRyxNQUFNLEdBQUcsV0FBVyxDQUFDO2dCQUN4QyxNQUFNLElBQUksQ0FBQyxDQUFDO2dCQUVaLHNDQUFzQztnQkFDdEMsSUFBSSxhQUFhLEdBQUcsTUFBTSxDQUFDO2dCQUMzQixPQUFPLE1BQU0sQ0FBQyxhQUFhLENBQUMsS0FBSyxDQUFDLElBQUksYUFBYSxHQUFHLFVBQVUsRUFBRSxDQUFDO29CQUNqRSxhQUFhLEVBQUUsQ0FBQztnQkFDbEIsQ0FBQztnQkFDRCxNQUFNLFVBQVUsR0FBRyxNQUFNLENBQUMsUUFBUSxDQUFDLE1BQU0sRUFBRSxhQUFhLENBQUMsQ0FBQyxRQUFRLENBQUMsTUFBTSxDQUFDLENBQUM7Z0JBQzNFLE1BQU0sR0FBRyxhQUFhLEdBQUcsQ0FBQyxDQUFDLENBQUMsdUJBQXVCO2dCQUVuRCxpQkFBaUI7Z0JBQ2pCLE1BQU0sU0FBUyxHQUE0QixFQUFFLENBQUM7Z0JBQzlDLE9BQU8sTUFBTSxHQUFHLFVBQVUsRUFBRSxDQUFDO29CQUMzQixNQUFNLE9BQU8sR0FBRyxNQUFNLENBQUMsV0FBVyxDQUFDLE1BQU0sQ0FBQyxDQUFDO29CQUMzQyxNQUFNLFNBQVMsR0FBRyxNQUFNLENBQUMsUUFBUSxDQUFDLE1BQU0sRUFBRSxNQUFNLEdBQUcsT0FBTyxDQUFDLENBQUM7b0JBQzVELFNBQVMsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQztvQkFDcEQsTUFBTSxJQUFJLE9BQU8sQ0FBQztnQkFDcEIsQ0FBQztnQkFFRCxRQUFRLENBQUMsSUFBSSxDQUFDO29CQUNaLElBQUksRUFBRSx5QkFBeUI7b0JBQy9CLE9BQU8sRUFBRSxFQUFFO29CQUNYLGtCQUFrQixFQUFFLFVBQVU7b0JBQzlCLFNBQVM7aUJBQ1YsQ0FBQyxDQUFDO2dCQUNILGlCQUFpQixDQUFDLEdBQUcsQ0FBQyxVQUFVLEVBQUUsU0FBUyxDQUFDLENBQUM7WUFDL0MsQ0FBQztpQkFBTSxDQUFDO2dCQUNOLE1BQU0sSUFBSSxLQUFLLENBQUMseUJBQXlCLFdBQVcsRUFBRSxDQUFDLENBQUM7WUFDMUQsQ0FBQztRQUNILENBQUM7UUFFRCw4Q0FBOEM7UUFDOUMsTUFBTSxjQUFjLEdBQUcsUUFBUSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxJQUFJLEtBQUssWUFBWSxDQUFDLENBQUM7UUFDbkUsSUFBSSxDQUFDLGNBQWMsRUFBRSxDQUFDO1lBQ3BCLE1BQU0sSUFBSSxLQUFLLENBQUMscUNBQXFDLENBQUMsQ0FBQztRQUN6RCxDQUFDO1FBRUQsTUFBTSxPQUFPLEdBQUcsY0FBYyxDQUFDLE9BQU8sQ0FBQztRQUN2QyxNQUFNLFdBQVcsR0FBRyxNQUFNLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQzVDLE1BQU0sUUFBUSxHQUFHLE9BQU8sQ0FBQyxHQUFHLElBQUksT0FBTyxDQUFDO1FBRXhDLE9BQU87WUFDTCxPQUFPLEVBQUU7Z0JBQ1AsV0FBVztnQkFDWCxPQUFPO2dCQUNQLFFBQVE7Z0JBQ1IsU0FBUyxFQUFFLE1BQU0sQ0FBQyxTQUFTO2dCQUMzQixNQUFNLEVBQUUsTUFBTSxDQUFDLE1BQU07Z0JBQ3JCLGlCQUFpQixFQUFFLGlCQUFpQixDQUFDLElBQUksR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDLGlCQUFpQixDQUFDLENBQUMsQ0FBQyxTQUFTO2FBQzlFO1lBQ0QsYUFBYSxFQUFFLE1BQU0sQ0FBQyxhQUFhO1NBQ3BDLENBQUM7SUFDSixDQUFDO0lBRUQ7O09BRUc7SUFDSyxNQUFNLENBQUMsWUFBWSxDQUFDLE1BQWMsRUFBRSxNQUFzQjtRQUNoRSxJQUFJLE1BQU0sR0FBRyxFQUFFLENBQUMsQ0FBQyxjQUFjO1FBRS9CLE1BQU0sS0FBSyxHQUFHLE1BQU0sQ0FBQyxXQUFXLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDekMsTUFBTSxJQUFJLENBQUMsQ0FBQztRQUVaLHVDQUF1QztRQUN2QyxJQUFJLE9BQU8sR0FBRyxNQUFNLENBQUM7UUFDckIsT0FBTyxNQUFNLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQyxJQUFJLE9BQU8sR0FBRyxNQUFNLENBQUMsTUFBTSxFQUFFLENBQUM7WUFDeEQsT0FBTyxFQUFFLENBQUM7UUFDWixDQUFDO1FBQ0QsTUFBTSxrQkFBa0IsR0FBRyxNQUFNLENBQUMsUUFBUSxDQUFDLE1BQU0sRUFBRSxPQUFPLENBQUMsQ0FBQyxRQUFRLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDN0UsTUFBTSxHQUFHLE9BQU8sR0FBRyxDQUFDLENBQUM7UUFFckIsTUFBTSxZQUFZLEdBQUcsTUFBTSxDQUFDLFdBQVcsQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUNoRCxNQUFNLElBQUksQ0FBQyxDQUFDO1FBRVosTUFBTSxjQUFjLEdBQUcsTUFBTSxDQUFDLFdBQVcsQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUNsRCxNQUFNLElBQUksQ0FBQyxDQUFDO1FBRVosc0JBQXNCO1FBQ3RCLE1BQU0sU0FBUyxHQUFHLE1BQU0sQ0FBQyxXQUFXLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDN0MsTUFBTSxXQUFXLEdBQUcsTUFBTSxDQUFDLFFBQVEsQ0FBQyxNQUFNLEVBQUUsTUFBTSxHQUFHLFNBQVMsQ0FBQyxDQUFDO1FBQ2hFLE1BQU0sS0FBSyxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBQ3BELE1BQU0sSUFBSSxTQUFTLENBQUM7UUFFcEIsdUZBQXVGO1FBQ3ZGLE1BQU0sS0FBSyxHQUFHLGtCQUFrQixDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUM1QyxNQUFNLFFBQVEsR0FBRyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFFMUIsa0RBQWtEO1FBQ2xELElBQUksV0FBVyxHQUFHLE1BQU0sQ0FBQztRQUN6QixJQUFJLE9BQU8sR0FBRyxLQUFLLENBQUM7UUFFcEIsSUFBSSxLQUFLLENBQUMsQ0FBQyxDQUFDLEtBQUssTUFBTSxFQUFFLENBQUM7WUFDeEIsb0JBQW9CO1lBQ3BCLFdBQVcsR0FBRyxNQUFNLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQ3BDLCtDQUErQztZQUMvQyxJQUFJLFdBQVcsS0FBSyxVQUFVLElBQUksV0FBVyxLQUFLLFVBQVUsRUFBRSxDQUFDO2dCQUM3RCxXQUFXLEdBQUcsT0FBTyxDQUFDO1lBQ3hCLENBQUM7UUFDSCxDQUFDO1FBRUQsT0FBTztZQUNMLE9BQU8sRUFBRTtnQkFDUCxXQUFXO2dCQUNYLE9BQU87Z0JBQ1AsUUFBUTtnQkFDUixTQUFTLEVBQUUsTUFBTSxDQUFDLFNBQVM7Z0JBQzNCLE1BQU0sRUFBRSxNQUFNLENBQUMsTUFBTTthQUN0QjtZQUNELGFBQWEsRUFBRSxNQUFNLENBQUMsYUFBYTtTQUNwQyxDQUFDO0lBQ0osQ0FBQztJQUVEOztPQUVHO0lBQ0gsTUFBTSxDQUFDLG1CQUFtQixDQUN4QixVQUFrQixFQUNsQixRQUErQixFQUMvQixZQUFvQixJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsR0FBRyxVQUFVLENBQUM7UUFFMUQsZ0RBQWdEO1FBQ2hELE1BQU0sV0FBVyxHQUFHLEVBQUUsR0FBRyxRQUFRLEVBQUUsQ0FBQztRQUVwQyxrQ0FBa0M7UUFDbEMsTUFBTSxRQUFRLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsV0FBVyxDQUFDLENBQUM7UUFFckQsMkJBQTJCO1FBQzNCLDREQUE0RDtRQUM1RCxNQUFNLGFBQWEsR0FBRyxFQUFFLEdBQUcsQ0FBQyxHQUFHLENBQUMsR0FBRyxRQUFRLENBQUMsTUFBTSxDQUFDO1FBRW5ELE1BQU0sTUFBTSxHQUFHLE1BQU0sQ0FBQyxLQUFLLENBQUMsYUFBYSxDQUFDLENBQUM7UUFDM0MsSUFBSSxNQUFNLEdBQUcsQ0FBQyxDQUFDO1FBRWYsZUFBZTtRQUNmLE1BQU0sQ0FBQyxZQUFZLENBQUMsYUFBYSxFQUFFLE1BQU0sQ0FBQyxDQUFDLENBQUssZ0JBQWdCO1FBQ2hFLE1BQU0sSUFBSSxDQUFDLENBQUM7UUFDWixNQUFNLENBQUMsWUFBWSxDQUFDLFNBQVMsRUFBRSxNQUFNLENBQUMsQ0FBQyxDQUFTLFlBQVk7UUFDNUQsTUFBTSxJQUFJLENBQUMsQ0FBQztRQUNaLE1BQU0sQ0FBQyxZQUFZLENBQUMsVUFBVSxFQUFFLE1BQU0sQ0FBQyxDQUFDLENBQVEsYUFBYTtRQUM3RCxNQUFNLElBQUksQ0FBQyxDQUFDO1FBQ1osTUFBTSxDQUFDLFlBQVksQ0FBQyxNQUFNLEVBQUUsTUFBTSxDQUFDLENBQUMsQ0FBWSxTQUFTO1FBQ3pELE1BQU0sSUFBSSxDQUFDLENBQUM7UUFFWixnQ0FBZ0M7UUFDaEMsTUFBTSxDQUFDLGFBQWEsQ0FBQyxDQUFDLEVBQUUsTUFBTSxDQUFDLENBQUM7UUFDaEMsTUFBTSxJQUFJLENBQUMsQ0FBQztRQUVaLDhCQUE4QjtRQUM5QixNQUFNLENBQUMsVUFBVSxDQUFDLFlBQVksRUFBRSxNQUFNLENBQUMsQ0FBQztRQUN4QyxNQUFNLElBQUksQ0FBQyxDQUFDO1FBRVosa0JBQWtCO1FBQ2xCLE1BQU0sQ0FBQyxJQUFJLENBQUMsUUFBUSxDQUFDLENBQUMsSUFBSSxDQUFDLE1BQU0sRUFBRSxNQUFNLENBQUMsQ0FBQztRQUUzQyxPQUFPLE1BQU0sQ0FBQztJQUNoQixDQUFDO0lBRUQ7O09BRUc7SUFDSCxNQUFNLENBQUMscUJBQXFCLENBQzFCLFVBQWtCLEVBQ2xCLFNBQWtDLEVBQ2xDLFlBQW9CLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLE1BQU0sRUFBRSxHQUFHLFVBQVUsQ0FBQyxFQUMxRCxXQUFtQixNQUFNLENBQUMsQ0FBQyxDQUFDO1FBRTVCLDBCQUEwQjtRQUMxQixNQUFNLFVBQVUsR0FBRyxTQUFTLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxTQUFTLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztRQUNyRSxNQUFNLGFBQWEsR0FBRyxVQUFVLENBQUMsTUFBTSxDQUFDLENBQUMsR0FBRyxFQUFFLEdBQUcsRUFBRSxFQUFFLENBQUMsR0FBRyxHQUFHLEdBQUcsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFFM0Usa0JBQWtCO1FBQ2xCLHFHQUFxRztRQUNyRyxNQUFNLGFBQWEsR0FBRyxFQUFFLEdBQUcsQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBQyxHQUFHLGFBQWEsQ0FBQztRQUV6RCxNQUFNLE1BQU0sR0FBRyxNQUFNLENBQUMsS0FBSyxDQUFDLGFBQWEsQ0FBQyxDQUFDO1FBQzNDLElBQUksTUFBTSxHQUFHLENBQUMsQ0FBQztRQUVmLGVBQWU7UUFDZixNQUFNLENBQUMsWUFBWSxDQUFDLGFBQWEsRUFBRSxNQUFNLENBQUMsQ0FBQyxDQUFLLGdCQUFnQjtRQUNoRSxNQUFNLElBQUksQ0FBQyxDQUFDO1FBQ1osTUFBTSxDQUFDLFlBQVksQ0FBQyxTQUFTLEVBQUUsTUFBTSxDQUFDLENBQUMsQ0FBUyxZQUFZO1FBQzVELE1BQU0sSUFBSSxDQUFDLENBQUM7UUFDWixNQUFNLENBQUMsWUFBWSxDQUFDLFVBQVUsRUFBRSxNQUFNLENBQUMsQ0FBQyxDQUFRLGFBQWE7UUFDN0QsTUFBTSxJQUFJLENBQUMsQ0FBQztRQUNaLE1BQU0sQ0FBQyxZQUFZLENBQUMsUUFBUSxFQUFFLE1BQU0sQ0FBQyxDQUFDLENBQVUsU0FBUztRQUN6RCxNQUFNLElBQUksQ0FBQyxDQUFDO1FBRVosd0JBQXdCO1FBQ3hCLE1BQU0sQ0FBQyxZQUFZLENBQUMsQ0FBQyxFQUFFLE1BQU0sQ0FBQyxDQUFDLENBQWlCLGdDQUFnQztRQUNoRixNQUFNLElBQUksQ0FBQyxDQUFDO1FBQ1osTUFBTSxDQUFDLGVBQWUsQ0FBQyxRQUFRLEVBQUUsTUFBTSxDQUFDLENBQUMsQ0FBTyxXQUFXO1FBQzNELE1BQU0sSUFBSSxDQUFDLENBQUM7UUFDWixNQUFNLENBQUMsWUFBWSxDQUFDLENBQUMsRUFBRSxNQUFNLENBQUMsQ0FBQyxDQUFpQixlQUFlO1FBQy9ELE1BQU0sSUFBSSxDQUFDLENBQUM7UUFDWixNQUFNLENBQUMsWUFBWSxDQUFDLFNBQVMsQ0FBQyxNQUFNLEVBQUUsTUFBTSxDQUFDLENBQUMsQ0FBRSxpQkFBaUI7UUFDakUsTUFBTSxJQUFJLENBQUMsQ0FBQztRQUVaLGtCQUFrQjtRQUNsQixLQUFLLE1BQU0sU0FBUyxJQUFJLFVBQVUsRUFBRSxDQUFDO1lBQ25DLE1BQU0sQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUMsSUFBSSxDQUFDLE1BQU0sRUFBRSxNQUFNLENBQUMsQ0FBQztZQUM1QyxNQUFNLElBQUksU0FBUyxDQUFDLE1BQU0sQ0FBQztRQUM3QixDQUFDO1FBRUQsT0FBTyxNQUFNLENBQUM7SUFDaEIsQ0FBQztJQUVEOztPQUVHO0lBQ0gsTUFBTSxDQUFDLG1CQUFtQixDQUN4QixVQUFrQixFQUNsQixTQUFpQixFQUNqQixZQUFvQixFQUNwQixXQUFvQjtRQUVwQixNQUFNLFFBQVEsR0FBMEI7WUFDdEMsRUFBRSxFQUFFLENBQUM7WUFDTCxNQUFNLEVBQUUsWUFBWTtZQUNwQixJQUFJLEVBQUUsU0FBUztZQUNmLFFBQVEsRUFBRSxJQUFJLENBQUMsZ0JBQWdCLENBQUMsU0FBUyxDQUFDO1NBQzNDLENBQUM7UUFFRixPQUFPLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxVQUFVLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDeEQsQ0FBQztJQUVEOztPQUVHO0lBQ0ssTUFBTSxDQUFDLGdCQUFnQixDQUFDLElBQVk7UUFDMUMsTUFBTSxVQUFVLEdBQTJCO1lBQ3pDLENBQUMsRUFBRSxJQUFJO1lBQ1AsQ0FBQyxFQUFFLGVBQWU7WUFDbEIsQ0FBQyxFQUFFLFVBQVU7WUFDYixLQUFLLEVBQUUsY0FBYztZQUNyQixLQUFLLEVBQUUsbUJBQW1CO1lBQzFCLEVBQUUsRUFBRSxjQUFjO1lBQ2xCLEVBQUUsRUFBRSxtQkFBbUI7WUFDdkIsRUFBRSxFQUFFLGVBQWU7WUFDbkIsRUFBRSxFQUFFLGlCQUFpQjtZQUNyQixFQUFFLEVBQUUsaUJBQWlCO1lBQ3JCLEVBQUUsRUFBRSxnQkFBZ0I7WUFDcEIsRUFBRSxFQUFFLGtCQUFrQjtZQUN0QixFQUFFLEVBQUUsc0JBQXNCO1lBQzFCLEdBQUcsRUFBRSxlQUFlO1lBQ3BCLEdBQUcsRUFBRSwyQkFBMkI7WUFDaEMsR0FBRyxFQUFFLGFBQWE7WUFDbEIsR0FBRyxFQUFFLG1CQUFtQjtTQUN6QixDQUFDO1FBRUYsT0FBTyxVQUFVLENBQUMsSUFBSSxDQUFDLElBQUksY0FBYyxDQUFDO0lBQzVDLENBQUM7Q0FDRiJ9