@sochdb/sochdb 0.4.2 → 0.4.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +356 -14
- package/_bin/aarch64-apple-darwin/libsochdb_storage.dylib +0 -0
- package/_bin/aarch64-apple-darwin/sochdb-bulk +0 -0
- package/_bin/aarch64-apple-darwin/sochdb-grpc-server +0 -0
- package/_bin/aarch64-apple-darwin/sochdb-server +0 -0
- package/_bin/x86_64-pc-windows-msvc/sochdb-bulk.exe +0 -0
- package/_bin/x86_64-pc-windows-msvc/sochdb-grpc-server.exe +0 -0
- package/_bin/x86_64-pc-windows-msvc/sochdb_storage.dll +0 -0
- package/_bin/x86_64-unknown-linux-gnu/libsochdb_storage.so +0 -0
- package/_bin/x86_64-unknown-linux-gnu/sochdb-bulk +0 -0
- package/_bin/x86_64-unknown-linux-gnu/sochdb-grpc-server +0 -0
- package/_bin/x86_64-unknown-linux-gnu/sochdb-server +0 -0
- package/dist/cjs/embedded/database.js +98 -4
- package/dist/cjs/embedded/ffi/bindings.js +46 -8
- package/dist/cjs/index.js +28 -6
- package/dist/cjs/mcp/client.js +115 -0
- package/dist/cjs/mcp/index.js +28 -0
- package/dist/cjs/mcp/server.js +242 -0
- package/dist/cjs/mcp/types.js +32 -0
- package/dist/cjs/namespace.js +147 -19
- package/dist/cjs/policy/index.js +26 -0
- package/dist/cjs/policy/service.js +394 -0
- package/dist/cjs/policy/types.js +8 -0
- package/dist/esm/embedded/database.js +98 -4
- package/dist/esm/embedded/ffi/bindings.js +48 -8
- package/dist/esm/index.js +28 -6
- package/dist/esm/mcp/client.js +116 -0
- package/dist/esm/mcp/index.js +28 -0
- package/dist/esm/mcp/server.js +244 -0
- package/dist/esm/mcp/types.js +34 -0
- package/dist/esm/namespace.js +150 -19
- package/dist/esm/policy/index.js +26 -0
- package/dist/esm/policy/service.js +396 -0
- package/dist/esm/policy/types.js +8 -0
- package/dist/types/embedded/database.d.ts +66 -1
- package/dist/types/embedded/database.d.ts.map +1 -1
- package/dist/types/embedded/ffi/bindings.d.ts +7 -0
- package/dist/types/embedded/ffi/bindings.d.ts.map +1 -1
- package/dist/types/index.d.ts +23 -5
- package/dist/types/index.d.ts.map +1 -1
- package/dist/types/mcp/client.d.ts +69 -0
- package/dist/types/mcp/client.d.ts.map +1 -0
- package/dist/types/mcp/index.d.ts +9 -0
- package/dist/types/mcp/index.d.ts.map +1 -0
- package/dist/types/mcp/server.d.ts +87 -0
- package/dist/types/mcp/server.d.ts.map +1 -0
- package/dist/types/mcp/types.d.ts +124 -0
- package/dist/types/mcp/types.d.ts.map +1 -0
- package/dist/types/namespace.d.ts +13 -0
- package/dist/types/namespace.d.ts.map +1 -1
- package/dist/types/policy/index.d.ts +8 -0
- package/dist/types/policy/index.d.ts.map +1 -0
- package/dist/types/policy/service.d.ts +115 -0
- package/dist/types/policy/service.d.ts.map +1 -0
- package/dist/types/policy/types.d.ts +102 -0
- package/dist/types/policy/types.d.ts.map +1 -0
- package/package.json +2 -2
package/README.md
CHANGED
|
@@ -1,8 +1,212 @@
|
|
|
1
1
|
# SochDB Node.js SDK
|
|
2
2
|
|
|
3
|
-
**
|
|
3
|
+
**LLM-Optimized Embedded Database with Native Vector Search**
|
|
4
|
+
|
|
5
|
+
---
|
|
6
|
+
|
|
7
|
+
## Installation
|
|
8
|
+
|
|
9
|
+
```bash
|
|
10
|
+
npm install @sochdb/sochdb
|
|
11
|
+
```
|
|
12
|
+
|
|
13
|
+
Or from source:
|
|
14
|
+
```bash
|
|
15
|
+
cd sochdb-typescript-sdk
|
|
16
|
+
npm install
|
|
17
|
+
```
|
|
18
|
+
|
|
19
|
+
---
|
|
20
|
+
|
|
21
|
+
## Architecture: Flexible Deployment
|
|
22
|
+
|
|
23
|
+
**Tri-mode architecture: Embedded + Concurrent + Server (gRPC/IPC)**
|
|
4
24
|
Choose the deployment mode that fits your needs.
|
|
5
25
|
|
|
26
|
+
---
|
|
27
|
+
|
|
28
|
+
# SochDB Node.js SDK Documentation
|
|
29
|
+
|
|
30
|
+
**LLM-Optimized Embedded Database with Native Vector Search**
|
|
31
|
+
|
|
32
|
+
---
|
|
33
|
+
|
|
34
|
+
## Table of Contents
|
|
35
|
+
|
|
36
|
+
1. [Quick Start](#1-quick-start)
|
|
37
|
+
2. [Installation](#2-installation)
|
|
38
|
+
3. [Features](#3-features)
|
|
39
|
+
- [Namespace API](#namespace-api---multi-tenant-isolation)
|
|
40
|
+
- [Priority Queue API](#priority-queue-api---task-processing)
|
|
41
|
+
4. [Architecture Overview](#4-architecture-overview)
|
|
42
|
+
5. [Core Key-Value Operations](#5-core-key-value-operations)
|
|
43
|
+
6. [Transactions (ACID with SSI)](#6-transactions-acid-with-ssi)
|
|
44
|
+
7. [Query Builder](#7-query-builder)
|
|
45
|
+
8. [Prefix Scanning](#8-prefix-scanning)
|
|
46
|
+
9. [SQL Operations](#9-sql-operations)
|
|
47
|
+
10. [Table Management & Index Policies](#10-table-management--index-policies)
|
|
48
|
+
11. [Namespaces & Collections](#11-namespaces--collections)
|
|
49
|
+
12. [Priority Queues](#12-priority-queues)
|
|
50
|
+
13. [Vector Search](#13-vector-search)
|
|
51
|
+
14. [Hybrid Search (Vector + BM25)](#14-hybrid-search-vector--bm25)
|
|
52
|
+
15. [Graph Operations](#15-graph-operations)
|
|
53
|
+
16. [Temporal Graph (Time-Travel)](#16-temporal-graph-time-travel)
|
|
54
|
+
17. [Semantic Cache](#17-semantic-cache)
|
|
55
|
+
18. [Memory System](#18-memory-system)
|
|
56
|
+
19. [Session Management](#19-session-management)
|
|
57
|
+
20. [Context Query Builder (LLM Optimization)](#20-context-query-builder-llm-optimization)
|
|
58
|
+
21. [Atomic Multi-Index Writes](#21-atomic-multi-index-writes)
|
|
59
|
+
22. [Recovery & WAL Management](#22-recovery--wal-management)
|
|
60
|
+
23. [Checkpoints & Snapshots](#23-checkpoints--snapshots)
|
|
61
|
+
24. [Compression & Storage](#24-compression--storage)
|
|
62
|
+
25. [Statistics & Monitoring](#25-statistics--monitoring)
|
|
63
|
+
26. [Distributed Tracing](#26-distributed-tracing)
|
|
64
|
+
27. [Workflow & Run Tracking](#27-workflow--run-tracking)
|
|
65
|
+
28. [Server Mode (gRPC Client)](#28-server-mode-grpc-client)
|
|
66
|
+
29. [IPC Client (Unix Sockets)](#29-ipc-client-unix-sockets)
|
|
67
|
+
30. [Standalone VectorIndex](#30-standalone-vectorindex)
|
|
68
|
+
31. [Vector Utilities](#31-vector-utilities)
|
|
69
|
+
32. [Data Formats (TOON/JSON/Columnar)](#32-data-formats-toonjsoncolumnar)
|
|
70
|
+
33. [Policy Service](#33-policy-service)
|
|
71
|
+
34. [MCP (Model Context Protocol)](#34-mcp-model-context-protocol)
|
|
72
|
+
35. [Configuration Reference](#35-configuration-reference)
|
|
73
|
+
36. [Error Handling](#36-error-handling)
|
|
74
|
+
37. [Async Support](#37-async-support)
|
|
75
|
+
38. [Building & Development](#38-building--development)
|
|
76
|
+
39. [Complete Examples](#39-complete-examples)
|
|
77
|
+
40. [Migration Guide](#40-migration-guide)
|
|
78
|
+
|
|
79
|
+
---
|
|
80
|
+
|
|
81
|
+
## 1. Quick Start
|
|
82
|
+
|
|
83
|
+
### Concurrent Embedded Mode
|
|
84
|
+
|
|
85
|
+
For web applications with multiple Node.js processes (PM2 cluster, multiple workers):
|
|
86
|
+
|
|
87
|
+
```typescript
|
|
88
|
+
import { EmbeddedDatabase } from '@sochdb/sochdb';
|
|
89
|
+
import express from 'express';
|
|
90
|
+
|
|
91
|
+
// Open in concurrent mode - multiple processes can access simultaneously
|
|
92
|
+
const db = EmbeddedDatabase.openConcurrent('./web_db');
|
|
93
|
+
|
|
94
|
+
const app = express();
|
|
95
|
+
|
|
96
|
+
app.get('/user/:id', async (req, res) => {
|
|
97
|
+
// Multiple concurrent requests can read simultaneously (~100ns)
|
|
98
|
+
const data = await db.get(Buffer.from(`user:${req.params.id}`));
|
|
99
|
+
if (!data) {
|
|
100
|
+
res.status(404).json({ error: 'not found' });
|
|
101
|
+
return;
|
|
102
|
+
}
|
|
103
|
+
res.send(data);
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
app.post('/user/:id', async (req, res) => {
|
|
107
|
+
// Writes are automatically coordinated (~60µs amortized)
|
|
108
|
+
await db.put(Buffer.from(`user:${req.params.id}`), req.body);
|
|
109
|
+
res.json({ status: 'ok' });
|
|
110
|
+
});
|
|
111
|
+
|
|
112
|
+
// Check concurrent mode status
|
|
113
|
+
console.log(`Concurrent mode: ${db.isConcurrent}`); // true
|
|
114
|
+
|
|
115
|
+
// Start with PM2 cluster mode (multiple workers can access DB)
|
|
116
|
+
// pm2 start app.js -i max
|
|
117
|
+
app.listen(3000);
|
|
118
|
+
```
|
|
119
|
+
|
|
120
|
+
### Performance
|
|
121
|
+
|
|
122
|
+
| Operation | Standard Mode | Concurrent Mode |
|
|
123
|
+
|-----------|---------------|-----------------|
|
|
124
|
+
| Read (single process) | ~100ns | ~100ns |
|
|
125
|
+
| Read (multi-process) | **Blocked** ❌ | ~100ns ✅ |
|
|
126
|
+
| Write | ~5ms (fsync) | ~60µs (amortized) |
|
|
127
|
+
| Max concurrent readers | 1 | 1024 |
|
|
128
|
+
|
|
129
|
+
### PM2 Cluster Example
|
|
130
|
+
|
|
131
|
+
```bash
|
|
132
|
+
# Install PM2
|
|
133
|
+
npm install -g pm2
|
|
134
|
+
|
|
135
|
+
# Start with automatic worker scaling
|
|
136
|
+
pm2 start server.js -i max
|
|
137
|
+
|
|
138
|
+
# All workers can access the same database concurrently!
|
|
139
|
+
pm2 logs
|
|
140
|
+
```
|
|
141
|
+
|
|
142
|
+
### PM2 Ecosystem File
|
|
143
|
+
|
|
144
|
+
```javascript
|
|
145
|
+
// ecosystem.config.js
|
|
146
|
+
module.exports = {
|
|
147
|
+
apps: [{
|
|
148
|
+
name: 'api-server',
|
|
149
|
+
script: './server.js',
|
|
150
|
+
instances: 'max', // Scale across all CPU cores
|
|
151
|
+
exec_mode: 'cluster',
|
|
152
|
+
env: {
|
|
153
|
+
NODE_ENV: 'production',
|
|
154
|
+
DB_PATH: './shared_db' // All workers use same DB
|
|
155
|
+
}
|
|
156
|
+
}]
|
|
157
|
+
};
|
|
158
|
+
```
|
|
159
|
+
|
|
160
|
+
```bash
|
|
161
|
+
# Deploy with ecosystem file
|
|
162
|
+
pm2 start ecosystem.config.js
|
|
163
|
+
|
|
164
|
+
# Monitor all workers
|
|
165
|
+
pm2 monit
|
|
166
|
+
```
|
|
167
|
+
|
|
168
|
+
### Docker Compose with PM2
|
|
169
|
+
|
|
170
|
+
```yaml
|
|
171
|
+
version: '3.8'
|
|
172
|
+
services:
|
|
173
|
+
app:
|
|
174
|
+
build: .
|
|
175
|
+
environment:
|
|
176
|
+
- NODE_ENV=production
|
|
177
|
+
- INSTANCES=4 # 4 PM2 workers
|
|
178
|
+
volumes:
|
|
179
|
+
- ./data:/app/data # Shared database volume
|
|
180
|
+
ports:
|
|
181
|
+
- "3000:3000"
|
|
182
|
+
command: pm2-runtime start ecosystem.config.js
|
|
183
|
+
```
|
|
184
|
+
|
|
185
|
+
### Kubernetes Deployment
|
|
186
|
+
|
|
187
|
+
```yaml
|
|
188
|
+
apiVersion: apps/v1
|
|
189
|
+
kind: Deployment
|
|
190
|
+
metadata:
|
|
191
|
+
name: sochdb-app
|
|
192
|
+
spec:
|
|
193
|
+
replicas: 4 # 4 pods share the database
|
|
194
|
+
template:
|
|
195
|
+
spec:
|
|
196
|
+
containers:
|
|
197
|
+
- name: app
|
|
198
|
+
image: myapp:latest
|
|
199
|
+
volumeMounts:
|
|
200
|
+
- name: db-storage
|
|
201
|
+
mountPath: /app/data
|
|
202
|
+
volumes:
|
|
203
|
+
- name: db-storage
|
|
204
|
+
persistentVolumeClaim:
|
|
205
|
+
claimName: sochdb-pvc # Shared PVC with ReadWriteMany
|
|
206
|
+
```
|
|
207
|
+
|
|
208
|
+
---
|
|
209
|
+
|
|
6
210
|
## Features
|
|
7
211
|
|
|
8
212
|
### Memory System - LLM-Native Memory for AI Agents
|
|
@@ -239,21 +443,161 @@ console.log(`Pending: ${stats.pending}, Completed: ${stats.completed}`);
|
|
|
239
443
|
|
|
240
444
|
---
|
|
241
445
|
|
|
242
|
-
|
|
446
|
+
---
|
|
447
|
+
|
|
448
|
+
## System Requirements
|
|
449
|
+
|
|
450
|
+
### For Concurrent Mode
|
|
451
|
+
|
|
452
|
+
- **SochDB Core**: Latest version
|
|
453
|
+
- **Node.js**: 14.0+ (18.0+ recommended)
|
|
454
|
+
- **Native Library**: `libsochdb_storage.{dylib,so}`
|
|
455
|
+
- **FFI**: Koffi (automatically installed)
|
|
456
|
+
|
|
457
|
+
**Operating Systems:**
|
|
458
|
+
- ✅ Linux (Ubuntu 20.04+, RHEL 8+)
|
|
459
|
+
- ✅ macOS (10.15+, both Intel and Apple Silicon)
|
|
460
|
+
- ⚠️ Windows (requires native builds)
|
|
461
|
+
|
|
462
|
+
**File Descriptors:**
|
|
463
|
+
- Default limit: 1024 (sufficient for most workloads)
|
|
464
|
+
- For high concurrency with PM2: `ulimit -n 4096`
|
|
465
|
+
|
|
466
|
+
**Memory:**
|
|
467
|
+
- Standard mode: ~50MB base + data
|
|
468
|
+
- Concurrent mode: +4KB per concurrent reader slot (1024 slots = ~4MB overhead)
|
|
469
|
+
- PM2 cluster: Each worker has independent memory
|
|
470
|
+
|
|
471
|
+
---
|
|
472
|
+
|
|
473
|
+
## Troubleshooting
|
|
474
|
+
|
|
475
|
+
### "Database is locked" Error (Standard Mode)
|
|
476
|
+
|
|
477
|
+
```
|
|
478
|
+
Error: SQLITE_BUSY: database is locked
|
|
479
|
+
```
|
|
480
|
+
|
|
481
|
+
**Solution**: Use concurrent mode for multi-process access:
|
|
482
|
+
|
|
483
|
+
```typescript
|
|
484
|
+
// ❌ Standard mode - PM2 cluster will fail
|
|
485
|
+
const db = new EmbeddedDatabase('./data.db');
|
|
243
486
|
|
|
487
|
+
// ✅ Concurrent mode - PM2 cluster works!
|
|
488
|
+
const db = EmbeddedDatabase.openConcurrent('./data.db');
|
|
489
|
+
```
|
|
490
|
+
|
|
491
|
+
### Library Not Found Error
|
|
492
|
+
|
|
493
|
+
```
|
|
494
|
+
Error: Dynamic library 'libsochdb_storage.dylib' not found
|
|
495
|
+
```
|
|
496
|
+
|
|
497
|
+
**macOS**:
|
|
244
498
|
```bash
|
|
245
|
-
|
|
499
|
+
# Build and install library
|
|
500
|
+
cd /path/to/sochdb
|
|
501
|
+
cargo build --release
|
|
502
|
+
sudo cp target/release/libsochdb_storage.dylib /usr/local/lib/
|
|
246
503
|
```
|
|
247
504
|
|
|
248
|
-
|
|
505
|
+
**Linux**:
|
|
249
506
|
```bash
|
|
250
|
-
cd sochdb
|
|
251
|
-
|
|
507
|
+
cd /path/to/sochdb
|
|
508
|
+
cargo build --release
|
|
509
|
+
sudo cp target/release/libsochdb_storage.so /usr/local/lib/
|
|
510
|
+
sudo ldconfig
|
|
511
|
+
```
|
|
512
|
+
|
|
513
|
+
**Development Mode** (no install):
|
|
514
|
+
```bash
|
|
515
|
+
export DYLD_LIBRARY_PATH=/path/to/sochdb/target/release # macOS
|
|
516
|
+
export LD_LIBRARY_PATH=/path/to/sochdb/target/release # Linux
|
|
517
|
+
```
|
|
518
|
+
|
|
519
|
+
### PM2 Cluster Issues
|
|
520
|
+
|
|
521
|
+
**Symptom**: Workers crash with "database locked"
|
|
522
|
+
|
|
523
|
+
**Solution**: Ensure concurrent mode is used:
|
|
524
|
+
```javascript
|
|
525
|
+
// ecosystem.config.js
|
|
526
|
+
module.exports = {
|
|
527
|
+
apps: [{
|
|
528
|
+
name: 'api',
|
|
529
|
+
script: './server.js',
|
|
530
|
+
instances: 4,
|
|
531
|
+
exec_mode: 'cluster',
|
|
532
|
+
env: {
|
|
533
|
+
USE_CONCURRENT_MODE: 'true' // Flag to use openConcurrent()
|
|
534
|
+
}
|
|
535
|
+
}]
|
|
536
|
+
};
|
|
537
|
+
```
|
|
538
|
+
|
|
539
|
+
```typescript
|
|
540
|
+
// server.ts
|
|
541
|
+
const db = process.env.USE_CONCURRENT_MODE
|
|
542
|
+
? EmbeddedDatabase.openConcurrent('./db')
|
|
543
|
+
: new EmbeddedDatabase('./db');
|
|
544
|
+
|
|
545
|
+
console.log('Concurrent mode:', db.isConcurrent); // Should be true
|
|
546
|
+
```
|
|
547
|
+
|
|
548
|
+
### Docker Volume Permissions
|
|
549
|
+
|
|
550
|
+
**Symptom**: `EACCES: permission denied` when opening database
|
|
551
|
+
|
|
552
|
+
**Solution**: Fix volume ownership:
|
|
553
|
+
```dockerfile
|
|
554
|
+
FROM node:18
|
|
555
|
+
WORKDIR /app
|
|
556
|
+
|
|
557
|
+
# Create data directory with correct permissions
|
|
558
|
+
RUN mkdir -p /app/data && chown -R node:node /app
|
|
559
|
+
|
|
560
|
+
# Switch to non-root user
|
|
561
|
+
USER node
|
|
562
|
+
|
|
563
|
+
COPY --chown=node:node . .
|
|
564
|
+
RUN npm install
|
|
565
|
+
|
|
566
|
+
CMD ["npm", "start"]
|
|
567
|
+
```
|
|
568
|
+
|
|
569
|
+
### Performance Issues
|
|
570
|
+
|
|
571
|
+
**Symptom**: Concurrent reads slower than expected
|
|
572
|
+
|
|
573
|
+
**Check 1** - Verify concurrent mode:
|
|
574
|
+
```typescript
|
|
575
|
+
if (!db.isConcurrent) {
|
|
576
|
+
console.error('Database is not in concurrent mode!');
|
|
577
|
+
process.exit(1);
|
|
578
|
+
}
|
|
579
|
+
```
|
|
580
|
+
|
|
581
|
+
**Check 2** - Monitor PM2 workers:
|
|
582
|
+
```bash
|
|
583
|
+
pm2 monit # Real-time monitoring
|
|
584
|
+
pm2 logs --lines 200 # Check for errors
|
|
585
|
+
```
|
|
586
|
+
|
|
587
|
+
**Check 3** - Batch writes:
|
|
588
|
+
```typescript
|
|
589
|
+
// ❌ Slow - individual writes
|
|
590
|
+
for (const item of items) {
|
|
591
|
+
await collection.insert(item);
|
|
592
|
+
}
|
|
593
|
+
|
|
594
|
+
// ✅ Fast - batch write
|
|
595
|
+
await collection.insertBatch(items);
|
|
252
596
|
```
|
|
253
597
|
|
|
254
598
|
---
|
|
255
599
|
|
|
256
|
-
## 🆕 Vector Search - Native HNSW
|
|
600
|
+
## 🆕 Vector Search - Native HNSW
|
|
257
601
|
|
|
258
602
|
SochDB now includes **native HNSW (Hierarchical Navigable Small World)** vector search for sub-millisecond similarity search across millions of vectors.
|
|
259
603
|
|
|
@@ -337,8 +681,6 @@ const results = await collection.search({ queryVector: [...], k: 10 });
|
|
|
337
681
|
- Direct HNSW API (above), OR
|
|
338
682
|
- gRPC Server Mode (see below)
|
|
339
683
|
|
|
340
|
-
**✅ Coming in v0.4.3:** Collection API will automatically use native HNSW
|
|
341
|
-
|
|
342
684
|
#### 3. gRPC Server Mode (Production-Ready)
|
|
343
685
|
|
|
344
686
|
For distributed systems, multi-language support:
|
|
@@ -440,15 +782,15 @@ class HnswIndex {
|
|
|
440
782
|
|
|
441
783
|
### Roadmap
|
|
442
784
|
|
|
443
|
-
- **
|
|
444
|
-
- **
|
|
445
|
-
- **
|
|
785
|
+
- **Current**: Direct HNSW FFI bindings
|
|
786
|
+
- **Next**: Collection API auto-uses HNSW in embedded mode
|
|
787
|
+
- **Future**: Persistent HNSW indexes with disk storage
|
|
446
788
|
|
|
447
789
|
---
|
|
448
790
|
|
|
449
791
|
# SochDB Node.js SDK Documentation
|
|
450
792
|
|
|
451
|
-
**
|
|
793
|
+
**LLM-Optimized Embedded Database with Native Vector Search**
|
|
452
794
|
|
|
453
795
|
---
|
|
454
796
|
|
|
@@ -3679,7 +4021,7 @@ A:
|
|
|
3679
4021
|
A: Yes! Both modes have the same API. Change `Database.open()` to `SochDBClient()` and vice versa.
|
|
3680
4022
|
|
|
3681
4023
|
**Q: Do temporal graphs work in embedded mode?**
|
|
3682
|
-
A: Yes!
|
|
4024
|
+
A: Yes! Temporal graphs work in both embedded and server modes with identical APIs.
|
|
3683
4025
|
|
|
3684
4026
|
**Q: Is embedded mode slower than server mode?**
|
|
3685
4027
|
A: Embedded mode is faster for single-process use (no network overhead). Server mode is better for distributed deployments.
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
@@ -24,14 +24,20 @@ const transaction_1 = require("./transaction");
|
|
|
24
24
|
* ```
|
|
25
25
|
*/
|
|
26
26
|
class EmbeddedDatabase {
|
|
27
|
-
constructor(path, handle) {
|
|
27
|
+
constructor(path, handle, concurrent = false, fallback = false) {
|
|
28
28
|
this.closed = false;
|
|
29
|
+
this.concurrent = false;
|
|
30
|
+
this._concurrentModeFallback = false;
|
|
29
31
|
this.path = path;
|
|
30
32
|
this.handle = handle;
|
|
33
|
+
this.concurrent = concurrent;
|
|
34
|
+
this._concurrentModeFallback = fallback;
|
|
31
35
|
this.bindings = bindings_1.NativeBindings.getInstance();
|
|
32
36
|
}
|
|
33
37
|
/**
|
|
34
|
-
* Open a database at the specified path
|
|
38
|
+
* Open a database at the specified path in standard mode
|
|
39
|
+
*
|
|
40
|
+
* For web applications with multiple processes, use `openConcurrent()` instead.
|
|
35
41
|
*
|
|
36
42
|
* @param path - Path to database directory
|
|
37
43
|
* @param config - Optional configuration
|
|
@@ -60,7 +66,95 @@ class EmbeddedDatabase {
|
|
|
60
66
|
if (!handle) {
|
|
61
67
|
throw new errors_1.DatabaseError(`Failed to open database at ${path}`);
|
|
62
68
|
}
|
|
63
|
-
return new EmbeddedDatabase(path, handle);
|
|
69
|
+
return new EmbeddedDatabase(path, handle, false);
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Open a database in concurrent mode for multi-process web applications
|
|
73
|
+
*
|
|
74
|
+
* This mode allows multiple Node.js processes (e.g., PM2 cluster workers,
|
|
75
|
+
* multiple Express instances) to access the database simultaneously.
|
|
76
|
+
*
|
|
77
|
+
* Features:
|
|
78
|
+
* - Lock-free reads with ~100ns latency
|
|
79
|
+
* - Multi-reader, single-writer coordination
|
|
80
|
+
* - Automatic write serialization
|
|
81
|
+
*
|
|
82
|
+
* @example
|
|
83
|
+
* ```typescript
|
|
84
|
+
* import { EmbeddedDatabase } from '@sochdb/sochdb';
|
|
85
|
+
* import express from 'express';
|
|
86
|
+
*
|
|
87
|
+
* // Open in concurrent mode - multiple workers can access
|
|
88
|
+
* const db = EmbeddedDatabase.openConcurrent('./web_db');
|
|
89
|
+
*
|
|
90
|
+
* const app = express();
|
|
91
|
+
*
|
|
92
|
+
* app.get('/user/:id', async (req, res) => {
|
|
93
|
+
* // Multiple concurrent requests can read simultaneously
|
|
94
|
+
* const data = await db.get(Buffer.from(`user:${req.params.id}`));
|
|
95
|
+
* if (!data) {
|
|
96
|
+
* res.status(404).json({ error: 'not found' });
|
|
97
|
+
* return;
|
|
98
|
+
* }
|
|
99
|
+
* res.send(data);
|
|
100
|
+
* });
|
|
101
|
+
*
|
|
102
|
+
* app.post('/user/:id', async (req, res) => {
|
|
103
|
+
* // Writes are serialized automatically
|
|
104
|
+
* await db.put(Buffer.from(`user:${req.params.id}`), req.body);
|
|
105
|
+
* res.json({ status: 'ok' });
|
|
106
|
+
* });
|
|
107
|
+
*
|
|
108
|
+
* // Start with PM2 cluster mode:
|
|
109
|
+
* // pm2 start app.js -i max
|
|
110
|
+
* app.listen(3000);
|
|
111
|
+
* ```
|
|
112
|
+
*
|
|
113
|
+
* @param path - Path to database directory
|
|
114
|
+
* @param options - Optional configuration for concurrent mode
|
|
115
|
+
* @returns EmbeddedDatabase instance in concurrent mode
|
|
116
|
+
*/
|
|
117
|
+
static openConcurrent(path, options) {
|
|
118
|
+
const bindings = bindings_1.NativeBindings.getInstance();
|
|
119
|
+
const fallbackToStandard = options?.fallbackToStandard ?? false;
|
|
120
|
+
if (!bindings.isConcurrentModeAvailable()) {
|
|
121
|
+
if (fallbackToStandard) {
|
|
122
|
+
console.warn('[SochDB] Concurrent mode not available in native library (requires v0.4.8+). ' +
|
|
123
|
+
'Falling back to standard mode. For production multi-process deployments, ' +
|
|
124
|
+
'please upgrade the SochDB native library.');
|
|
125
|
+
const handle = bindings.sochdb_open(path);
|
|
126
|
+
if (!handle) {
|
|
127
|
+
throw new errors_1.DatabaseError(`Failed to open database at ${path}`);
|
|
128
|
+
}
|
|
129
|
+
return new EmbeddedDatabase(path, handle, false, true);
|
|
130
|
+
}
|
|
131
|
+
throw new errors_1.DatabaseError('Concurrent mode not supported. Please upgrade the SochDB native library to v0.4.8+ ' +
|
|
132
|
+
'or use openConcurrent(path, { fallbackToStandard: true }) to fall back to standard mode.');
|
|
133
|
+
}
|
|
134
|
+
const handle = bindings.sochdb_open_concurrent(path);
|
|
135
|
+
if (!handle) {
|
|
136
|
+
throw new errors_1.DatabaseError(`Failed to open database in concurrent mode at ${path}`);
|
|
137
|
+
}
|
|
138
|
+
const isConcurrent = bindings.sochdb_is_concurrent?.(handle) === 1;
|
|
139
|
+
return new EmbeddedDatabase(path, handle, isConcurrent, false);
|
|
140
|
+
}
|
|
141
|
+
/**
|
|
142
|
+
* Check if database is opened in concurrent mode
|
|
143
|
+
*/
|
|
144
|
+
get isConcurrent() {
|
|
145
|
+
return this.concurrent;
|
|
146
|
+
}
|
|
147
|
+
/**
|
|
148
|
+
* Check if concurrent mode fell back to standard mode
|
|
149
|
+
*/
|
|
150
|
+
get isConcurrentFallback() {
|
|
151
|
+
return this._concurrentModeFallback;
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* Check if concurrent mode is available in the native library
|
|
155
|
+
*/
|
|
156
|
+
static isConcurrentModeAvailable() {
|
|
157
|
+
return bindings_1.NativeBindings.getInstance().isConcurrentModeAvailable();
|
|
64
158
|
}
|
|
65
159
|
/**
|
|
66
160
|
* Put a key-value pair (auto-transaction)
|
|
@@ -233,4 +327,4 @@ class EmbeddedDatabase {
|
|
|
233
327
|
}
|
|
234
328
|
}
|
|
235
329
|
exports.EmbeddedDatabase = EmbeddedDatabase;
|
|
236
|
-
//# 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\n    private constructor(path: string, handle: any) {\n        this.path = path;\n        this.handle = handle;\n        this.bindings = NativeBindings.getInstance();\n    }\n\n    /**\n     * Open a database at the specified path\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);\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"]}
|
|
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"]}
|