@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.
Files changed (69) hide show
  1. package/README.md +220 -33
  2. package/_bin/aarch64-apple-darwin/libsochdb_storage.dylib +0 -0
  3. package/_bin/aarch64-apple-darwin/sochdb-bulk +0 -0
  4. package/_bin/aarch64-apple-darwin/sochdb-grpc-server +0 -0
  5. package/_bin/aarch64-apple-darwin/sochdb-server +0 -0
  6. package/_bin/x86_64-pc-windows-msvc/sochdb-bulk.exe +0 -0
  7. package/_bin/x86_64-pc-windows-msvc/sochdb-grpc-server.exe +0 -0
  8. package/_bin/x86_64-pc-windows-msvc/sochdb_storage.dll +0 -0
  9. package/_bin/x86_64-unknown-linux-gnu/libsochdb_storage.so +0 -0
  10. package/_bin/x86_64-unknown-linux-gnu/sochdb-bulk +0 -0
  11. package/_bin/x86_64-unknown-linux-gnu/sochdb-grpc-server +0 -0
  12. package/_bin/x86_64-unknown-linux-gnu/sochdb-server +0 -0
  13. package/bin/sochdb-bulk.js +1 -1
  14. package/bin/sochdb-grpc-server.js +1 -1
  15. package/bin/sochdb-server.js +1 -1
  16. package/dist/cjs/context-builder.js +280 -0
  17. package/dist/cjs/database.js +2 -2
  18. package/dist/cjs/embedded/database.js +2 -2
  19. package/dist/cjs/errors.js +99 -7
  20. package/dist/cjs/index.js +40 -3
  21. package/dist/cjs/ipc-client.js +2 -2
  22. package/dist/cjs/memory/consolidation.js +202 -0
  23. package/dist/cjs/memory/extraction.js +181 -0
  24. package/dist/cjs/memory/index.js +26 -0
  25. package/dist/cjs/memory/retrieval.js +232 -0
  26. package/dist/cjs/memory/types.js +69 -0
  27. package/dist/cjs/namespace.js +255 -0
  28. package/dist/cjs/queue.js +289 -0
  29. package/dist/cjs/semantic-cache.js +220 -0
  30. package/dist/esm/context-builder.js +280 -0
  31. package/dist/esm/database.js +2 -2
  32. package/dist/esm/embedded/database.js +2 -2
  33. package/dist/esm/errors.js +107 -7
  34. package/dist/esm/index.js +40 -3
  35. package/dist/esm/ipc-client.js +2 -2
  36. package/dist/esm/memory/consolidation.js +206 -0
  37. package/dist/esm/memory/extraction.js +185 -0
  38. package/dist/esm/memory/index.js +26 -0
  39. package/dist/esm/memory/retrieval.js +243 -0
  40. package/dist/esm/memory/types.js +72 -0
  41. package/dist/esm/namespace.js +262 -0
  42. package/dist/esm/queue.js +291 -0
  43. package/dist/esm/semantic-cache.js +223 -0
  44. package/dist/types/context-builder.d.ts +97 -0
  45. package/dist/types/context-builder.d.ts.map +1 -0
  46. package/dist/types/database.d.ts +1 -1
  47. package/dist/types/embedded/database.d.ts +1 -1
  48. package/dist/types/errors.d.ts +57 -1
  49. package/dist/types/errors.d.ts.map +1 -1
  50. package/dist/types/index.d.ts +12 -2
  51. package/dist/types/index.d.ts.map +1 -1
  52. package/dist/types/ipc-client.d.ts +1 -1
  53. package/dist/types/memory/consolidation.d.ts +66 -0
  54. package/dist/types/memory/consolidation.d.ts.map +1 -0
  55. package/dist/types/memory/extraction.d.ts +82 -0
  56. package/dist/types/memory/extraction.d.ts.map +1 -0
  57. package/dist/types/memory/index.d.ts +10 -0
  58. package/dist/types/memory/index.d.ts.map +1 -0
  59. package/dist/types/memory/retrieval.d.ts +46 -0
  60. package/dist/types/memory/retrieval.d.ts.map +1 -0
  61. package/dist/types/memory/types.d.ts +147 -0
  62. package/dist/types/memory/types.d.ts.map +1 -0
  63. package/dist/types/namespace.d.ts +129 -0
  64. package/dist/types/namespace.d.ts.map +1 -0
  65. package/dist/types/queue.d.ts +120 -0
  66. package/dist/types/queue.d.ts.map +1 -0
  67. package/dist/types/semantic-cache.d.ts +84 -0
  68. package/dist/types/semantic-cache.d.ts.map +1 -0
  69. package/package.json +1 -1
package/README.md CHANGED
@@ -1,8 +1,202 @@
1
- # SochDB Node.js SDK v0.4.0
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. [Architecture Overview](#3-architecture-overview)
71
- 4. [Core Key-Value Operations](#4-core-key-value-operations)
72
- 5. [Transactions (ACID with SSI)](#5-transactions-acid-with-ssi)
73
- 6. [Query Builder](#6-query-builder)
74
- 7. [Prefix Scanning](#7-prefix-scanning)
75
- 8. [SQL Operations](#8-sql-operations)
76
- 9. [Table Management & Index Policies](#9-table-management--index-policies)
77
- 10. [Namespaces & Multi-Tenancy](#10-namespaces--multi-tenancy)
78
- 11. [Collections & Vector Search](#11-collections--vector-search)
79
- 12. [Hybrid Search (Vector + BM25)](#12-hybrid-search-vector--bm25)
80
- 13. [Graph Operations](#13-graph-operations)
81
- 14. [Temporal Graph (Time-Travel)](#14-temporal-graph-time-travel)
82
- 15. [Semantic Cache](#15-semantic-cache)
83
- 16. [Context Query Builder (LLM Optimization) and Session](#16-context-query-builder-llm-optimization)
84
- 17. [Atomic Multi-Index Writes](#17-atomic-multi-index-writes)
85
- 18. [Recovery & WAL Management](#18-recovery--wal-management)
86
- 19. [Checkpoints & Snapshots](#19-checkpoints--snapshots)
87
- 20. [Compression & Storage](#20-compression--storage)
88
- 21. [Statistics & Monitoring](#21-statistics--monitoring)
89
- 22. [Distributed Tracing](#22-distributed-tracing)
90
- 23. [Workflow & Run Tracking](#23-workflow--run-tracking)
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. [Standalone VectorIndex](#26-standalone-vectorindex)
94
- 27. [Vector Utilities](#27-vector-utilities)
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
 
@@ -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 @sushanth/sochdb
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 @sushanth/sochdb
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
  }
@@ -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 @sushanth/sochdb
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"]}