@sochdb/sochdb 0.4.0 → 0.4.1
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 +220 -33
- 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/bin/sochdb-bulk.js +1 -1
- package/bin/sochdb-grpc-server.js +1 -1
- package/bin/sochdb-server.js +1 -1
- package/dist/cjs/context-builder.js +280 -0
- package/dist/cjs/database.js +2 -2
- package/dist/cjs/embedded/database.js +2 -2
- package/dist/cjs/errors.js +99 -7
- package/dist/cjs/index.js +40 -3
- package/dist/cjs/ipc-client.js +2 -2
- package/dist/cjs/memory/consolidation.js +202 -0
- package/dist/cjs/memory/extraction.js +181 -0
- package/dist/cjs/memory/index.js +26 -0
- package/dist/cjs/memory/retrieval.js +232 -0
- package/dist/cjs/memory/types.js +69 -0
- package/dist/cjs/namespace.js +255 -0
- package/dist/cjs/queue.js +289 -0
- package/dist/cjs/semantic-cache.js +220 -0
- package/dist/esm/context-builder.js +280 -0
- package/dist/esm/database.js +2 -2
- package/dist/esm/embedded/database.js +2 -2
- package/dist/esm/errors.js +107 -7
- package/dist/esm/index.js +40 -3
- package/dist/esm/ipc-client.js +2 -2
- package/dist/esm/memory/consolidation.js +206 -0
- package/dist/esm/memory/extraction.js +185 -0
- package/dist/esm/memory/index.js +26 -0
- package/dist/esm/memory/retrieval.js +243 -0
- package/dist/esm/memory/types.js +72 -0
- package/dist/esm/namespace.js +262 -0
- package/dist/esm/queue.js +291 -0
- package/dist/esm/semantic-cache.js +223 -0
- package/dist/types/context-builder.d.ts +97 -0
- package/dist/types/context-builder.d.ts.map +1 -0
- package/dist/types/database.d.ts +1 -1
- package/dist/types/embedded/database.d.ts +1 -1
- package/dist/types/errors.d.ts +57 -1
- package/dist/types/errors.d.ts.map +1 -1
- package/dist/types/index.d.ts +12 -2
- package/dist/types/index.d.ts.map +1 -1
- package/dist/types/ipc-client.d.ts +1 -1
- package/dist/types/memory/consolidation.d.ts +66 -0
- package/dist/types/memory/consolidation.d.ts.map +1 -0
- package/dist/types/memory/extraction.d.ts +82 -0
- package/dist/types/memory/extraction.d.ts.map +1 -0
- package/dist/types/memory/index.d.ts +10 -0
- package/dist/types/memory/index.d.ts.map +1 -0
- package/dist/types/memory/retrieval.d.ts +46 -0
- package/dist/types/memory/retrieval.d.ts.map +1 -0
- package/dist/types/memory/types.d.ts +147 -0
- package/dist/types/memory/types.d.ts.map +1 -0
- package/dist/types/namespace.d.ts +129 -0
- package/dist/types/namespace.d.ts.map +1 -0
- package/dist/types/queue.d.ts +120 -0
- package/dist/types/queue.d.ts.map +1 -0
- package/dist/types/semantic-cache.d.ts +84 -0
- package/dist/types/semantic-cache.d.ts.map +1 -0
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -1,8 +1,202 @@
|
|
|
1
|
-
# SochDB Node.js SDK
|
|
1
|
+
# SochDB Node.js SDK
|
|
2
2
|
|
|
3
3
|
**Dual-mode architecture: Embedded (FFI) + Server (gRPC/IPC)**
|
|
4
4
|
Choose the deployment mode that fits your needs.
|
|
5
5
|
|
|
6
|
+
## Features
|
|
7
|
+
|
|
8
|
+
### Memory System - LLM-Native Memory for AI Agents
|
|
9
|
+
Complete memory system with extraction, consolidation, and hybrid retrieval:
|
|
10
|
+
|
|
11
|
+
```typescript
|
|
12
|
+
import {
|
|
13
|
+
EmbeddedDatabase,
|
|
14
|
+
ExtractionPipeline,
|
|
15
|
+
Consolidator,
|
|
16
|
+
HybridRetriever,
|
|
17
|
+
AllowedSet,
|
|
18
|
+
} from '@sochdb/sochdb';
|
|
19
|
+
|
|
20
|
+
const db = await EmbeddedDatabase.open('./memory_db');
|
|
21
|
+
|
|
22
|
+
// Extract entities and relations from text
|
|
23
|
+
const pipeline = ExtractionPipeline.fromDatabase(db, 'user_123', {
|
|
24
|
+
entityTypes: ['person', 'organization', 'location'],
|
|
25
|
+
minConfidence: 0.7,
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
const result = await pipeline.extractAndCommit(
|
|
29
|
+
'Alice works at Acme Corp',
|
|
30
|
+
myLLMExtractor // Your LLM integration
|
|
31
|
+
);
|
|
32
|
+
console.log(`Extracted ${result.entities.length} entities`);
|
|
33
|
+
|
|
34
|
+
// Consolidate facts with event sourcing
|
|
35
|
+
const consolidator = Consolidator.fromDatabase(db, 'user_123');
|
|
36
|
+
await consolidator.add({
|
|
37
|
+
fact: { subject: 'Alice', predicate: 'lives_in', object: 'SF' },
|
|
38
|
+
source: 'conversation_1',
|
|
39
|
+
confidence: 0.9,
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
const updated = await consolidator.consolidate();
|
|
43
|
+
const facts = await consolidator.getCanonicalFacts();
|
|
44
|
+
|
|
45
|
+
// Hybrid retrieval with RRF fusion
|
|
46
|
+
const retriever = HybridRetriever.fromDatabase(db, 'user_123', 'documents');
|
|
47
|
+
await retriever.indexDocuments(docs);
|
|
48
|
+
|
|
49
|
+
const results = await retriever.retrieve(
|
|
50
|
+
'machine learning papers',
|
|
51
|
+
queryEmbedding,
|
|
52
|
+
AllowedSet.fromNamespace('user_123')
|
|
53
|
+
);
|
|
54
|
+
```
|
|
55
|
+
|
|
56
|
+
**[→ See Full Example](./examples/memory-system-example.ts)**
|
|
57
|
+
|
|
58
|
+
**Key Features:**
|
|
59
|
+
- ✅ Extraction Pipeline: Compile LLM outputs into typed facts
|
|
60
|
+
- ✅ Event-Sourced Consolidation: Append-only with temporal updates
|
|
61
|
+
- ✅ Hybrid Retrieval: RRF fusion of vector + keyword search
|
|
62
|
+
- ✅ Namespace Isolation: Multi-tenant security with pre-filtering
|
|
63
|
+
- ✅ Schema Validation: Type checking and confidence thresholds
|
|
64
|
+
|
|
65
|
+
### Semantic Cache - LLM Response Caching
|
|
66
|
+
Vector similarity-based caching for LLM responses to reduce costs and latency:
|
|
67
|
+
|
|
68
|
+
```typescript
|
|
69
|
+
import { EmbeddedDatabase, SemanticCache } from '@sochdb/sochdb';
|
|
70
|
+
|
|
71
|
+
const db = await EmbeddedDatabase.open('./mydb');
|
|
72
|
+
const cache = new SemanticCache(db, 'llm_responses');
|
|
73
|
+
|
|
74
|
+
// Store LLM response with embedding
|
|
75
|
+
await cache.put(
|
|
76
|
+
'What is machine learning?',
|
|
77
|
+
'Machine learning is a subset of AI...',
|
|
78
|
+
embedding, // 384-dim vector
|
|
79
|
+
3600, // TTL in seconds
|
|
80
|
+
{ model: 'gpt-4', tokens: 150 }
|
|
81
|
+
);
|
|
82
|
+
|
|
83
|
+
// Check cache before calling LLM
|
|
84
|
+
const hit = await cache.get(queryEmbedding, 0.85);
|
|
85
|
+
if (hit) {
|
|
86
|
+
console.log(`Cache HIT! Similarity: ${hit.score.toFixed(4)}`);
|
|
87
|
+
console.log(`Response: ${hit.value}`);
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
// Get statistics
|
|
91
|
+
const stats = await cache.stats();
|
|
92
|
+
console.log(`Hit rate: ${(stats.hitRate * 100).toFixed(1)}%`);
|
|
93
|
+
```
|
|
94
|
+
|
|
95
|
+
**[→ See Full Example](./examples/semantic-cache-example.ts)**
|
|
96
|
+
|
|
97
|
+
**Key Benefits:**
|
|
98
|
+
- ✅ Cosine similarity matching (0-1 threshold)
|
|
99
|
+
- ✅ TTL-based expiration
|
|
100
|
+
- ✅ Hit/miss statistics tracking
|
|
101
|
+
- ✅ Memory usage monitoring
|
|
102
|
+
- ✅ Automatic expired entry purging
|
|
103
|
+
|
|
104
|
+
### Context Query Builder - Token-Aware LLM Context
|
|
105
|
+
Assemble LLM context with priority-based truncation and token budgeting:
|
|
106
|
+
|
|
107
|
+
```typescript
|
|
108
|
+
import { ContextQueryBuilder, ContextOutputFormat, TruncationStrategy } from '@sochdb/sochdb';
|
|
109
|
+
|
|
110
|
+
const builder = new ContextQueryBuilder()
|
|
111
|
+
.withBudget(4096) // Token limit
|
|
112
|
+
.setFormat(ContextOutputFormat.TOON)
|
|
113
|
+
.setTruncation(TruncationStrategy.TAIL_DROP);
|
|
114
|
+
|
|
115
|
+
builder
|
|
116
|
+
.literal('SYSTEM', 0, 'You are a helpful AI assistant.')
|
|
117
|
+
.literal('USER_PROFILE', 1, 'User: Alice, Role: Engineer')
|
|
118
|
+
.literal('HISTORY', 2, 'Recent conversation context...')
|
|
119
|
+
.literal('KNOWLEDGE', 3, 'Retrieved documents...');
|
|
120
|
+
|
|
121
|
+
const result = builder.execute();
|
|
122
|
+
console.log(`Tokens: ${result.tokenCount}/${4096}`);
|
|
123
|
+
console.log(`Context:\n${result.text}`);
|
|
124
|
+
```
|
|
125
|
+
|
|
126
|
+
**[→ See Full Example](./examples/context-builder-example.ts)**
|
|
127
|
+
|
|
128
|
+
**Key Benefits:**
|
|
129
|
+
- ✅ Priority-based section ordering (lower = higher priority)
|
|
130
|
+
- ✅ Token budget enforcement
|
|
131
|
+
- ✅ Multiple truncation strategies (tail drop, head drop, proportional)
|
|
132
|
+
- ✅ Multiple output formats (TOON, JSON, Markdown)
|
|
133
|
+
- ✅ Token count estimation
|
|
134
|
+
|
|
135
|
+
### Namespace API - Multi-Tenant Isolation
|
|
136
|
+
First-class namespace handles for secure multi-tenancy and data isolation:
|
|
137
|
+
|
|
138
|
+
```typescript
|
|
139
|
+
import { Database, Namespace, Collection, DistanceMetric } from '@sochdb/sochdb';
|
|
140
|
+
|
|
141
|
+
const db = await Database.open('./mydb');
|
|
142
|
+
|
|
143
|
+
// Create isolated namespace for each tenant
|
|
144
|
+
const namespace = new Namespace(db, 'tenant_acme', {
|
|
145
|
+
name: 'tenant_acme',
|
|
146
|
+
displayName: 'ACME Corporation',
|
|
147
|
+
labels: { plan: 'enterprise', region: 'us-west' }
|
|
148
|
+
});
|
|
149
|
+
|
|
150
|
+
// Create vector collection
|
|
151
|
+
const collection = await namespace.createCollection({
|
|
152
|
+
name: 'documents',
|
|
153
|
+
dimension: 384,
|
|
154
|
+
metric: DistanceMetric.Cosine,
|
|
155
|
+
indexed: true
|
|
156
|
+
});
|
|
157
|
+
|
|
158
|
+
// Insert and search vectors
|
|
159
|
+
await collection.insert([1.0, 2.0, ...], { title: 'Doc 1' });
|
|
160
|
+
const results = await collection.search({ queryVector: [...], k: 10 });
|
|
161
|
+
```
|
|
162
|
+
|
|
163
|
+
**[→ See Full Example](./examples/namespace-example.ts)**
|
|
164
|
+
|
|
165
|
+
### Priority Queue API - Task Processing
|
|
166
|
+
Efficient priority queue with ordered-key storage (no O(N) blob rewrites):
|
|
167
|
+
|
|
168
|
+
```typescript
|
|
169
|
+
import { Database, PriorityQueue } from '@sochdb/sochdb';
|
|
170
|
+
|
|
171
|
+
const db = await Database.open('./queue_db');
|
|
172
|
+
const queue = PriorityQueue.fromDatabase(db, 'tasks');
|
|
173
|
+
|
|
174
|
+
// Enqueue with priority (lower = higher urgency)
|
|
175
|
+
await queue.enqueue(1, Buffer.from('urgent task'), { type: 'payment' });
|
|
176
|
+
|
|
177
|
+
// Worker processes tasks
|
|
178
|
+
const task = await queue.dequeue('worker-1');
|
|
179
|
+
if (task) {
|
|
180
|
+
// Process task...
|
|
181
|
+
await queue.ack(task.taskId);
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
// Get statistics
|
|
185
|
+
const stats = await queue.stats();
|
|
186
|
+
console.log(`Pending: ${stats.pending}, Completed: ${stats.completed}`);
|
|
187
|
+
```
|
|
188
|
+
|
|
189
|
+
**[→ See Full Example](./examples/queue-example.ts)**
|
|
190
|
+
|
|
191
|
+
**Key Benefits:**
|
|
192
|
+
- ✅ O(log N) enqueue/dequeue with ordered scans
|
|
193
|
+
- ✅ Atomic claim protocol for concurrent workers
|
|
194
|
+
- ✅ Visibility timeout for crash recovery
|
|
195
|
+
- ✅ Dead letter queue for failed tasks
|
|
196
|
+
- ✅ Multiple queues per database
|
|
197
|
+
|
|
198
|
+
---
|
|
199
|
+
|
|
6
200
|
## Architecture: Flexible Deployment
|
|
7
201
|
|
|
8
202
|
```
|
|
@@ -67,40 +261,33 @@ npm install
|
|
|
67
261
|
|
|
68
262
|
1. [Quick Start](#1-quick-start)
|
|
69
263
|
2. [Installation](#2-installation)
|
|
70
|
-
3. [
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
264
|
+
3. [Features](#3-features)
|
|
265
|
+
- [Namespace API](#namespace-api---multi-tenant-isolation)
|
|
266
|
+
- [Priority Queue API](#priority-queue-api---task-processing)
|
|
267
|
+
4. [Architecture Overview](#4-architecture-overview)
|
|
268
|
+
5. [Core Key-Value Operations](#5-core-key-value-operations)
|
|
269
|
+
6. [Transactions (ACID with SSI)](#6-transactions-acid-with-ssi)
|
|
270
|
+
7. [Query Builder](#7-query-builder)
|
|
271
|
+
8. [Prefix Scanning](#8-prefix-scanning)
|
|
272
|
+
9. [SQL Operations](#9-sql-operations)
|
|
273
|
+
10. [Table Management & Index Policies](#10-table-management--index-policies)
|
|
274
|
+
11. [Namespaces & Collections](#11-namespaces--collections)
|
|
275
|
+
12. [Priority Queues](#12-priority-queues)
|
|
276
|
+
13. [Vector Search](#13-vector-search)
|
|
277
|
+
14. [Hybrid Search (Vector + BM25)](#14-hybrid-search-vector--bm25)
|
|
278
|
+
15. [Graph Operations](#15-graph-operations)
|
|
279
|
+
16. [Temporal Graph (Time-Travel)](#16-temporal-graph-time-travel)
|
|
280
|
+
17. [Semantic Cache](#17-semantic-cache)
|
|
281
|
+
18. [Context Query Builder (LLM Optimization)](#18-context-query-builder-llm-optimization)
|
|
282
|
+
19. [Atomic Multi-Index Writes](#19-atomic-multi-index-writes)
|
|
283
|
+
20. [Recovery & WAL Management](#20-recovery--wal-management)
|
|
284
|
+
21. [Checkpoints & Snapshots](#21-checkpoints--snapshots)
|
|
285
|
+
22. [Compression & Storage](#22-compression--storage)
|
|
286
|
+
23. [Statistics & Monitoring](#23-statistics--monitoring)
|
|
91
287
|
24. [Server Mode (gRPC Client)](#24-server-mode-grpc-client)
|
|
92
288
|
25. [IPC Client (Unix Sockets)](#25-ipc-client-unix-sockets)
|
|
93
|
-
26. [
|
|
94
|
-
27. [
|
|
95
|
-
28. [Data Formats (TOON/JSON/Columnar)](#28-data-formats-toonjsoncolumnar)
|
|
96
|
-
29. [Policy Service](#29-policy-service)
|
|
97
|
-
30. [MCP (Model Context Protocol)](#30-mcp-model-context-protocol)
|
|
98
|
-
31. [Configuration Reference](#31-configuration-reference)
|
|
99
|
-
32. [Error Handling](#32-error-handling)
|
|
100
|
-
33. [Async Support](#33-async-support)
|
|
101
|
-
34. [Building & Development](#34-building--development)
|
|
102
|
-
35. [Complete Examples](#35-complete-examples)
|
|
103
|
-
36. [Migration Guide](#36-migration-guide)
|
|
289
|
+
26. [Error Handling](#26-error-handling)
|
|
290
|
+
27. [Complete Examples](#27-complete-examples)
|
|
104
291
|
|
|
105
292
|
---
|
|
106
293
|
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
package/bin/sochdb-bulk.js
CHANGED
|
@@ -46,7 +46,7 @@ sochdb-bulk binary not found!
|
|
|
46
46
|
Searched at: ${bundledPath}
|
|
47
47
|
|
|
48
48
|
To fix:
|
|
49
|
-
1. Reinstall the package: npm install --force @
|
|
49
|
+
1. Reinstall the package: npm install --force @sochdb/sochdb
|
|
50
50
|
2. Or set SOCHDB_BULK_PATH environment variable
|
|
51
51
|
`);
|
|
52
52
|
}
|
|
@@ -46,7 +46,7 @@ sochdb-grpc-server binary not found!
|
|
|
46
46
|
Searched at: ${bundledPath}
|
|
47
47
|
|
|
48
48
|
To fix:
|
|
49
|
-
1. Reinstall the package: npm install --force @
|
|
49
|
+
1. Reinstall the package: npm install --force @sochdb/sochdb
|
|
50
50
|
2. Or set SOCHDB_GRPC_SERVER_PATH environment variable
|
|
51
51
|
`);
|
|
52
52
|
}
|
package/bin/sochdb-server.js
CHANGED
|
@@ -46,7 +46,7 @@ sochdb-server binary not found!
|
|
|
46
46
|
Searched at: ${bundledPath}
|
|
47
47
|
|
|
48
48
|
To fix:
|
|
49
|
-
1. Reinstall the package: npm install --force @
|
|
49
|
+
1. Reinstall the package: npm install --force @sochdb/sochdb
|
|
50
50
|
2. Or set SOCHDB_SERVER_PATH environment variable
|
|
51
51
|
`);
|
|
52
52
|
}
|
|
@@ -0,0 +1,280 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Context Query Builder for LLM Context Assembly
|
|
4
|
+
*
|
|
5
|
+
* Token-aware context assembly with priority-based truncation.
|
|
6
|
+
*/
|
|
7
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
|
+
exports.ContextQueryBuilder = exports.TruncationStrategy = exports.ContextOutputFormat = void 0;
|
|
9
|
+
exports.createContextBuilder = createContextBuilder;
|
|
10
|
+
var ContextOutputFormat;
|
|
11
|
+
(function (ContextOutputFormat) {
|
|
12
|
+
ContextOutputFormat["TOON"] = "toon";
|
|
13
|
+
ContextOutputFormat["JSON"] = "json";
|
|
14
|
+
ContextOutputFormat["MARKDOWN"] = "markdown";
|
|
15
|
+
})(ContextOutputFormat || (exports.ContextOutputFormat = ContextOutputFormat = {}));
|
|
16
|
+
var TruncationStrategy;
|
|
17
|
+
(function (TruncationStrategy) {
|
|
18
|
+
TruncationStrategy["TAIL_DROP"] = "tail_drop";
|
|
19
|
+
TruncationStrategy["HEAD_DROP"] = "head_drop";
|
|
20
|
+
TruncationStrategy["PROPORTIONAL"] = "proportional";
|
|
21
|
+
})(TruncationStrategy || (exports.TruncationStrategy = TruncationStrategy = {}));
|
|
22
|
+
/**
|
|
23
|
+
* Context Query Builder for assembling LLM context
|
|
24
|
+
*/
|
|
25
|
+
class ContextQueryBuilder {
|
|
26
|
+
constructor() {
|
|
27
|
+
this.tokenBudget = 4096;
|
|
28
|
+
this.format = ContextOutputFormat.TOON;
|
|
29
|
+
this.truncation = TruncationStrategy.TAIL_DROP;
|
|
30
|
+
this.sections = [];
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Set session ID for context
|
|
34
|
+
*/
|
|
35
|
+
forSession(sessionId) {
|
|
36
|
+
this.sessionId = sessionId;
|
|
37
|
+
return this;
|
|
38
|
+
}
|
|
39
|
+
/**
|
|
40
|
+
* Set token budget
|
|
41
|
+
*/
|
|
42
|
+
withBudget(tokens) {
|
|
43
|
+
this.tokenBudget = tokens;
|
|
44
|
+
return this;
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Set output format
|
|
48
|
+
*/
|
|
49
|
+
setFormat(format) {
|
|
50
|
+
this.format = format;
|
|
51
|
+
return this;
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* Set truncation strategy
|
|
55
|
+
*/
|
|
56
|
+
setTruncation(strategy) {
|
|
57
|
+
this.truncation = strategy;
|
|
58
|
+
return this;
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Add literal text section
|
|
62
|
+
*/
|
|
63
|
+
literal(name, priority, text) {
|
|
64
|
+
const tokenCount = this.estimateTokens(text);
|
|
65
|
+
this.sections.push({
|
|
66
|
+
name,
|
|
67
|
+
priority,
|
|
68
|
+
content: text,
|
|
69
|
+
tokenCount,
|
|
70
|
+
});
|
|
71
|
+
return this;
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Start a new section
|
|
75
|
+
*/
|
|
76
|
+
section(name, priority) {
|
|
77
|
+
this.currentSection = {
|
|
78
|
+
name,
|
|
79
|
+
priority,
|
|
80
|
+
content: '',
|
|
81
|
+
tokenCount: 0,
|
|
82
|
+
};
|
|
83
|
+
return this;
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Add content to current section
|
|
87
|
+
*/
|
|
88
|
+
get(path) {
|
|
89
|
+
if (!this.currentSection) {
|
|
90
|
+
throw new Error('No active section. Call section() first.');
|
|
91
|
+
}
|
|
92
|
+
this.currentSection.content += `GET ${path}\n`;
|
|
93
|
+
return this;
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Add last N records query
|
|
97
|
+
*/
|
|
98
|
+
last(n, table) {
|
|
99
|
+
if (!this.currentSection) {
|
|
100
|
+
throw new Error('No active section. Call section() first.');
|
|
101
|
+
}
|
|
102
|
+
this.currentSection.content += `LAST ${n} FROM ${table}\n`;
|
|
103
|
+
return this;
|
|
104
|
+
}
|
|
105
|
+
/**
|
|
106
|
+
* Add where equals condition
|
|
107
|
+
*/
|
|
108
|
+
whereEq(field, value) {
|
|
109
|
+
if (!this.currentSection) {
|
|
110
|
+
throw new Error('No active section. Call section() first.');
|
|
111
|
+
}
|
|
112
|
+
this.currentSection.content += `WHERE ${field} = ${JSON.stringify(value)}\n`;
|
|
113
|
+
return this;
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* Add vector search
|
|
117
|
+
*/
|
|
118
|
+
search(collection, embedding, k) {
|
|
119
|
+
if (!this.currentSection) {
|
|
120
|
+
throw new Error('No active section. Call section() first.');
|
|
121
|
+
}
|
|
122
|
+
this.currentSection.content += `SEARCH ${collection} WITH ${embedding} LIMIT ${k}\n`;
|
|
123
|
+
return this;
|
|
124
|
+
}
|
|
125
|
+
/**
|
|
126
|
+
* Add SQL query
|
|
127
|
+
*/
|
|
128
|
+
sql(query) {
|
|
129
|
+
if (!this.currentSection) {
|
|
130
|
+
throw new Error('No active section. Call section() first.');
|
|
131
|
+
}
|
|
132
|
+
this.currentSection.content += `SQL: ${query}\n`;
|
|
133
|
+
return this;
|
|
134
|
+
}
|
|
135
|
+
/**
|
|
136
|
+
* Finish current section
|
|
137
|
+
*/
|
|
138
|
+
done() {
|
|
139
|
+
if (this.currentSection) {
|
|
140
|
+
this.currentSection.tokenCount = this.estimateTokens(this.currentSection.content);
|
|
141
|
+
this.sections.push(this.currentSection);
|
|
142
|
+
this.currentSection = undefined;
|
|
143
|
+
}
|
|
144
|
+
return this;
|
|
145
|
+
}
|
|
146
|
+
/**
|
|
147
|
+
* Execute and build context
|
|
148
|
+
*/
|
|
149
|
+
execute() {
|
|
150
|
+
// Finish any pending section
|
|
151
|
+
if (this.currentSection) {
|
|
152
|
+
this.done();
|
|
153
|
+
}
|
|
154
|
+
// Sort sections by priority (lower = higher priority)
|
|
155
|
+
const sortedSections = [...this.sections].sort((a, b) => a.priority - b.priority);
|
|
156
|
+
// Calculate total tokens
|
|
157
|
+
let totalTokens = sortedSections.reduce((sum, s) => sum + s.tokenCount, 0);
|
|
158
|
+
// Truncate if needed
|
|
159
|
+
const truncatedSections = [];
|
|
160
|
+
const includedSections = [];
|
|
161
|
+
if (totalTokens <= this.tokenBudget) {
|
|
162
|
+
// No truncation needed
|
|
163
|
+
for (const section of sortedSections) {
|
|
164
|
+
includedSections.push(section);
|
|
165
|
+
truncatedSections.push({
|
|
166
|
+
name: section.name,
|
|
167
|
+
tokenCount: section.tokenCount,
|
|
168
|
+
truncated: false,
|
|
169
|
+
});
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
else {
|
|
173
|
+
// Apply truncation strategy
|
|
174
|
+
let remainingBudget = this.tokenBudget;
|
|
175
|
+
if (this.truncation === TruncationStrategy.TAIL_DROP) {
|
|
176
|
+
// Include sections in priority order until budget exhausted
|
|
177
|
+
for (const section of sortedSections) {
|
|
178
|
+
if (section.tokenCount <= remainingBudget) {
|
|
179
|
+
includedSections.push(section);
|
|
180
|
+
remainingBudget -= section.tokenCount;
|
|
181
|
+
truncatedSections.push({
|
|
182
|
+
name: section.name,
|
|
183
|
+
tokenCount: section.tokenCount,
|
|
184
|
+
truncated: false,
|
|
185
|
+
});
|
|
186
|
+
}
|
|
187
|
+
else {
|
|
188
|
+
truncatedSections.push({
|
|
189
|
+
name: section.name,
|
|
190
|
+
tokenCount: 0,
|
|
191
|
+
truncated: true,
|
|
192
|
+
});
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
else if (this.truncation === TruncationStrategy.PROPORTIONAL) {
|
|
197
|
+
// Proportionally reduce all sections
|
|
198
|
+
const ratio = this.tokenBudget / totalTokens;
|
|
199
|
+
for (const section of sortedSections) {
|
|
200
|
+
const allocatedTokens = Math.floor(section.tokenCount * ratio);
|
|
201
|
+
const truncatedContent = this.truncateText(section.content, allocatedTokens);
|
|
202
|
+
includedSections.push({
|
|
203
|
+
...section,
|
|
204
|
+
content: truncatedContent,
|
|
205
|
+
tokenCount: allocatedTokens,
|
|
206
|
+
});
|
|
207
|
+
truncatedSections.push({
|
|
208
|
+
name: section.name,
|
|
209
|
+
tokenCount: allocatedTokens,
|
|
210
|
+
truncated: allocatedTokens < section.tokenCount,
|
|
211
|
+
});
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
// Build final context based on format
|
|
216
|
+
let text = '';
|
|
217
|
+
let actualTokens = 0;
|
|
218
|
+
if (this.format === ContextOutputFormat.TOON) {
|
|
219
|
+
text = this.buildToonFormat(includedSections);
|
|
220
|
+
}
|
|
221
|
+
else if (this.format === ContextOutputFormat.JSON) {
|
|
222
|
+
text = this.buildJsonFormat(includedSections);
|
|
223
|
+
}
|
|
224
|
+
else {
|
|
225
|
+
text = this.buildMarkdownFormat(includedSections);
|
|
226
|
+
}
|
|
227
|
+
actualTokens = this.estimateTokens(text);
|
|
228
|
+
return {
|
|
229
|
+
text,
|
|
230
|
+
tokenCount: actualTokens,
|
|
231
|
+
sections: truncatedSections,
|
|
232
|
+
};
|
|
233
|
+
}
|
|
234
|
+
// Helper methods
|
|
235
|
+
estimateTokens(text) {
|
|
236
|
+
// Rough estimate: ~4 characters per token for English
|
|
237
|
+
return Math.ceil(text.length / 4);
|
|
238
|
+
}
|
|
239
|
+
truncateText(text, maxTokens) {
|
|
240
|
+
const maxChars = maxTokens * 4;
|
|
241
|
+
if (text.length <= maxChars) {
|
|
242
|
+
return text;
|
|
243
|
+
}
|
|
244
|
+
return text.substring(0, maxChars) + '...';
|
|
245
|
+
}
|
|
246
|
+
buildToonFormat(sections) {
|
|
247
|
+
const lines = [];
|
|
248
|
+
for (const section of sections) {
|
|
249
|
+
lines.push(`[${section.name}]`);
|
|
250
|
+
lines.push(section.content);
|
|
251
|
+
lines.push('');
|
|
252
|
+
}
|
|
253
|
+
return lines.join('\n');
|
|
254
|
+
}
|
|
255
|
+
buildJsonFormat(sections) {
|
|
256
|
+
const obj = {};
|
|
257
|
+
for (const section of sections) {
|
|
258
|
+
obj[section.name] = section.content;
|
|
259
|
+
}
|
|
260
|
+
return JSON.stringify(obj, null, 2);
|
|
261
|
+
}
|
|
262
|
+
buildMarkdownFormat(sections) {
|
|
263
|
+
const lines = [];
|
|
264
|
+
for (const section of sections) {
|
|
265
|
+
lines.push(`## ${section.name}`);
|
|
266
|
+
lines.push('');
|
|
267
|
+
lines.push(section.content);
|
|
268
|
+
lines.push('');
|
|
269
|
+
}
|
|
270
|
+
return lines.join('\n');
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
exports.ContextQueryBuilder = ContextQueryBuilder;
|
|
274
|
+
/**
|
|
275
|
+
* Create a context query builder
|
|
276
|
+
*/
|
|
277
|
+
function createContextBuilder() {
|
|
278
|
+
return new ContextQueryBuilder();
|
|
279
|
+
}
|
|
280
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context-builder.js","sourceRoot":"","sources":["../../src/context-builder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Context Query Builder for LLM Context Assembly\n * \n * Token-aware context assembly with priority-based truncation.\n */\n\nexport enum ContextOutputFormat {\n  TOON = 'toon',\n  JSON = 'json',\n  MARKDOWN = 'markdown',\n}\n\nexport enum TruncationStrategy {\n  TAIL_DROP = 'tail_drop',       // Drop from end\n  HEAD_DROP = 'head_drop',       // Drop from beginning\n  PROPORTIONAL = 'proportional', // Proportional across sections\n}\n\ninterface Section {\n  name: string;\n  priority: number;\n  content: string;\n  tokenCount: number;\n}\n\nexport interface ContextResult {\n  text: string;\n  tokenCount: number;\n  sections: Array<{ name: string; tokenCount: number; truncated: boolean }>;\n}\n\n/**\n * Context Query Builder for assembling LLM context\n */\nexport class ContextQueryBuilder {\n  private sessionId?: string;\n  private tokenBudget: number = 4096;\n  private format: ContextOutputFormat = ContextOutputFormat.TOON;\n  private truncation: TruncationStrategy = TruncationStrategy.TAIL_DROP;\n  private sections: Section[] = [];\n  private currentSection?: Section;\n\n  /**\n   * Set session ID for context\n   */\n  forSession(sessionId: string): this {\n    this.sessionId = sessionId;\n    return this;\n  }\n\n  /**\n   * Set token budget\n   */\n  withBudget(tokens: number): this {\n    this.tokenBudget = tokens;\n    return this;\n  }\n\n  /**\n   * Set output format\n   */\n  setFormat(format: ContextOutputFormat): this {\n    this.format = format;\n    return this;\n  }\n\n  /**\n   * Set truncation strategy\n   */\n  setTruncation(strategy: TruncationStrategy): this {\n    this.truncation = strategy;\n    return this;\n  }\n\n  /**\n   * Add literal text section\n   */\n  literal(name: string, priority: number, text: string): this {\n    const tokenCount = this.estimateTokens(text);\n    this.sections.push({\n      name,\n      priority,\n      content: text,\n      tokenCount,\n    });\n    return this;\n  }\n\n  /**\n   * Start a new section\n   */\n  section(name: string, priority: number): this {\n    this.currentSection = {\n      name,\n      priority,\n      content: '',\n      tokenCount: 0,\n    };\n    return this;\n  }\n\n  /**\n   * Add content to current section\n   */\n  get(path: string): this {\n    if (!this.currentSection) {\n      throw new Error('No active section. Call section() first.');\n    }\n    this.currentSection.content += `GET ${path}\\n`;\n    return this;\n  }\n\n  /**\n   * Add last N records query\n   */\n  last(n: number, table: string): this {\n    if (!this.currentSection) {\n      throw new Error('No active section. Call section() first.');\n    }\n    this.currentSection.content += `LAST ${n} FROM ${table}\\n`;\n    return this;\n  }\n\n  /**\n   * Add where equals condition\n   */\n  whereEq(field: string, value: any): this {\n    if (!this.currentSection) {\n      throw new Error('No active section. Call section() first.');\n    }\n    this.currentSection.content += `WHERE ${field} = ${JSON.stringify(value)}\\n`;\n    return this;\n  }\n\n  /**\n   * Add vector search\n   */\n  search(collection: string, embedding: string, k: number): this {\n    if (!this.currentSection) {\n      throw new Error('No active section. Call section() first.');\n    }\n    this.currentSection.content += `SEARCH ${collection} WITH ${embedding} LIMIT ${k}\\n`;\n    return this;\n  }\n\n  /**\n   * Add SQL query\n   */\n  sql(query: string): this {\n    if (!this.currentSection) {\n      throw new Error('No active section. Call section() first.');\n    }\n    this.currentSection.content += `SQL: ${query}\\n`;\n    return this;\n  }\n\n  /**\n   * Finish current section\n   */\n  done(): this {\n    if (this.currentSection) {\n      this.currentSection.tokenCount = this.estimateTokens(this.currentSection.content);\n      this.sections.push(this.currentSection);\n      this.currentSection = undefined;\n    }\n    return this;\n  }\n\n  /**\n   * Execute and build context\n   */\n  execute(): ContextResult {\n    // Finish any pending section\n    if (this.currentSection) {\n      this.done();\n    }\n\n    // Sort sections by priority (lower = higher priority)\n    const sortedSections = [...this.sections].sort((a, b) => a.priority - b.priority);\n\n    // Calculate total tokens\n    let totalTokens = sortedSections.reduce((sum, s) => sum + s.tokenCount, 0);\n\n    // Truncate if needed\n    const truncatedSections: Array<{ name: string; tokenCount: number; truncated: boolean }> = [];\n    const includedSections: Section[] = [];\n\n    if (totalTokens <= this.tokenBudget) {\n      // No truncation needed\n      for (const section of sortedSections) {\n        includedSections.push(section);\n        truncatedSections.push({\n          name: section.name,\n          tokenCount: section.tokenCount,\n          truncated: false,\n        });\n      }\n    } else {\n      // Apply truncation strategy\n      let remainingBudget = this.tokenBudget;\n\n      if (this.truncation === TruncationStrategy.TAIL_DROP) {\n        // Include sections in priority order until budget exhausted\n        for (const section of sortedSections) {\n          if (section.tokenCount <= remainingBudget) {\n            includedSections.push(section);\n            remainingBudget -= section.tokenCount;\n            truncatedSections.push({\n              name: section.name,\n              tokenCount: section.tokenCount,\n              truncated: false,\n            });\n          } else {\n            truncatedSections.push({\n              name: section.name,\n              tokenCount: 0,\n              truncated: true,\n            });\n          }\n        }\n      } else if (this.truncation === TruncationStrategy.PROPORTIONAL) {\n        // Proportionally reduce all sections\n        const ratio = this.tokenBudget / totalTokens;\n        for (const section of sortedSections) {\n          const allocatedTokens = Math.floor(section.tokenCount * ratio);\n          const truncatedContent = this.truncateText(section.content, allocatedTokens);\n          includedSections.push({\n            ...section,\n            content: truncatedContent,\n            tokenCount: allocatedTokens,\n          });\n          truncatedSections.push({\n            name: section.name,\n            tokenCount: allocatedTokens,\n            truncated: allocatedTokens < section.tokenCount,\n          });\n        }\n      }\n    }\n\n    // Build final context based on format\n    let text = '';\n    let actualTokens = 0;\n\n    if (this.format === ContextOutputFormat.TOON) {\n      text = this.buildToonFormat(includedSections);\n    } else if (this.format === ContextOutputFormat.JSON) {\n      text = this.buildJsonFormat(includedSections);\n    } else {\n      text = this.buildMarkdownFormat(includedSections);\n    }\n\n    actualTokens = this.estimateTokens(text);\n\n    return {\n      text,\n      tokenCount: actualTokens,\n      sections: truncatedSections,\n    };\n  }\n\n  // Helper methods\n  private estimateTokens(text: string): number {\n    // Rough estimate: ~4 characters per token for English\n    return Math.ceil(text.length / 4);\n  }\n\n  private truncateText(text: string, maxTokens: number): string {\n    const maxChars = maxTokens * 4;\n    if (text.length <= maxChars) {\n      return text;\n    }\n    return text.substring(0, maxChars) + '...';\n  }\n\n  private buildToonFormat(sections: Section[]): string {\n    const lines: string[] = [];\n    \n    for (const section of sections) {\n      lines.push(`[${section.name}]`);\n      lines.push(section.content);\n      lines.push('');\n    }\n    \n    return lines.join('\\n');\n  }\n\n  private buildJsonFormat(sections: Section[]): string {\n    const obj: Record<string, string> = {};\n    \n    for (const section of sections) {\n      obj[section.name] = section.content;\n    }\n    \n    return JSON.stringify(obj, null, 2);\n  }\n\n  private buildMarkdownFormat(sections: Section[]): string {\n    const lines: string[] = [];\n    \n    for (const section of sections) {\n      lines.push(`## ${section.name}`);\n      lines.push('');\n      lines.push(section.content);\n      lines.push('');\n    }\n    \n    return lines.join('\\n');\n  }\n}\n\n/**\n * Create a context query builder\n */\nexport function createContextBuilder(): ContextQueryBuilder {\n  return new ContextQueryBuilder();\n}\n"]}
|