@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
@@ -0,0 +1,181 @@
1
+ "use strict";
2
+ /**
3
+ * Extraction Pipeline for Memory System
4
+ *
5
+ * Compiles LLM outputs into typed, validated facts (Entity, Relation, Assertion).
6
+ */
7
+ Object.defineProperty(exports, "__esModule", { value: true });
8
+ exports.ExtractionPipeline = void 0;
9
+ const crypto_1 = require("crypto");
10
+ /**
11
+ * Extraction Pipeline
12
+ */
13
+ class ExtractionPipeline {
14
+ constructor(db, namespace, schema) {
15
+ this.db = db;
16
+ this.namespace = namespace;
17
+ this.schema = schema;
18
+ this.prefix = Buffer.from(`memory:${namespace}:`);
19
+ }
20
+ /**
21
+ * Create pipeline from database
22
+ */
23
+ static fromDatabase(db, namespace, schema) {
24
+ return new ExtractionPipeline(db, namespace, schema);
25
+ }
26
+ /**
27
+ * Extract entities and relations from text
28
+ */
29
+ async extract(text, extractor) {
30
+ const rawResult = await extractor(text);
31
+ const timestamp = Date.now();
32
+ // Normalize entities
33
+ const entities = (rawResult.entities || []).map(e => ({
34
+ id: this.generateEntityId(e.name, e.entity_type),
35
+ name: e.name,
36
+ entityType: e.entity_type,
37
+ properties: e.properties,
38
+ confidence: e.confidence || 1.0,
39
+ provenance: text.substring(0, 100),
40
+ timestamp,
41
+ }));
42
+ // Validate entities
43
+ if (this.schema?.entityTypes) {
44
+ const validTypes = new Set(this.schema.entityTypes);
45
+ const filteredEntities = entities.filter(e => validTypes.has(e.entityType));
46
+ if (filteredEntities.length < entities.length) {
47
+ console.warn(`Filtered ${entities.length - filteredEntities.length} entities with invalid types`);
48
+ }
49
+ entities.splice(0, entities.length, ...filteredEntities);
50
+ }
51
+ // Normalize relations
52
+ const relations = (rawResult.relations || []).map(r => ({
53
+ id: this.generateRelationId(r.from_entity, r.relation_type, r.to_entity),
54
+ fromEntity: r.from_entity,
55
+ relationType: r.relation_type,
56
+ toEntity: r.to_entity,
57
+ properties: r.properties,
58
+ confidence: r.confidence || 1.0,
59
+ provenance: text.substring(0, 100),
60
+ timestamp,
61
+ }));
62
+ // Validate relations
63
+ if (this.schema?.relationTypes) {
64
+ const validTypes = new Set(this.schema.relationTypes);
65
+ const filteredRelations = relations.filter(r => validTypes.has(r.relationType));
66
+ if (filteredRelations.length < relations.length) {
67
+ console.warn(`Filtered ${relations.length - filteredRelations.length} relations with invalid types`);
68
+ }
69
+ relations.splice(0, relations.length, ...filteredRelations);
70
+ }
71
+ // Normalize assertions
72
+ const assertions = (rawResult.assertions || []).map(a => ({
73
+ id: this.generateAssertionId(a.subject, a.predicate, a.object),
74
+ subject: a.subject,
75
+ predicate: a.predicate,
76
+ object: a.object,
77
+ confidence: a.confidence || 1.0,
78
+ provenance: text.substring(0, 100),
79
+ timestamp,
80
+ }));
81
+ // Apply min confidence filter
82
+ if (this.schema?.minConfidence) {
83
+ const minConf = this.schema.minConfidence;
84
+ const filterByConfidence = (items) => items.filter(item => (item.confidence || 0) >= minConf);
85
+ entities.splice(0, entities.length, ...filterByConfidence(entities));
86
+ relations.splice(0, relations.length, ...filterByConfidence(relations));
87
+ assertions.splice(0, assertions.length, ...filterByConfidence(assertions));
88
+ }
89
+ return { entities, relations, assertions };
90
+ }
91
+ /**
92
+ * Extract and immediately commit to database
93
+ */
94
+ async extractAndCommit(text, extractor) {
95
+ const result = await this.extract(text, extractor);
96
+ await this.commit(result);
97
+ return result;
98
+ }
99
+ /**
100
+ * Commit extraction result to database
101
+ */
102
+ async commit(result) {
103
+ // Store entities
104
+ for (const entity of result.entities) {
105
+ const key = Buffer.concat([this.prefix, Buffer.from(`entity:${entity.id}`)]);
106
+ await this.db.put(key, Buffer.from(JSON.stringify(entity)));
107
+ }
108
+ // Store relations
109
+ for (const relation of result.relations) {
110
+ const key = Buffer.concat([this.prefix, Buffer.from(`relation:${relation.id}`)]);
111
+ await this.db.put(key, Buffer.from(JSON.stringify(relation)));
112
+ }
113
+ // Store assertions
114
+ for (const assertion of result.assertions) {
115
+ const key = Buffer.concat([this.prefix, Buffer.from(`assertion:${assertion.id}`)]);
116
+ await this.db.put(key, Buffer.from(JSON.stringify(assertion)));
117
+ }
118
+ }
119
+ /**
120
+ * Get all entities
121
+ */
122
+ async getEntities() {
123
+ const entities = [];
124
+ const entityPrefix = Buffer.concat([this.prefix, Buffer.from('entity:')]);
125
+ for await (const [_, value] of this.db.scanPrefix(entityPrefix)) {
126
+ entities.push(JSON.parse(value.toString()));
127
+ }
128
+ return entities;
129
+ }
130
+ /**
131
+ * Get all relations
132
+ */
133
+ async getRelations() {
134
+ const relations = [];
135
+ const relationPrefix = Buffer.concat([this.prefix, Buffer.from('relation:')]);
136
+ for await (const [_, value] of this.db.scanPrefix(relationPrefix)) {
137
+ relations.push(JSON.parse(value.toString()));
138
+ }
139
+ return relations;
140
+ }
141
+ /**
142
+ * Get all assertions
143
+ */
144
+ async getAssertions() {
145
+ const assertions = [];
146
+ const assertionPrefix = Buffer.concat([this.prefix, Buffer.from('assertion:')]);
147
+ for await (const [_, value] of this.db.scanPrefix(assertionPrefix)) {
148
+ assertions.push(JSON.parse(value.toString()));
149
+ }
150
+ return assertions;
151
+ }
152
+ /**
153
+ * Generate deterministic entity ID
154
+ */
155
+ generateEntityId(name, entityType) {
156
+ return (0, crypto_1.createHash)('sha256')
157
+ .update(`${name}:${entityType}`)
158
+ .digest('hex')
159
+ .substring(0, 16);
160
+ }
161
+ /**
162
+ * Generate deterministic relation ID
163
+ */
164
+ generateRelationId(from, relationType, to) {
165
+ return (0, crypto_1.createHash)('sha256')
166
+ .update(`${from}:${relationType}:${to}`)
167
+ .digest('hex')
168
+ .substring(0, 16);
169
+ }
170
+ /**
171
+ * Generate deterministic assertion ID
172
+ */
173
+ generateAssertionId(subject, predicate, object) {
174
+ return (0, crypto_1.createHash)('sha256')
175
+ .update(`${subject}:${predicate}:${object}`)
176
+ .digest('hex')
177
+ .substring(0, 16);
178
+ }
179
+ }
180
+ exports.ExtractionPipeline = ExtractionPipeline;
181
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"extraction.js","sourceRoot":"","sources":["../../../src/memory/extraction.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Extraction Pipeline for Memory System\n * \n * Compiles LLM outputs into typed, validated facts (Entity, Relation, Assertion).\n */\n\nimport { EmbeddedDatabase } from '../embedded';\nimport {\n  Entity,\n  Relation,\n  Assertion,\n  ExtractionResult,\n  ExtractionSchema,\n} from './types';\nimport { createHash } from 'crypto';\n\n/**\n * Extractor function type - user provides this to call their LLM\n */\nexport type ExtractorFunction = (text: string) => Promise<{\n  entities?: Array<{ name: string; entity_type: string; properties?: Record<string, any>; confidence?: number }>;\n  relations?: Array<{ from_entity: string; relation_type: string; to_entity: string; properties?: Record<string, any>; confidence?: number }>;\n  assertions?: Array<{ subject: string; predicate: string; object: string; confidence?: number }>;\n}>;\n\n/**\n * Extraction Pipeline\n */\nexport class ExtractionPipeline {\n  private db: EmbeddedDatabase;\n  private namespace: string;\n  private schema?: ExtractionSchema;\n  private prefix: Buffer;\n\n  constructor(db: EmbeddedDatabase, namespace: string, schema?: ExtractionSchema) {\n    this.db = db;\n    this.namespace = namespace;\n    this.schema = schema;\n    this.prefix = Buffer.from(`memory:${namespace}:`);\n  }\n\n  /**\n   * Create pipeline from database\n   */\n  static fromDatabase(db: EmbeddedDatabase, namespace: string, schema?: ExtractionSchema): ExtractionPipeline {\n    return new ExtractionPipeline(db, namespace, schema);\n  }\n\n  /**\n   * Extract entities and relations from text\n   */\n  async extract(text: string, extractor: ExtractorFunction): Promise<ExtractionResult> {\n    const rawResult = await extractor(text);\n    const timestamp = Date.now();\n\n    // Normalize entities\n    const entities: Entity[] = (rawResult.entities || []).map(e => ({\n      id: this.generateEntityId(e.name, e.entity_type),\n      name: e.name,\n      entityType: e.entity_type,\n      properties: e.properties,\n      confidence: e.confidence || 1.0,\n      provenance: text.substring(0, 100),\n      timestamp,\n    }));\n\n    // Validate entities\n    if (this.schema?.entityTypes) {\n      const validTypes = new Set(this.schema.entityTypes);\n      const filteredEntities = entities.filter(e => validTypes.has(e.entityType));\n      if (filteredEntities.length < entities.length) {\n        console.warn(`Filtered ${entities.length - filteredEntities.length} entities with invalid types`);\n      }\n      entities.splice(0, entities.length, ...filteredEntities);\n    }\n\n    // Normalize relations\n    const relations: Relation[] = (rawResult.relations || []).map(r => ({\n      id: this.generateRelationId(r.from_entity, r.relation_type, r.to_entity),\n      fromEntity: r.from_entity,\n      relationType: r.relation_type,\n      toEntity: r.to_entity,\n      properties: r.properties,\n      confidence: r.confidence || 1.0,\n      provenance: text.substring(0, 100),\n      timestamp,\n    }));\n\n    // Validate relations\n    if (this.schema?.relationTypes) {\n      const validTypes = new Set(this.schema.relationTypes);\n      const filteredRelations = relations.filter(r => validTypes.has(r.relationType));\n      if (filteredRelations.length < relations.length) {\n        console.warn(`Filtered ${relations.length - filteredRelations.length} relations with invalid types`);\n      }\n      relations.splice(0, relations.length, ...filteredRelations);\n    }\n\n    // Normalize assertions\n    const assertions: Assertion[] = (rawResult.assertions || []).map(a => ({\n      id: this.generateAssertionId(a.subject, a.predicate, a.object),\n      subject: a.subject,\n      predicate: a.predicate,\n      object: a.object,\n      confidence: a.confidence || 1.0,\n      provenance: text.substring(0, 100),\n      timestamp,\n    }));\n\n    // Apply min confidence filter\n    if (this.schema?.minConfidence) {\n      const minConf = this.schema.minConfidence;\n      const filterByConfidence = <T extends { confidence?: number }>(items: T[]): T[] =>\n        items.filter(item => (item.confidence || 0) >= minConf);\n\n      entities.splice(0, entities.length, ...filterByConfidence(entities));\n      relations.splice(0, relations.length, ...filterByConfidence(relations));\n      assertions.splice(0, assertions.length, ...filterByConfidence(assertions));\n    }\n\n    return { entities, relations, assertions };\n  }\n\n  /**\n   * Extract and immediately commit to database\n   */\n  async extractAndCommit(text: string, extractor: ExtractorFunction): Promise<ExtractionResult> {\n    const result = await this.extract(text, extractor);\n    await this.commit(result);\n    return result;\n  }\n\n  /**\n   * Commit extraction result to database\n   */\n  async commit(result: ExtractionResult): Promise<void> {\n    // Store entities\n    for (const entity of result.entities) {\n      const key = Buffer.concat([this.prefix, Buffer.from(`entity:${entity.id}`)]);\n      await this.db.put(key, Buffer.from(JSON.stringify(entity)));\n    }\n\n    // Store relations\n    for (const relation of result.relations) {\n      const key = Buffer.concat([this.prefix, Buffer.from(`relation:${relation.id}`)]);\n      await this.db.put(key, Buffer.from(JSON.stringify(relation)));\n    }\n\n    // Store assertions\n    for (const assertion of result.assertions) {\n      const key = Buffer.concat([this.prefix, Buffer.from(`assertion:${assertion.id}`)]);\n      await this.db.put(key, Buffer.from(JSON.stringify(assertion)));\n    }\n  }\n\n  /**\n   * Get all entities\n   */\n  async getEntities(): Promise<Entity[]> {\n    const entities: Entity[] = [];\n    const entityPrefix = Buffer.concat([this.prefix, Buffer.from('entity:')]);\n\n    for await (const [_, value] of this.db.scanPrefix(entityPrefix)) {\n      entities.push(JSON.parse(value.toString()));\n    }\n\n    return entities;\n  }\n\n  /**\n   * Get all relations\n   */\n  async getRelations(): Promise<Relation[]> {\n    const relations: Relation[] = [];\n    const relationPrefix = Buffer.concat([this.prefix, Buffer.from('relation:')]);\n\n    for await (const [_, value] of this.db.scanPrefix(relationPrefix)) {\n      relations.push(JSON.parse(value.toString()));\n    }\n\n    return relations;\n  }\n\n  /**\n   * Get all assertions\n   */\n  async getAssertions(): Promise<Assertion[]> {\n    const assertions: Assertion[] = [];\n    const assertionPrefix = Buffer.concat([this.prefix, Buffer.from('assertion:')]);\n\n    for await (const [_, value] of this.db.scanPrefix(assertionPrefix)) {\n      assertions.push(JSON.parse(value.toString()));\n    }\n\n    return assertions;\n  }\n\n  /**\n   * Generate deterministic entity ID\n   */\n  private generateEntityId(name: string, entityType: string): string {\n    return createHash('sha256')\n      .update(`${name}:${entityType}`)\n      .digest('hex')\n      .substring(0, 16);\n  }\n\n  /**\n   * Generate deterministic relation ID\n   */\n  private generateRelationId(from: string, relationType: string, to: string): string {\n    return createHash('sha256')\n      .update(`${from}:${relationType}:${to}`)\n      .digest('hex')\n      .substring(0, 16);\n  }\n\n  /**\n   * Generate deterministic assertion ID\n   */\n  private generateAssertionId(subject: string, predicate: string, object: string): string {\n    return createHash('sha256')\n      .update(`${subject}:${predicate}:${object}`)\n      .digest('hex')\n      .substring(0, 16);\n  }\n}\n"]}
@@ -0,0 +1,26 @@
1
+ "use strict";
2
+ /**
3
+ * Memory System - Main Export
4
+ *
5
+ * LLM-native memory system with extraction, consolidation, and retrieval.
6
+ */
7
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
8
+ if (k2 === undefined) k2 = k;
9
+ var desc = Object.getOwnPropertyDescriptor(m, k);
10
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
11
+ desc = { enumerable: true, get: function() { return m[k]; } };
12
+ }
13
+ Object.defineProperty(o, k2, desc);
14
+ }) : (function(o, m, k, k2) {
15
+ if (k2 === undefined) k2 = k;
16
+ o[k2] = m[k];
17
+ }));
18
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
19
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
20
+ };
21
+ Object.defineProperty(exports, "__esModule", { value: true });
22
+ __exportStar(require("./types"), exports);
23
+ __exportStar(require("./extraction"), exports);
24
+ __exportStar(require("./consolidation"), exports);
25
+ __exportStar(require("./retrieval"), exports);
26
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbWVtb3J5L2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7OztHQUlHOzs7Ozs7Ozs7Ozs7Ozs7O0FBRUgsMENBQXdCO0FBQ3hCLCtDQUE2QjtBQUM3QixrREFBZ0M7QUFDaEMsOENBQTRCIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBNZW1vcnkgU3lzdGVtIC0gTWFpbiBFeHBvcnRcbiAqIFxuICogTExNLW5hdGl2ZSBtZW1vcnkgc3lzdGVtIHdpdGggZXh0cmFjdGlvbiwgY29uc29saWRhdGlvbiwgYW5kIHJldHJpZXZhbC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL3R5cGVzJztcbmV4cG9ydCAqIGZyb20gJy4vZXh0cmFjdGlvbic7XG5leHBvcnQgKiBmcm9tICcuL2NvbnNvbGlkYXRpb24nO1xuZXhwb3J0ICogZnJvbSAnLi9yZXRyaWV2YWwnO1xuIl19
@@ -0,0 +1,232 @@
1
+ "use strict";
2
+ /**
3
+ * Hybrid Retriever for Memory System
4
+ *
5
+ * Combines vector and keyword search with RRF (Reciprocal Rank Fusion).
6
+ */
7
+ Object.defineProperty(exports, "__esModule", { value: true });
8
+ exports.HybridRetriever = void 0;
9
+ /**
10
+ * Simple BM25 scorer for keyword matching
11
+ */
12
+ class BM25Scorer {
13
+ constructor() {
14
+ this.k1 = 1.5;
15
+ this.b = 0.75;
16
+ this.avgDocLength = 0;
17
+ this.docLengths = new Map();
18
+ this.termFreqs = new Map();
19
+ this.docFreqs = new Map();
20
+ this.numDocs = 0;
21
+ }
22
+ /**
23
+ * Index a document
24
+ */
25
+ indexDocument(docId, text) {
26
+ const terms = this.tokenize(text);
27
+ const termCounts = new Map();
28
+ for (const term of terms) {
29
+ termCounts.set(term, (termCounts.get(term) || 0) + 1);
30
+ }
31
+ this.docLengths.set(docId, terms.length);
32
+ this.termFreqs.set(docId, termCounts);
33
+ for (const term of termCounts.keys()) {
34
+ this.docFreqs.set(term, (this.docFreqs.get(term) || 0) + 1);
35
+ }
36
+ this.numDocs++;
37
+ this.updateAvgDocLength();
38
+ }
39
+ /**
40
+ * Score a query against a document
41
+ */
42
+ score(docId, queryTerms) {
43
+ const termCounts = this.termFreqs.get(docId);
44
+ if (!termCounts)
45
+ return 0;
46
+ const docLength = this.docLengths.get(docId) || 0;
47
+ let score = 0;
48
+ for (const term of queryTerms) {
49
+ const tf = termCounts.get(term) || 0;
50
+ const df = this.docFreqs.get(term) || 0;
51
+ if (tf === 0)
52
+ continue;
53
+ const idf = Math.log((this.numDocs - df + 0.5) / (df + 0.5) + 1);
54
+ const norm = tf / (tf + this.k1 * (1 - this.b + this.b * (docLength / this.avgDocLength)));
55
+ score += idf * norm;
56
+ }
57
+ return score;
58
+ }
59
+ /**
60
+ * Tokenize text into terms
61
+ */
62
+ tokenize(text) {
63
+ return text
64
+ .toLowerCase()
65
+ .replace(/[^\w\s]/g, ' ')
66
+ .split(/\s+/)
67
+ .filter(t => t.length > 0);
68
+ }
69
+ /**
70
+ * Update average document length
71
+ */
72
+ updateAvgDocLength() {
73
+ const totalLength = Array.from(this.docLengths.values()).reduce((a, b) => a + b, 0);
74
+ this.avgDocLength = totalLength / this.numDocs;
75
+ }
76
+ }
77
+ /**
78
+ * Cosine similarity for vector search
79
+ */
80
+ function cosineSimilarity(a, b) {
81
+ if (a.length !== b.length)
82
+ return 0;
83
+ let dotProduct = 0;
84
+ let normA = 0;
85
+ let normB = 0;
86
+ for (let i = 0; i < a.length; i++) {
87
+ dotProduct += a[i] * b[i];
88
+ normA += a[i] * a[i];
89
+ normB += b[i] * b[i];
90
+ }
91
+ if (normA === 0 || normB === 0)
92
+ return 0;
93
+ return dotProduct / (Math.sqrt(normA) * Math.sqrt(normB));
94
+ }
95
+ /**
96
+ * Hybrid Retriever with RRF fusion
97
+ */
98
+ class HybridRetriever {
99
+ constructor(db, namespace, collection, config) {
100
+ this.indexed = false;
101
+ this.db = db;
102
+ this.namespace = namespace;
103
+ this.collection = collection;
104
+ this.config = {
105
+ k: config?.k || 10,
106
+ alpha: config?.alpha !== undefined ? config.alpha : 0.5,
107
+ enableRerank: config?.enableRerank || false,
108
+ rerankK: config?.rerankK || 100,
109
+ };
110
+ this.prefix = Buffer.from(`retrieval:${namespace}:${collection}:`);
111
+ this.bm25 = new BM25Scorer();
112
+ }
113
+ /**
114
+ * Create retriever from database
115
+ */
116
+ static fromDatabase(db, namespace, collection, config) {
117
+ return new HybridRetriever(db, namespace, collection, config);
118
+ }
119
+ /**
120
+ * Index documents for retrieval
121
+ */
122
+ async indexDocuments(documents) {
123
+ for (const doc of documents) {
124
+ const key = Buffer.concat([this.prefix, Buffer.from(doc.id)]);
125
+ await this.db.put(key, Buffer.from(JSON.stringify({
126
+ id: doc.id,
127
+ content: doc.content,
128
+ embedding: doc.embedding,
129
+ metadata: doc.metadata,
130
+ })));
131
+ // Index for BM25
132
+ this.bm25.indexDocument(doc.id, doc.content);
133
+ }
134
+ this.indexed = true;
135
+ }
136
+ /**
137
+ * Retrieve documents with hybrid search
138
+ */
139
+ async retrieve(queryText, queryVector, allowed, k) {
140
+ const startTime = Date.now();
141
+ const targetK = k || this.config.k;
142
+ // Get all documents
143
+ const documents = [];
144
+ for await (const [_, value] of this.db.scanPrefix(this.prefix)) {
145
+ const doc = JSON.parse(value.toString());
146
+ // Apply pre-filtering with AllowedSet
147
+ if (allowed.contains(doc.id, doc.metadata)) {
148
+ documents.push(doc);
149
+ }
150
+ }
151
+ // Vector search scores
152
+ const vectorScores = new Map();
153
+ documents
154
+ .map(doc => ({
155
+ id: doc.id,
156
+ score: cosineSimilarity(queryVector, doc.embedding),
157
+ }))
158
+ .sort((a, b) => b.score - a.score)
159
+ .forEach((item, rank) => {
160
+ vectorScores.set(item.id, { score: item.score, rank });
161
+ });
162
+ // Keyword search scores (BM25)
163
+ const queryTerms = queryText
164
+ .toLowerCase()
165
+ .replace(/[^\w\s]/g, ' ')
166
+ .split(/\s+/)
167
+ .filter(t => t.length > 0);
168
+ const keywordScores = new Map();
169
+ documents
170
+ .map(doc => ({
171
+ id: doc.id,
172
+ score: this.bm25.score(doc.id, queryTerms),
173
+ }))
174
+ .sort((a, b) => b.score - a.score)
175
+ .forEach((item, rank) => {
176
+ keywordScores.set(item.id, { score: item.score, rank });
177
+ });
178
+ // RRF (Reciprocal Rank Fusion)
179
+ const k_rrf = 60; // RRF constant
180
+ const alpha = this.config.alpha;
181
+ const fusedScores = documents.map(doc => {
182
+ const vectorData = vectorScores.get(doc.id);
183
+ const keywordData = keywordScores.get(doc.id);
184
+ const vectorScore = vectorData ? 1 / (k_rrf + vectorData.rank) : 0;
185
+ const keywordScore = keywordData ? 1 / (k_rrf + keywordData.rank) : 0;
186
+ // Weighted combination
187
+ const finalScore = alpha * vectorScore + (1 - alpha) * keywordScore;
188
+ return {
189
+ id: doc.id,
190
+ score: finalScore,
191
+ content: doc.content,
192
+ metadata: doc.metadata,
193
+ vectorRank: vectorData?.rank,
194
+ keywordRank: keywordData?.rank,
195
+ };
196
+ });
197
+ // Sort by fused score and take top k
198
+ fusedScores.sort((a, b) => b.score - a.score);
199
+ const results = fusedScores.slice(0, targetK);
200
+ return {
201
+ results,
202
+ queryTime: Date.now() - startTime,
203
+ totalResults: documents.length,
204
+ };
205
+ }
206
+ /**
207
+ * Explain ranking for a specific document
208
+ */
209
+ async explain(queryText, queryVector, docId) {
210
+ // Simplified version - full implementation would require re-running retrieval
211
+ const key = Buffer.concat([this.prefix, Buffer.from(docId)]);
212
+ const value = await this.db.get(key);
213
+ if (!value) {
214
+ return {};
215
+ }
216
+ const doc = JSON.parse(value.toString());
217
+ const vectorScore = cosineSimilarity(queryVector, doc.embedding);
218
+ const queryTerms = queryText
219
+ .toLowerCase()
220
+ .replace(/[^\w\s]/g, ' ')
221
+ .split(/\s+/)
222
+ .filter(t => t.length > 0);
223
+ const keywordScore = this.bm25.score(docId, queryTerms);
224
+ return {
225
+ vectorRank: undefined, // Would need full ranking
226
+ keywordRank: undefined,
227
+ expectedRrfScore: vectorScore + keywordScore, // Simplified
228
+ };
229
+ }
230
+ }
231
+ exports.HybridRetriever = HybridRetriever;
232
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"retrieval.js","sourceRoot":"","sources":["../../../src/memory/retrieval.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Hybrid Retriever for Memory System\n * \n * Combines vector and keyword search with RRF (Reciprocal Rank Fusion).\n */\n\nimport { EmbeddedDatabase } from '../embedded';\nimport {\n  AllowedSet,\n  RetrievalConfig,\n  RetrievalResponse,\n  RetrievalResult,\n} from './types';\n\n/**\n * Simple BM25 scorer for keyword matching\n */\nclass BM25Scorer {\n  private k1 = 1.5;\n  private b = 0.75;\n  private avgDocLength: number;\n  private docLengths: Map<string, number>;\n  private termFreqs: Map<string, Map<string, number>>;\n  private docFreqs: Map<string, number>;\n  private numDocs: number;\n\n  constructor() {\n    this.avgDocLength = 0;\n    this.docLengths = new Map();\n    this.termFreqs = new Map();\n    this.docFreqs = new Map();\n    this.numDocs = 0;\n  }\n\n  /**\n   * Index a document\n   */\n  indexDocument(docId: string, text: string): void {\n    const terms = this.tokenize(text);\n    const termCounts = new Map<string, number>();\n\n    for (const term of terms) {\n      termCounts.set(term, (termCounts.get(term) || 0) + 1);\n    }\n\n    this.docLengths.set(docId, terms.length);\n    this.termFreqs.set(docId, termCounts);\n\n    for (const term of termCounts.keys()) {\n      this.docFreqs.set(term, (this.docFreqs.get(term) || 0) + 1);\n    }\n\n    this.numDocs++;\n    this.updateAvgDocLength();\n  }\n\n  /**\n   * Score a query against a document\n   */\n  score(docId: string, queryTerms: string[]): number {\n    const termCounts = this.termFreqs.get(docId);\n    if (!termCounts) return 0;\n\n    const docLength = this.docLengths.get(docId) || 0;\n    let score = 0;\n\n    for (const term of queryTerms) {\n      const tf = termCounts.get(term) || 0;\n      const df = this.docFreqs.get(term) || 0;\n      \n      if (tf === 0) continue;\n\n      const idf = Math.log((this.numDocs - df + 0.5) / (df + 0.5) + 1);\n      const norm = tf / (tf + this.k1 * (1 - this.b + this.b * (docLength / this.avgDocLength)));\n      \n      score += idf * norm;\n    }\n\n    return score;\n  }\n\n  /**\n   * Tokenize text into terms\n   */\n  private tokenize(text: string): string[] {\n    return text\n      .toLowerCase()\n      .replace(/[^\\w\\s]/g, ' ')\n      .split(/\\s+/)\n      .filter(t => t.length > 0);\n  }\n\n  /**\n   * Update average document length\n   */\n  private updateAvgDocLength(): void {\n    const totalLength = Array.from(this.docLengths.values()).reduce((a, b) => a + b, 0);\n    this.avgDocLength = totalLength / this.numDocs;\n  }\n}\n\n/**\n * Cosine similarity for vector search\n */\nfunction cosineSimilarity(a: number[], b: number[]): number {\n  if (a.length !== b.length) return 0;\n\n  let dotProduct = 0;\n  let normA = 0;\n  let normB = 0;\n\n  for (let i = 0; i < a.length; i++) {\n    dotProduct += a[i] * b[i];\n    normA += a[i] * a[i];\n    normB += b[i] * b[i];\n  }\n\n  if (normA === 0 || normB === 0) return 0;\n  return dotProduct / (Math.sqrt(normA) * Math.sqrt(normB));\n}\n\n/**\n * Hybrid Retriever with RRF fusion\n */\nexport class HybridRetriever {\n  private db: EmbeddedDatabase;\n  private namespace: string;\n  private collection: string;\n  private config: RetrievalConfig;\n  private prefix: Buffer;\n  private bm25: BM25Scorer;\n  private indexed: boolean = false;\n\n  constructor(\n    db: EmbeddedDatabase,\n    namespace: string,\n    collection: string,\n    config?: RetrievalConfig\n  ) {\n    this.db = db;\n    this.namespace = namespace;\n    this.collection = collection;\n    this.config = {\n      k: config?.k || 10,\n      alpha: config?.alpha !== undefined ? config.alpha : 0.5,\n      enableRerank: config?.enableRerank || false,\n      rerankK: config?.rerankK || 100,\n    };\n    this.prefix = Buffer.from(`retrieval:${namespace}:${collection}:`);\n    this.bm25 = new BM25Scorer();\n  }\n\n  /**\n   * Create retriever from database\n   */\n  static fromDatabase(\n    db: EmbeddedDatabase,\n    namespace: string,\n    collection: string,\n    config?: RetrievalConfig\n  ): HybridRetriever {\n    return new HybridRetriever(db, namespace, collection, config);\n  }\n\n  /**\n   * Index documents for retrieval\n   */\n  async indexDocuments(documents: Array<{ id: string; content: string; embedding: number[]; metadata?: Record<string, any> }>): Promise<void> {\n    for (const doc of documents) {\n      const key = Buffer.concat([this.prefix, Buffer.from(doc.id)]);\n      await this.db.put(key, Buffer.from(JSON.stringify({\n        id: doc.id,\n        content: doc.content,\n        embedding: doc.embedding,\n        metadata: doc.metadata,\n      })));\n\n      // Index for BM25\n      this.bm25.indexDocument(doc.id, doc.content);\n    }\n\n    this.indexed = true;\n  }\n\n  /**\n   * Retrieve documents with hybrid search\n   */\n  async retrieve(\n    queryText: string,\n    queryVector: number[],\n    allowed: AllowedSet,\n    k?: number\n  ): Promise<RetrievalResponse> {\n    const startTime = Date.now();\n    const targetK = k || this.config.k!;\n\n    // Get all documents\n    const documents: Array<{\n      id: string;\n      content: string;\n      embedding: number[];\n      metadata?: Record<string, any>;\n    }> = [];\n\n    for await (const [_, value] of this.db.scanPrefix(this.prefix)) {\n      const doc = JSON.parse(value.toString());\n      \n      // Apply pre-filtering with AllowedSet\n      if (allowed.contains(doc.id, doc.metadata)) {\n        documents.push(doc);\n      }\n    }\n\n    // Vector search scores\n    const vectorScores = new Map<string, { score: number; rank: number }>();\n    documents\n      .map(doc => ({\n        id: doc.id,\n        score: cosineSimilarity(queryVector, doc.embedding),\n      }))\n      .sort((a, b) => b.score - a.score)\n      .forEach((item, rank) => {\n        vectorScores.set(item.id, { score: item.score, rank });\n      });\n\n    // Keyword search scores (BM25)\n    const queryTerms = queryText\n      .toLowerCase()\n      .replace(/[^\\w\\s]/g, ' ')\n      .split(/\\s+/)\n      .filter(t => t.length > 0);\n\n    const keywordScores = new Map<string, { score: number; rank: number }>();\n    documents\n      .map(doc => ({\n        id: doc.id,\n        score: this.bm25.score(doc.id, queryTerms),\n      }))\n      .sort((a, b) => b.score - a.score)\n      .forEach((item, rank) => {\n        keywordScores.set(item.id, { score: item.score, rank });\n      });\n\n    // RRF (Reciprocal Rank Fusion)\n    const k_rrf = 60; // RRF constant\n    const alpha = this.config.alpha!;\n    \n    const fusedScores = documents.map(doc => {\n      const vectorData = vectorScores.get(doc.id);\n      const keywordData = keywordScores.get(doc.id);\n      \n      const vectorScore = vectorData ? 1 / (k_rrf + vectorData.rank) : 0;\n      const keywordScore = keywordData ? 1 / (k_rrf + keywordData.rank) : 0;\n      \n      // Weighted combination\n      const finalScore = alpha * vectorScore + (1 - alpha) * keywordScore;\n\n      return {\n        id: doc.id,\n        score: finalScore,\n        content: doc.content,\n        metadata: doc.metadata,\n        vectorRank: vectorData?.rank,\n        keywordRank: keywordData?.rank,\n      };\n    });\n\n    // Sort by fused score and take top k\n    fusedScores.sort((a, b) => b.score - a.score);\n    const results = fusedScores.slice(0, targetK);\n\n    return {\n      results,\n      queryTime: Date.now() - startTime,\n      totalResults: documents.length,\n    };\n  }\n\n  /**\n   * Explain ranking for a specific document\n   */\n  async explain(\n    queryText: string,\n    queryVector: number[],\n    docId: string\n  ): Promise<{ vectorRank?: number; keywordRank?: number; expectedRrfScore?: number }> {\n    // Simplified version - full implementation would require re-running retrieval\n    const key = Buffer.concat([this.prefix, Buffer.from(docId)]);\n    const value = await this.db.get(key);\n    \n    if (!value) {\n      return {};\n    }\n\n    const doc = JSON.parse(value.toString());\n    \n    const vectorScore = cosineSimilarity(queryVector, doc.embedding);\n    const queryTerms = queryText\n      .toLowerCase()\n      .replace(/[^\\w\\s]/g, ' ')\n      .split(/\\s+/)\n      .filter(t => t.length > 0);\n    const keywordScore = this.bm25.score(docId, queryTerms);\n\n    return {\n      vectorRank: undefined, // Would need full ranking\n      keywordRank: undefined,\n      expectedRrfScore: vectorScore + keywordScore, // Simplified\n    };\n  }\n}\n"]}
@@ -0,0 +1,69 @@
1
+ "use strict";
2
+ /**
3
+ * Memory System Core Types
4
+ *
5
+ * Type definitions for LLM-native memory system with extraction,
6
+ * consolidation, and retrieval capabilities.
7
+ */
8
+ Object.defineProperty(exports, "__esModule", { value: true });
9
+ exports.NamespacePolicy = exports.AllowedSet = void 0;
10
+ /**
11
+ * AllowedSet for pre-filtering
12
+ */
13
+ class AllowedSet {
14
+ static fromIds(ids) {
15
+ return new IdsAllowedSet(new Set(ids));
16
+ }
17
+ static fromNamespace(namespace) {
18
+ return new NamespaceAllowedSet(namespace);
19
+ }
20
+ static fromFilter(filterFn) {
21
+ return new FilterAllowedSet(filterFn);
22
+ }
23
+ static allowAll() {
24
+ return new AllAllowedSet();
25
+ }
26
+ }
27
+ exports.AllowedSet = AllowedSet;
28
+ class IdsAllowedSet extends AllowedSet {
29
+ constructor(ids) {
30
+ super();
31
+ this.ids = ids;
32
+ }
33
+ contains(id) {
34
+ return this.ids.has(id);
35
+ }
36
+ }
37
+ class NamespaceAllowedSet extends AllowedSet {
38
+ constructor(namespace) {
39
+ super();
40
+ this.namespace = namespace;
41
+ }
42
+ contains(id) {
43
+ return id.startsWith(`${this.namespace}_`) || id.startsWith(`${this.namespace}:`);
44
+ }
45
+ }
46
+ class FilterAllowedSet extends AllowedSet {
47
+ constructor(filterFn) {
48
+ super();
49
+ this.filterFn = filterFn;
50
+ }
51
+ contains(id, metadata) {
52
+ return this.filterFn(id, metadata);
53
+ }
54
+ }
55
+ class AllAllowedSet extends AllowedSet {
56
+ contains() {
57
+ return true;
58
+ }
59
+ }
60
+ /**
61
+ * Namespace policy
62
+ */
63
+ var NamespacePolicy;
64
+ (function (NamespacePolicy) {
65
+ NamespacePolicy["STRICT"] = "strict";
66
+ NamespacePolicy["EXPLICIT"] = "explicit";
67
+ NamespacePolicy["PERMISSIVE"] = "permissive";
68
+ })(NamespacePolicy || (exports.NamespacePolicy = NamespacePolicy = {}));
69
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbWVtb3J5L3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7Ozs7R0FLRzs7O0FBNEhIOztHQUVHO0FBQ0gsTUFBc0IsVUFBVTtJQUc5QixNQUFNLENBQUMsT0FBTyxDQUFDLEdBQWE7UUFDMUIsT0FBTyxJQUFJLGFBQWEsQ0FBQyxJQUFJLEdBQUcsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDO0lBQ3pDLENBQUM7SUFFRCxNQUFNLENBQUMsYUFBYSxDQUFDLFNBQWlCO1FBQ3BDLE9BQU8sSUFBSSxtQkFBbUIsQ0FBQyxTQUFTLENBQUMsQ0FBQztJQUM1QyxDQUFDO0lBRUQsTUFBTSxDQUFDLFVBQVUsQ0FBQyxRQUFpRTtRQUNqRixPQUFPLElBQUksZ0JBQWdCLENBQUMsUUFBUSxDQUFDLENBQUM7SUFDeEMsQ0FBQztJQUVELE1BQU0sQ0FBQyxRQUFRO1FBQ2IsT0FBTyxJQUFJLGFBQWEsRUFBRSxDQUFDO0lBQzdCLENBQUM7Q0FDRjtBQWxCRCxnQ0FrQkM7QUFFRCxNQUFNLGFBQWMsU0FBUSxVQUFVO0lBQ3BDLFlBQW9CLEdBQWdCO1FBQ2xDLEtBQUssRUFBRSxDQUFDO1FBRFUsUUFBRyxHQUFILEdBQUcsQ0FBYTtJQUVwQyxDQUFDO0lBRUQsUUFBUSxDQUFDLEVBQVU7UUFDakIsT0FBTyxJQUFJLENBQUMsR0FBRyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsQ0FBQztJQUMxQixDQUFDO0NBQ0Y7QUFFRCxNQUFNLG1CQUFvQixTQUFRLFVBQVU7SUFDMUMsWUFBb0IsU0FBaUI7UUFDbkMsS0FBSyxFQUFFLENBQUM7UUFEVSxjQUFTLEdBQVQsU0FBUyxDQUFRO0lBRXJDLENBQUM7SUFFRCxRQUFRLENBQUMsRUFBVTtRQUNqQixPQUFPLEVBQUUsQ0FBQyxVQUFVLENBQUMsR0FBRyxJQUFJLENBQUMsU0FBUyxHQUFHLENBQUMsSUFBSSxFQUFFLENBQUMsVUFBVSxDQUFDLEdBQUcsSUFBSSxDQUFDLFNBQVMsR0FBRyxDQUFDLENBQUM7SUFDcEYsQ0FBQztDQUNGO0FBRUQsTUFBTSxnQkFBaUIsU0FBUSxVQUFVO0lBQ3ZDLFlBQW9CLFFBQWlFO1FBQ25GLEtBQUssRUFBRSxDQUFDO1FBRFUsYUFBUSxHQUFSLFFBQVEsQ0FBeUQ7SUFFckYsQ0FBQztJQUVELFFBQVEsQ0FBQyxFQUFVLEVBQUUsUUFBOEI7UUFDakQsT0FBTyxJQUFJLENBQUMsUUFBUSxDQUFDLEVBQUUsRUFBRSxRQUFRLENBQUMsQ0FBQztJQUNyQyxDQUFDO0NBQ0Y7QUFFRCxNQUFNLGFBQWMsU0FBUSxVQUFVO0lBQ3BDLFFBQVE7UUFDTixPQUFPLElBQUksQ0FBQztJQUNkLENBQUM7Q0FDRjtBQUVEOztHQUVHO0FBQ0gsSUFBWSxlQUlYO0FBSkQsV0FBWSxlQUFlO0lBQ3pCLG9DQUFpQixDQUFBO0lBQ2pCLHdDQUFxQixDQUFBO0lBQ3JCLDRDQUF5QixDQUFBO0FBQzNCLENBQUMsRUFKVyxlQUFlLCtCQUFmLGVBQWUsUUFJMUIiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIE1lbW9yeSBTeXN0ZW0gQ29yZSBUeXBlc1xuICogXG4gKiBUeXBlIGRlZmluaXRpb25zIGZvciBMTE0tbmF0aXZlIG1lbW9yeSBzeXN0ZW0gd2l0aCBleHRyYWN0aW9uLFxuICogY29uc29saWRhdGlvbiwgYW5kIHJldHJpZXZhbCBjYXBhYmlsaXRpZXMuXG4gKi9cblxuLyoqXG4gKiBFbnRpdHkgZXh0cmFjdGVkIGZyb20gdGV4dFxuICovXG5leHBvcnQgaW50ZXJmYWNlIEVudGl0eSB7XG4gIGlkPzogc3RyaW5nO1xuICBuYW1lOiBzdHJpbmc7XG4gIGVudGl0eVR5cGU6IHN0cmluZztcbiAgcHJvcGVydGllcz86IFJlY29yZDxzdHJpbmcsIGFueT47XG4gIGNvbmZpZGVuY2U/OiBudW1iZXI7XG4gIHByb3ZlbmFuY2U/OiBzdHJpbmc7XG4gIHRpbWVzdGFtcD86IG51bWJlcjtcbn1cblxuLyoqXG4gKiBSZWxhdGlvbiBiZXR3ZWVuIHR3byBlbnRpdGllc1xuICovXG5leHBvcnQgaW50ZXJmYWNlIFJlbGF0aW9uIHtcbiAgaWQ/OiBzdHJpbmc7XG4gIGZyb21FbnRpdHk6IHN0cmluZztcbiAgcmVsYXRpb25UeXBlOiBzdHJpbmc7XG4gIHRvRW50aXR5OiBzdHJpbmc7XG4gIHByb3BlcnRpZXM/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+O1xuICBjb25maWRlbmNlPzogbnVtYmVyO1xuICBwcm92ZW5hbmNlPzogc3RyaW5nO1xuICB0aW1lc3RhbXA/OiBudW1iZXI7XG59XG5cbi8qKlxuICogQXNzZXJ0aW9uIChzdWJqZWN0LXByZWRpY2F0ZS1vYmplY3QgdHJpcGxlKVxuICovXG5leHBvcnQgaW50ZXJmYWNlIEFzc2VydGlvbiB7XG4gIGlkPzogc3RyaW5nO1xuICBzdWJqZWN0OiBzdHJpbmc7XG4gIHByZWRpY2F0ZTogc3RyaW5nO1xuICBvYmplY3Q6IHN0cmluZztcbiAgY29uZmlkZW5jZT86IG51bWJlcjtcbiAgcHJvdmVuYW5jZT86IHN0cmluZztcbiAgdGltZXN0YW1wPzogbnVtYmVyO1xufVxuXG4vKipcbiAqIFJhdyBhc3NlcnRpb24gZm9yIGNvbnNvbGlkYXRpb25cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBSYXdBc3NlcnRpb24ge1xuICBpZD86IHN0cmluZztcbiAgZmFjdDogUmVjb3JkPHN0cmluZywgYW55PjtcbiAgc291cmNlOiBzdHJpbmc7XG4gIGNvbmZpZGVuY2U6IG51bWJlcjtcbiAgdGltZXN0YW1wPzogbnVtYmVyO1xufVxuXG4vKipcbiAqIENhbm9uaWNhbCBmYWN0IGFmdGVyIGNvbnNvbGlkYXRpb25cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBDYW5vbmljYWxGYWN0IHtcbiAgaWQ6IHN0cmluZztcbiAgbWVyZ2VkRmFjdDogUmVjb3JkPHN0cmluZywgYW55PjtcbiAgY29uZmlkZW5jZTogbnVtYmVyO1xuICBzb3VyY2VzOiBzdHJpbmdbXTtcbiAgdmFsaWRGcm9tOiBudW1iZXI7XG4gIHZhbGlkVW50aWw/OiBudW1iZXI7XG59XG5cbi8qKlxuICogRXh0cmFjdGlvbiByZXN1bHQgZnJvbSBMTE1cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBFeHRyYWN0aW9uUmVzdWx0IHtcbiAgZW50aXRpZXM6IEVudGl0eVtdO1xuICByZWxhdGlvbnM6IFJlbGF0aW9uW107XG4gIGFzc2VydGlvbnM6IEFzc2VydGlvbltdO1xufVxuXG4vKipcbiAqIEV4dHJhY3Rpb24gc2NoZW1hIGZvciB2YWxpZGF0aW9uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgRXh0cmFjdGlvblNjaGVtYSB7XG4gIGVudGl0eVR5cGVzPzogc3RyaW5nW107XG4gIHJlbGF0aW9uVHlwZXM/OiBzdHJpbmdbXTtcbiAgbWluQ29uZmlkZW5jZT86IG51bWJlcjtcbiAgcmVxdWlyZVByb3ZlbmFuY2U/OiBib29sZWFuO1xufVxuXG4vKipcbiAqIENvbnNvbGlkYXRpb24gY29uZmlndXJhdGlvblxuICovXG5leHBvcnQgaW50ZXJmYWNlIENvbnNvbGlkYXRpb25Db25maWcge1xuICBzaW1pbGFyaXR5VGhyZXNob2xkPzogbnVtYmVyO1xuICB1c2VUZW1wb3JhbFVwZGF0ZXM/OiBib29sZWFuO1xuICBtYXhDb25mbGljdEFnZT86IG51bWJlcjsgLy8gc2Vjb25kc1xufVxuXG4vKipcbiAqIFJldHJpZXZhbCBjb25maWd1cmF0aW9uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUmV0cmlldmFsQ29uZmlnIHtcbiAgaz86IG51bWJlcjtcbiAgYWxwaGE/OiBudW1iZXI7IC8vIDA9a2V5d29yZCBvbmx5LCAxPXZlY3RvciBvbmx5LCAwLjU9YmFsYW5jZWRcbiAgZW5hYmxlUmVyYW5rPzogYm9vbGVhbjtcbiAgcmVyYW5rSz86IG51bWJlcjtcbn1cblxuLyoqXG4gKiBSZXRyaWV2YWwgcmVzdWx0XG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUmV0cmlldmFsUmVzdWx0IHtcbiAgaWQ6IHN0cmluZztcbiAgc2NvcmU6IG51bWJlcjtcbiAgY29udGVudDogc3RyaW5nO1xuICBtZXRhZGF0YT86IFJlY29yZDxzdHJpbmcsIGFueT47XG4gIHZlY3RvclJhbms/OiBudW1iZXI7XG4gIGtleXdvcmRSYW5rPzogbnVtYmVyO1xufVxuXG4vKipcbiAqIFJldHJpZXZhbCByZXNwb25zZVxuICovXG5leHBvcnQgaW50ZXJmYWNlIFJldHJpZXZhbFJlc3BvbnNlIHtcbiAgcmVzdWx0czogUmV0cmlldmFsUmVzdWx0W107XG4gIHF1ZXJ5VGltZTogbnVtYmVyO1xuICB0b3RhbFJlc3VsdHM6IG51bWJlcjtcbn1cblxuLyoqXG4gKiBBbGxvd2VkU2V0IGZvciBwcmUtZmlsdGVyaW5nXG4gKi9cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBBbGxvd2VkU2V0IHtcbiAgYWJzdHJhY3QgY29udGFpbnMoaWQ6IHN0cmluZywgbWV0YWRhdGE/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+KTogYm9vbGVhbjtcblxuICBzdGF0aWMgZnJvbUlkcyhpZHM6IHN0cmluZ1tdKTogQWxsb3dlZFNldCB7XG4gICAgcmV0dXJuIG5ldyBJZHNBbGxvd2VkU2V0KG5ldyBTZXQoaWRzKSk7XG4gIH1cblxuICBzdGF0aWMgZnJvbU5hbWVzcGFjZShuYW1lc3BhY2U6IHN0cmluZyk6IEFsbG93ZWRTZXQge1xuICAgIHJldHVybiBuZXcgTmFtZXNwYWNlQWxsb3dlZFNldChuYW1lc3BhY2UpO1xuICB9XG5cbiAgc3RhdGljIGZyb21GaWx0ZXIoZmlsdGVyRm46IChpZDogc3RyaW5nLCBtZXRhZGF0YT86IFJlY29yZDxzdHJpbmcsIGFueT4pID0+IGJvb2xlYW4pOiBBbGxvd2VkU2V0IHtcbiAgICByZXR1cm4gbmV3IEZpbHRlckFsbG93ZWRTZXQoZmlsdGVyRm4pO1xuICB9XG5cbiAgc3RhdGljIGFsbG93QWxsKCk6IEFsbG93ZWRTZXQge1xuICAgIHJldHVybiBuZXcgQWxsQWxsb3dlZFNldCgpO1xuICB9XG59XG5cbmNsYXNzIElkc0FsbG93ZWRTZXQgZXh0ZW5kcyBBbGxvd2VkU2V0IHtcbiAgY29uc3RydWN0b3IocHJpdmF0ZSBpZHM6IFNldDxzdHJpbmc+KSB7XG4gICAgc3VwZXIoKTtcbiAgfVxuXG4gIGNvbnRhaW5zKGlkOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5pZHMuaGFzKGlkKTtcbiAgfVxufVxuXG5jbGFzcyBOYW1lc3BhY2VBbGxvd2VkU2V0IGV4dGVuZHMgQWxsb3dlZFNldCB7XG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgbmFtZXNwYWNlOiBzdHJpbmcpIHtcbiAgICBzdXBlcigpO1xuICB9XG5cbiAgY29udGFpbnMoaWQ6IHN0cmluZyk6IGJvb2xlYW4ge1xuICAgIHJldHVybiBpZC5zdGFydHNXaXRoKGAke3RoaXMubmFtZXNwYWNlfV9gKSB8fCBpZC5zdGFydHNXaXRoKGAke3RoaXMubmFtZXNwYWNlfTpgKTtcbiAgfVxufVxuXG5jbGFzcyBGaWx0ZXJBbGxvd2VkU2V0IGV4dGVuZHMgQWxsb3dlZFNldCB7XG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgZmlsdGVyRm46IChpZDogc3RyaW5nLCBtZXRhZGF0YT86IFJlY29yZDxzdHJpbmcsIGFueT4pID0+IGJvb2xlYW4pIHtcbiAgICBzdXBlcigpO1xuICB9XG5cbiAgY29udGFpbnMoaWQ6IHN0cmluZywgbWV0YWRhdGE/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+KTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMuZmlsdGVyRm4oaWQsIG1ldGFkYXRhKTtcbiAgfVxufVxuXG5jbGFzcyBBbGxBbGxvd2VkU2V0IGV4dGVuZHMgQWxsb3dlZFNldCB7XG4gIGNvbnRhaW5zKCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0cnVlO1xuICB9XG59XG5cbi8qKlxuICogTmFtZXNwYWNlIHBvbGljeVxuICovXG5leHBvcnQgZW51bSBOYW1lc3BhY2VQb2xpY3kge1xuICBTVFJJQ1QgPSAnc3RyaWN0JyxcbiAgRVhQTElDSVQgPSAnZXhwbGljaXQnLFxuICBQRVJNSVNTSVZFID0gJ3Blcm1pc3NpdmUnLFxufVxuXG4vKipcbiAqIENyb3NzLW5hbWVzcGFjZSBncmFudFxuICovXG5leHBvcnQgaW50ZXJmYWNlIE5hbWVzcGFjZUdyYW50IHtcbiAgaWQ6IHN0cmluZztcbiAgZnJvbU5hbWVzcGFjZTogc3RyaW5nO1xuICB0b05hbWVzcGFjZTogc3RyaW5nO1xuICBvcGVyYXRpb25zOiBzdHJpbmdbXTtcbiAgZXhwaXJlc0F0PzogbnVtYmVyO1xuICByZWFzb24/OiBzdHJpbmc7XG59XG4iXX0=