@aztec/bb.js 1.2.0 → 2.0.0-nightly.20250813
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/dest/browser/barretenberg/backend.d.ts.map +1 -1
- package/dest/browser/barretenberg/backend.js +56 -45
- package/dest/browser/barretenberg/index.d.ts +10 -0
- package/dest/browser/barretenberg/index.d.ts.map +1 -1
- package/dest/browser/barretenberg/index.js +30 -2
- package/dest/browser/barretenberg/verifier.js +2 -2
- package/dest/browser/barretenberg_api/index.d.ts +2 -2
- package/dest/browser/barretenberg_api/index.d.ts.map +1 -1
- package/dest/browser/barretenberg_api/index.js +5 -5
- package/dest/browser/barretenberg_wasm/barretenberg_wasm_base/index.d.ts +1 -0
- package/dest/browser/barretenberg_wasm/barretenberg_wasm_base/index.d.ts.map +1 -1
- package/dest/browser/barretenberg_wasm/barretenberg_wasm_base/index.js +5 -4
- package/dest/browser/barretenberg_wasm/barretenberg_wasm_main/index.d.ts +2 -0
- package/dest/browser/barretenberg_wasm/barretenberg_wasm_main/index.d.ts.map +1 -1
- package/dest/browser/barretenberg_wasm/barretenberg_wasm_main/index.js +17 -1
- package/dest/browser/barretenberg_wasm/barretenberg_wasm_thread/index.d.ts +1 -0
- package/dest/browser/barretenberg_wasm/barretenberg_wasm_thread/index.d.ts.map +1 -1
- package/dest/browser/barretenberg_wasm/fetch_code/browser/barretenberg-threads.js +1 -1
- package/dest/browser/barretenberg_wasm/fetch_code/browser/barretenberg.js +1 -1
- package/dest/browser/cbind/generate.d.ts +5 -0
- package/dest/browser/cbind/generate.d.ts.map +1 -0
- package/dest/browser/cbind/generate.js +64 -0
- package/dest/browser/cbind/generated/api_types.d.ts +801 -0
- package/dest/browser/cbind/generated/api_types.d.ts.map +1 -0
- package/dest/browser/cbind/generated/api_types.js +1099 -0
- package/dest/browser/cbind/generated/async.d.ts +27 -0
- package/dest/browser/cbind/generated/async.d.ts.map +1 -0
- package/dest/browser/cbind/generated/async.js +184 -0
- package/dest/browser/cbind/generated/native.d.ts +35 -0
- package/dest/browser/cbind/generated/native.d.ts.map +1 -0
- package/dest/browser/cbind/generated/native.js +270 -0
- package/dest/browser/cbind/generated/sync.d.ts +27 -0
- package/dest/browser/cbind/generated/sync.d.ts.map +1 -0
- package/dest/browser/cbind/generated/sync.js +165 -0
- package/dest/browser/cbind/schema_compiler.d.ts +70 -0
- package/dest/browser/cbind/schema_compiler.d.ts.map +1 -0
- package/dest/browser/cbind/schema_compiler.js +683 -0
- package/dest/node/barretenberg/backend.d.ts.map +1 -1
- package/dest/node/barretenberg/backend.js +56 -45
- package/dest/node/barretenberg/index.d.ts +10 -0
- package/dest/node/barretenberg/index.d.ts.map +1 -1
- package/dest/node/barretenberg/index.js +30 -2
- package/dest/node/barretenberg/verifier.js +2 -2
- package/dest/node/barretenberg_api/index.d.ts +2 -2
- package/dest/node/barretenberg_api/index.d.ts.map +1 -1
- package/dest/node/barretenberg_api/index.js +5 -5
- package/dest/node/barretenberg_wasm/barretenberg-threads.wasm.gz +0 -0
- package/dest/node/barretenberg_wasm/barretenberg_wasm_base/index.d.ts +1 -0
- package/dest/node/barretenberg_wasm/barretenberg_wasm_base/index.d.ts.map +1 -1
- package/dest/node/barretenberg_wasm/barretenberg_wasm_base/index.js +5 -4
- package/dest/node/barretenberg_wasm/barretenberg_wasm_main/index.d.ts +2 -0
- package/dest/node/barretenberg_wasm/barretenberg_wasm_main/index.d.ts.map +1 -1
- package/dest/node/barretenberg_wasm/barretenberg_wasm_main/index.js +17 -1
- package/dest/node/barretenberg_wasm/barretenberg_wasm_thread/index.d.ts +1 -0
- package/dest/node/barretenberg_wasm/barretenberg_wasm_thread/index.d.ts.map +1 -1
- package/dest/node/cbind/generate.d.ts +5 -0
- package/dest/node/cbind/generate.d.ts.map +1 -0
- package/dest/node/cbind/generate.js +64 -0
- package/dest/node/cbind/generated/api_types.d.ts +801 -0
- package/dest/node/cbind/generated/api_types.d.ts.map +1 -0
- package/dest/node/cbind/generated/api_types.js +1099 -0
- package/dest/node/cbind/generated/async.d.ts +27 -0
- package/dest/node/cbind/generated/async.d.ts.map +1 -0
- package/dest/node/cbind/generated/async.js +184 -0
- package/dest/node/cbind/generated/native.d.ts +35 -0
- package/dest/node/cbind/generated/native.d.ts.map +1 -0
- package/dest/node/cbind/generated/native.js +270 -0
- package/dest/node/cbind/generated/sync.d.ts +27 -0
- package/dest/node/cbind/generated/sync.d.ts.map +1 -0
- package/dest/node/cbind/generated/sync.js +165 -0
- package/dest/node/cbind/schema_compiler.d.ts +70 -0
- package/dest/node/cbind/schema_compiler.d.ts.map +1 -0
- package/dest/node/cbind/schema_compiler.js +683 -0
- package/dest/node/main.js +3 -3
- package/dest/node-cjs/barretenberg/backend.d.ts.map +1 -1
- package/dest/node-cjs/barretenberg/backend.js +58 -47
- package/dest/node-cjs/barretenberg/index.d.ts +10 -0
- package/dest/node-cjs/barretenberg/index.d.ts.map +1 -1
- package/dest/node-cjs/barretenberg/index.js +30 -2
- package/dest/node-cjs/barretenberg/verifier.js +2 -2
- package/dest/node-cjs/barretenberg_api/index.d.ts +2 -2
- package/dest/node-cjs/barretenberg_api/index.d.ts.map +1 -1
- package/dest/node-cjs/barretenberg_api/index.js +5 -5
- package/dest/node-cjs/barretenberg_wasm/barretenberg-threads.wasm.gz +0 -0
- package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_base/index.d.ts +1 -0
- package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_base/index.d.ts.map +1 -1
- package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_base/index.js +5 -4
- package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_main/index.d.ts +2 -0
- package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_main/index.d.ts.map +1 -1
- package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_main/index.js +17 -1
- package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_thread/index.d.ts +1 -0
- package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_thread/index.d.ts.map +1 -1
- package/dest/node-cjs/cbind/generate.d.ts +5 -0
- package/dest/node-cjs/cbind/generate.d.ts.map +1 -0
- package/dest/node-cjs/cbind/generate.js +66 -0
- package/dest/node-cjs/cbind/generated/api_types.d.ts +801 -0
- package/dest/node-cjs/cbind/generated/api_types.d.ts.map +1 -0
- package/dest/node-cjs/cbind/generated/api_types.js +1189 -0
- package/dest/node-cjs/cbind/generated/async.d.ts +27 -0
- package/dest/node-cjs/cbind/generated/async.d.ts.map +1 -0
- package/dest/node-cjs/cbind/generated/async.js +188 -0
- package/dest/node-cjs/cbind/generated/native.d.ts +35 -0
- package/dest/node-cjs/cbind/generated/native.d.ts.map +1 -0
- package/dest/node-cjs/cbind/generated/native.js +274 -0
- package/dest/node-cjs/cbind/generated/sync.d.ts +27 -0
- package/dest/node-cjs/cbind/generated/sync.d.ts.map +1 -0
- package/dest/node-cjs/cbind/generated/sync.js +169 -0
- package/dest/node-cjs/cbind/schema_compiler.d.ts +70 -0
- package/dest/node-cjs/cbind/schema_compiler.d.ts.map +1 -0
- package/dest/node-cjs/cbind/schema_compiler.js +691 -0
- package/dest/node-cjs/main.js +3 -3
- package/package.json +4 -3
- package/src/barretenberg/backend.ts +62 -63
- package/src/barretenberg/index.ts +39 -1
- package/src/barretenberg/verifier.ts +1 -1
- package/src/barretenberg_api/index.ts +4 -4
- package/src/barretenberg_wasm/barretenberg_wasm_base/index.ts +5 -3
- package/src/barretenberg_wasm/barretenberg_wasm_main/index.ts +23 -0
- package/src/cbind/README.md +1 -0
- package/src/cbind/generate.ts +89 -0
- package/src/cbind/schema_compiler.ts +832 -0
- package/src/main.ts +2 -2
|
@@ -0,0 +1,691 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Unified schema compiler with integrated strategies
|
|
4
|
+
*/
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.SchemaCompiler = void 0;
|
|
7
|
+
exports.createSharedTypesCompiler = createSharedTypesCompiler;
|
|
8
|
+
exports.createSyncApiCompiler = createSyncApiCompiler;
|
|
9
|
+
exports.createAsyncApiCompiler = createAsyncApiCompiler;
|
|
10
|
+
exports.createNativeApiCompiler = createNativeApiCompiler;
|
|
11
|
+
// Helper functions
|
|
12
|
+
function capitalize(s) {
|
|
13
|
+
return s.charAt(0).toUpperCase() + s.substring(1);
|
|
14
|
+
}
|
|
15
|
+
function camelCase(s) {
|
|
16
|
+
return s
|
|
17
|
+
.split('_')
|
|
18
|
+
.map((part, index) => (index === 0 ? part.charAt(0).toLowerCase() + part.substring(1) : capitalize(part)))
|
|
19
|
+
.join('');
|
|
20
|
+
}
|
|
21
|
+
function pascalCase(s) {
|
|
22
|
+
return s.split('_').map(capitalize).join('');
|
|
23
|
+
}
|
|
24
|
+
class SchemaCompiler {
|
|
25
|
+
constructor(config) {
|
|
26
|
+
this.config = config;
|
|
27
|
+
this.typeCache = new Map();
|
|
28
|
+
this.functionMetadata = [];
|
|
29
|
+
// WORKTODO(bbapi): AI slop fixup - redundant with typeCache, remove
|
|
30
|
+
this.referencedTypes = new Set();
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Process API schema and extract function metadata
|
|
34
|
+
*/
|
|
35
|
+
processApiSchema(commandsSchema, responsesSchema) {
|
|
36
|
+
// Process types
|
|
37
|
+
this.processSchema(commandsSchema);
|
|
38
|
+
this.processSchema(responsesSchema);
|
|
39
|
+
// Extract function metadata from named unions
|
|
40
|
+
if (!Array.isArray(commandsSchema) || commandsSchema[0] !== 'named_union' ||
|
|
41
|
+
!Array.isArray(responsesSchema) || responsesSchema[0] !== 'named_union') {
|
|
42
|
+
throw new Error('Expected named_union schema format');
|
|
43
|
+
}
|
|
44
|
+
const commands = commandsSchema[1];
|
|
45
|
+
const responses = responsesSchema[1];
|
|
46
|
+
for (let i = 0; i < commands.length; i++) {
|
|
47
|
+
const [commandName] = commands[i];
|
|
48
|
+
const [responseName] = responses[i];
|
|
49
|
+
this.functionMetadata.push({
|
|
50
|
+
name: camelCase(commandName),
|
|
51
|
+
commandType: pascalCase(commandName),
|
|
52
|
+
responseType: pascalCase(responseName),
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Process a schema and populate type cache
|
|
58
|
+
*/
|
|
59
|
+
processSchema(schema) {
|
|
60
|
+
const key = this.getSchemaKey(schema);
|
|
61
|
+
if (this.typeCache.has(key)) {
|
|
62
|
+
return this.typeCache.get(key);
|
|
63
|
+
}
|
|
64
|
+
const typeInfo = this.generateTypeInfo(schema);
|
|
65
|
+
this.typeCache.set(key, typeInfo);
|
|
66
|
+
return typeInfo;
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Generate the complete output
|
|
70
|
+
*/
|
|
71
|
+
compile() {
|
|
72
|
+
const parts = [
|
|
73
|
+
'// AUTOGENERATED FILE - DO NOT EDIT',
|
|
74
|
+
'',
|
|
75
|
+
];
|
|
76
|
+
// Generate imports
|
|
77
|
+
parts.push(...this.generateImports());
|
|
78
|
+
parts.push('');
|
|
79
|
+
// Generate type declarations only for 'types' mode
|
|
80
|
+
if (this.config.mode === 'types') {
|
|
81
|
+
const sortedTypes = Array.from(this.typeCache.values())
|
|
82
|
+
.filter(t => t.declaration)
|
|
83
|
+
.sort((a, b) => a.typeName.localeCompare(b.typeName));
|
|
84
|
+
// Group declarations
|
|
85
|
+
const typeAliases = sortedTypes.filter(t => t.declaration?.startsWith('export type') && !t.declaration?.includes('interface'));
|
|
86
|
+
const publicInterfaces = sortedTypes.filter(t => t.declaration?.includes('export interface'));
|
|
87
|
+
const privateInterfaces = sortedTypes.filter(t => t.declaration?.includes('interface Msgpack'));
|
|
88
|
+
// Add type aliases if needed
|
|
89
|
+
if (typeAliases.length > 0) {
|
|
90
|
+
parts.push('// Type aliases');
|
|
91
|
+
for (const type of typeAliases) {
|
|
92
|
+
parts.push(type.declaration);
|
|
93
|
+
}
|
|
94
|
+
parts.push('');
|
|
95
|
+
}
|
|
96
|
+
// Add tuple helper if needed
|
|
97
|
+
if (this.needsTupleHelper()) {
|
|
98
|
+
parts.push('// Tuple type for fixed-size arrays', 'type Tuple<T, N extends number> = N extends N ? (number extends N ? T[] : _TupleOf<T, N, []>) : never;', 'type _TupleOf<T, N extends number, R extends unknown[]> = R[\'length\'] extends N ? R : _TupleOf<T, N, [T, ...R]>;', '', '// Helper functions', 'function mapTuple<T, S, N extends number>(tuple: Tuple<T, N>, fn: (item: T) => S): Tuple<S, N> {', ' return tuple.map(fn) as Tuple<S, N>;', '}', '');
|
|
99
|
+
}
|
|
100
|
+
// Add public interfaces
|
|
101
|
+
if (publicInterfaces.length > 0) {
|
|
102
|
+
parts.push('// Public interfaces (exported)');
|
|
103
|
+
for (const type of publicInterfaces) {
|
|
104
|
+
parts.push(type.declaration, '');
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
// Add private interfaces
|
|
108
|
+
if (privateInterfaces.length > 0) {
|
|
109
|
+
parts.push('// Private Msgpack interfaces (not exported)');
|
|
110
|
+
for (const type of privateInterfaces) {
|
|
111
|
+
parts.push(type.declaration, '');
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
// Add conversion functions (only for api_types.ts)
|
|
115
|
+
const conversions = sortedTypes.filter(t => t.toMethod || t.fromMethod);
|
|
116
|
+
if (conversions.length > 0) {
|
|
117
|
+
parts.push('// Conversion functions (exported)');
|
|
118
|
+
for (const type of conversions) {
|
|
119
|
+
if (type.toMethod) {
|
|
120
|
+
parts.push('export ' + type.toMethod, '');
|
|
121
|
+
}
|
|
122
|
+
if (type.fromMethod) {
|
|
123
|
+
parts.push('export ' + type.fromMethod, '');
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
// Add BbApiBase interface
|
|
128
|
+
if (this.functionMetadata.length > 0) {
|
|
129
|
+
parts.push('', '// Base API interface');
|
|
130
|
+
parts.push(this.generateBbApiBaseInterface());
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
// Add API class for non-types modes
|
|
134
|
+
if (this.config.mode !== 'types' && this.functionMetadata.length > 0) {
|
|
135
|
+
parts.push(this.generateApiClass());
|
|
136
|
+
}
|
|
137
|
+
return parts.join('\n') + '\n';
|
|
138
|
+
}
|
|
139
|
+
getSchemaKey(schema) {
|
|
140
|
+
if (typeof schema === 'string')
|
|
141
|
+
return schema;
|
|
142
|
+
if (Array.isArray(schema))
|
|
143
|
+
return JSON.stringify(schema);
|
|
144
|
+
if (typeof schema === 'object')
|
|
145
|
+
return schema.__typename || JSON.stringify(schema);
|
|
146
|
+
return String(schema);
|
|
147
|
+
}
|
|
148
|
+
needsTupleHelper() {
|
|
149
|
+
return Array.from(this.typeCache.values()).some(t => t.typeName.includes('Tuple<'));
|
|
150
|
+
}
|
|
151
|
+
trackTypeUsage(typeName) {
|
|
152
|
+
// Only track for API modes
|
|
153
|
+
if (this.config.mode === 'types')
|
|
154
|
+
return;
|
|
155
|
+
// Extract base types from complex types
|
|
156
|
+
const baseTypes = this.extractBaseTypes(typeName);
|
|
157
|
+
for (const type of baseTypes) {
|
|
158
|
+
// Skip built-in types
|
|
159
|
+
if (['string', 'number', 'boolean', 'Uint8Array'].includes(type)) {
|
|
160
|
+
continue;
|
|
161
|
+
}
|
|
162
|
+
this.referencedTypes.add(type);
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
extractBaseTypes(typeName) {
|
|
166
|
+
const types = [];
|
|
167
|
+
// Handle arrays
|
|
168
|
+
const arrayMatch = typeName.match(/^(.+)\[\]$/);
|
|
169
|
+
if (arrayMatch) {
|
|
170
|
+
types.push(...this.extractBaseTypes(arrayMatch[1]));
|
|
171
|
+
return types;
|
|
172
|
+
}
|
|
173
|
+
// Handle Tuple
|
|
174
|
+
const tupleMatch = typeName.match(/^Tuple<(.+),\s*\d+>$/);
|
|
175
|
+
if (tupleMatch) {
|
|
176
|
+
types.push(...this.extractBaseTypes(tupleMatch[1]));
|
|
177
|
+
return types;
|
|
178
|
+
}
|
|
179
|
+
// Handle Record
|
|
180
|
+
const recordMatch = typeName.match(/^Record<(.+),\s*(.+)>$/);
|
|
181
|
+
if (recordMatch) {
|
|
182
|
+
types.push(...this.extractBaseTypes(recordMatch[1]));
|
|
183
|
+
types.push(...this.extractBaseTypes(recordMatch[2]));
|
|
184
|
+
return types;
|
|
185
|
+
}
|
|
186
|
+
// Handle union types
|
|
187
|
+
if (typeName.includes(' | ')) {
|
|
188
|
+
const parts = typeName.split(' | ');
|
|
189
|
+
for (const part of parts) {
|
|
190
|
+
types.push(...this.extractBaseTypes(part.trim()));
|
|
191
|
+
}
|
|
192
|
+
return types;
|
|
193
|
+
}
|
|
194
|
+
// Base case - simple type
|
|
195
|
+
types.push(typeName);
|
|
196
|
+
return types;
|
|
197
|
+
}
|
|
198
|
+
generateTypeInfo(schema) {
|
|
199
|
+
if (Array.isArray(schema)) {
|
|
200
|
+
return this.processArraySchema(schema);
|
|
201
|
+
}
|
|
202
|
+
else if (typeof schema === 'string') {
|
|
203
|
+
return this.processPrimitiveSchema(schema);
|
|
204
|
+
}
|
|
205
|
+
else if (typeof schema === 'object') {
|
|
206
|
+
return this.processObjectSchema(schema);
|
|
207
|
+
}
|
|
208
|
+
throw new Error(`Unsupported schema type: ${schema}`);
|
|
209
|
+
}
|
|
210
|
+
processArraySchema(schema) {
|
|
211
|
+
const [type, ...args] = schema;
|
|
212
|
+
switch (type) {
|
|
213
|
+
case 'array': {
|
|
214
|
+
const [subtype, size] = args[0];
|
|
215
|
+
const subtypeInfo = this.processSchema(subtype);
|
|
216
|
+
return {
|
|
217
|
+
typeName: `Tuple<${subtypeInfo.typeName}, ${size}>`,
|
|
218
|
+
msgpackTypeName: `Tuple<${subtypeInfo.msgpackTypeName || subtypeInfo.typeName}, ${size}>`,
|
|
219
|
+
};
|
|
220
|
+
}
|
|
221
|
+
case 'variant': {
|
|
222
|
+
const variants = args[0];
|
|
223
|
+
const variantInfos = variants.map(v => this.processSchema(v));
|
|
224
|
+
const typeName = variantInfos.map(v => v.typeName).join(' | ');
|
|
225
|
+
const msgpackUnion = variantInfos.map(v => v.msgpackTypeName || v.typeName).join(' | ');
|
|
226
|
+
return {
|
|
227
|
+
typeName,
|
|
228
|
+
msgpackTypeName: `[number, ${msgpackUnion}]`,
|
|
229
|
+
};
|
|
230
|
+
}
|
|
231
|
+
case 'named_union': {
|
|
232
|
+
const namedTypes = args[0];
|
|
233
|
+
const tupleTypes = [];
|
|
234
|
+
for (const [name, schemaOrName] of namedTypes) {
|
|
235
|
+
let typeInfo;
|
|
236
|
+
if (typeof schemaOrName === 'string') {
|
|
237
|
+
const typeName = pascalCase(schemaOrName);
|
|
238
|
+
typeInfo = this.getOrCreateEmptyType(typeName);
|
|
239
|
+
}
|
|
240
|
+
else {
|
|
241
|
+
typeInfo = this.processSchema(schemaOrName);
|
|
242
|
+
}
|
|
243
|
+
// Track usage of the type
|
|
244
|
+
this.trackTypeUsage(typeInfo.typeName);
|
|
245
|
+
tupleTypes.push(`["${name}", ${typeInfo.typeName}]`);
|
|
246
|
+
}
|
|
247
|
+
return {
|
|
248
|
+
typeName: tupleTypes.join(' | '),
|
|
249
|
+
msgpackTypeName: tupleTypes.join(' | '),
|
|
250
|
+
};
|
|
251
|
+
}
|
|
252
|
+
case 'vector': {
|
|
253
|
+
const [subtype] = args[0];
|
|
254
|
+
if (subtype === 'unsigned char') {
|
|
255
|
+
return { typeName: 'Uint8Array' };
|
|
256
|
+
}
|
|
257
|
+
const subtypeInfo = this.processSchema(subtype);
|
|
258
|
+
return {
|
|
259
|
+
typeName: `${subtypeInfo.typeName}[]`,
|
|
260
|
+
msgpackTypeName: `${subtypeInfo.msgpackTypeName || subtypeInfo.typeName}[]`,
|
|
261
|
+
};
|
|
262
|
+
}
|
|
263
|
+
case 'alias': {
|
|
264
|
+
const [rawTypeName, msgpackName] = args[0];
|
|
265
|
+
const typeName = pascalCase(rawTypeName);
|
|
266
|
+
let targetType;
|
|
267
|
+
if (msgpackName.startsWith('bin')) {
|
|
268
|
+
targetType = 'Uint8Array';
|
|
269
|
+
}
|
|
270
|
+
else if (['int', 'unsigned int', 'unsigned short'].includes(msgpackName)) {
|
|
271
|
+
targetType = 'number';
|
|
272
|
+
}
|
|
273
|
+
else {
|
|
274
|
+
throw new Error(`Unsupported alias type: ${msgpackName}`);
|
|
275
|
+
}
|
|
276
|
+
// Create a proper type alias declaration
|
|
277
|
+
return {
|
|
278
|
+
typeName,
|
|
279
|
+
msgpackTypeName: targetType,
|
|
280
|
+
declaration: `export type ${typeName} = ${targetType};`,
|
|
281
|
+
};
|
|
282
|
+
}
|
|
283
|
+
case 'shared_ptr': {
|
|
284
|
+
const [subtype] = args[0];
|
|
285
|
+
return this.processSchema(subtype);
|
|
286
|
+
}
|
|
287
|
+
case 'map': {
|
|
288
|
+
const [keyType, valueType] = args[0];
|
|
289
|
+
const keyInfo = this.processSchema(keyType);
|
|
290
|
+
const valueInfo = this.processSchema(valueType);
|
|
291
|
+
return {
|
|
292
|
+
typeName: `Record<${keyInfo.typeName}, ${valueInfo.typeName}>`,
|
|
293
|
+
msgpackTypeName: `Record<${keyInfo.msgpackTypeName || keyInfo.typeName}, ${valueInfo.msgpackTypeName || valueInfo.typeName}>`,
|
|
294
|
+
};
|
|
295
|
+
}
|
|
296
|
+
default:
|
|
297
|
+
throw new Error(`Unsupported array schema type: ${type}`);
|
|
298
|
+
}
|
|
299
|
+
}
|
|
300
|
+
processPrimitiveSchema(schema) {
|
|
301
|
+
switch (schema) {
|
|
302
|
+
case 'bool':
|
|
303
|
+
return { typeName: 'boolean' };
|
|
304
|
+
case 'int':
|
|
305
|
+
case 'unsigned int':
|
|
306
|
+
case 'unsigned short':
|
|
307
|
+
case 'unsigned long':
|
|
308
|
+
case 'double':
|
|
309
|
+
return { typeName: 'number' };
|
|
310
|
+
case 'string':
|
|
311
|
+
return { typeName: 'string' };
|
|
312
|
+
case 'bin32':
|
|
313
|
+
return { typeName: 'Uint8Array' };
|
|
314
|
+
default:
|
|
315
|
+
return { typeName: pascalCase(schema) };
|
|
316
|
+
}
|
|
317
|
+
}
|
|
318
|
+
processObjectSchema(schema) {
|
|
319
|
+
const typeName = pascalCase(schema.__typename);
|
|
320
|
+
const msgpackTypeName = 'Msgpack' + typeName;
|
|
321
|
+
const declaration = this.generateInterfaces(typeName, schema);
|
|
322
|
+
const toMethod = this.generateToMethod(typeName, schema);
|
|
323
|
+
const fromMethod = this.generateFromMethod(typeName, schema);
|
|
324
|
+
return {
|
|
325
|
+
typeName,
|
|
326
|
+
msgpackTypeName,
|
|
327
|
+
declaration,
|
|
328
|
+
toMethod,
|
|
329
|
+
fromMethod,
|
|
330
|
+
};
|
|
331
|
+
}
|
|
332
|
+
getOrCreateEmptyType(typeName) {
|
|
333
|
+
const key = `empty_${typeName}`;
|
|
334
|
+
if (this.typeCache.has(key)) {
|
|
335
|
+
return this.typeCache.get(key);
|
|
336
|
+
}
|
|
337
|
+
const typeInfo = {
|
|
338
|
+
typeName,
|
|
339
|
+
msgpackTypeName: 'Msgpack' + typeName,
|
|
340
|
+
declaration: `export interface ${typeName} {}\n\ninterface Msgpack${typeName} {}`,
|
|
341
|
+
toMethod: `function to${typeName}(o: Msgpack${typeName}): ${typeName} {\n return {};\n}`,
|
|
342
|
+
fromMethod: `function from${typeName}(o: ${typeName}): Msgpack${typeName} {\n return {};\n}`,
|
|
343
|
+
};
|
|
344
|
+
this.typeCache.set(key, typeInfo);
|
|
345
|
+
return typeInfo;
|
|
346
|
+
}
|
|
347
|
+
generateInterfaces(name, schema) {
|
|
348
|
+
const publicInterface = this.generatePublicInterface(name, schema);
|
|
349
|
+
const msgpackInterface = this.generateMsgpackInterface(name, schema);
|
|
350
|
+
return publicInterface + '\n\n' + msgpackInterface;
|
|
351
|
+
}
|
|
352
|
+
generatePublicInterface(name, schema) {
|
|
353
|
+
let result = `export interface ${name} {\n`;
|
|
354
|
+
for (const [key, value] of Object.entries(schema)) {
|
|
355
|
+
if (key === '__typename')
|
|
356
|
+
continue;
|
|
357
|
+
const typeInfo = this.processSchema(value);
|
|
358
|
+
// Track type usage
|
|
359
|
+
this.trackTypeUsage(typeInfo.typeName);
|
|
360
|
+
result += ` ${camelCase(key)}: ${typeInfo.typeName};\n`;
|
|
361
|
+
}
|
|
362
|
+
result += '}';
|
|
363
|
+
return result;
|
|
364
|
+
}
|
|
365
|
+
generateMsgpackInterface(name, schema) {
|
|
366
|
+
let result = `interface Msgpack${name} {\n`;
|
|
367
|
+
for (const [key, value] of Object.entries(schema)) {
|
|
368
|
+
if (key === '__typename')
|
|
369
|
+
continue;
|
|
370
|
+
const typeInfo = this.processSchema(value);
|
|
371
|
+
result += ` ${key}: ${typeInfo.msgpackTypeName || typeInfo.typeName};\n`;
|
|
372
|
+
}
|
|
373
|
+
result += '}';
|
|
374
|
+
return result;
|
|
375
|
+
}
|
|
376
|
+
generateToMethod(name, schema) {
|
|
377
|
+
const fields = Object.entries(schema).filter(([key]) => key !== '__typename');
|
|
378
|
+
if (fields.length === 0) {
|
|
379
|
+
return `function to${name}(o: Msgpack${name}): ${name} {\n return {};\n}`;
|
|
380
|
+
}
|
|
381
|
+
const checks = fields.map(([key]) => ` if (o.${key} === undefined) { throw new Error("Expected ${key} in ${name} deserialization"); }`).join('\n');
|
|
382
|
+
const conversions = fields.map(([key, value]) => {
|
|
383
|
+
const typeInfo = this.processSchema(value);
|
|
384
|
+
const converter = this.generateConverter(typeInfo, `o.${key}`, 'to');
|
|
385
|
+
return ` ${camelCase(key)}: ${converter},`;
|
|
386
|
+
}).join('\n');
|
|
387
|
+
return `function to${name}(o: Msgpack${name}): ${name} {
|
|
388
|
+
${checks};
|
|
389
|
+
return {
|
|
390
|
+
${conversions}
|
|
391
|
+
};
|
|
392
|
+
}`;
|
|
393
|
+
}
|
|
394
|
+
generateFromMethod(name, schema) {
|
|
395
|
+
const fields = Object.entries(schema).filter(([key]) => key !== '__typename');
|
|
396
|
+
if (fields.length === 0) {
|
|
397
|
+
return `function from${name}(o: ${name}): Msgpack${name} {\n return {};\n}`;
|
|
398
|
+
}
|
|
399
|
+
const checks = fields.map(([key]) => ` if (o.${camelCase(key)} === undefined) { throw new Error("Expected ${camelCase(key)} in ${name} serialization"); }`).join('\n');
|
|
400
|
+
const conversions = fields.map(([key, value]) => {
|
|
401
|
+
const typeInfo = this.processSchema(value);
|
|
402
|
+
const converter = this.generateConverter(typeInfo, `o.${camelCase(key)}`, 'from');
|
|
403
|
+
return ` ${key}: ${converter},`;
|
|
404
|
+
}).join('\n');
|
|
405
|
+
return `function from${name}(o: ${name}): Msgpack${name} {
|
|
406
|
+
${checks};
|
|
407
|
+
return {
|
|
408
|
+
${conversions}
|
|
409
|
+
};
|
|
410
|
+
}`;
|
|
411
|
+
}
|
|
412
|
+
generateConverter(typeInfo, value, direction) {
|
|
413
|
+
// Handle arrays/tuples
|
|
414
|
+
if (typeInfo.typeName.includes('[]') || typeInfo.typeName.includes('Tuple<')) {
|
|
415
|
+
const elementType = typeInfo.typeName.match(/^(.+)\[\]$/) || typeInfo.typeName.match(/^Tuple<(.+),/);
|
|
416
|
+
if (elementType) {
|
|
417
|
+
const isTuple = typeInfo.typeName.includes('Tuple<');
|
|
418
|
+
const mapFn = isTuple ? 'mapTuple' : 'map';
|
|
419
|
+
return `${value}.${mapFn}((v: any) => v)`; // Simplified for now
|
|
420
|
+
}
|
|
421
|
+
}
|
|
422
|
+
// Handle custom types
|
|
423
|
+
if (typeInfo.declaration) {
|
|
424
|
+
return `${direction}${typeInfo.typeName}(${value})`;
|
|
425
|
+
}
|
|
426
|
+
return value;
|
|
427
|
+
}
|
|
428
|
+
generateImports() {
|
|
429
|
+
const imports = [];
|
|
430
|
+
// Base imports
|
|
431
|
+
if (this.config.imports) {
|
|
432
|
+
imports.push(...this.config.imports);
|
|
433
|
+
}
|
|
434
|
+
// For API modes, import from api_types
|
|
435
|
+
if (this.config.mode !== 'types') {
|
|
436
|
+
const neededImports = new Set();
|
|
437
|
+
// Add types and conversion functions from function metadata
|
|
438
|
+
for (const metadata of this.functionMetadata) {
|
|
439
|
+
neededImports.add(metadata.commandType);
|
|
440
|
+
neededImports.add(metadata.responseType);
|
|
441
|
+
neededImports.add(`from${metadata.commandType}`);
|
|
442
|
+
neededImports.add(`to${metadata.responseType}`);
|
|
443
|
+
}
|
|
444
|
+
// Add referenced types
|
|
445
|
+
for (const type of this.referencedTypes) {
|
|
446
|
+
neededImports.add(type);
|
|
447
|
+
}
|
|
448
|
+
// Add BbApiBase interface
|
|
449
|
+
neededImports.add('BbApiBase');
|
|
450
|
+
if (neededImports.size > 0) {
|
|
451
|
+
const sortedImports = Array.from(neededImports).sort();
|
|
452
|
+
imports.push(`import { ${sortedImports.join(', ')} } from './api_types.js';`);
|
|
453
|
+
}
|
|
454
|
+
}
|
|
455
|
+
return imports;
|
|
456
|
+
}
|
|
457
|
+
generateBbApiBaseInterface() {
|
|
458
|
+
const methods = this.functionMetadata.map(m => ` ${m.name}(command: ${m.commandType}): Promise<${m.responseType}>;`).join('\n');
|
|
459
|
+
return `export interface BbApiBase {
|
|
460
|
+
${methods}
|
|
461
|
+
destroy(): Promise<void>;
|
|
462
|
+
}`;
|
|
463
|
+
}
|
|
464
|
+
generateApiClass() {
|
|
465
|
+
const className = this.getApiClassName();
|
|
466
|
+
const methods = this.functionMetadata.map(m => this.generateApiMethod(m)).join('\n\n');
|
|
467
|
+
if (this.config.mode === 'native') {
|
|
468
|
+
return this.generateNativeApiClass(methods);
|
|
469
|
+
}
|
|
470
|
+
// For sync API, don't implement BbApiBase since methods are synchronous
|
|
471
|
+
const implementsClause = this.config.mode === 'sync' ? '' : ' implements BbApiBase';
|
|
472
|
+
const msgpackCallHelper = `${this.config.mode === 'async' ? 'async ' : ''}function msgpackCall(wasm: ${this.getWasmType()}, cbind: string, input: any[]) {` +
|
|
473
|
+
` const inputBuffer = new Encoder({ useRecords: false }).pack(input);` +
|
|
474
|
+
` const encodedResult = ${this.config.mode === 'async' ? 'await ' : ''}wasm.cbindCall(cbind, inputBuffer);` +
|
|
475
|
+
` return new Decoder({ useRecords: false }).unpack(encodedResult);` +
|
|
476
|
+
`}\n`;
|
|
477
|
+
return (msgpackCallHelper +
|
|
478
|
+
`export class ${className}${implementsClause} {
|
|
479
|
+
constructor(protected wasm: ${this.getWasmType()}) {}
|
|
480
|
+
|
|
481
|
+
${methods}
|
|
482
|
+
|
|
483
|
+
destroy(): Promise<void> {
|
|
484
|
+
return this.wasm.destroy();
|
|
485
|
+
}
|
|
486
|
+
}`);
|
|
487
|
+
}
|
|
488
|
+
getApiClassName() {
|
|
489
|
+
switch (this.config.mode) {
|
|
490
|
+
case 'sync': return 'SyncApi';
|
|
491
|
+
case 'async': return 'AsyncApi';
|
|
492
|
+
case 'native': return 'NativeApi';
|
|
493
|
+
default: throw new Error(`Invalid mode: ${this.config.mode}`);
|
|
494
|
+
}
|
|
495
|
+
}
|
|
496
|
+
getWasmType() {
|
|
497
|
+
switch (this.config.mode) {
|
|
498
|
+
case 'sync': return 'BarretenbergWasmMain';
|
|
499
|
+
case 'async': return 'BarretenbergWasmMainWorker';
|
|
500
|
+
default: return '';
|
|
501
|
+
}
|
|
502
|
+
}
|
|
503
|
+
generateApiMethod(metadata) {
|
|
504
|
+
const { name, commandType, responseType } = metadata;
|
|
505
|
+
if (this.config.mode === 'native') {
|
|
506
|
+
return ` ${name}(command: ${commandType}): Promise<${responseType}> {
|
|
507
|
+
const msgpackCommand = from${commandType}(command);
|
|
508
|
+
return this.sendCommand(['${metadata.commandType}', msgpackCommand]).then(([variantName, result]: [string, any]) => {
|
|
509
|
+
if (variantName !== '${responseType}') {
|
|
510
|
+
throw new Error(\`Expected variant name '${responseType}' but got '\${variantName}'\`);
|
|
511
|
+
}
|
|
512
|
+
return to${responseType}(result);
|
|
513
|
+
});
|
|
514
|
+
}`;
|
|
515
|
+
}
|
|
516
|
+
// For async mode, queue immediately and return promise
|
|
517
|
+
if (this.config.mode === 'async') {
|
|
518
|
+
return ` ${name}(command: ${commandType}): Promise<${responseType}> {
|
|
519
|
+
const msgpackCommand = from${commandType}(command);
|
|
520
|
+
return msgpackCall(this.wasm, 'bbapi', [["${capitalize(name)}", msgpackCommand]]).then(([variantName, result]: [string, any]) => {
|
|
521
|
+
if (variantName !== '${responseType}') {
|
|
522
|
+
throw new Error(\`Expected variant name '${responseType}' but got '\${variantName}'\`);
|
|
523
|
+
}
|
|
524
|
+
return to${responseType}(result);
|
|
525
|
+
});
|
|
526
|
+
}`;
|
|
527
|
+
}
|
|
528
|
+
// For sync mode, keep the synchronous behavior
|
|
529
|
+
return ` ${name}(command: ${commandType}): ${responseType} {
|
|
530
|
+
const msgpackCommand = from${commandType}(command);
|
|
531
|
+
const [variantName, result] = msgpackCall(this.wasm, 'bbapi', [["${capitalize(name)}", msgpackCommand]]);
|
|
532
|
+
if (variantName !== '${responseType}') {
|
|
533
|
+
throw new Error(\`Expected variant name '${responseType}' but got '\${variantName}'\`);
|
|
534
|
+
}
|
|
535
|
+
return to${responseType}(result);
|
|
536
|
+
}`;
|
|
537
|
+
}
|
|
538
|
+
generateNativeApiClass(methods) {
|
|
539
|
+
return `interface NativeApiRequest {
|
|
540
|
+
resolve: (value: any) => void;
|
|
541
|
+
reject: (error: any) => void;
|
|
542
|
+
}
|
|
543
|
+
|
|
544
|
+
class StreamBuffer {
|
|
545
|
+
private buffer = Buffer.alloc(0);
|
|
546
|
+
private expectedLength: number | null = null;
|
|
547
|
+
|
|
548
|
+
addData(data: Buffer): Buffer[] {
|
|
549
|
+
// Create buffer to grow as needed
|
|
550
|
+
const newBuffer = Buffer.allocUnsafe(this.buffer.length + data.length);
|
|
551
|
+
this.buffer.copy(newBuffer, 0);
|
|
552
|
+
data.copy(newBuffer, this.buffer.length);
|
|
553
|
+
this.buffer = newBuffer;
|
|
554
|
+
|
|
555
|
+
const messages: Buffer[] = [];
|
|
556
|
+
|
|
557
|
+
while (true) {
|
|
558
|
+
if (this.expectedLength === null) {
|
|
559
|
+
if (this.buffer.length < 4) break;
|
|
560
|
+
this.expectedLength = this.buffer.readUInt32LE(0);
|
|
561
|
+
this.buffer = this.buffer.subarray(4);
|
|
562
|
+
}
|
|
563
|
+
|
|
564
|
+
if (this.buffer.length < this.expectedLength) break;
|
|
565
|
+
|
|
566
|
+
// Extract complete message
|
|
567
|
+
const messageBuffer = this.buffer.subarray(0, this.expectedLength);
|
|
568
|
+
messages.push(messageBuffer);
|
|
569
|
+
this.buffer = this.buffer.subarray(this.expectedLength);
|
|
570
|
+
this.expectedLength = null;
|
|
571
|
+
}
|
|
572
|
+
|
|
573
|
+
return messages;
|
|
574
|
+
}
|
|
575
|
+
}
|
|
576
|
+
|
|
577
|
+
export class NativeApi implements BbApiBase {
|
|
578
|
+
private decoder = new Decoder({ useRecords: false });
|
|
579
|
+
private encoder = new Encoder({ useRecords: false });
|
|
580
|
+
private pendingRequests: NativeApiRequest[] = [];
|
|
581
|
+
|
|
582
|
+
private constructor(private proc: ChildProcess) {}
|
|
583
|
+
|
|
584
|
+
static async new(bbPath = 'bb', logger = console.log): Promise<NativeApi> {
|
|
585
|
+
const proc = spawn(bbPath, ['msgpack', 'run'], {
|
|
586
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
587
|
+
});
|
|
588
|
+
|
|
589
|
+
if (!proc.stdout || !proc.stdin) {
|
|
590
|
+
throw new Error('Failed to initialize bb process');
|
|
591
|
+
}
|
|
592
|
+
|
|
593
|
+
const api = new NativeApi(proc);
|
|
594
|
+
const streamBuffer = new StreamBuffer();
|
|
595
|
+
|
|
596
|
+
proc.stdout.on('data', (data: Buffer) => {
|
|
597
|
+
const messages = streamBuffer.addData(data);
|
|
598
|
+
|
|
599
|
+
for (const messageBuffer of messages) {
|
|
600
|
+
const pendingRequest = api.pendingRequests.shift();
|
|
601
|
+
if (!pendingRequest) {
|
|
602
|
+
throw new Error('Received response without a pending request');
|
|
603
|
+
}
|
|
604
|
+
|
|
605
|
+
try {
|
|
606
|
+
const decoded = api.decoder.decode(messageBuffer);
|
|
607
|
+
if (!Array.isArray(decoded) || decoded.length !== 2) {
|
|
608
|
+
throw new Error(\`Invalid response format: \${JSON.stringify(decoded)}\`);
|
|
609
|
+
}
|
|
610
|
+
const [variantName, result] = decoded;
|
|
611
|
+
pendingRequest.resolve([variantName, result]);
|
|
612
|
+
} catch (error) {
|
|
613
|
+
pendingRequest.reject(error);
|
|
614
|
+
break;
|
|
615
|
+
}
|
|
616
|
+
}
|
|
617
|
+
});
|
|
618
|
+
|
|
619
|
+
proc.stderr.on('data', (data: Buffer) => {
|
|
620
|
+
logger(data.toString().trim());
|
|
621
|
+
});
|
|
622
|
+
|
|
623
|
+
proc.on('error', err => {
|
|
624
|
+
throw new Error(err.message);
|
|
625
|
+
});
|
|
626
|
+
return api;
|
|
627
|
+
}
|
|
628
|
+
|
|
629
|
+
private sendCommand(command: any): Promise<any> {
|
|
630
|
+
return new Promise((resolve, reject) => {
|
|
631
|
+
this.pendingRequests.push({ resolve, reject });
|
|
632
|
+
const encoded = this.encoder.encode(command);
|
|
633
|
+
|
|
634
|
+
// Write length prefix (4 bytes, little-endian)
|
|
635
|
+
const lengthBuffer = Buffer.allocUnsafe(4);
|
|
636
|
+
lengthBuffer.writeUInt32LE(encoded.length, 0);
|
|
637
|
+
|
|
638
|
+
// Write length prefix followed by the encoded data
|
|
639
|
+
this.proc.stdin!.write(lengthBuffer);
|
|
640
|
+
this.proc.stdin!.write(encoded);
|
|
641
|
+
});
|
|
642
|
+
}
|
|
643
|
+
|
|
644
|
+
async close(): Promise<void> {
|
|
645
|
+
this.proc.kill();
|
|
646
|
+
}
|
|
647
|
+
|
|
648
|
+
destroy(): Promise<void> {
|
|
649
|
+
return this.close();
|
|
650
|
+
}
|
|
651
|
+
|
|
652
|
+
${methods}
|
|
653
|
+
}`;
|
|
654
|
+
}
|
|
655
|
+
}
|
|
656
|
+
exports.SchemaCompiler = SchemaCompiler;
|
|
657
|
+
// Factory methods for creating configured compilers
|
|
658
|
+
function createSharedTypesCompiler() {
|
|
659
|
+
return new SchemaCompiler({
|
|
660
|
+
mode: 'types',
|
|
661
|
+
imports: [],
|
|
662
|
+
});
|
|
663
|
+
}
|
|
664
|
+
function createSyncApiCompiler() {
|
|
665
|
+
return new SchemaCompiler({
|
|
666
|
+
mode: 'sync',
|
|
667
|
+
imports: [
|
|
668
|
+
`import { BarretenbergWasmMain } from "../../barretenberg_wasm/barretenberg_wasm_main/index.js";`,
|
|
669
|
+
`import { Decoder, Encoder } from 'msgpackr';`,
|
|
670
|
+
],
|
|
671
|
+
});
|
|
672
|
+
}
|
|
673
|
+
function createAsyncApiCompiler() {
|
|
674
|
+
return new SchemaCompiler({
|
|
675
|
+
mode: 'async',
|
|
676
|
+
imports: [
|
|
677
|
+
`import { BarretenbergWasmMainWorker } from "../../barretenberg_wasm/barretenberg_wasm_main/index.js";`,
|
|
678
|
+
`import { Decoder, Encoder } from 'msgpackr';`
|
|
679
|
+
],
|
|
680
|
+
});
|
|
681
|
+
}
|
|
682
|
+
function createNativeApiCompiler() {
|
|
683
|
+
return new SchemaCompiler({
|
|
684
|
+
mode: 'native',
|
|
685
|
+
imports: [
|
|
686
|
+
`import { spawn, ChildProcess } from 'child_process';`,
|
|
687
|
+
`import { Decoder, Encoder } from 'msgpackr';`
|
|
688
|
+
],
|
|
689
|
+
});
|
|
690
|
+
}
|
|
691
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2NoZW1hX2NvbXBpbGVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2NiaW5kL3NjaGVtYV9jb21waWxlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7O0dBRUc7OztBQTB4QkgsOERBS0M7QUFFRCxzREFRQztBQUVELHdEQVFDO0FBRUQsMERBUUM7QUF0eEJELG1CQUFtQjtBQUNuQixTQUFTLFVBQVUsQ0FBQyxDQUFTO0lBQzNCLE9BQU8sQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQyxXQUFXLEVBQUUsR0FBRyxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDO0FBQ3BELENBQUM7QUFFRCxTQUFTLFNBQVMsQ0FBQyxDQUFTO0lBQzFCLE9BQU8sQ0FBQztTQUNMLEtBQUssQ0FBQyxHQUFHLENBQUM7U0FDVixHQUFHLENBQUMsQ0FBQyxJQUFJLEVBQUUsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFDLEtBQUssS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUMsV0FBVyxFQUFFLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUM7U0FDekcsSUFBSSxDQUFDLEVBQUUsQ0FBQyxDQUFDO0FBQ2QsQ0FBQztBQUVELFNBQVMsVUFBVSxDQUFDLENBQVM7SUFDM0IsT0FBTyxDQUFDLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxVQUFVLENBQUMsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLENBQUM7QUFDL0MsQ0FBQztBQUVELE1BQWEsY0FBYztJQU16QixZQUFvQixNQUFzQjtRQUF0QixXQUFNLEdBQU4sTUFBTSxDQUFnQjtRQUxsQyxjQUFTLEdBQUcsSUFBSSxHQUFHLEVBQW9CLENBQUM7UUFDeEMscUJBQWdCLEdB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
|