@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.
Files changed (122) hide show
  1. package/dest/browser/barretenberg/backend.d.ts.map +1 -1
  2. package/dest/browser/barretenberg/backend.js +56 -45
  3. package/dest/browser/barretenberg/index.d.ts +10 -0
  4. package/dest/browser/barretenberg/index.d.ts.map +1 -1
  5. package/dest/browser/barretenberg/index.js +30 -2
  6. package/dest/browser/barretenberg/verifier.js +2 -2
  7. package/dest/browser/barretenberg_api/index.d.ts +2 -2
  8. package/dest/browser/barretenberg_api/index.d.ts.map +1 -1
  9. package/dest/browser/barretenberg_api/index.js +5 -5
  10. package/dest/browser/barretenberg_wasm/barretenberg_wasm_base/index.d.ts +1 -0
  11. package/dest/browser/barretenberg_wasm/barretenberg_wasm_base/index.d.ts.map +1 -1
  12. package/dest/browser/barretenberg_wasm/barretenberg_wasm_base/index.js +5 -4
  13. package/dest/browser/barretenberg_wasm/barretenberg_wasm_main/index.d.ts +2 -0
  14. package/dest/browser/barretenberg_wasm/barretenberg_wasm_main/index.d.ts.map +1 -1
  15. package/dest/browser/barretenberg_wasm/barretenberg_wasm_main/index.js +17 -1
  16. package/dest/browser/barretenberg_wasm/barretenberg_wasm_thread/index.d.ts +1 -0
  17. package/dest/browser/barretenberg_wasm/barretenberg_wasm_thread/index.d.ts.map +1 -1
  18. package/dest/browser/barretenberg_wasm/fetch_code/browser/barretenberg-threads.js +1 -1
  19. package/dest/browser/barretenberg_wasm/fetch_code/browser/barretenberg.js +1 -1
  20. package/dest/browser/cbind/generate.d.ts +5 -0
  21. package/dest/browser/cbind/generate.d.ts.map +1 -0
  22. package/dest/browser/cbind/generate.js +64 -0
  23. package/dest/browser/cbind/generated/api_types.d.ts +801 -0
  24. package/dest/browser/cbind/generated/api_types.d.ts.map +1 -0
  25. package/dest/browser/cbind/generated/api_types.js +1099 -0
  26. package/dest/browser/cbind/generated/async.d.ts +27 -0
  27. package/dest/browser/cbind/generated/async.d.ts.map +1 -0
  28. package/dest/browser/cbind/generated/async.js +184 -0
  29. package/dest/browser/cbind/generated/native.d.ts +35 -0
  30. package/dest/browser/cbind/generated/native.d.ts.map +1 -0
  31. package/dest/browser/cbind/generated/native.js +270 -0
  32. package/dest/browser/cbind/generated/sync.d.ts +27 -0
  33. package/dest/browser/cbind/generated/sync.d.ts.map +1 -0
  34. package/dest/browser/cbind/generated/sync.js +165 -0
  35. package/dest/browser/cbind/schema_compiler.d.ts +70 -0
  36. package/dest/browser/cbind/schema_compiler.d.ts.map +1 -0
  37. package/dest/browser/cbind/schema_compiler.js +683 -0
  38. package/dest/node/barretenberg/backend.d.ts.map +1 -1
  39. package/dest/node/barretenberg/backend.js +56 -45
  40. package/dest/node/barretenberg/index.d.ts +10 -0
  41. package/dest/node/barretenberg/index.d.ts.map +1 -1
  42. package/dest/node/barretenberg/index.js +30 -2
  43. package/dest/node/barretenberg/verifier.js +2 -2
  44. package/dest/node/barretenberg_api/index.d.ts +2 -2
  45. package/dest/node/barretenberg_api/index.d.ts.map +1 -1
  46. package/dest/node/barretenberg_api/index.js +5 -5
  47. package/dest/node/barretenberg_wasm/barretenberg-threads.wasm.gz +0 -0
  48. package/dest/node/barretenberg_wasm/barretenberg_wasm_base/index.d.ts +1 -0
  49. package/dest/node/barretenberg_wasm/barretenberg_wasm_base/index.d.ts.map +1 -1
  50. package/dest/node/barretenberg_wasm/barretenberg_wasm_base/index.js +5 -4
  51. package/dest/node/barretenberg_wasm/barretenberg_wasm_main/index.d.ts +2 -0
  52. package/dest/node/barretenberg_wasm/barretenberg_wasm_main/index.d.ts.map +1 -1
  53. package/dest/node/barretenberg_wasm/barretenberg_wasm_main/index.js +17 -1
  54. package/dest/node/barretenberg_wasm/barretenberg_wasm_thread/index.d.ts +1 -0
  55. package/dest/node/barretenberg_wasm/barretenberg_wasm_thread/index.d.ts.map +1 -1
  56. package/dest/node/cbind/generate.d.ts +5 -0
  57. package/dest/node/cbind/generate.d.ts.map +1 -0
  58. package/dest/node/cbind/generate.js +64 -0
  59. package/dest/node/cbind/generated/api_types.d.ts +801 -0
  60. package/dest/node/cbind/generated/api_types.d.ts.map +1 -0
  61. package/dest/node/cbind/generated/api_types.js +1099 -0
  62. package/dest/node/cbind/generated/async.d.ts +27 -0
  63. package/dest/node/cbind/generated/async.d.ts.map +1 -0
  64. package/dest/node/cbind/generated/async.js +184 -0
  65. package/dest/node/cbind/generated/native.d.ts +35 -0
  66. package/dest/node/cbind/generated/native.d.ts.map +1 -0
  67. package/dest/node/cbind/generated/native.js +270 -0
  68. package/dest/node/cbind/generated/sync.d.ts +27 -0
  69. package/dest/node/cbind/generated/sync.d.ts.map +1 -0
  70. package/dest/node/cbind/generated/sync.js +165 -0
  71. package/dest/node/cbind/schema_compiler.d.ts +70 -0
  72. package/dest/node/cbind/schema_compiler.d.ts.map +1 -0
  73. package/dest/node/cbind/schema_compiler.js +683 -0
  74. package/dest/node/main.js +3 -3
  75. package/dest/node-cjs/barretenberg/backend.d.ts.map +1 -1
  76. package/dest/node-cjs/barretenberg/backend.js +58 -47
  77. package/dest/node-cjs/barretenberg/index.d.ts +10 -0
  78. package/dest/node-cjs/barretenberg/index.d.ts.map +1 -1
  79. package/dest/node-cjs/barretenberg/index.js +30 -2
  80. package/dest/node-cjs/barretenberg/verifier.js +2 -2
  81. package/dest/node-cjs/barretenberg_api/index.d.ts +2 -2
  82. package/dest/node-cjs/barretenberg_api/index.d.ts.map +1 -1
  83. package/dest/node-cjs/barretenberg_api/index.js +5 -5
  84. package/dest/node-cjs/barretenberg_wasm/barretenberg-threads.wasm.gz +0 -0
  85. package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_base/index.d.ts +1 -0
  86. package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_base/index.d.ts.map +1 -1
  87. package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_base/index.js +5 -4
  88. package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_main/index.d.ts +2 -0
  89. package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_main/index.d.ts.map +1 -1
  90. package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_main/index.js +17 -1
  91. package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_thread/index.d.ts +1 -0
  92. package/dest/node-cjs/barretenberg_wasm/barretenberg_wasm_thread/index.d.ts.map +1 -1
  93. package/dest/node-cjs/cbind/generate.d.ts +5 -0
  94. package/dest/node-cjs/cbind/generate.d.ts.map +1 -0
  95. package/dest/node-cjs/cbind/generate.js +66 -0
  96. package/dest/node-cjs/cbind/generated/api_types.d.ts +801 -0
  97. package/dest/node-cjs/cbind/generated/api_types.d.ts.map +1 -0
  98. package/dest/node-cjs/cbind/generated/api_types.js +1189 -0
  99. package/dest/node-cjs/cbind/generated/async.d.ts +27 -0
  100. package/dest/node-cjs/cbind/generated/async.d.ts.map +1 -0
  101. package/dest/node-cjs/cbind/generated/async.js +188 -0
  102. package/dest/node-cjs/cbind/generated/native.d.ts +35 -0
  103. package/dest/node-cjs/cbind/generated/native.d.ts.map +1 -0
  104. package/dest/node-cjs/cbind/generated/native.js +274 -0
  105. package/dest/node-cjs/cbind/generated/sync.d.ts +27 -0
  106. package/dest/node-cjs/cbind/generated/sync.d.ts.map +1 -0
  107. package/dest/node-cjs/cbind/generated/sync.js +169 -0
  108. package/dest/node-cjs/cbind/schema_compiler.d.ts +70 -0
  109. package/dest/node-cjs/cbind/schema_compiler.d.ts.map +1 -0
  110. package/dest/node-cjs/cbind/schema_compiler.js +691 -0
  111. package/dest/node-cjs/main.js +3 -3
  112. package/package.json +4 -3
  113. package/src/barretenberg/backend.ts +62 -63
  114. package/src/barretenberg/index.ts +39 -1
  115. package/src/barretenberg/verifier.ts +1 -1
  116. package/src/barretenberg_api/index.ts +4 -4
  117. package/src/barretenberg_wasm/barretenberg_wasm_base/index.ts +5 -3
  118. package/src/barretenberg_wasm/barretenberg_wasm_main/index.ts +23 -0
  119. package/src/cbind/README.md +1 -0
  120. package/src/cbind/generate.ts +89 -0
  121. package/src/cbind/schema_compiler.ts +832 -0
  122. 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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