@cmdoss/memwal-sdk 0.6.2 → 0.7.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 (119) hide show
  1. package/ARCHITECTURE.md +547 -547
  2. package/BENCHMARKS.md +238 -238
  3. package/README.md +181 -181
  4. package/dist/ai-sdk/tools.d.ts +2 -2
  5. package/dist/ai-sdk/tools.js +2 -2
  6. package/dist/client/PersonalDataWallet.d.ts.map +1 -1
  7. package/dist/client/SimplePDWClient.d.ts +1 -1
  8. package/dist/client/SimplePDWClient.d.ts.map +1 -1
  9. package/dist/client/SimplePDWClient.js +16 -7
  10. package/dist/client/SimplePDWClient.js.map +1 -1
  11. package/dist/client/namespaces/EmbeddingsNamespace.d.ts +1 -1
  12. package/dist/client/namespaces/EmbeddingsNamespace.js +1 -1
  13. package/dist/client/namespaces/MemoryNamespace.d.ts +27 -0
  14. package/dist/client/namespaces/MemoryNamespace.d.ts.map +1 -1
  15. package/dist/client/namespaces/MemoryNamespace.js +104 -0
  16. package/dist/client/namespaces/MemoryNamespace.js.map +1 -1
  17. package/dist/client/namespaces/consolidated/AINamespace.d.ts +2 -2
  18. package/dist/client/namespaces/consolidated/AINamespace.js +2 -2
  19. package/dist/client/namespaces/consolidated/BlockchainNamespace.d.ts.map +1 -1
  20. package/dist/client/namespaces/consolidated/BlockchainNamespace.js +22 -2
  21. package/dist/client/namespaces/consolidated/BlockchainNamespace.js.map +1 -1
  22. package/dist/graph/GraphService.js +1 -1
  23. package/dist/graph/GraphService.js.map +1 -1
  24. package/dist/index.d.ts +3 -1
  25. package/dist/index.d.ts.map +1 -1
  26. package/dist/index.js +3 -1
  27. package/dist/index.js.map +1 -1
  28. package/dist/retrieval/MemoryRetrievalService.d.ts +31 -0
  29. package/dist/retrieval/MemoryRetrievalService.d.ts.map +1 -1
  30. package/dist/retrieval/MemoryRetrievalService.js +44 -4
  31. package/dist/retrieval/MemoryRetrievalService.js.map +1 -1
  32. package/dist/services/EmbeddingService.d.ts +28 -1
  33. package/dist/services/EmbeddingService.d.ts.map +1 -1
  34. package/dist/services/EmbeddingService.js +54 -0
  35. package/dist/services/EmbeddingService.js.map +1 -1
  36. package/dist/services/IndexManager.d.ts +5 -1
  37. package/dist/services/IndexManager.d.ts.map +1 -1
  38. package/dist/services/IndexManager.js +17 -40
  39. package/dist/services/IndexManager.js.map +1 -1
  40. package/dist/services/QueryService.js +1 -1
  41. package/dist/services/QueryService.js.map +1 -1
  42. package/dist/services/StorageService.d.ts +10 -0
  43. package/dist/services/StorageService.d.ts.map +1 -1
  44. package/dist/services/StorageService.js +13 -0
  45. package/dist/services/StorageService.js.map +1 -1
  46. package/dist/services/storage/QuiltBatchManager.d.ts +101 -1
  47. package/dist/services/storage/QuiltBatchManager.d.ts.map +1 -1
  48. package/dist/services/storage/QuiltBatchManager.js +410 -20
  49. package/dist/services/storage/QuiltBatchManager.js.map +1 -1
  50. package/dist/services/storage/index.d.ts +1 -1
  51. package/dist/services/storage/index.d.ts.map +1 -1
  52. package/dist/services/storage/index.js.map +1 -1
  53. package/dist/utils/LRUCache.d.ts +106 -0
  54. package/dist/utils/LRUCache.d.ts.map +1 -0
  55. package/dist/utils/LRUCache.js +281 -0
  56. package/dist/utils/LRUCache.js.map +1 -0
  57. package/dist/utils/index.d.ts +1 -0
  58. package/dist/utils/index.d.ts.map +1 -1
  59. package/dist/utils/index.js +2 -0
  60. package/dist/utils/index.js.map +1 -1
  61. package/dist/utils/memoryIndexOnChain.d.ts +212 -0
  62. package/dist/utils/memoryIndexOnChain.d.ts.map +1 -0
  63. package/dist/utils/memoryIndexOnChain.js +312 -0
  64. package/dist/utils/memoryIndexOnChain.js.map +1 -0
  65. package/dist/utils/rebuildIndexNode.d.ts +29 -0
  66. package/dist/utils/rebuildIndexNode.d.ts.map +1 -1
  67. package/dist/utils/rebuildIndexNode.js +366 -98
  68. package/dist/utils/rebuildIndexNode.js.map +1 -1
  69. package/dist/vector/HnswWasmService.d.ts +20 -5
  70. package/dist/vector/HnswWasmService.d.ts.map +1 -1
  71. package/dist/vector/HnswWasmService.js +73 -40
  72. package/dist/vector/HnswWasmService.js.map +1 -1
  73. package/dist/vector/IHnswService.d.ts +10 -1
  74. package/dist/vector/IHnswService.d.ts.map +1 -1
  75. package/dist/vector/IHnswService.js.map +1 -1
  76. package/dist/vector/NodeHnswService.d.ts +16 -0
  77. package/dist/vector/NodeHnswService.d.ts.map +1 -1
  78. package/dist/vector/NodeHnswService.js +84 -5
  79. package/dist/vector/NodeHnswService.js.map +1 -1
  80. package/dist/vector/createHnswService.d.ts +1 -1
  81. package/dist/vector/createHnswService.js +1 -1
  82. package/dist/vector/index.d.ts +1 -1
  83. package/dist/vector/index.js +1 -1
  84. package/package.json +157 -157
  85. package/src/access/index.ts +8 -8
  86. package/src/aggregation/index.ts +8 -8
  87. package/src/ai-sdk/tools.ts +2 -2
  88. package/src/client/SimplePDWClient.ts +23 -8
  89. package/src/client/namespaces/EmbeddingsNamespace.ts +1 -1
  90. package/src/client/namespaces/MemoryNamespace.ts +137 -0
  91. package/src/client/namespaces/consolidated/AINamespace.ts +2 -2
  92. package/src/client/namespaces/consolidated/BlockchainNamespace.ts +20 -2
  93. package/src/client/signers/DappKitSigner.ts +207 -207
  94. package/src/core/types/index.ts +1 -1
  95. package/src/generated/pdw/deps/sui/object.ts +12 -12
  96. package/src/generated/pdw/deps/sui/vec_map.ts +32 -32
  97. package/src/generated/pdw/memory.ts +1087 -1087
  98. package/src/generated/pdw/wallet.ts +123 -123
  99. package/src/generated/utils/index.ts +159 -159
  100. package/src/graph/GraphService.ts +1 -1
  101. package/src/index.ts +25 -1
  102. package/src/permissions/index.ts +9 -9
  103. package/src/retrieval/MemoryRetrievalService.ts +78 -4
  104. package/src/services/EmbeddingService.ts +66 -1
  105. package/src/services/IndexManager.ts +18 -45
  106. package/src/services/QueryService.ts +1 -1
  107. package/src/services/StorageService.ts +15 -0
  108. package/src/services/storage/QuiltBatchManager.ts +492 -22
  109. package/src/services/storage/index.ts +6 -1
  110. package/src/utils/LRUCache.ts +378 -0
  111. package/src/utils/index.ts +8 -0
  112. package/src/utils/memoryIndexOnChain.ts +507 -0
  113. package/src/utils/rebuildIndexNode.ts +453 -106
  114. package/src/vector/HnswWasmService.ts +95 -43
  115. package/src/vector/IHnswService.ts +10 -1
  116. package/src/vector/NodeHnswService.ts +103 -5
  117. package/src/vector/createHnswService.ts +1 -1
  118. package/src/vector/index.ts +1 -1
  119. package/src/wallet/index.ts +17 -17
