@sochdb/sochdb 0.5.1 → 0.5.3

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.
@@ -10,6 +10,7 @@ exports.EmbeddedDatabase = void 0;
10
10
  const errors_1 = require("../errors");
11
11
  const bindings_1 = require("./ffi/bindings");
12
12
  const transaction_1 = require("./transaction");
13
+ const namespace_1 = require("../namespace");
13
14
  /**
14
15
  * Embedded Database using direct FFI
15
16
  *
@@ -44,6 +45,9 @@ class EmbeddedDatabase {
44
45
  * @returns EmbeddedDatabase instance
45
46
  */
46
47
  static open(path, config) {
48
+ if (path.includes('\x00')) {
49
+ throw new errors_1.DatabaseError('Database path must not contain null bytes');
50
+ }
47
51
  const bindings = bindings_1.NativeBindings.getInstance();
48
52
  let handle;
49
53
  if (config) {
@@ -115,6 +119,9 @@ class EmbeddedDatabase {
115
119
  * @returns EmbeddedDatabase instance in concurrent mode
116
120
  */
117
121
  static openConcurrent(path, options) {
122
+ if (path.includes('\x00')) {
123
+ throw new errors_1.DatabaseError('Database path must not contain null bytes');
124
+ }
118
125
  const bindings = bindings_1.NativeBindings.getInstance();
119
126
  const fallbackToStandard = options?.fallbackToStandard ?? false;
120
127
  if (!bindings.isConcurrentModeAvailable()) {
@@ -325,6 +332,277 @@ class EmbeddedDatabase {
325
332
  getBindings() {
326
333
  return this.bindings;
327
334
  }
335
+ // ========================================================================
336
+ // Namespace Operations
337
+ // ========================================================================
338
+ /**
339
+ * Create a new namespace
340
+ */
341
+ async createNamespace(name, config) {
342
+ this.ensureOpen();
343
+ const metaKey = `_namespace/${name}/metadata`;
344
+ const existing = await this.get(Buffer.from(metaKey));
345
+ if (existing) {
346
+ throw new namespace_1.NamespaceExistsError(name);
347
+ }
348
+ const nsConfig = {
349
+ name,
350
+ displayName: config?.displayName,
351
+ labels: config?.labels,
352
+ readOnly: config?.readOnly ?? false,
353
+ };
354
+ await this.put(Buffer.from(metaKey), Buffer.from(JSON.stringify(nsConfig)));
355
+ return new namespace_1.Namespace(this, name, nsConfig);
356
+ }
357
+ /**
358
+ * Get an existing namespace
359
+ */
360
+ async namespace(name) {
361
+ this.ensureOpen();
362
+ const metaKey = `_namespace/${name}/metadata`;
363
+ const data = await this.get(Buffer.from(metaKey));
364
+ if (!data) {
365
+ throw new namespace_1.NamespaceNotFoundError(name);
366
+ }
367
+ const nsConfig = JSON.parse(data.toString());
368
+ return new namespace_1.Namespace(this, name, nsConfig);
369
+ }
370
+ /**
371
+ * Get or create a namespace
372
+ */
373
+ async getOrCreateNamespace(name, config) {
374
+ try {
375
+ return await this.namespace(name);
376
+ }
377
+ catch (error) {
378
+ if (error instanceof namespace_1.NamespaceNotFoundError) {
379
+ return await this.createNamespace(name, config);
380
+ }
381
+ throw error;
382
+ }
383
+ }
384
+ /**
385
+ * Delete a namespace and all its data
386
+ */
387
+ async deleteNamespace(name) {
388
+ this.ensureOpen();
389
+ const prefix = Buffer.from(`_namespace/${name}/`);
390
+ const collPrefix = Buffer.from(`_collection/${name}/`);
391
+ const toDelete = [];
392
+ try {
393
+ for await (const [keyBuf] of this.scanPrefix(prefix)) {
394
+ toDelete.push(keyBuf);
395
+ }
396
+ for await (const [keyBuf] of this.scanPrefix(collPrefix)) {
397
+ toDelete.push(keyBuf);
398
+ }
399
+ }
400
+ catch { /* scan not available */ }
401
+ for (const key of toDelete) {
402
+ await this.delete(key);
403
+ }
404
+ return toDelete.length > 0;
405
+ }
406
+ /**
407
+ * List all namespace names
408
+ */
409
+ async listNamespaces() {
410
+ this.ensureOpen();
411
+ const prefix = Buffer.from('_namespace/');
412
+ const names = new Set();
413
+ try {
414
+ for await (const [keyBuf] of this.scanPrefix(prefix)) {
415
+ const key = keyBuf.toString();
416
+ const afterPrefix = key.substring('_namespace/'.length);
417
+ const nsName = afterPrefix.split('/')[0];
418
+ if (nsName)
419
+ names.add(nsName);
420
+ }
421
+ }
422
+ catch { /* scan not available */ }
423
+ return Array.from(names);
424
+ }
425
+ // ========================================================================
426
+ // Graph Overlay Operations (thin wrappers using KV)
427
+ // ========================================================================
428
+ /**
429
+ * Add a node to the graph overlay
430
+ */
431
+ async addNode(namespace, nodeId, nodeType, properties) {
432
+ this.ensureOpen();
433
+ const key = `_graph/${namespace}/nodes/${nodeId}`;
434
+ const value = JSON.stringify({ id: nodeId, node_type: nodeType, properties: properties || {} });
435
+ await this.put(Buffer.from(key), Buffer.from(value));
436
+ }
437
+ /**
438
+ * Add an edge between nodes
439
+ */
440
+ async addEdge(namespace, fromId, edgeType, toId, properties) {
441
+ this.ensureOpen();
442
+ const key = `_graph/${namespace}/edges/${fromId}/${edgeType}/${toId}`;
443
+ const value = JSON.stringify({ from_id: fromId, edge_type: edgeType, to_id: toId, properties: properties || {} });
444
+ await this.put(Buffer.from(key), Buffer.from(value));
445
+ }
446
+ /**
447
+ * Traverse the graph from a starting node
448
+ */
449
+ async traverse(namespace, startNode, maxDepth = 10, order = 'bfs') {
450
+ this.ensureOpen();
451
+ const visited = new Set();
452
+ const nodes = [];
453
+ const edges = [];
454
+ const frontier = [[startNode, 0]];
455
+ while (frontier.length > 0) {
456
+ const [currentNode, depth] = order === 'bfs' ? frontier.shift() : frontier.pop();
457
+ if (depth > maxDepth || visited.has(currentNode))
458
+ continue;
459
+ visited.add(currentNode);
460
+ const nodeData = await this.get(Buffer.from(`_graph/${namespace}/nodes/${currentNode}`));
461
+ if (nodeData)
462
+ nodes.push(JSON.parse(nodeData.toString()));
463
+ const edgePrefix = Buffer.from(`_graph/${namespace}/edges/${currentNode}/`);
464
+ try {
465
+ for await (const [, valueBuf] of this.scanPrefix(edgePrefix)) {
466
+ const edge = JSON.parse(valueBuf.toString());
467
+ edges.push(edge);
468
+ if (!visited.has(edge.to_id))
469
+ frontier.push([edge.to_id, depth + 1]);
470
+ }
471
+ }
472
+ catch { /* scan not available */ }
473
+ }
474
+ return { nodes, edges };
475
+ }
476
+ // ========================================================================
477
+ // Semantic Cache Operations (thin wrappers using KV)
478
+ // ========================================================================
479
+ /**
480
+ * Put a value in the semantic cache
481
+ */
482
+ async cachePut(cacheName, key, value, embedding, ttlSeconds = 0) {
483
+ this.ensureOpen();
484
+ const keyHash = Buffer.from(key).toString('hex').slice(0, 16);
485
+ const cacheKey = `_cache/${cacheName}/${keyHash}`;
486
+ const expiresAt = ttlSeconds > 0 ? Math.floor(Date.now() / 1000) + ttlSeconds : 0;
487
+ const cacheValue = JSON.stringify({ key, value, embedding, expires_at: expiresAt });
488
+ await this.put(Buffer.from(cacheKey), Buffer.from(cacheValue));
489
+ }
490
+ /**
491
+ * Get a value from the semantic cache by embedding similarity
492
+ */
493
+ async cacheGet(cacheName, queryEmbedding, threshold = 0.85) {
494
+ this.ensureOpen();
495
+ const prefix = Buffer.from(`_cache/${cacheName}/`);
496
+ const now = Math.floor(Date.now() / 1000);
497
+ let bestMatch = null;
498
+ try {
499
+ for await (const [, valueBuf] of this.scanPrefix(prefix)) {
500
+ const entry = JSON.parse(valueBuf.toString());
501
+ if (entry.expires_at > 0 && now > entry.expires_at)
502
+ continue;
503
+ if (entry.embedding && entry.embedding.length === queryEmbedding.length) {
504
+ const similarity = EmbeddedDatabase.cosineSimilarity(queryEmbedding, entry.embedding);
505
+ if (similarity >= threshold) {
506
+ if (!bestMatch || similarity > bestMatch.similarity) {
507
+ bestMatch = { similarity, value: entry.value };
508
+ }
509
+ }
510
+ }
511
+ }
512
+ }
513
+ catch { /* scan not available */ }
514
+ return bestMatch?.value ?? null;
515
+ }
516
+ /**
517
+ * Delete a cache entry
518
+ */
519
+ async cacheDelete(cacheName, key) {
520
+ this.ensureOpen();
521
+ const keyHash = Buffer.from(key).toString('hex').slice(0, 16);
522
+ await this.delete(Buffer.from(`_cache/${cacheName}/${keyHash}`));
523
+ }
524
+ /**
525
+ * Clear all entries in a semantic cache
526
+ */
527
+ async cacheClear(cacheName) {
528
+ this.ensureOpen();
529
+ const prefix = Buffer.from(`_cache/${cacheName}/`);
530
+ let deleted = 0;
531
+ try {
532
+ const toDelete = [];
533
+ for await (const [keyBuf] of this.scanPrefix(prefix)) {
534
+ toDelete.push(keyBuf);
535
+ }
536
+ for (const k of toDelete) {
537
+ await this.delete(k);
538
+ deleted++;
539
+ }
540
+ }
541
+ catch { /* scan not available */ }
542
+ return deleted;
543
+ }
544
+ // ========================================================================
545
+ // Static Format Utilities
546
+ // ========================================================================
547
+ /**
548
+ * Convert records to TOON format
549
+ */
550
+ static toToon(tableName, records, fields) {
551
+ if (!records || records.length === 0)
552
+ return `${tableName}[0]{}:`;
553
+ const useFields = fields ?? Object.keys(records[0]);
554
+ const header = `${tableName}[${records.length}]{${useFields.join(',')}}:`;
555
+ const escapeValue = (v) => {
556
+ const s = v != null ? String(v) : '';
557
+ if (s.includes(',') || s.includes(';') || s.includes('\n'))
558
+ return `"${s}"`;
559
+ return s;
560
+ };
561
+ const rows = records.map(r => useFields.map(f => escapeValue(r[f])).join(',')).join(';');
562
+ return header + rows;
563
+ }
564
+ /**
565
+ * Convert records to JSON format
566
+ */
567
+ static toJson(tableName, records, fields, compact = true) {
568
+ if (!records || records.length === 0) {
569
+ return JSON.stringify({ table: tableName, count: 0, records: [] });
570
+ }
571
+ const filteredRecords = fields
572
+ ? records.map(r => {
573
+ const filtered = {};
574
+ for (const f of fields)
575
+ filtered[f] = r[f];
576
+ return filtered;
577
+ })
578
+ : records;
579
+ const output = { table: tableName, count: filteredRecords.length, records: filteredRecords };
580
+ return compact ? JSON.stringify(output) : JSON.stringify(output, null, 2);
581
+ }
582
+ /**
583
+ * Parse JSON format
584
+ */
585
+ static fromJson(jsonStr) {
586
+ const data = JSON.parse(jsonStr);
587
+ const table = data.table ?? 'unknown';
588
+ const records = data.records ?? [];
589
+ const fields = records.length > 0 ? Object.keys(records[0]) : [];
590
+ return { table, fields, records };
591
+ }
592
+ // ========================================================================
593
+ // Private Helpers
594
+ // ========================================================================
595
+ static cosineSimilarity(a, b) {
596
+ let dot = 0, normA = 0, normB = 0;
597
+ for (let i = 0; i < a.length; i++) {
598
+ dot += a[i] * b[i];
599
+ normA += a[i] * a[i];
600
+ normB += b[i] * b[i];
601
+ }
602
+ normA = Math.sqrt(normA);
603
+ normB = Math.sqrt(normB);
604
+ return normA === 0 || normB === 0 ? 0 : dot / (normA * normB);
605
+ }
328
606
  }
329
607
  exports.EmbeddedDatabase = EmbeddedDatabase;
330
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"database.js","sourceRoot":"","sources":["../../../src/embedded/database.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Embedded Database - FFI Mode\n * \n * Direct FFI access to SochDB native library.\n * No server required - similar to Python SDK's Database class.\n */\n\nimport { DatabaseError } from '../errors';\nimport { NativeBindings } from './ffi/bindings';\nimport { EmbeddedTransaction } from './transaction';\nimport * as koffi from 'koffi';\n\nexport interface EmbeddedDatabaseConfig {\n    walEnabled?: boolean;\n    syncMode?: 'full' | 'normal' | 'off';\n    memtableSizeBytes?: number;\n    groupCommit?: boolean;\n    indexPolicy?: 'write_optimized' | 'balanced' | 'scan_optimized' | 'append_only';\n}\n\n/**\n * Embedded Database using direct FFI\n * \n * @example\n * ```typescript\n * import { EmbeddedDatabase } from '@sochdb/sochdb';\n * \n * const db = await EmbeddedDatabase.open('./mydb');\n * await db.put(Buffer.from('key'), Buffer.from('value'));\n * const value = await db.get(Buffer.from('key'));\n * await db.close();\n * ```\n */\nexport class EmbeddedDatabase {\n    private handle: any;\n    private bindings: NativeBindings;\n    private closed = false;\n    private path: string;\n    private concurrent = false;\n    private _concurrentModeFallback = false;\n\n    private constructor(path: string, handle: any, concurrent = false, fallback = false) {\n        this.path = path;\n        this.handle = handle;\n        this.concurrent = concurrent;\n        this._concurrentModeFallback = fallback;\n        this.bindings = NativeBindings.getInstance();\n    }\n\n    /**\n     * Open a database at the specified path in standard mode\n     * \n     * For web applications with multiple processes, use `openConcurrent()` instead.\n     * \n     * @param path - Path to database directory\n     * @param config - Optional configuration\n     * @returns EmbeddedDatabase instance\n     */\n    static open(path: string, config?: EmbeddedDatabaseConfig): EmbeddedDatabase {\n        const bindings = NativeBindings.getInstance();\n        let handle;\n\n        if (config) {\n            const cConfig = {\n                wal_enabled: config.walEnabled ?? false,\n                wal_enabled_set: config.walEnabled !== undefined,\n                sync_mode: config.syncMode === 'full' ? 2 : (config.syncMode === 'normal' ? 1 : 0),\n                sync_mode_set: config.syncMode !== undefined,\n                memtable_size_bytes: BigInt(config.memtableSizeBytes ?? 0),\n                group_commit: config.groupCommit ?? false,\n                group_commit_set: config.groupCommit !== undefined,\n                default_index_policy: 1, // Default to Balanced\n                default_index_policy_set: false\n            };\n            handle = bindings.sochdb_open_with_config(path, cConfig);\n        } else {\n            handle = bindings.sochdb_open(path);\n        }\n\n        if (!handle) {\n            throw new DatabaseError(`Failed to open database at ${path}`);\n        }\n\n        return new EmbeddedDatabase(path, handle, false);\n    }\n\n    /**\n     * Open a database in concurrent mode for multi-process web applications\n     * \n     * This mode allows multiple Node.js processes (e.g., PM2 cluster workers,\n     * multiple Express instances) to access the database simultaneously.\n     * \n     * Features:\n     * - Lock-free reads with ~100ns latency\n     * - Multi-reader, single-writer coordination\n     * - Automatic write serialization\n     * \n     * @example\n     * ```typescript\n     * import { EmbeddedDatabase } from '@sochdb/sochdb';\n     * import express from 'express';\n     * \n     * // Open in concurrent mode - multiple workers can access\n     * const db = EmbeddedDatabase.openConcurrent('./web_db');\n     * \n     * const app = express();\n     * \n     * app.get('/user/:id', async (req, res) => {\n     *   // Multiple concurrent requests can read simultaneously\n     *   const data = await db.get(Buffer.from(`user:${req.params.id}`));\n     *   if (!data) {\n     *     res.status(404).json({ error: 'not found' });\n     *     return;\n     *   }\n     *   res.send(data);\n     * });\n     * \n     * app.post('/user/:id', async (req, res) => {\n     *   // Writes are serialized automatically\n     *   await db.put(Buffer.from(`user:${req.params.id}`), req.body);\n     *   res.json({ status: 'ok' });\n     * });\n     * \n     * // Start with PM2 cluster mode:\n     * // pm2 start app.js -i max\n     * app.listen(3000);\n     * ```\n     * \n     * @param path - Path to database directory\n     * @param options - Optional configuration for concurrent mode\n     * @returns EmbeddedDatabase instance in concurrent mode\n     */\n    static openConcurrent(path: string, options?: { fallbackToStandard?: boolean }): EmbeddedDatabase {\n        const bindings = NativeBindings.getInstance();\n        const fallbackToStandard = options?.fallbackToStandard ?? false;\n        \n        if (!bindings.isConcurrentModeAvailable()) {\n            if (fallbackToStandard) {\n                console.warn(\n                    '[SochDB] Concurrent mode not available in native library (requires v0.4.8+). ' +\n                    'Falling back to standard mode. For production multi-process deployments, ' +\n                    'please upgrade the SochDB native library.'\n                );\n                const handle = bindings.sochdb_open(path);\n                if (!handle) {\n                    throw new DatabaseError(`Failed to open database at ${path}`);\n                }\n                return new EmbeddedDatabase(path, handle, false, true);\n            }\n            throw new DatabaseError(\n                'Concurrent mode not supported. Please upgrade the SochDB native library to v0.4.8+ ' +\n                'or use openConcurrent(path, { fallbackToStandard: true }) to fall back to standard mode.'\n            );\n        }\n\n        const handle = bindings.sochdb_open_concurrent(path);\n        if (!handle) {\n            throw new DatabaseError(`Failed to open database in concurrent mode at ${path}`);\n        }\n\n        const isConcurrent = bindings.sochdb_is_concurrent?.(handle) === 1;\n        return new EmbeddedDatabase(path, handle, isConcurrent, false);\n    }\n\n    /**\n     * Check if database is opened in concurrent mode\n     */\n    get isConcurrent(): boolean {\n        return this.concurrent;\n    }\n\n    /**\n     * Check if concurrent mode fell back to standard mode\n     */\n    get isConcurrentFallback(): boolean {\n        return this._concurrentModeFallback;\n    }\n\n    /**\n     * Check if concurrent mode is available in the native library\n     */\n    static isConcurrentModeAvailable(): boolean {\n        return NativeBindings.getInstance().isConcurrentModeAvailable();\n    }\n\n    /**\n     * Put a key-value pair (auto-transaction)\n     */\n    async put(key: Buffer, value: Buffer): Promise<void> {\n        this.ensureOpen();\n\n        const txn = this.transaction();\n        try {\n            await txn.put(key, value);\n            await txn.commit();\n        } catch (error) {\n            await txn.abort();\n            throw error;\n        }\n    }\n\n    /**\n     * Get a value by key (auto-transaction)\n     */\n    async get(key: Buffer): Promise<Buffer | null> {\n        this.ensureOpen();\n\n        const txn = this.transaction();\n        try {\n            const value = await txn.get(key);\n            await txn.commit();\n            return value;\n        } catch (error) {\n            await txn.abort();\n            throw error;\n        }\n    }\n\n    /**\n     * Delete a key (auto-transaction)\n     */\n    async delete(key: Buffer): Promise<void> {\n        this.ensureOpen();\n\n        const txn = this.transaction();\n        try {\n            await txn.delete(key);\n            await txn.commit();\n        } catch (error) {\n            await txn.abort();\n            throw error;\n        }\n    }\n\n    /**\n     * Put value at path (auto-transaction)\n     */\n    async putPath(path: string, value: Buffer): Promise<void> {\n        this.ensureOpen();\n\n        const txn = this.transaction();\n        try {\n            await txn.putPath(path, value);\n            await txn.commit();\n        } catch (error) {\n            await txn.abort();\n            throw error;\n        }\n    }\n\n    /**\n     * Get value at path (auto-transaction)\n     */\n    async getPath(path: string): Promise<Buffer | null> {\n        this.ensureOpen();\n\n        const txn = this.transaction();\n        try {\n            const value = await txn.getPath(path);\n            await txn.commit();\n            return value;\n        } catch (error) {\n            await txn.abort();\n            throw error;\n        }\n    }\n\n    /**\n     * Scan keys with prefix\n     */\n    async *scanPrefix(prefix: Buffer): AsyncGenerator<[Buffer, Buffer]> {\n        this.ensureOpen();\n\n        const txn = this.transaction();\n        try {\n            for await (const entry of txn.scanPrefix(prefix)) {\n                yield entry;\n            }\n            await txn.commit();\n        } catch (error) {\n            await txn.abort();\n            throw error;\n        }\n    }\n\n    /**\n     * Begin a transaction\n     */\n    transaction(): EmbeddedTransaction {\n        this.ensureOpen();\n\n        const txnHandle = this.bindings.sochdb_begin_txn(this.handle);\n        return new EmbeddedTransaction(this, this.handle, txnHandle);\n    }\n\n    /**\n     * Execute operations within a transaction (with auto-commit/abort)\n     */\n    async withTransaction<T>(fn: (txn: EmbeddedTransaction) => Promise<T>): Promise<T> {\n        const txn = this.transaction();\n        try {\n            const result = await fn(txn);\n            await txn.commit();\n            return result;\n        } catch (error) {\n            await txn.abort();\n            throw error;\n        }\n    }\n\n    /**\n     * Force a checkpoint\n     */\n    async checkpoint(): Promise<bigint> {\n        this.ensureOpen();\n        const lsn = this.bindings.sochdb_checkpoint(this.handle);\n        return BigInt(lsn);\n    }\n\n    /**\n     * Get storage statistics\n     */\n    async stats(): Promise<{\n        memtableSizeBytes: bigint;\n        walSizeBytes: bigint;\n        activeTransactions: number;\n        minActiveSnapshot: bigint;\n        lastCheckpointLsn: bigint;\n    }> {\n        this.ensureOpen();\n\n        // Returns struct by value (automatically decoded)\n        const stats = this.bindings.sochdb_stats(this.handle);\n\n        const result = {\n            memtableSizeBytes: BigInt(stats.memtable_size_bytes),\n            walSizeBytes: BigInt(stats.wal_size_bytes),\n            activeTransactions: stats.active_transactions,\n            minActiveSnapshot: BigInt(stats.min_active_snapshot),\n            lastCheckpointLsn: BigInt(stats.last_checkpoint_lsn),\n        };\n\n        return result;\n    }\n\n    /**\n     * Close the database\n     */\n    close(): void {\n        if (!this.closed) {\n            this.bindings.sochdb_close(this.handle);\n            this.closed = true;\n        }\n    }\n\n    private ensureOpen(): void {\n        if (this.closed) {\n            throw new DatabaseError('Database is closed');\n        }\n    }\n\n    /**\n     * Get internal handle (for transactions)\n     * @internal\n     */\n    getHandle(): any {\n        return this.handle;\n    }\n\n    /**\n     * Get bindings instance (for transactions)\n     * @internal\n     */\n    getBindings(): NativeBindings {\n        return this.bindings;\n    }\n}\n"]}
608
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"database.js","sourceRoot":"","sources":["../../../src/embedded/database.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Embedded Database - FFI Mode\n * \n * Direct FFI access to SochDB native library.\n * No server required - similar to Python SDK's Database class.\n */\n\nimport { DatabaseError } from '../errors';\nimport { NativeBindings } from './ffi/bindings';\nimport { EmbeddedTransaction } from './transaction';\nimport { Namespace, NamespaceConfig, NamespaceNotFoundError, NamespaceExistsError } from '../namespace';\nimport * as koffi from 'koffi';\n\nexport interface EmbeddedDatabaseConfig {\n    walEnabled?: boolean;\n    syncMode?: 'full' | 'normal' | 'off';\n    memtableSizeBytes?: number;\n    groupCommit?: boolean;\n    indexPolicy?: 'write_optimized' | 'balanced' | 'scan_optimized' | 'append_only';\n}\n\n/**\n * Embedded Database using direct FFI\n * \n * @example\n * ```typescript\n * import { EmbeddedDatabase } from '@sochdb/sochdb';\n * \n * const db = await EmbeddedDatabase.open('./mydb');\n * await db.put(Buffer.from('key'), Buffer.from('value'));\n * const value = await db.get(Buffer.from('key'));\n * await db.close();\n * ```\n */\nexport class EmbeddedDatabase {\n    private handle: any;\n    private bindings: NativeBindings;\n    private closed = false;\n    private path: string;\n    private concurrent = false;\n    private _concurrentModeFallback = false;\n\n    private constructor(path: string, handle: any, concurrent = false, fallback = false) {\n        this.path = path;\n        this.handle = handle;\n        this.concurrent = concurrent;\n        this._concurrentModeFallback = fallback;\n        this.bindings = NativeBindings.getInstance();\n    }\n\n    /**\n     * Open a database at the specified path in standard mode\n     * \n     * For web applications with multiple processes, use `openConcurrent()` instead.\n     * \n     * @param path - Path to database directory\n     * @param config - Optional configuration\n     * @returns EmbeddedDatabase instance\n     */\n    static open(path: string, config?: EmbeddedDatabaseConfig): EmbeddedDatabase {\n        if (path.includes('\\x00')) {\n            throw new DatabaseError('Database path must not contain null bytes');\n        }\n        const bindings = NativeBindings.getInstance();\n        let handle;\n\n        if (config) {\n            const cConfig = {\n                wal_enabled: config.walEnabled ?? false,\n                wal_enabled_set: config.walEnabled !== undefined,\n                sync_mode: config.syncMode === 'full' ? 2 : (config.syncMode === 'normal' ? 1 : 0),\n                sync_mode_set: config.syncMode !== undefined,\n                memtable_size_bytes: BigInt(config.memtableSizeBytes ?? 0),\n                group_commit: config.groupCommit ?? false,\n                group_commit_set: config.groupCommit !== undefined,\n                default_index_policy: 1, // Default to Balanced\n                default_index_policy_set: false\n            };\n            handle = bindings.sochdb_open_with_config(path, cConfig);\n        } else {\n            handle = bindings.sochdb_open(path);\n        }\n\n        if (!handle) {\n            throw new DatabaseError(`Failed to open database at ${path}`);\n        }\n\n        return new EmbeddedDatabase(path, handle, false);\n    }\n\n    /**\n     * Open a database in concurrent mode for multi-process web applications\n     * \n     * This mode allows multiple Node.js processes (e.g., PM2 cluster workers,\n     * multiple Express instances) to access the database simultaneously.\n     * \n     * Features:\n     * - Lock-free reads with ~100ns latency\n     * - Multi-reader, single-writer coordination\n     * - Automatic write serialization\n     * \n     * @example\n     * ```typescript\n     * import { EmbeddedDatabase } from '@sochdb/sochdb';\n     * import express from 'express';\n     * \n     * // Open in concurrent mode - multiple workers can access\n     * const db = EmbeddedDatabase.openConcurrent('./web_db');\n     * \n     * const app = express();\n     * \n     * app.get('/user/:id', async (req, res) => {\n     *   // Multiple concurrent requests can read simultaneously\n     *   const data = await db.get(Buffer.from(`user:${req.params.id}`));\n     *   if (!data) {\n     *     res.status(404).json({ error: 'not found' });\n     *     return;\n     *   }\n     *   res.send(data);\n     * });\n     * \n     * app.post('/user/:id', async (req, res) => {\n     *   // Writes are serialized automatically\n     *   await db.put(Buffer.from(`user:${req.params.id}`), req.body);\n     *   res.json({ status: 'ok' });\n     * });\n     * \n     * // Start with PM2 cluster mode:\n     * // pm2 start app.js -i max\n     * app.listen(3000);\n     * ```\n     * \n     * @param path - Path to database directory\n     * @param options - Optional configuration for concurrent mode\n     * @returns EmbeddedDatabase instance in concurrent mode\n     */\n    static openConcurrent(path: string, options?: { fallbackToStandard?: boolean }): EmbeddedDatabase {\n        if (path.includes('\\x00')) {\n            throw new DatabaseError('Database path must not contain null bytes');\n        }\n        const bindings = NativeBindings.getInstance();\n        const fallbackToStandard = options?.fallbackToStandard ?? false;\n        \n        if (!bindings.isConcurrentModeAvailable()) {\n            if (fallbackToStandard) {\n                console.warn(\n                    '[SochDB] Concurrent mode not available in native library (requires v0.4.8+). ' +\n                    'Falling back to standard mode. For production multi-process deployments, ' +\n                    'please upgrade the SochDB native library.'\n                );\n                const handle = bindings.sochdb_open(path);\n                if (!handle) {\n                    throw new DatabaseError(`Failed to open database at ${path}`);\n                }\n                return new EmbeddedDatabase(path, handle, false, true);\n            }\n            throw new DatabaseError(\n                'Concurrent mode not supported. Please upgrade the SochDB native library to v0.4.8+ ' +\n                'or use openConcurrent(path, { fallbackToStandard: true }) to fall back to standard mode.'\n            );\n        }\n\n        const handle = bindings.sochdb_open_concurrent(path);\n        if (!handle) {\n            throw new DatabaseError(`Failed to open database in concurrent mode at ${path}`);\n        }\n\n        const isConcurrent = bindings.sochdb_is_concurrent?.(handle) === 1;\n        return new EmbeddedDatabase(path, handle, isConcurrent, false);\n    }\n\n    /**\n     * Check if database is opened in concurrent mode\n     */\n    get isConcurrent(): boolean {\n        return this.concurrent;\n    }\n\n    /**\n     * Check if concurrent mode fell back to standard mode\n     */\n    get isConcurrentFallback(): boolean {\n        return this._concurrentModeFallback;\n    }\n\n    /**\n     * Check if concurrent mode is available in the native library\n     */\n    static isConcurrentModeAvailable(): boolean {\n        return NativeBindings.getInstance().isConcurrentModeAvailable();\n    }\n\n    /**\n     * Put a key-value pair (auto-transaction)\n     */\n    async put(key: Buffer, value: Buffer): Promise<void> {\n        this.ensureOpen();\n\n        const txn = this.transaction();\n        try {\n            await txn.put(key, value);\n            await txn.commit();\n        } catch (error) {\n            await txn.abort();\n            throw error;\n        }\n    }\n\n    /**\n     * Get a value by key (auto-transaction)\n     */\n    async get(key: Buffer): Promise<Buffer | null> {\n        this.ensureOpen();\n\n        const txn = this.transaction();\n        try {\n            const value = await txn.get(key);\n            await txn.commit();\n            return value;\n        } catch (error) {\n            await txn.abort();\n            throw error;\n        }\n    }\n\n    /**\n     * Delete a key (auto-transaction)\n     */\n    async delete(key: Buffer): Promise<void> {\n        this.ensureOpen();\n\n        const txn = this.transaction();\n        try {\n            await txn.delete(key);\n            await txn.commit();\n        } catch (error) {\n            await txn.abort();\n            throw error;\n        }\n    }\n\n    /**\n     * Put value at path (auto-transaction)\n     */\n    async putPath(path: string, value: Buffer): Promise<void> {\n        this.ensureOpen();\n\n        const txn = this.transaction();\n        try {\n            await txn.putPath(path, value);\n            await txn.commit();\n        } catch (error) {\n            await txn.abort();\n            throw error;\n        }\n    }\n\n    /**\n     * Get value at path (auto-transaction)\n     */\n    async getPath(path: string): Promise<Buffer | null> {\n        this.ensureOpen();\n\n        const txn = this.transaction();\n        try {\n            const value = await txn.getPath(path);\n            await txn.commit();\n            return value;\n        } catch (error) {\n            await txn.abort();\n            throw error;\n        }\n    }\n\n    /**\n     * Scan keys with prefix\n     */\n    async *scanPrefix(prefix: Buffer): AsyncGenerator<[Buffer, Buffer]> {\n        this.ensureOpen();\n\n        const txn = this.transaction();\n        try {\n            for await (const entry of txn.scanPrefix(prefix)) {\n                yield entry;\n            }\n            await txn.commit();\n        } catch (error) {\n            await txn.abort();\n            throw error;\n        }\n    }\n\n    /**\n     * Begin a transaction\n     */\n    transaction(): EmbeddedTransaction {\n        this.ensureOpen();\n\n        const txnHandle = this.bindings.sochdb_begin_txn(this.handle);\n        return new EmbeddedTransaction(this, this.handle, txnHandle);\n    }\n\n    /**\n     * Execute operations within a transaction (with auto-commit/abort)\n     */\n    async withTransaction<T>(fn: (txn: EmbeddedTransaction) => Promise<T>): Promise<T> {\n        const txn = this.transaction();\n        try {\n            const result = await fn(txn);\n            await txn.commit();\n            return result;\n        } catch (error) {\n            await txn.abort();\n            throw error;\n        }\n    }\n\n    /**\n     * Force a checkpoint\n     */\n    async checkpoint(): Promise<bigint> {\n        this.ensureOpen();\n        const lsn = this.bindings.sochdb_checkpoint(this.handle);\n        return BigInt(lsn);\n    }\n\n    /**\n     * Get storage statistics\n     */\n    async stats(): Promise<{\n        memtableSizeBytes: bigint;\n        walSizeBytes: bigint;\n        activeTransactions: number;\n        minActiveSnapshot: bigint;\n        lastCheckpointLsn: bigint;\n    }> {\n        this.ensureOpen();\n\n        // Returns struct by value (automatically decoded)\n        const stats = this.bindings.sochdb_stats(this.handle);\n\n        const result = {\n            memtableSizeBytes: BigInt(stats.memtable_size_bytes),\n            walSizeBytes: BigInt(stats.wal_size_bytes),\n            activeTransactions: stats.active_transactions,\n            minActiveSnapshot: BigInt(stats.min_active_snapshot),\n            lastCheckpointLsn: BigInt(stats.last_checkpoint_lsn),\n        };\n\n        return result;\n    }\n\n    /**\n     * Close the database\n     */\n    close(): void {\n        if (!this.closed) {\n            this.bindings.sochdb_close(this.handle);\n            this.closed = true;\n        }\n    }\n\n    private ensureOpen(): void {\n        if (this.closed) {\n            throw new DatabaseError('Database is closed');\n        }\n    }\n\n    /**\n     * Get internal handle (for transactions)\n     * @internal\n     */\n    getHandle(): any {\n        return this.handle;\n    }\n\n    /**\n     * Get bindings instance (for transactions)\n     * @internal\n     */\n    getBindings(): NativeBindings {\n        return this.bindings;\n    }\n\n    // ========================================================================\n    // Namespace Operations\n    // ========================================================================\n\n    /**\n     * Create a new namespace\n     */\n    async createNamespace(name: string, config?: Partial<NamespaceConfig>): Promise<Namespace> {\n        this.ensureOpen();\n        const metaKey = `_namespace/${name}/metadata`;\n        const existing = await this.get(Buffer.from(metaKey));\n        if (existing) {\n            throw new NamespaceExistsError(name);\n        }\n\n        const nsConfig: NamespaceConfig = {\n            name,\n            displayName: config?.displayName,\n            labels: config?.labels,\n            readOnly: config?.readOnly ?? false,\n        };\n        await this.put(Buffer.from(metaKey), Buffer.from(JSON.stringify(nsConfig)));\n        return new Namespace(this, name, nsConfig);\n    }\n\n    /**\n     * Get an existing namespace\n     */\n    async namespace(name: string): Promise<Namespace> {\n        this.ensureOpen();\n        const metaKey = `_namespace/${name}/metadata`;\n        const data = await this.get(Buffer.from(metaKey));\n        if (!data) {\n            throw new NamespaceNotFoundError(name);\n        }\n        const nsConfig = JSON.parse(data.toString());\n        return new Namespace(this, name, nsConfig);\n    }\n\n    /**\n     * Get or create a namespace\n     */\n    async getOrCreateNamespace(name: string, config?: Partial<NamespaceConfig>): Promise<Namespace> {\n        try {\n            return await this.namespace(name);\n        } catch (error) {\n            if (error instanceof NamespaceNotFoundError) {\n                return await this.createNamespace(name, config);\n            }\n            throw error;\n        }\n    }\n\n    /**\n     * Delete a namespace and all its data\n     */\n    async deleteNamespace(name: string): Promise<boolean> {\n        this.ensureOpen();\n        const prefix = Buffer.from(`_namespace/${name}/`);\n        const collPrefix = Buffer.from(`_collection/${name}/`);\n\n        const toDelete: Buffer[] = [];\n        try {\n            for await (const [keyBuf] of this.scanPrefix(prefix)) {\n                toDelete.push(keyBuf);\n            }\n            for await (const [keyBuf] of this.scanPrefix(collPrefix)) {\n                toDelete.push(keyBuf);\n            }\n        } catch { /* scan not available */ }\n\n        for (const key of toDelete) {\n            await this.delete(key);\n        }\n        return toDelete.length > 0;\n    }\n\n    /**\n     * List all namespace names\n     */\n    async listNamespaces(): Promise<string[]> {\n        this.ensureOpen();\n        const prefix = Buffer.from('_namespace/');\n        const names = new Set<string>();\n        try {\n            for await (const [keyBuf] of this.scanPrefix(prefix)) {\n                const key = keyBuf.toString();\n                const afterPrefix = key.substring('_namespace/'.length);\n                const nsName = afterPrefix.split('/')[0];\n                if (nsName) names.add(nsName);\n            }\n        } catch { /* scan not available */ }\n        return Array.from(names);\n    }\n\n    // ========================================================================\n    // Graph Overlay Operations (thin wrappers using KV)\n    // ========================================================================\n\n    /**\n     * Add a node to the graph overlay\n     */\n    async addNode(\n        namespace: string, nodeId: string, nodeType: string,\n        properties?: Record<string, string>\n    ): Promise<void> {\n        this.ensureOpen();\n        const key = `_graph/${namespace}/nodes/${nodeId}`;\n        const value = JSON.stringify({ id: nodeId, node_type: nodeType, properties: properties || {} });\n        await this.put(Buffer.from(key), Buffer.from(value));\n    }\n\n    /**\n     * Add an edge between nodes\n     */\n    async addEdge(\n        namespace: string, fromId: string, edgeType: string, toId: string,\n        properties?: Record<string, string>\n    ): Promise<void> {\n        this.ensureOpen();\n        const key = `_graph/${namespace}/edges/${fromId}/${edgeType}/${toId}`;\n        const value = JSON.stringify({ from_id: fromId, edge_type: edgeType, to_id: toId, properties: properties || {} });\n        await this.put(Buffer.from(key), Buffer.from(value));\n    }\n\n    /**\n     * Traverse the graph from a starting node\n     */\n    async traverse(\n        namespace: string, startNode: string, maxDepth = 10,\n        order: 'bfs' | 'dfs' = 'bfs'\n    ): Promise<{ nodes: any[]; edges: any[] }> {\n        this.ensureOpen();\n        const visited = new Set<string>();\n        const nodes: any[] = [];\n        const edges: any[] = [];\n        const frontier: Array<[string, number]> = [[startNode, 0]];\n\n        while (frontier.length > 0) {\n            const [currentNode, depth] = order === 'bfs' ? frontier.shift()! : frontier.pop()!;\n            if (depth > maxDepth || visited.has(currentNode)) continue;\n            visited.add(currentNode);\n\n            const nodeData = await this.get(Buffer.from(`_graph/${namespace}/nodes/${currentNode}`));\n            if (nodeData) nodes.push(JSON.parse(nodeData.toString()));\n\n            const edgePrefix = Buffer.from(`_graph/${namespace}/edges/${currentNode}/`);\n            try {\n                for await (const [, valueBuf] of this.scanPrefix(edgePrefix)) {\n                    const edge = JSON.parse(valueBuf.toString());\n                    edges.push(edge);\n                    if (!visited.has(edge.to_id)) frontier.push([edge.to_id, depth + 1]);\n                }\n            } catch { /* scan not available */ }\n        }\n        return { nodes, edges };\n    }\n\n    // ========================================================================\n    // Semantic Cache Operations (thin wrappers using KV)\n    // ========================================================================\n\n    /**\n     * Put a value in the semantic cache\n     */\n    async cachePut(\n        cacheName: string, key: string, value: string,\n        embedding: number[], ttlSeconds = 0\n    ): Promise<void> {\n        this.ensureOpen();\n        const keyHash = Buffer.from(key).toString('hex').slice(0, 16);\n        const cacheKey = `_cache/${cacheName}/${keyHash}`;\n        const expiresAt = ttlSeconds > 0 ? Math.floor(Date.now() / 1000) + ttlSeconds : 0;\n        const cacheValue = JSON.stringify({ key, value, embedding, expires_at: expiresAt });\n        await this.put(Buffer.from(cacheKey), Buffer.from(cacheValue));\n    }\n\n    /**\n     * Get a value from the semantic cache by embedding similarity\n     */\n    async cacheGet(\n        cacheName: string, queryEmbedding: number[], threshold = 0.85\n    ): Promise<string | null> {\n        this.ensureOpen();\n        const prefix = Buffer.from(`_cache/${cacheName}/`);\n        const now = Math.floor(Date.now() / 1000);\n        let bestMatch: { similarity: number; value: string } | null = null;\n\n        try {\n            for await (const [, valueBuf] of this.scanPrefix(prefix)) {\n                const entry = JSON.parse(valueBuf.toString());\n                if (entry.expires_at > 0 && now > entry.expires_at) continue;\n                if (entry.embedding && entry.embedding.length === queryEmbedding.length) {\n                    const similarity = EmbeddedDatabase.cosineSimilarity(queryEmbedding, entry.embedding);\n                    if (similarity >= threshold) {\n                        if (!bestMatch || similarity > bestMatch.similarity) {\n                            bestMatch = { similarity, value: entry.value };\n                        }\n                    }\n                }\n            }\n        } catch { /* scan not available */ }\n\n        return bestMatch?.value ?? null;\n    }\n\n    /**\n     * Delete a cache entry\n     */\n    async cacheDelete(cacheName: string, key: string): Promise<void> {\n        this.ensureOpen();\n        const keyHash = Buffer.from(key).toString('hex').slice(0, 16);\n        await this.delete(Buffer.from(`_cache/${cacheName}/${keyHash}`));\n    }\n\n    /**\n     * Clear all entries in a semantic cache\n     */\n    async cacheClear(cacheName: string): Promise<number> {\n        this.ensureOpen();\n        const prefix = Buffer.from(`_cache/${cacheName}/`);\n        let deleted = 0;\n        try {\n            const toDelete: Buffer[] = [];\n            for await (const [keyBuf] of this.scanPrefix(prefix)) {\n                toDelete.push(keyBuf);\n            }\n            for (const k of toDelete) {\n                await this.delete(k);\n                deleted++;\n            }\n        } catch { /* scan not available */ }\n        return deleted;\n    }\n\n    // ========================================================================\n    // Static Format Utilities\n    // ========================================================================\n\n    /**\n     * Convert records to TOON format\n     */\n    static toToon(\n        tableName: string,\n        records: Array<Record<string, any>>,\n        fields?: string[]\n    ): string {\n        if (!records || records.length === 0) return `${tableName}[0]{}:`;\n        const useFields = fields ?? Object.keys(records[0]);\n        const header = `${tableName}[${records.length}]{${useFields.join(',')}}:`;\n        const escapeValue = (v: any): string => {\n            const s = v != null ? String(v) : '';\n            if (s.includes(',') || s.includes(';') || s.includes('\\n')) return `\"${s}\"`;\n            return s;\n        };\n        const rows = records.map(r => useFields.map(f => escapeValue(r[f])).join(',')).join(';');\n        return header + rows;\n    }\n\n    /**\n     * Convert records to JSON format\n     */\n    static toJson(\n        tableName: string,\n        records: Array<Record<string, any>>,\n        fields?: string[],\n        compact = true\n    ): string {\n        if (!records || records.length === 0) {\n            return JSON.stringify({ table: tableName, count: 0, records: [] });\n        }\n        const filteredRecords = fields\n            ? records.map(r => {\n                const filtered: Record<string, any> = {};\n                for (const f of fields) filtered[f] = r[f];\n                return filtered;\n            })\n            : records;\n        const output = { table: tableName, count: filteredRecords.length, records: filteredRecords };\n        return compact ? JSON.stringify(output) : JSON.stringify(output, null, 2);\n    }\n\n    /**\n     * Parse JSON format\n     */\n    static fromJson(jsonStr: string): {\n        table: string; fields: string[]; records: Array<Record<string, any>>;\n    } {\n        const data = JSON.parse(jsonStr);\n        const table = data.table ?? 'unknown';\n        const records = data.records ?? [];\n        const fields = records.length > 0 ? Object.keys(records[0]) : [];\n        return { table, fields, records };\n    }\n\n    // ========================================================================\n    // Private Helpers\n    // ========================================================================\n\n    private static cosineSimilarity(a: number[], b: number[]): number {\n        let dot = 0, normA = 0, normB = 0;\n        for (let i = 0; i < a.length; i++) {\n            dot += a[i] * b[i]; normA += a[i] * a[i]; normB += b[i] * b[i];\n        }\n        normA = Math.sqrt(normA); normB = Math.sqrt(normB);\n        return normA === 0 || normB === 0 ? 0 : dot / (normA * normB);\n    }\n}\n"]}
package/dist/cjs/index.js CHANGED
@@ -1,6 +1,6 @@
1
1
  "use strict";
2
2
  /**
3
- * SochDB Node.js SDK v0.5.1
3
+ * SochDB Node.js SDK v0.5.3
4
4
  *
5
5
  * Dual-mode architecture: Embedded (FFI) + Server (gRPC/IPC)
6
6
  *
@@ -55,7 +55,7 @@ exports.GrpcClient = exports.SplitBrainError = exports.EpochMismatchError = expo
55
55
  exports.openConcurrent = openConcurrent;
56
56
  exports.open = open;
57
57
  // Version
58
- exports.VERSION = '0.5.1';
58
+ exports.VERSION = '0.5.3';
59
59
  // Embedded mode (FFI) - NEW
60
60
  var embedded_1 = require("./embedded");
61
61
  Object.defineProperty(exports, "EmbeddedDatabase", { enumerable: true, get: function () { return embedded_1.EmbeddedDatabase; } });
@@ -152,4 +152,4 @@ Object.defineProperty(exports, "SplitBrainError", { enumerable: true, get: funct
152
152
  // Convenience alias
153
153
  var grpc_client_2 = require("./grpc-client");
154
154
  Object.defineProperty(exports, "GrpcClient", { enumerable: true, get: function () { return grpc_client_2.SochDBClient; } });
155
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAkDG;;;AAqBH,wCAEC;AAMD,oBAEC;AA7BD,UAAU;AACG,QAAA,OAAO,GAAG,OAAO,CAAC;AAE/B,4BAA4B;AAC5B,uCAAsE;AAA7D,4GAAA,gBAAgB,OAAA;AACzB,uCAAiD;AAAxC,+GAAA,mBAAmB,OAAA;AAC5B,uCAAiE;AAAxD,qGAAA,SAAS,OAAA;AAAc,wGAAA,YAAY,OAAA;AAG5C,0CAA0C;AAC1C,uCAA0D;AAAjD,oGAAA,gBAAgB,OAAY;AAErC,kEAAkE;AAClE,yCAAsE;AAEtE;;;GAGG;AACH,SAAgB,cAAc,CAAC,IAAY,EAAE,OAA0C;IACrF,OAAO,2BAAgB,CAAC,cAAc,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;AACxD,CAAC;AAED;;;GAGG;AACH,SAAgB,IAAI,CAAC,IAAY,EAAE,MAA+B;IAChE,OAAO,2BAAgB,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;AAC7C,CAAC;AAED,yBAAyB;AACzB,yCAQqB;AAPnB,sGAAA,SAAS,OAAA;AACT,uGAAA,UAAU,OAAA;AACV,2GAAA,cAAc,OAAA;AACd,mHAAA,sBAAsB,OAAA;AACtB,iHAAA,oBAAoB,OAAA;AACpB,oHAAA,uBAAuB,OAAA;AACvB,kHAAA,qBAAqB,OAAA;AASvB,qBAAqB;AACrB,iCAIiB;AAHf,sGAAA,aAAa,OAAA;AACb,kGAAA,SAAS,OAAA;AACT,oGAAA,WAAW,OAAA;AASb,0BAA0B;AAC1B,mDAE0B;AADxB,+GAAA,aAAa,OAAA;AAQf,2BAA2B;AAC3B,qDAK2B;AAJzB,sHAAA,mBAAmB,OAAA;AACnB,sHAAA,mBAAmB,OAAA;AACnB,qHAAA,kBAAkB,OAAA;AAClB,uHAAA,oBAAoB,OAAA;AAMtB,yBAAyB;AACzB,mCAKkB;AAJhB,4GAAA,kBAAkB,OAAA;AAClB,sGAAA,YAAY,OAAA;AACZ,yGAAA,eAAe,OAAA;AACf,oGAAA,UAAU,OAAA;AAgBZ,wCAAwC;AACxC,6BAKe;AAJb,gGAAA,SAAS,OAAA;AACT,gGAAA,SAAS,OAAA;AACT,+FAAA,QAAQ,OAAA;AACR,sGAAA,eAAe,OAAA;AAiBjB,0BAA0B;AAC1B,mCAEkB;AADhB,uGAAA,aAAa,OAAA;AAef,yBAAyB;AACzB,6CAA6C;AAApC,2GAAA,YAAY,OAAA;AAQrB,2CAAyC;AAAhC,uGAAA,SAAS,OAAA;AAElB,mBAAmB;AACnB,mCAMkB;AALhB,oGAAA,UAAU,OAAA;AACV,uGAAA,aAAa,OAAA;AACb,yGAAA,eAAe,OAAA;AACf,4GAAA,kBAAkB,OAAA;AAClB,+GAAA,qBAAqB,OAAA;AAGvB,mBAAmB;AACnB,iCAAgC;AAAvB,8FAAA,KAAK,OAAA;AAGd,mCAakB;AAZhB,qGAAA,WAAW,OAAA;AACX,yGAAA,eAAe,OAAA;AACf,0GAAA,gBAAgB,OAAA;AAChB,uGAAA,aAAa,OAAA;AACb,uGAAA,aAAa,OAAA;AACb,uBAAuB;AACvB,mGAAA,SAAS,OAAA;AACT,mGAAA,SAAS,OAAA;AACT,6GAAA,mBAAmB,OAAA;AACnB,0GAAA,gBAAgB,OAAA;AAChB,4GAAA,kBAAkB,OAAA;AAClB,yGAAA,eAAe,OAAA;AAGjB,oBAAoB;AACpB,6CAA2D;AAAlD,yGAAA,YAAY,OAAc","sourcesContent":["/**\n * SochDB Node.js SDK v0.5.1\n * \n * Dual-mode architecture: Embedded (FFI) + Server (gRPC/IPC)\n * \n * Architecture: Flexible Deployment\n * ==================================\n * This SDK supports BOTH modes:\n * \n * 1. Embedded Mode (FFI) - For single-process apps:\n *    - Direct FFI bindings to Rust libraries\n *    - No server required - just npm install and run\n *    - Best for: Local development, simple apps\n * \n * 2. Concurrent Mode (FFI) - For multi-process apps:\n *    - Same FFI bindings with MVCC for concurrent access\n *    - Multiple Node.js processes can access same database\n *    - Best for: PM2 cluster, Express workers\n * \n * 3. Server Mode (gRPC/IPC) - For distributed systems:\n *    - Thin client connecting to sochdb-grpc server\n *    - Best for: Production, multi-language, scalability\n * \n * @example Embedded Mode\n * ```typescript\n * import { Database } from '@sochdb/sochdb';\n * \n * // Direct FFI - no server needed\n * const db = await Database.open('./mydb');\n * await db.put(Buffer.from('key'), Buffer.from('value'));\n * await db.close();\n * ```\n * \n * @example Concurrent Mode\n * ```typescript\n * import { Database } from '@sochdb/sochdb';\n * \n * // Multiple processes can access simultaneously\n * const db = Database.openConcurrent('./mydb');\n * console.log(`Concurrent: ${db.isConcurrent}`);\n * ```\n * \n * @example Server Mode\n * ```typescript\n * import { SochDBClient } from '@sochdb/sochdb';\n * \n * // Connect to server\n * const client = new SochDBClient({ address: 'localhost:50051' });\n * await client.putKv('key', Buffer.from('value'));\n * ```\n */\n\n// Version\nexport const VERSION = '0.5.1';\n\n// Embedded mode (FFI) - NEW\nexport { EmbeddedDatabase, EmbeddedDatabaseConfig } from './embedded';\nexport { EmbeddedTransaction } from './embedded';\nexport { HnswIndex, HnswConfig, HnswBindings } from './embedded';\nexport { SearchResult as HnswSearchResult } from './embedded';\n\n// Embedded mode (FFI) - Convenience alias\nexport { EmbeddedDatabase as Database } from './embedded';\n\n// Embedded mode (FFI) - Convenience functions for concurrent mode\nimport { EmbeddedDatabase, EmbeddedDatabaseConfig } from './embedded';\n\n/**\n * Open a database in concurrent mode (multi-process safe).\n * Alias for EmbeddedDatabase.openConcurrent()\n */\nexport function openConcurrent(path: string, options?: { fallbackToStandard?: boolean }): EmbeddedDatabase {\n  return EmbeddedDatabase.openConcurrent(path, options);\n}\n\n/**\n * Open a database in standard mode.\n * Alias for EmbeddedDatabase.open()\n */\nexport function open(path: string, config?: EmbeddedDatabaseConfig): EmbeddedDatabase {\n  return EmbeddedDatabase.open(path, config);\n}\n\n// Namespace API (v0.4.1)\nexport {\n  Namespace,\n  Collection,\n  DistanceMetric,\n  NamespaceNotFoundError,\n  NamespaceExistsError,\n  CollectionNotFoundError,\n  CollectionExistsError,\n} from './namespace';\nexport type {\n  NamespaceConfig,\n  CollectionConfig,\n  SearchRequest,\n  SearchResult as NamespaceSearchResult,\n} from './namespace';\n\n// Queue API (v0.4.1)\nexport {\n  PriorityQueue,\n  TaskState,\n  createQueue,\n} from './queue';\nexport type {\n  QueueConfig,\n  QueueKey,\n  Task,\n  QueueStats,\n} from './queue';\n\n// Semantic Cache (v0.4.1)\nexport {\n  SemanticCache,\n} from './semantic-cache';\nexport type {\n  CacheEntry,\n  CacheHit,\n  CacheStats,\n} from './semantic-cache';\n\n// Context Builder (v0.4.1)\nexport {\n  ContextQueryBuilder,\n  ContextOutputFormat,\n  TruncationStrategy,\n  createContextBuilder,\n} from './context-builder';\nexport type {\n  ContextResult,\n} from './context-builder';\n\n// Memory System (v0.4.2)\nexport {\n  ExtractionPipeline,\n  Consolidator,\n  HybridRetriever,\n  AllowedSet,\n} from './memory';\nexport type {\n  Entity,\n  Relation,\n  Assertion,\n  RawAssertion,\n  CanonicalFact,\n  ExtractionResult,\n  ExtractionSchema,\n  ConsolidationConfig,\n  RetrievalConfig,\n  RetrievalResult,\n  RetrievalResponse,\n} from './memory';\n\n// MCP - Model Context Protocol (v0.4.3)\nexport {\n  McpServer,\n  McpClient,\n  McpError,\n  MCP_ERROR_CODES,\n} from './mcp';\nexport type {\n  McpTool,\n  McpToolCall,\n  McpToolResult,\n  McpResource,\n  McpResourceContent,\n  McpPrompt,\n  McpPromptArgument,\n  McpPromptMessage,\n  McpServerCapabilities,\n  McpServerConfig,\n  McpClientConfig,\n  McpTransport,\n} from './mcp';\n\n// Policy Service (v0.4.3)\nexport {\n  PolicyService,\n} from './policy';\nexport type {\n  PolicyRule,\n  PolicyCondition,\n  PolicyEvaluation,\n  NamespacePolicy,\n  NamespaceGrant,\n  NamespacePermission,\n  PolicyAction,\n  PolicyRequest,\n  PolicySet,\n  PolicyAuditEntry,\n} from './policy';\n\n// Server mode (gRPC/IPC)\nexport { SochDBClient } from './grpc-client';\nexport type {\n  SearchResult,\n  Document,\n  GraphNode,\n  GraphEdge,\n} from './grpc-client';\n\nexport { IpcClient } from './ipc-client';\n\n// Format utilities\nexport {\n  WireFormat,\n  ContextFormat,\n  CanonicalFormat,\n  FormatCapabilities,\n  FormatConversionError,\n} from './format';\n\n// Type definitions\nexport { Query } from './query';\nexport type { QueryResult } from './query';\n\nexport {\n  SochDBError,\n  ConnectionError,\n  TransactionError,\n  ProtocolError,\n  DatabaseError,\n  // Lock errors (v0.4.1)\n  ErrorCode,\n  LockError,\n  DatabaseLockedError,\n  LockTimeoutError,\n  EpochMismatchError,\n  SplitBrainError,\n} from './errors';\n\n// Convenience alias\nexport { SochDBClient as GrpcClient } from './grpc-client';\n"]}
155
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAkDG;;;AAqBH,wCAEC;AAMD,oBAEC;AA7BD,UAAU;AACG,QAAA,OAAO,GAAG,OAAO,CAAC;AAE/B,4BAA4B;AAC5B,uCAAsE;AAA7D,4GAAA,gBAAgB,OAAA;AACzB,uCAAiD;AAAxC,+GAAA,mBAAmB,OAAA;AAC5B,uCAAiE;AAAxD,qGAAA,SAAS,OAAA;AAAc,wGAAA,YAAY,OAAA;AAG5C,0CAA0C;AAC1C,uCAA0D;AAAjD,oGAAA,gBAAgB,OAAY;AAErC,kEAAkE;AAClE,yCAAsE;AAEtE;;;GAGG;AACH,SAAgB,cAAc,CAAC,IAAY,EAAE,OAA0C;IACrF,OAAO,2BAAgB,CAAC,cAAc,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;AACxD,CAAC;AAED;;;GAGG;AACH,SAAgB,IAAI,CAAC,IAAY,EAAE,MAA+B;IAChE,OAAO,2BAAgB,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;AAC7C,CAAC;AAED,yBAAyB;AACzB,yCAQqB;AAPnB,sGAAA,SAAS,OAAA;AACT,uGAAA,UAAU,OAAA;AACV,2GAAA,cAAc,OAAA;AACd,mHAAA,sBAAsB,OAAA;AACtB,iHAAA,oBAAoB,OAAA;AACpB,oHAAA,uBAAuB,OAAA;AACvB,kHAAA,qBAAqB,OAAA;AASvB,qBAAqB;AACrB,iCAIiB;AAHf,sGAAA,aAAa,OAAA;AACb,kGAAA,SAAS,OAAA;AACT,oGAAA,WAAW,OAAA;AASb,0BAA0B;AAC1B,mDAE0B;AADxB,+GAAA,aAAa,OAAA;AAQf,2BAA2B;AAC3B,qDAK2B;AAJzB,sHAAA,mBAAmB,OAAA;AACnB,sHAAA,mBAAmB,OAAA;AACnB,qHAAA,kBAAkB,OAAA;AAClB,uHAAA,oBAAoB,OAAA;AAMtB,yBAAyB;AACzB,mCAKkB;AAJhB,4GAAA,kBAAkB,OAAA;AAClB,sGAAA,YAAY,OAAA;AACZ,yGAAA,eAAe,OAAA;AACf,oGAAA,UAAU,OAAA;AAgBZ,wCAAwC;AACxC,6BAKe;AAJb,gGAAA,SAAS,OAAA;AACT,gGAAA,SAAS,OAAA;AACT,+FAAA,QAAQ,OAAA;AACR,sGAAA,eAAe,OAAA;AAiBjB,0BAA0B;AAC1B,mCAEkB;AADhB,uGAAA,aAAa,OAAA;AAef,yBAAyB;AACzB,6CAA6C;AAApC,2GAAA,YAAY,OAAA;AAQrB,2CAAyC;AAAhC,uGAAA,SAAS,OAAA;AAElB,mBAAmB;AACnB,mCAMkB;AALhB,oGAAA,UAAU,OAAA;AACV,uGAAA,aAAa,OAAA;AACb,yGAAA,eAAe,OAAA;AACf,4GAAA,kBAAkB,OAAA;AAClB,+GAAA,qBAAqB,OAAA;AAGvB,mBAAmB;AACnB,iCAAgC;AAAvB,8FAAA,KAAK,OAAA;AAGd,mCAakB;AAZhB,qGAAA,WAAW,OAAA;AACX,yGAAA,eAAe,OAAA;AACf,0GAAA,gBAAgB,OAAA;AAChB,uGAAA,aAAa,OAAA;AACb,uGAAA,aAAa,OAAA;AACb,uBAAuB;AACvB,mGAAA,SAAS,OAAA;AACT,mGAAA,SAAS,OAAA;AACT,6GAAA,mBAAmB,OAAA;AACnB,0GAAA,gBAAgB,OAAA;AAChB,4GAAA,kBAAkB,OAAA;AAClB,yGAAA,eAAe,OAAA;AAGjB,oBAAoB;AACpB,6CAA2D;AAAlD,yGAAA,YAAY,OAAc","sourcesContent":["/**\n * SochDB Node.js SDK v0.5.3\n * \n * Dual-mode architecture: Embedded (FFI) + Server (gRPC/IPC)\n * \n * Architecture: Flexible Deployment\n * ==================================\n * This SDK supports BOTH modes:\n * \n * 1. Embedded Mode (FFI) - For single-process apps:\n *    - Direct FFI bindings to Rust libraries\n *    - No server required - just npm install and run\n *    - Best for: Local development, simple apps\n * \n * 2. Concurrent Mode (FFI) - For multi-process apps:\n *    - Same FFI bindings with MVCC for concurrent access\n *    - Multiple Node.js processes can access same database\n *    - Best for: PM2 cluster, Express workers\n * \n * 3. Server Mode (gRPC/IPC) - For distributed systems:\n *    - Thin client connecting to sochdb-grpc server\n *    - Best for: Production, multi-language, scalability\n * \n * @example Embedded Mode\n * ```typescript\n * import { Database } from '@sochdb/sochdb';\n * \n * // Direct FFI - no server needed\n * const db = await Database.open('./mydb');\n * await db.put(Buffer.from('key'), Buffer.from('value'));\n * await db.close();\n * ```\n * \n * @example Concurrent Mode\n * ```typescript\n * import { Database } from '@sochdb/sochdb';\n * \n * // Multiple processes can access simultaneously\n * const db = Database.openConcurrent('./mydb');\n * console.log(`Concurrent: ${db.isConcurrent}`);\n * ```\n * \n * @example Server Mode\n * ```typescript\n * import { SochDBClient } from '@sochdb/sochdb';\n * \n * // Connect to server\n * const client = new SochDBClient({ address: 'localhost:50051' });\n * await client.putKv('key', Buffer.from('value'));\n * ```\n */\n\n// Version\nexport const VERSION = '0.5.3';\n\n// Embedded mode (FFI) - NEW\nexport { EmbeddedDatabase, EmbeddedDatabaseConfig } from './embedded';\nexport { EmbeddedTransaction } from './embedded';\nexport { HnswIndex, HnswConfig, HnswBindings } from './embedded';\nexport { SearchResult as HnswSearchResult } from './embedded';\n\n// Embedded mode (FFI) - Convenience alias\nexport { EmbeddedDatabase as Database } from './embedded';\n\n// Embedded mode (FFI) - Convenience functions for concurrent mode\nimport { EmbeddedDatabase, EmbeddedDatabaseConfig } from './embedded';\n\n/**\n * Open a database in concurrent mode (multi-process safe).\n * Alias for EmbeddedDatabase.openConcurrent()\n */\nexport function openConcurrent(path: string, options?: { fallbackToStandard?: boolean }): EmbeddedDatabase {\n  return EmbeddedDatabase.openConcurrent(path, options);\n}\n\n/**\n * Open a database in standard mode.\n * Alias for EmbeddedDatabase.open()\n */\nexport function open(path: string, config?: EmbeddedDatabaseConfig): EmbeddedDatabase {\n  return EmbeddedDatabase.open(path, config);\n}\n\n// Namespace API (v0.4.1)\nexport {\n  Namespace,\n  Collection,\n  DistanceMetric,\n  NamespaceNotFoundError,\n  NamespaceExistsError,\n  CollectionNotFoundError,\n  CollectionExistsError,\n} from './namespace';\nexport type {\n  NamespaceConfig,\n  CollectionConfig,\n  SearchRequest,\n  SearchResult as NamespaceSearchResult,\n} from './namespace';\n\n// Queue API (v0.4.1)\nexport {\n  PriorityQueue,\n  TaskState,\n  createQueue,\n} from './queue';\nexport type {\n  QueueConfig,\n  QueueKey,\n  Task,\n  QueueStats,\n} from './queue';\n\n// Semantic Cache (v0.4.1)\nexport {\n  SemanticCache,\n} from './semantic-cache';\nexport type {\n  CacheEntry,\n  CacheHit,\n  CacheStats,\n} from './semantic-cache';\n\n// Context Builder (v0.4.1)\nexport {\n  ContextQueryBuilder,\n  ContextOutputFormat,\n  TruncationStrategy,\n  createContextBuilder,\n} from './context-builder';\nexport type {\n  ContextResult,\n} from './context-builder';\n\n// Memory System (v0.4.2)\nexport {\n  ExtractionPipeline,\n  Consolidator,\n  HybridRetriever,\n  AllowedSet,\n} from './memory';\nexport type {\n  Entity,\n  Relation,\n  Assertion,\n  RawAssertion,\n  CanonicalFact,\n  ExtractionResult,\n  ExtractionSchema,\n  ConsolidationConfig,\n  RetrievalConfig,\n  RetrievalResult,\n  RetrievalResponse,\n} from './memory';\n\n// MCP - Model Context Protocol (v0.4.3)\nexport {\n  McpServer,\n  McpClient,\n  McpError,\n  MCP_ERROR_CODES,\n} from './mcp';\nexport type {\n  McpTool,\n  McpToolCall,\n  McpToolResult,\n  McpResource,\n  McpResourceContent,\n  McpPrompt,\n  McpPromptArgument,\n  McpPromptMessage,\n  McpServerCapabilities,\n  McpServerConfig,\n  McpClientConfig,\n  McpTransport,\n} from './mcp';\n\n// Policy Service (v0.4.3)\nexport {\n  PolicyService,\n} from './policy';\nexport type {\n  PolicyRule,\n  PolicyCondition,\n  PolicyEvaluation,\n  NamespacePolicy,\n  NamespaceGrant,\n  NamespacePermission,\n  PolicyAction,\n  PolicyRequest,\n  PolicySet,\n  PolicyAuditEntry,\n} from './policy';\n\n// Server mode (gRPC/IPC)\nexport { SochDBClient } from './grpc-client';\nexport type {\n  SearchResult,\n  Document,\n  GraphNode,\n  GraphEdge,\n} from './grpc-client';\n\nexport { IpcClient } from './ipc-client';\n\n// Format utilities\nexport {\n  WireFormat,\n  ContextFormat,\n  CanonicalFormat,\n  FormatCapabilities,\n  FormatConversionError,\n} from './format';\n\n// Type definitions\nexport { Query } from './query';\nexport type { QueryResult } from './query';\n\nexport {\n  SochDBError,\n  ConnectionError,\n  TransactionError,\n  ProtocolError,\n  DatabaseError,\n  // Lock errors (v0.4.1)\n  ErrorCode,\n  LockError,\n  DatabaseLockedError,\n  LockTimeoutError,\n  EpochMismatchError,\n  SplitBrainError,\n} from './errors';\n\n// Convenience alias\nexport { SochDBClient as GrpcClient } from './grpc-client';\n"]}