@@ -0,0 +1,507 @@
1
+ /**
2
+ * Memory Index On-Chain Utilities
3
+ *
4
+ * Utilities for managing MemoryIndex object on Sui blockchain:
5
+ * - Reading current MemoryIndex state from chain
6
+ * - Updating blob IDs after Walrus sync
7
+ * - Creating new MemoryIndex if doesn't exist
8
+ *
9
+ * Environment Variables:
10
+ * - MEMORY_INDEX_ID: Sui object ID of the MemoryIndex
11
+ * - INDEX_BLOB_ID: Current Walrus blob ID for HNSW index (updated after sync)
12
+ * - GRAPH_BLOB_ID: Current Walrus blob ID for knowledge graph
13
+ */
14
+
15
+ import { SuiClient } from '@mysten/sui/client';
16
+ import { Transaction } from '@mysten/sui/transactions';
17
+ import type { Signer } from '@mysten/sui/cryptography';
18
+
19
+ /**
20
+ * On-chain MemoryIndex object structure
21
+ */
22
+ export interface OnChainMemoryIndex {
23
+ /** Sui object ID */
24
+ objectId: string;
25
+ /** Owner address */
26
+ owner: string;
27
+ /** Version number for optimistic locking */
28
+ version: number;
29
+ /** Walrus blob ID for HNSW index */
30
+ indexBlobId: string;
31
+ /** Walrus blob ID for knowledge graph */
32
+ graphBlobId: string;
33
+ }
34
+
35
+ /**
36
+ * Options for getting MemoryIndex from chain
37
+ */
38
+ export interface GetMemoryIndexOptions {
39
+ /** Sui client instance */
40
+ client: SuiClient;
41
+ /** MemoryIndex object ID (from MEMORY_INDEX_ID env var) */
42
+ memoryIndexId: string;
43
+ }
44
+
45
+ /**
46
+ * Options for updating MemoryIndex on-chain
47
+ */
48
+ export interface UpdateMemoryIndexOnChainOptions {
49
+ /** Sui client instance */
50
+ client: SuiClient;
51
+ /** Signer for transaction */
52
+ signer: Signer;
53
+ /** Package ID of the PDW contract */
54
+ packageId: string;
55
+ /** MemoryIndex object ID */
56
+ memoryIndexId: string;
57
+ /** Current version for optimistic locking */
58
+ expectedVersion: number;
59
+ /** New index blob ID from Walrus */
60
+ newIndexBlobId: string;
61
+ /** New graph blob ID from Walrus */
62
+ newGraphBlobId: string;
63
+ /** Optional gas budget */
64
+ gasBudget?: number;
65
+ }
66
+
67
+ /**
68
+ * Options for creating a new MemoryIndex
69
+ */
70
+ export interface CreateMemoryIndexOnChainOptions {
71
+ /** Sui client instance */
72
+ client: SuiClient;
73
+ /** Signer for transaction */
74
+ signer: Signer;
75
+ /** Package ID of the PDW contract */
76
+ packageId: string;
77
+ /** Initial index blob ID (can be placeholder) */
78
+ indexBlobId: string;
79
+ /** Initial graph blob ID (can be placeholder) */
80
+ graphBlobId: string;
81
+ /** Optional gas budget */
82
+ gasBudget?: number;
83
+ }
84
+
85
+ /**
86
+ * Result of updating MemoryIndex
87
+ */
88
+ export interface UpdateMemoryIndexResult {
89
+ success: boolean;
90
+ /** Transaction digest */
91
+ digest?: string;
92
+ /** New version number after update */
93
+ newVersion?: number;
94
+ /** Error message if failed */
95
+ error?: string;
96
+ }
97
+
98
+ /**
99
+ * Result of creating MemoryIndex
100
+ */
101
+ export interface CreateMemoryIndexResult {
102
+ success: boolean;
103
+ /** Created MemoryIndex object ID */
104
+ memoryIndexId?: string;
105
+ /** Transaction digest */
106
+ digest?: string;
107
+ /** Error message if failed */
108
+ error?: string;
109
+ }
110
+
111
+ /**
112
+ * Read MemoryIndex object from Sui blockchain
113
+ *
114
+ * @param options - Options containing client and memoryIndexId
115
+ * @returns The MemoryIndex object data or null if not found
116
+ *
117
+ * @example
118
+ * ```typescript
119
+ * const memoryIndex = await getMemoryIndex({
120
+ * client: suiClient,
121
+ * memoryIndexId: process.env.MEMORY_INDEX_ID!
122
+ * });
123
+ *
124
+ * if (memoryIndex) {
125
+ * console.log(`Version: ${memoryIndex.version}`);
126
+ * console.log(`Index Blob: ${memoryIndex.indexBlobId}`);
127
+ * }
128
+ * ```
129
+ */
130
+ export async function getMemoryIndex(
131
+ options: GetMemoryIndexOptions
132
+ ): Promise<OnChainMemoryIndex | null> {
133
+ const { client, memoryIndexId } = options;
134
+
135
+ if (!memoryIndexId) {
136
+ console.warn('⚠️ MEMORY_INDEX_ID not provided');
137
+ return null;
138
+ }
139
+
140
+ try {
141
+ const object = await client.getObject({
142
+ id: memoryIndexId,
143
+ options: {
144
+ showContent: true,
145
+ showOwner: true,
146
+ }
147
+ });
148
+
149
+ if (!object.data?.content || object.data.content.dataType !== 'moveObject') {
150
+ console.warn(`⚠️ MemoryIndex object ${memoryIndexId} not found or not a Move object`);
151
+ return null;
152
+ }
153
+
154
+ const fields = (object.data.content as any).fields;
155
+ const owner = object.data.owner;
156
+
157
+ // Extract owner address
158
+ let ownerAddress = '';
159
+ if (owner && typeof owner === 'object' && 'AddressOwner' in owner) {
160
+ ownerAddress = (owner as { AddressOwner: string }).AddressOwner;
161
+ }
162
+
163
+ return {
164
+ objectId: memoryIndexId,
165
+ owner: ownerAddress,
166
+ version: Number(fields.version || 1),
167
+ indexBlobId: fields.index_blob_id || '',
168
+ graphBlobId: fields.graph_blob_id || '',
169
+ };
170
+ } catch (error) {
171
+ console.error(`❌ Failed to read MemoryIndex ${memoryIndexId}:`, error);
172
+ return null;
173
+ }
174
+ }
175
+
176
+ /**
177
+ * Update MemoryIndex on-chain with new blob IDs
178
+ *
179
+ * Call this after uploading index/graph to Walrus to update the on-chain reference.
180
+ * Uses optimistic locking (expectedVersion must match current version).
181
+ *
182
+ * @param options - Update options
183
+ * @returns Result with success status and new version
184
+ *
185
+ * @example
186
+ * ```typescript
187
+ * // First, read current state
188
+ * const current = await getMemoryIndex({ client, memoryIndexId });
189
+ *
190
+ * // Then upload to Walrus and get new blob ID
191
+ * const newBlobId = await uploadIndexToWalrus();
192
+ *
193
+ * // Finally, update on-chain
194
+ * const result = await updateMemoryIndexOnChain({
195
+ * client,
196
+ * signer: keypair,
197
+ * packageId: process.env.PACKAGE_ID!,
198
+ * memoryIndexId: process.env.MEMORY_INDEX_ID!,
199
+ * expectedVersion: current.version,
200
+ * newIndexBlobId: newBlobId,
201
+ * newGraphBlobId: current.graphBlobId, // Keep existing if not changed
202
+ * });
203
+ * ```
204
+ */
205
+ export async function updateMemoryIndexOnChain(
206
+ options: UpdateMemoryIndexOnChainOptions
207
+ ): Promise<UpdateMemoryIndexResult> {
208
+ const {
209
+ client,
210
+ signer,
211
+ packageId,
212
+ memoryIndexId,
213
+ expectedVersion,
214
+ newIndexBlobId,
215
+ newGraphBlobId,
216
+ gasBudget = 10_000_000,
217
+ } = options;
218
+
219
+ try {
220
+ console.log(`\n📝 Updating MemoryIndex on-chain...`);
221
+ console.log(` Object ID: ${memoryIndexId}`);
222
+ console.log(` Expected version: ${expectedVersion}`);
223
+ console.log(` New index blob: ${newIndexBlobId}`);
224
+ console.log(` New graph blob: ${newGraphBlobId}`);
225
+
226
+ const tx = new Transaction();
227
+ tx.setGasBudget(gasBudget);
228
+
229
+ // Convert strings to vector<u8> for Move
230
+ const newIndexBlobIdBytes = Array.from(new TextEncoder().encode(newIndexBlobId));
231
+ const newGraphBlobIdBytes = Array.from(new TextEncoder().encode(newGraphBlobId));
232
+
233
+ tx.moveCall({
234
+ target: `${packageId}::memory::update_memory_index`,
235
+ arguments: [
236
+ tx.object(memoryIndexId), // &mut MemoryIndex
237
+ tx.pure.u64(expectedVersion), // expected_version: u64
238
+ tx.pure.vector('u8', newIndexBlobIdBytes), // new_index_blob_id: vector<u8>
239
+ tx.pure.vector('u8', newGraphBlobIdBytes), // new_graph_blob_id: vector<u8>
240
+ ]
241
+ });
242
+
243
+ const result = await client.signAndExecuteTransaction({
244
+ transaction: tx,
245
+ signer,
246
+ options: {
247
+ showEffects: true,
248
+ showObjectChanges: true,
249
+ },
250
+ });
251
+
252
+ if (result.effects?.status?.status !== 'success') {
253
+ const error = result.effects?.status?.error || 'Unknown error';
254
+ console.error(`❌ Transaction failed: ${error}`);
255
+ return { success: false, error };
256
+ }
257
+
258
+ const newVersion = expectedVersion + 1;
259
+ console.log(`✅ MemoryIndex updated successfully`);
260
+ console.log(` Transaction: ${result.digest}`);
261
+ console.log(` New version: ${newVersion}`);
262
+
263
+ return {
264
+ success: true,
265
+ digest: result.digest,
266
+ newVersion,
267
+ };
268
+ } catch (error) {
269
+ const errorMsg = error instanceof Error ? error.message : String(error);
270
+ console.error(`❌ Failed to update MemoryIndex:`, errorMsg);
271
+ return { success: false, error: errorMsg };
272
+ }
273
+ }
274
+
275
+ /**
276
+ * Create a new MemoryIndex on-chain
277
+ *
278
+ * Use this when setting up a new wallet or if MEMORY_INDEX_ID doesn't exist.
279
+ *
280
+ * @param options - Creation options
281
+ * @returns Result with created object ID
282
+ *
283
+ * @example
284
+ * ```typescript
285
+ * const result = await createMemoryIndexOnChain({
286
+ * client,
287
+ * signer: keypair,
288
+ * packageId: process.env.PACKAGE_ID!,
289
+ * indexBlobId: 'placeholder-index',
290
+ * graphBlobId: 'placeholder-graph',
291
+ * });
292
+ *
293
+ * if (result.success) {
294
+ * console.log(`Created MemoryIndex: ${result.memoryIndexId}`);
295
+ * // Save to .env: MEMORY_INDEX_ID=${result.memoryIndexId}
296
+ * }
297
+ * ```
298
+ */
299
+ export async function createMemoryIndexOnChain(
300
+ options: CreateMemoryIndexOnChainOptions
301
+ ): Promise<CreateMemoryIndexResult> {
302
+ const {
303
+ client,
304
+ signer,
305
+ packageId,
306
+ indexBlobId,
307
+ graphBlobId,
308
+ gasBudget = 10_000_000,
309
+ } = options;
310
+
311
+ try {
312
+ console.log(`\n📝 Creating new MemoryIndex on-chain...`);
313
+ console.log(` Package ID: ${packageId}`);
314
+ console.log(` Index blob: ${indexBlobId}`);
315
+ console.log(` Graph blob: ${graphBlobId}`);
316
+
317
+ const tx = new Transaction();
318
+ tx.setGasBudget(gasBudget);
319
+
320
+ // Convert strings to vector<u8> for Move
321
+ const indexBlobIdBytes = Array.from(new TextEncoder().encode(indexBlobId));
322
+ const graphBlobIdBytes = Array.from(new TextEncoder().encode(graphBlobId));
323
+
324
+ tx.moveCall({
325
+ target: `${packageId}::memory::create_memory_index`,
326
+ arguments: [
327
+ tx.pure.vector('u8', indexBlobIdBytes),
328
+ tx.pure.vector('u8', graphBlobIdBytes),
329
+ ]
330
+ });
331
+
332
+ const result = await client.signAndExecuteTransaction({
333
+ transaction: tx,
334
+ signer,
335
+ options: {
336
+ showEffects: true,
337
+ showObjectChanges: true,
338
+ },
339
+ });
340
+
341
+ if (result.effects?.status?.status !== 'success') {
342
+ const error = result.effects?.status?.error || 'Unknown error';
343
+ console.error(`❌ Transaction failed: ${error}`);
344
+ return { success: false, error };
345
+ }
346
+
347
+ // Find the created MemoryIndex object
348
+ const createdObject = result.objectChanges?.find(
349
+ (change) => change.type === 'created' && change.objectType?.includes('::memory::MemoryIndex')
350
+ );
351
+
352
+ if (!createdObject || createdObject.type !== 'created') {
353
+ return { success: false, error: 'MemoryIndex object not found in transaction result' };
354
+ }
355
+
356
+ const memoryIndexId = createdObject.objectId;
357
+ console.log(`✅ MemoryIndex created successfully`);
358
+ console.log(` Object ID: ${memoryIndexId}`);
359
+ console.log(` Transaction: ${result.digest}`);
360
+ console.log(`\n Add to .env:`);
361
+ console.log(` MEMORY_INDEX_ID=${memoryIndexId}`);
362
+
363
+ return {
364
+ success: true,
365
+ memoryIndexId,
366
+ digest: result.digest,
367
+ };
368
+ } catch (error) {
369
+ const errorMsg = error instanceof Error ? error.message : String(error);
370
+ console.error(`❌ Failed to create MemoryIndex:`, errorMsg);
371
+ return { success: false, error: errorMsg };
372
+ }
373
+ }
374
+
375
+ /**
376
+ * Sync index to Walrus and update on-chain MemoryIndex
377
+ *
378
+ * Convenience function that combines:
379
+ * 1. Reading current MemoryIndex state
380
+ * 2. Uploading index to Walrus
381
+ * 3. Updating on-chain blob ID
382
+ *
383
+ * @param options - Combined sync options
384
+ * @returns Result with new blob ID and transaction digest
385
+ */
386
+ export interface SyncAndUpdateOptions {
387
+ client: SuiClient;
388
+ signer: Signer;
389
+ packageId: string;
390
+ memoryIndexId: string;
391
+ /** Function to upload index to Walrus, returns new blob ID */
392
+ uploadToWalrus: () => Promise<string>;
393
+ /** Optional: also update graph blob ID */
394
+ newGraphBlobId?: string;
395
+ }
396
+
397
+ export interface SyncAndUpdateResult {
398
+ success: boolean;
399
+ newIndexBlobId?: string;
400
+ newGraphBlobId?: string;
401
+ digest?: string;
402
+ newVersion?: number;
403
+ error?: string;
404
+ }
405
+
406
+ export async function syncIndexAndUpdateOnChain(
407
+ options: SyncAndUpdateOptions
408
+ ): Promise<SyncAndUpdateResult> {
409
+ const {
410
+ client,
411
+ signer,
412
+ packageId,
413
+ memoryIndexId,
414
+ uploadToWalrus,
415
+ newGraphBlobId,
416
+ } = options;
417
+
418
+ try {
419
+ // Step 1: Read current state
420
+ console.log(`\n🔍 Reading current MemoryIndex state...`);
421
+ const current = await getMemoryIndex({ client, memoryIndexId });
422
+
423
+ if (!current) {
424
+ return { success: false, error: 'MemoryIndex not found on-chain' };
425
+ }
426
+
427
+ console.log(` Current version: ${current.version}`);
428
+ console.log(` Current index blob: ${current.indexBlobId}`);
429
+
430
+ // Step 2: Upload to Walrus
431
+ console.log(`\n☁️ Uploading index to Walrus...`);
432
+ const newIndexBlobId = await uploadToWalrus();
433
+ console.log(` New index blob: ${newIndexBlobId}`);
434
+
435
+ // Step 3: Update on-chain
436
+ const graphBlobId = newGraphBlobId || current.graphBlobId;
437
+ const updateResult = await updateMemoryIndexOnChain({
438
+ client,
439
+ signer,
440
+ packageId,
441
+ memoryIndexId,
442
+ expectedVersion: current.version,
443
+ newIndexBlobId,
444
+ newGraphBlobId: graphBlobId,
445
+ });
446
+
447
+ if (!updateResult.success) {
448
+ return { success: false, error: updateResult.error };
449
+ }
450
+
451
+ return {
452
+ success: true,
453
+ newIndexBlobId,
454
+ newGraphBlobId: graphBlobId,
455
+ digest: updateResult.digest,
456
+ newVersion: updateResult.newVersion,
457
+ };
458
+ } catch (error) {
459
+ const errorMsg = error instanceof Error ? error.message : String(error);
460
+ return { success: false, error: errorMsg };
461
+ }
462
+ }
463
+
464
+ /**
465
+ * Upload placeholder blob to Walrus
466
+ * Used when creating initial MemoryIndex
467
+ */
468
+ export async function uploadPlaceholderToWalrus(
469
+ walrusPublisherUrl: string,
470
+ type: 'index' | 'graph'
471
+ ): Promise<string> {
472
+ const content = JSON.stringify({
473
+ type: type === 'index' ? 'hnsw-index-placeholder' : 'knowledge-graph',
474
+ version: 1,
475
+ created: new Date().toISOString(),
476
+ note: type === 'index'
477
+ ? 'Placeholder for HNSW index. Actual index synced after adding memories.'
478
+ : 'Empty knowledge graph. Will be populated as memories are processed.',
479
+ ...(type === 'graph' ? { nodes: [], edges: [] } : {}),
480
+ });
481
+
482
+ const response = await fetch(walrusPublisherUrl, {
483
+ method: 'PUT',
484
+ headers: { 'Content-Type': 'application/octet-stream' },
485
+ body: new TextEncoder().encode(content),
486
+ });
487
+
488
+ if (!response.ok) {
489
+ throw new Error(`Walrus upload failed: ${response.status}`);
490
+ }
491
+
492
+ const result = await response.json();
493
+
494
+ // Handle different response formats
495
+ let blobId: string;
496
+ if (result.newlyCreated?.blobObject?.blobId) {
497
+ blobId = result.newlyCreated.blobObject.blobId;
498
+ } else if (result.alreadyCertified?.blobId) {
499
+ blobId = result.alreadyCertified.blobId;
500
+ } else if (result.blobId) {
501
+ blobId = result.blobId;
502
+ } else {
503
+ throw new Error('Could not extract blobId from Walrus response');
504
+ }
505
+
506
+ return blobId;
507
+ }