@soleri/core 2.4.0 → 2.6.0
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/dist/brain/brain.d.ts +7 -0
- package/dist/brain/brain.d.ts.map +1 -1
- package/dist/brain/brain.js +56 -9
- package/dist/brain/brain.js.map +1 -1
- package/dist/brain/intelligence.d.ts +1 -0
- package/dist/brain/intelligence.d.ts.map +1 -1
- package/dist/brain/intelligence.js +164 -148
- package/dist/brain/intelligence.js.map +1 -1
- package/dist/brain/types.d.ts +2 -2
- package/dist/brain/types.d.ts.map +1 -1
- package/dist/cognee/client.d.ts +3 -0
- package/dist/cognee/client.d.ts.map +1 -1
- package/dist/cognee/client.js +17 -0
- package/dist/cognee/client.js.map +1 -1
- package/dist/cognee/sync-manager.d.ts +94 -0
- package/dist/cognee/sync-manager.d.ts.map +1 -0
- package/dist/cognee/sync-manager.js +293 -0
- package/dist/cognee/sync-manager.js.map +1 -0
- package/dist/control/identity-manager.d.ts +3 -1
- package/dist/control/identity-manager.d.ts.map +1 -1
- package/dist/control/identity-manager.js +49 -51
- package/dist/control/identity-manager.js.map +1 -1
- package/dist/control/intent-router.d.ts +1 -0
- package/dist/control/intent-router.d.ts.map +1 -1
- package/dist/control/intent-router.js +32 -32
- package/dist/control/intent-router.js.map +1 -1
- package/dist/curator/curator.d.ts +9 -1
- package/dist/curator/curator.d.ts.map +1 -1
- package/dist/curator/curator.js +104 -92
- package/dist/curator/curator.js.map +1 -1
- package/dist/errors/classify.d.ts +13 -0
- package/dist/errors/classify.d.ts.map +1 -0
- package/dist/errors/classify.js +97 -0
- package/dist/errors/classify.js.map +1 -0
- package/dist/errors/index.d.ts +6 -0
- package/dist/errors/index.d.ts.map +1 -0
- package/dist/errors/index.js +4 -0
- package/dist/errors/index.js.map +1 -0
- package/dist/errors/retry.d.ts +40 -0
- package/dist/errors/retry.d.ts.map +1 -0
- package/dist/errors/retry.js +97 -0
- package/dist/errors/retry.js.map +1 -0
- package/dist/errors/types.d.ts +48 -0
- package/dist/errors/types.d.ts.map +1 -0
- package/dist/errors/types.js +59 -0
- package/dist/errors/types.js.map +1 -0
- package/dist/governance/governance.d.ts +1 -0
- package/dist/governance/governance.d.ts.map +1 -1
- package/dist/governance/governance.js +51 -68
- package/dist/governance/governance.js.map +1 -1
- package/dist/index.d.ts +26 -5
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +22 -3
- package/dist/index.js.map +1 -1
- package/dist/intake/content-classifier.d.ts +14 -0
- package/dist/intake/content-classifier.d.ts.map +1 -0
- package/dist/intake/content-classifier.js +125 -0
- package/dist/intake/content-classifier.js.map +1 -0
- package/dist/intake/dedup-gate.d.ts +17 -0
- package/dist/intake/dedup-gate.d.ts.map +1 -0
- package/dist/intake/dedup-gate.js +66 -0
- package/dist/intake/dedup-gate.js.map +1 -0
- package/dist/intake/intake-pipeline.d.ts +63 -0
- package/dist/intake/intake-pipeline.d.ts.map +1 -0
- package/dist/intake/intake-pipeline.js +373 -0
- package/dist/intake/intake-pipeline.js.map +1 -0
- package/dist/intake/types.d.ts +65 -0
- package/dist/intake/types.d.ts.map +1 -0
- package/dist/intake/types.js +3 -0
- package/dist/intake/types.js.map +1 -0
- package/dist/intelligence/loader.js +1 -1
- package/dist/intelligence/loader.js.map +1 -1
- package/dist/intelligence/types.d.ts +3 -1
- package/dist/intelligence/types.d.ts.map +1 -1
- package/dist/loop/loop-manager.d.ts +58 -7
- package/dist/loop/loop-manager.d.ts.map +1 -1
- package/dist/loop/loop-manager.js +280 -6
- package/dist/loop/loop-manager.js.map +1 -1
- package/dist/loop/types.d.ts +69 -1
- package/dist/loop/types.d.ts.map +1 -1
- package/dist/loop/types.js +4 -1
- package/dist/loop/types.js.map +1 -1
- package/dist/persistence/index.d.ts +4 -0
- package/dist/persistence/index.d.ts.map +1 -0
- package/dist/persistence/index.js +3 -0
- package/dist/persistence/index.js.map +1 -0
- package/dist/persistence/postgres-provider.d.ts +46 -0
- package/dist/persistence/postgres-provider.d.ts.map +1 -0
- package/dist/persistence/postgres-provider.js +115 -0
- package/dist/persistence/postgres-provider.js.map +1 -0
- package/dist/persistence/sqlite-provider.d.ts +28 -0
- package/dist/persistence/sqlite-provider.d.ts.map +1 -0
- package/dist/persistence/sqlite-provider.js +97 -0
- package/dist/persistence/sqlite-provider.js.map +1 -0
- package/dist/persistence/types.d.ts +58 -0
- package/dist/persistence/types.d.ts.map +1 -0
- package/dist/persistence/types.js +8 -0
- package/dist/persistence/types.js.map +1 -0
- package/dist/planning/gap-analysis.d.ts +47 -4
- package/dist/planning/gap-analysis.d.ts.map +1 -1
- package/dist/planning/gap-analysis.js +190 -13
- package/dist/planning/gap-analysis.js.map +1 -1
- package/dist/planning/gap-types.d.ts +1 -1
- package/dist/planning/gap-types.d.ts.map +1 -1
- package/dist/planning/gap-types.js.map +1 -1
- package/dist/planning/planner.d.ts +277 -9
- package/dist/planning/planner.d.ts.map +1 -1
- package/dist/planning/planner.js +611 -46
- package/dist/planning/planner.js.map +1 -1
- package/dist/playbooks/generic/brainstorming.d.ts +9 -0
- package/dist/playbooks/generic/brainstorming.d.ts.map +1 -0
- package/dist/playbooks/generic/brainstorming.js +105 -0
- package/dist/playbooks/generic/brainstorming.js.map +1 -0
- package/dist/playbooks/generic/code-review.d.ts +11 -0
- package/dist/playbooks/generic/code-review.d.ts.map +1 -0
- package/dist/playbooks/generic/code-review.js +176 -0
- package/dist/playbooks/generic/code-review.js.map +1 -0
- package/dist/playbooks/generic/subagent-execution.d.ts +9 -0
- package/dist/playbooks/generic/subagent-execution.d.ts.map +1 -0
- package/dist/playbooks/generic/subagent-execution.js +68 -0
- package/dist/playbooks/generic/subagent-execution.js.map +1 -0
- package/dist/playbooks/generic/systematic-debugging.d.ts +9 -0
- package/dist/playbooks/generic/systematic-debugging.d.ts.map +1 -0
- package/dist/playbooks/generic/systematic-debugging.js +87 -0
- package/dist/playbooks/generic/systematic-debugging.js.map +1 -0
- package/dist/playbooks/generic/tdd.d.ts +9 -0
- package/dist/playbooks/generic/tdd.d.ts.map +1 -0
- package/dist/playbooks/generic/tdd.js +70 -0
- package/dist/playbooks/generic/tdd.js.map +1 -0
- package/dist/playbooks/generic/verification.d.ts +9 -0
- package/dist/playbooks/generic/verification.d.ts.map +1 -0
- package/dist/playbooks/generic/verification.js +74 -0
- package/dist/playbooks/generic/verification.js.map +1 -0
- package/dist/playbooks/index.d.ts +4 -0
- package/dist/playbooks/index.d.ts.map +1 -0
- package/dist/playbooks/index.js +5 -0
- package/dist/playbooks/index.js.map +1 -0
- package/dist/playbooks/playbook-registry.d.ts +42 -0
- package/dist/playbooks/playbook-registry.d.ts.map +1 -0
- package/dist/playbooks/playbook-registry.js +227 -0
- package/dist/playbooks/playbook-registry.js.map +1 -0
- package/dist/playbooks/playbook-seeder.d.ts +47 -0
- package/dist/playbooks/playbook-seeder.d.ts.map +1 -0
- package/dist/playbooks/playbook-seeder.js +104 -0
- package/dist/playbooks/playbook-seeder.js.map +1 -0
- package/dist/playbooks/playbook-types.d.ts +132 -0
- package/dist/playbooks/playbook-types.d.ts.map +1 -0
- package/dist/playbooks/playbook-types.js +12 -0
- package/dist/playbooks/playbook-types.js.map +1 -0
- package/dist/project/project-registry.d.ts +4 -4
- package/dist/project/project-registry.d.ts.map +1 -1
- package/dist/project/project-registry.js +30 -57
- package/dist/project/project-registry.js.map +1 -1
- package/dist/prompts/index.d.ts +4 -0
- package/dist/prompts/index.d.ts.map +1 -0
- package/dist/prompts/index.js +3 -0
- package/dist/prompts/index.js.map +1 -0
- package/dist/prompts/parser.d.ts +17 -0
- package/dist/prompts/parser.d.ts.map +1 -0
- package/dist/prompts/parser.js +47 -0
- package/dist/prompts/parser.js.map +1 -0
- package/dist/prompts/template-manager.d.ts +25 -0
- package/dist/prompts/template-manager.d.ts.map +1 -0
- package/dist/prompts/template-manager.js +71 -0
- package/dist/prompts/template-manager.js.map +1 -0
- package/dist/prompts/types.d.ts +26 -0
- package/dist/prompts/types.d.ts.map +1 -0
- package/dist/prompts/types.js +5 -0
- package/dist/prompts/types.js.map +1 -0
- package/dist/runtime/admin-extra-ops.d.ts +5 -3
- package/dist/runtime/admin-extra-ops.d.ts.map +1 -1
- package/dist/runtime/admin-extra-ops.js +348 -11
- package/dist/runtime/admin-extra-ops.js.map +1 -1
- package/dist/runtime/admin-ops.d.ts.map +1 -1
- package/dist/runtime/admin-ops.js +10 -3
- package/dist/runtime/admin-ops.js.map +1 -1
- package/dist/runtime/capture-ops.d.ts.map +1 -1
- package/dist/runtime/capture-ops.js +20 -2
- package/dist/runtime/capture-ops.js.map +1 -1
- package/dist/runtime/cognee-sync-ops.d.ts +12 -0
- package/dist/runtime/cognee-sync-ops.d.ts.map +1 -0
- package/dist/runtime/cognee-sync-ops.js +55 -0
- package/dist/runtime/cognee-sync-ops.js.map +1 -0
- package/dist/runtime/core-ops.d.ts +8 -6
- package/dist/runtime/core-ops.d.ts.map +1 -1
- package/dist/runtime/core-ops.js +226 -9
- package/dist/runtime/core-ops.js.map +1 -1
- package/dist/runtime/curator-extra-ops.d.ts +2 -2
- package/dist/runtime/curator-extra-ops.d.ts.map +1 -1
- package/dist/runtime/curator-extra-ops.js +15 -3
- package/dist/runtime/curator-extra-ops.js.map +1 -1
- package/dist/runtime/domain-ops.js +2 -2
- package/dist/runtime/domain-ops.js.map +1 -1
- package/dist/runtime/grading-ops.d.ts.map +1 -1
- package/dist/runtime/grading-ops.js.map +1 -1
- package/dist/runtime/intake-ops.d.ts +14 -0
- package/dist/runtime/intake-ops.d.ts.map +1 -0
- package/dist/runtime/intake-ops.js +110 -0
- package/dist/runtime/intake-ops.js.map +1 -0
- package/dist/runtime/loop-ops.d.ts +5 -4
- package/dist/runtime/loop-ops.d.ts.map +1 -1
- package/dist/runtime/loop-ops.js +84 -12
- package/dist/runtime/loop-ops.js.map +1 -1
- package/dist/runtime/memory-cross-project-ops.d.ts.map +1 -1
- package/dist/runtime/memory-cross-project-ops.js.map +1 -1
- package/dist/runtime/memory-extra-ops.js +5 -5
- package/dist/runtime/memory-extra-ops.js.map +1 -1
- package/dist/runtime/orchestrate-ops.d.ts.map +1 -1
- package/dist/runtime/orchestrate-ops.js +8 -2
- package/dist/runtime/orchestrate-ops.js.map +1 -1
- package/dist/runtime/planning-extra-ops.d.ts +13 -5
- package/dist/runtime/planning-extra-ops.d.ts.map +1 -1
- package/dist/runtime/planning-extra-ops.js +381 -18
- package/dist/runtime/planning-extra-ops.js.map +1 -1
- package/dist/runtime/playbook-ops.d.ts +14 -0
- package/dist/runtime/playbook-ops.d.ts.map +1 -0
- package/dist/runtime/playbook-ops.js +141 -0
- package/dist/runtime/playbook-ops.js.map +1 -0
- package/dist/runtime/project-ops.d.ts.map +1 -1
- package/dist/runtime/project-ops.js +7 -2
- package/dist/runtime/project-ops.js.map +1 -1
- package/dist/runtime/runtime.d.ts.map +1 -1
- package/dist/runtime/runtime.js +28 -9
- package/dist/runtime/runtime.js.map +1 -1
- package/dist/runtime/types.d.ts +8 -0
- package/dist/runtime/types.d.ts.map +1 -1
- package/dist/runtime/vault-extra-ops.d.ts +4 -2
- package/dist/runtime/vault-extra-ops.d.ts.map +1 -1
- package/dist/runtime/vault-extra-ops.js +383 -4
- package/dist/runtime/vault-extra-ops.js.map +1 -1
- package/dist/vault/playbook.d.ts +34 -0
- package/dist/vault/playbook.d.ts.map +1 -0
- package/dist/vault/playbook.js +60 -0
- package/dist/vault/playbook.js.map +1 -0
- package/dist/vault/vault.d.ts +52 -32
- package/dist/vault/vault.d.ts.map +1 -1
- package/dist/vault/vault.js +300 -181
- package/dist/vault/vault.js.map +1 -1
- package/package.json +9 -3
- package/src/__tests__/admin-extra-ops.test.ts +62 -15
- package/src/__tests__/admin-ops.test.ts +2 -2
- package/src/__tests__/brain.test.ts +3 -3
- package/src/__tests__/cognee-integration.test.ts +80 -0
- package/src/__tests__/cognee-sync-manager.test.ts +103 -0
- package/src/__tests__/core-ops.test.ts +36 -4
- package/src/__tests__/curator-extra-ops.test.ts +24 -2
- package/src/__tests__/errors.test.ts +388 -0
- package/src/__tests__/grading-ops.test.ts +28 -7
- package/src/__tests__/intake-pipeline.test.ts +162 -0
- package/src/__tests__/loop-ops.test.ts +74 -3
- package/src/__tests__/memory-cross-project-ops.test.ts +3 -1
- package/src/__tests__/orchestrate-ops.test.ts +8 -3
- package/src/__tests__/persistence.test.ts +291 -0
- package/src/__tests__/planner.test.ts +99 -21
- package/src/__tests__/planning-extra-ops.test.ts +168 -10
- package/src/__tests__/playbook-registry.test.ts +326 -0
- package/src/__tests__/playbook-seeder.test.ts +163 -0
- package/src/__tests__/playbook.test.ts +389 -0
- package/src/__tests__/postgres-provider.test.ts +58 -0
- package/src/__tests__/project-ops.test.ts +18 -4
- package/src/__tests__/template-manager.test.ts +222 -0
- package/src/__tests__/vault-extra-ops.test.ts +82 -7
- package/src/__tests__/vault.test.ts +184 -0
- package/src/brain/brain.ts +71 -9
- package/src/brain/intelligence.ts +258 -307
- package/src/brain/types.ts +2 -2
- package/src/cognee/client.ts +18 -0
- package/src/cognee/sync-manager.ts +389 -0
- package/src/control/identity-manager.ts +77 -75
- package/src/control/intent-router.ts +55 -57
- package/src/curator/curator.ts +199 -139
- package/src/errors/classify.ts +102 -0
- package/src/errors/index.ts +5 -0
- package/src/errors/retry.ts +132 -0
- package/src/errors/types.ts +81 -0
- package/src/governance/governance.ts +90 -107
- package/src/index.ts +116 -3
- package/src/intake/content-classifier.ts +146 -0
- package/src/intake/dedup-gate.ts +92 -0
- package/src/intake/intake-pipeline.ts +503 -0
- package/src/intake/types.ts +69 -0
- package/src/intelligence/loader.ts +1 -1
- package/src/intelligence/types.ts +3 -1
- package/src/loop/loop-manager.ts +325 -7
- package/src/loop/types.ts +72 -1
- package/src/persistence/index.ts +9 -0
- package/src/persistence/postgres-provider.ts +157 -0
- package/src/persistence/sqlite-provider.ts +115 -0
- package/src/persistence/types.ts +74 -0
- package/src/planning/gap-analysis.ts +286 -17
- package/src/planning/gap-types.ts +4 -1
- package/src/planning/planner.ts +828 -55
- package/src/playbooks/generic/brainstorming.ts +110 -0
- package/src/playbooks/generic/code-review.ts +181 -0
- package/src/playbooks/generic/subagent-execution.ts +74 -0
- package/src/playbooks/generic/systematic-debugging.ts +92 -0
- package/src/playbooks/generic/tdd.ts +75 -0
- package/src/playbooks/generic/verification.ts +79 -0
- package/src/playbooks/index.ts +27 -0
- package/src/playbooks/playbook-registry.ts +284 -0
- package/src/playbooks/playbook-seeder.ts +119 -0
- package/src/playbooks/playbook-types.ts +162 -0
- package/src/project/project-registry.ts +81 -74
- package/src/prompts/index.ts +3 -0
- package/src/prompts/parser.ts +59 -0
- package/src/prompts/template-manager.ts +77 -0
- package/src/prompts/types.ts +28 -0
- package/src/runtime/admin-extra-ops.ts +391 -13
- package/src/runtime/admin-ops.ts +17 -6
- package/src/runtime/capture-ops.ts +25 -6
- package/src/runtime/cognee-sync-ops.ts +63 -0
- package/src/runtime/core-ops.ts +258 -8
- package/src/runtime/curator-extra-ops.ts +17 -3
- package/src/runtime/domain-ops.ts +2 -2
- package/src/runtime/grading-ops.ts +11 -2
- package/src/runtime/intake-ops.ts +126 -0
- package/src/runtime/loop-ops.ts +96 -13
- package/src/runtime/memory-cross-project-ops.ts +1 -2
- package/src/runtime/memory-extra-ops.ts +5 -5
- package/src/runtime/orchestrate-ops.ts +8 -2
- package/src/runtime/planning-extra-ops.ts +414 -23
- package/src/runtime/playbook-ops.ts +169 -0
- package/src/runtime/project-ops.ts +9 -3
- package/src/runtime/runtime.ts +36 -10
- package/src/runtime/types.ts +8 -0
- package/src/runtime/vault-extra-ops.ts +425 -4
- package/src/vault/playbook.ts +87 -0
- package/src/vault/vault.ts +419 -235
package/src/brain/types.ts
CHANGED
|
@@ -6,7 +6,7 @@ export interface ScoringWeights {
|
|
|
6
6
|
semantic: number;
|
|
7
7
|
vector: number;
|
|
8
8
|
severity: number;
|
|
9
|
-
|
|
9
|
+
temporalDecay: number;
|
|
10
10
|
tagOverlap: number;
|
|
11
11
|
domainMatch: number;
|
|
12
12
|
}
|
|
@@ -15,7 +15,7 @@ export interface ScoreBreakdown {
|
|
|
15
15
|
semantic: number;
|
|
16
16
|
vector: number;
|
|
17
17
|
severity: number;
|
|
18
|
-
|
|
18
|
+
temporalDecay: number;
|
|
19
19
|
tagOverlap: number;
|
|
20
20
|
domainMatch: number;
|
|
21
21
|
total: number;
|
package/src/cognee/client.ts
CHANGED
|
@@ -254,6 +254,24 @@ export class CogneeClient {
|
|
|
254
254
|
return this.healthCache?.status ?? null;
|
|
255
255
|
}
|
|
256
256
|
|
|
257
|
+
// ─── Delete ──────────────────────────────────────────────────────
|
|
258
|
+
|
|
259
|
+
async deleteEntries(entryIds: string[]): Promise<{ deleted: number }> {
|
|
260
|
+
if (!this.isAvailable || entryIds.length === 0) return { deleted: 0 };
|
|
261
|
+
|
|
262
|
+
try {
|
|
263
|
+
const res = await this.post('/api/v1/delete', {
|
|
264
|
+
datasetName: this.config.dataset,
|
|
265
|
+
entryIds,
|
|
266
|
+
});
|
|
267
|
+
|
|
268
|
+
if (!res.ok) return { deleted: 0 };
|
|
269
|
+
return { deleted: entryIds.length };
|
|
270
|
+
} catch {
|
|
271
|
+
return { deleted: 0 };
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
|
|
257
275
|
// ─── Auth ──────────────────────────────────────────────────────
|
|
258
276
|
// Auto-register + login pattern from Salvador MCP.
|
|
259
277
|
// Tries login first (account may already exist), falls back to register.
|
|
@@ -0,0 +1,389 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* CogneeSyncManager — queued, resilient sync between the Vault SQLite DB and Cognee.
|
|
3
|
+
*
|
|
4
|
+
* Maintains a persistent queue (`cognee_sync_queue`) so that ingestions, updates,
|
|
5
|
+
* and deletions survive process restarts. Drain is idempotent and retry-safe.
|
|
6
|
+
* Health-flip detection auto-drains when Cognee comes back online.
|
|
7
|
+
*
|
|
8
|
+
* Ported from Salvador MCP's battle-tested cognee-sync module.
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import { createHash } from 'node:crypto';
|
|
12
|
+
import type { PersistenceProvider } from '../persistence/types.js';
|
|
13
|
+
import type { CogneeClient } from './client.js';
|
|
14
|
+
import type { IntelligenceEntry } from '../intelligence/types.js';
|
|
15
|
+
|
|
16
|
+
// ─── Types ──────────────────────────────────────────────────────────
|
|
17
|
+
|
|
18
|
+
export type SyncOp = 'ingest' | 'update' | 'delete';
|
|
19
|
+
export type SyncStatus = 'pending' | 'processing' | 'completed' | 'failed';
|
|
20
|
+
|
|
21
|
+
export interface SyncQueueItem {
|
|
22
|
+
id: number;
|
|
23
|
+
op: SyncOp;
|
|
24
|
+
entryId: string;
|
|
25
|
+
dataset: string;
|
|
26
|
+
contentHash: string | null;
|
|
27
|
+
status: SyncStatus;
|
|
28
|
+
attempts: number;
|
|
29
|
+
error: string | null;
|
|
30
|
+
createdAt: number;
|
|
31
|
+
processedAt: number | null;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
export interface SyncManagerStats {
|
|
35
|
+
pending: number;
|
|
36
|
+
processing: number;
|
|
37
|
+
completed: number;
|
|
38
|
+
failed: number;
|
|
39
|
+
queueSize: number;
|
|
40
|
+
lastDrainAt: number | null;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
// ─── Constants ──────────────────────────────────────────────────────
|
|
44
|
+
|
|
45
|
+
const MAX_BATCH = 10;
|
|
46
|
+
const MAX_RETRIES = 3;
|
|
47
|
+
|
|
48
|
+
// ─── CogneeSyncManager ─────────────────────────────────────────────
|
|
49
|
+
|
|
50
|
+
export class CogneeSyncManager {
|
|
51
|
+
private db: PersistenceProvider;
|
|
52
|
+
private cognee: CogneeClient;
|
|
53
|
+
private dataset: string;
|
|
54
|
+
private lastDrainAt: number | null = null;
|
|
55
|
+
private drainTimer: ReturnType<typeof setInterval> | null = null;
|
|
56
|
+
private wasAvailable: boolean = false;
|
|
57
|
+
|
|
58
|
+
constructor(db: PersistenceProvider, cognee: CogneeClient, dataset: string) {
|
|
59
|
+
this.db = db;
|
|
60
|
+
this.cognee = cognee;
|
|
61
|
+
this.dataset = dataset;
|
|
62
|
+
this.initSchema();
|
|
63
|
+
this.wasAvailable = cognee.isAvailable;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
// ─── Schema ────────────────────────────────────────────────────
|
|
67
|
+
|
|
68
|
+
private initSchema(): void {
|
|
69
|
+
this.db.execSql(`
|
|
70
|
+
CREATE TABLE IF NOT EXISTS cognee_sync_queue (
|
|
71
|
+
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
72
|
+
op TEXT NOT NULL,
|
|
73
|
+
entry_id TEXT NOT NULL,
|
|
74
|
+
dataset TEXT NOT NULL,
|
|
75
|
+
content_hash TEXT,
|
|
76
|
+
status TEXT NOT NULL DEFAULT 'pending',
|
|
77
|
+
attempts INTEGER NOT NULL DEFAULT 0,
|
|
78
|
+
error TEXT,
|
|
79
|
+
created_at INTEGER NOT NULL DEFAULT (unixepoch()),
|
|
80
|
+
processed_at INTEGER
|
|
81
|
+
)
|
|
82
|
+
`);
|
|
83
|
+
|
|
84
|
+
// Add cognee_ingested_hash column to entries table for reconciliation.
|
|
85
|
+
// ALTER TABLE ... ADD COLUMN is a no-op error when the column already exists.
|
|
86
|
+
try {
|
|
87
|
+
this.db.run('ALTER TABLE entries ADD COLUMN cognee_ingested_hash TEXT');
|
|
88
|
+
} catch {
|
|
89
|
+
// Column already exists — expected on subsequent runs.
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
// ─── Content hashing ──────────────────────────────────────────
|
|
94
|
+
|
|
95
|
+
/**
|
|
96
|
+
* SHA-256 of the serialized entry fields, truncated to 16 hex characters.
|
|
97
|
+
* Deterministic for identical content — used to detect drift.
|
|
98
|
+
*/
|
|
99
|
+
static contentHash(entry: IntelligenceEntry): string {
|
|
100
|
+
const payload = JSON.stringify({
|
|
101
|
+
id: entry.id,
|
|
102
|
+
type: entry.type,
|
|
103
|
+
domain: entry.domain,
|
|
104
|
+
title: entry.title,
|
|
105
|
+
severity: entry.severity,
|
|
106
|
+
description: entry.description,
|
|
107
|
+
context: entry.context ?? null,
|
|
108
|
+
example: entry.example ?? null,
|
|
109
|
+
counterExample: entry.counterExample ?? null,
|
|
110
|
+
why: entry.why ?? null,
|
|
111
|
+
tags: entry.tags,
|
|
112
|
+
appliesTo: entry.appliesTo ?? [],
|
|
113
|
+
});
|
|
114
|
+
return createHash('sha256').update(payload).digest('hex').slice(0, 16);
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
// ─── Enqueue ──────────────────────────────────────────────────
|
|
118
|
+
|
|
119
|
+
/**
|
|
120
|
+
* Add an operation to the sync queue.
|
|
121
|
+
*
|
|
122
|
+
* @param op The operation type (ingest | update | delete).
|
|
123
|
+
* @param entryId The vault entry ID.
|
|
124
|
+
* @param entry Optional entry for hash computation. If omitted, hash is null.
|
|
125
|
+
*/
|
|
126
|
+
enqueue(op: SyncOp, entryId: string, entry?: IntelligenceEntry): void {
|
|
127
|
+
const contentHash = entry ? CogneeSyncManager.contentHash(entry) : null;
|
|
128
|
+
this.db.run(
|
|
129
|
+
`INSERT INTO cognee_sync_queue (op, entry_id, dataset, content_hash)
|
|
130
|
+
VALUES (@op, @entryId, @dataset, @contentHash)`,
|
|
131
|
+
{
|
|
132
|
+
op,
|
|
133
|
+
entryId,
|
|
134
|
+
dataset: this.dataset,
|
|
135
|
+
contentHash,
|
|
136
|
+
},
|
|
137
|
+
);
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
// ─── Drain ────────────────────────────────────────────────────
|
|
141
|
+
|
|
142
|
+
/**
|
|
143
|
+
* Process up to MAX_BATCH pending items from the queue.
|
|
144
|
+
* Returns the number of items successfully processed.
|
|
145
|
+
*
|
|
146
|
+
* If Cognee is not available, returns 0 without touching the queue.
|
|
147
|
+
*/
|
|
148
|
+
async drain(): Promise<number> {
|
|
149
|
+
if (!this.cognee.isAvailable) return 0;
|
|
150
|
+
|
|
151
|
+
// Claim a batch: move pending → processing
|
|
152
|
+
const items = this.db.all<Record<string, unknown>>(
|
|
153
|
+
`SELECT * FROM cognee_sync_queue
|
|
154
|
+
WHERE status = 'pending' AND dataset = @dataset
|
|
155
|
+
ORDER BY created_at ASC
|
|
156
|
+
LIMIT @limit`,
|
|
157
|
+
{ dataset: this.dataset, limit: MAX_BATCH },
|
|
158
|
+
);
|
|
159
|
+
|
|
160
|
+
if (items.length === 0) return 0;
|
|
161
|
+
|
|
162
|
+
let processed = 0;
|
|
163
|
+
|
|
164
|
+
for (const raw of items) {
|
|
165
|
+
const item = rowToQueueItem(raw);
|
|
166
|
+
|
|
167
|
+
// Mark as processing
|
|
168
|
+
this.db.run(
|
|
169
|
+
`UPDATE cognee_sync_queue SET status = 'processing', attempts = attempts + 1 WHERE id = @id`,
|
|
170
|
+
{ id: item.id },
|
|
171
|
+
);
|
|
172
|
+
|
|
173
|
+
try {
|
|
174
|
+
if (item.op === 'ingest' || item.op === 'update') {
|
|
175
|
+
const entry = this.readEntry(item.entryId);
|
|
176
|
+
if (!entry) {
|
|
177
|
+
// Entry was deleted from vault before we could sync — mark completed
|
|
178
|
+
this.markCompleted(item.id);
|
|
179
|
+
processed++;
|
|
180
|
+
continue;
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
const result = await this.cognee.addEntries([entry]);
|
|
184
|
+
if (result.added === 0) {
|
|
185
|
+
throw new Error('Cognee addEntries returned 0 added');
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
// Update the ingested hash on the entries table
|
|
189
|
+
const hash = CogneeSyncManager.contentHash(entry);
|
|
190
|
+
this.db.run(`UPDATE entries SET cognee_ingested_hash = @hash WHERE id = @id`, {
|
|
191
|
+
hash,
|
|
192
|
+
id: item.entryId,
|
|
193
|
+
});
|
|
194
|
+
|
|
195
|
+
this.markCompleted(item.id);
|
|
196
|
+
processed++;
|
|
197
|
+
} else if (item.op === 'delete') {
|
|
198
|
+
// deleteEntries may not exist yet — graceful degradation
|
|
199
|
+
const client = this.cognee as unknown as Record<string, unknown>;
|
|
200
|
+
if (typeof client.deleteEntries === 'function') {
|
|
201
|
+
await (client.deleteEntries as (ids: string[]) => Promise<unknown>)([item.entryId]);
|
|
202
|
+
}
|
|
203
|
+
// Clear the ingested hash (entry may already be gone from entries table)
|
|
204
|
+
this.db.run(`UPDATE entries SET cognee_ingested_hash = NULL WHERE id = @id`, {
|
|
205
|
+
id: item.entryId,
|
|
206
|
+
});
|
|
207
|
+
this.markCompleted(item.id);
|
|
208
|
+
processed++;
|
|
209
|
+
}
|
|
210
|
+
} catch (err) {
|
|
211
|
+
const errorMsg = err instanceof Error ? err.message : String(err);
|
|
212
|
+
const attempts = item.attempts + 1; // Already incremented above
|
|
213
|
+
if (attempts >= MAX_RETRIES) {
|
|
214
|
+
this.db.run(
|
|
215
|
+
`UPDATE cognee_sync_queue SET status = 'failed', error = @error, processed_at = unixepoch() WHERE id = @id`,
|
|
216
|
+
{ id: item.id, error: errorMsg },
|
|
217
|
+
);
|
|
218
|
+
} else {
|
|
219
|
+
// Back to pending for retry
|
|
220
|
+
this.db.run(
|
|
221
|
+
`UPDATE cognee_sync_queue SET status = 'pending', error = @error WHERE id = @id`,
|
|
222
|
+
{ id: item.id, error: errorMsg },
|
|
223
|
+
);
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
this.lastDrainAt = Math.floor(Date.now() / 1000);
|
|
229
|
+
return processed;
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
// ─── Reconciliation ───────────────────────────────────────────
|
|
233
|
+
|
|
234
|
+
/**
|
|
235
|
+
* Find entries whose cognee_ingested_hash is NULL or doesn't match the
|
|
236
|
+
* current content hash. Enqueue dirty entries for re-ingestion.
|
|
237
|
+
*
|
|
238
|
+
* Returns the number of entries enqueued.
|
|
239
|
+
*/
|
|
240
|
+
reconcile(): number {
|
|
241
|
+
// Get all entries that either have never been ingested or whose content changed
|
|
242
|
+
const rows = this.db.all<Record<string, unknown>>(
|
|
243
|
+
`SELECT * FROM entries WHERE cognee_ingested_hash IS NULL
|
|
244
|
+
UNION ALL
|
|
245
|
+
SELECT * FROM entries WHERE cognee_ingested_hash IS NOT NULL`,
|
|
246
|
+
);
|
|
247
|
+
|
|
248
|
+
let enqueued = 0;
|
|
249
|
+
|
|
250
|
+
for (const raw of rows) {
|
|
251
|
+
const entry = this.rowToEntry(raw);
|
|
252
|
+
const currentHash = CogneeSyncManager.contentHash(entry);
|
|
253
|
+
const ingestedHash = raw.cognee_ingested_hash as string | null;
|
|
254
|
+
|
|
255
|
+
if (ingestedHash === currentHash) continue;
|
|
256
|
+
|
|
257
|
+
// Determine op: null hash means never ingested, mismatched means update
|
|
258
|
+
const op: SyncOp = ingestedHash === null ? 'ingest' : 'update';
|
|
259
|
+
|
|
260
|
+
// Avoid duplicate pending items for the same entry
|
|
261
|
+
const existing = this.db.get<{ id: number }>(
|
|
262
|
+
`SELECT id FROM cognee_sync_queue
|
|
263
|
+
WHERE entry_id = @entryId AND dataset = @dataset AND status = 'pending'`,
|
|
264
|
+
{ entryId: entry.id, dataset: this.dataset },
|
|
265
|
+
);
|
|
266
|
+
|
|
267
|
+
if (!existing) {
|
|
268
|
+
this.enqueue(op, entry.id, entry);
|
|
269
|
+
enqueued++;
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
return enqueued;
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
// ─── Stats ────────────────────────────────────────────────────
|
|
277
|
+
|
|
278
|
+
getStats(): SyncManagerStats {
|
|
279
|
+
const countByStatus = (status: SyncStatus): number => {
|
|
280
|
+
const row = this.db.get<{ count: number }>(
|
|
281
|
+
`SELECT COUNT(*) as count FROM cognee_sync_queue WHERE status = @status AND dataset = @dataset`,
|
|
282
|
+
{ status, dataset: this.dataset },
|
|
283
|
+
);
|
|
284
|
+
return row?.count ?? 0;
|
|
285
|
+
};
|
|
286
|
+
|
|
287
|
+
const pending = countByStatus('pending');
|
|
288
|
+
const processing = countByStatus('processing');
|
|
289
|
+
const completed = countByStatus('completed');
|
|
290
|
+
const failed = countByStatus('failed');
|
|
291
|
+
|
|
292
|
+
return {
|
|
293
|
+
pending,
|
|
294
|
+
processing,
|
|
295
|
+
completed,
|
|
296
|
+
failed,
|
|
297
|
+
queueSize: pending + processing,
|
|
298
|
+
lastDrainAt: this.lastDrainAt,
|
|
299
|
+
};
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
// ─── Health-flip detection ────────────────────────────────────
|
|
303
|
+
|
|
304
|
+
/**
|
|
305
|
+
* Detects an unavailable-to-available transition on the Cognee client.
|
|
306
|
+
* When Cognee comes back online, automatically triggers a drain.
|
|
307
|
+
*
|
|
308
|
+
* Call this periodically (e.g. after each health check).
|
|
309
|
+
*/
|
|
310
|
+
async checkHealthFlip(): Promise<void> {
|
|
311
|
+
const nowAvailable = this.cognee.isAvailable;
|
|
312
|
+
if (nowAvailable && !this.wasAvailable) {
|
|
313
|
+
// Cognee just came back online — drain the queue
|
|
314
|
+
await this.drain();
|
|
315
|
+
}
|
|
316
|
+
this.wasAvailable = nowAvailable;
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
// ─── Cleanup ──────────────────────────────────────────────────
|
|
320
|
+
|
|
321
|
+
/**
|
|
322
|
+
* Clear any periodic drain timer.
|
|
323
|
+
*/
|
|
324
|
+
close(): void {
|
|
325
|
+
if (this.drainTimer) {
|
|
326
|
+
clearInterval(this.drainTimer);
|
|
327
|
+
this.drainTimer = null;
|
|
328
|
+
}
|
|
329
|
+
}
|
|
330
|
+
|
|
331
|
+
// ─── Private helpers ──────────────────────────────────────────
|
|
332
|
+
|
|
333
|
+
private markCompleted(id: number): void {
|
|
334
|
+
this.db.run(
|
|
335
|
+
`UPDATE cognee_sync_queue SET status = 'completed', processed_at = unixepoch() WHERE id = @id`,
|
|
336
|
+
{ id },
|
|
337
|
+
);
|
|
338
|
+
}
|
|
339
|
+
|
|
340
|
+
/**
|
|
341
|
+
* Read an entry from the entries table by ID.
|
|
342
|
+
* Returns null if the entry doesn't exist.
|
|
343
|
+
*/
|
|
344
|
+
private readEntry(id: string): IntelligenceEntry | null {
|
|
345
|
+
const row = this.db.get<Record<string, unknown>>('SELECT * FROM entries WHERE id = @id', {
|
|
346
|
+
id,
|
|
347
|
+
});
|
|
348
|
+
return row ? this.rowToEntry(row) : null;
|
|
349
|
+
}
|
|
350
|
+
|
|
351
|
+
/**
|
|
352
|
+
* Convert a raw DB row into an IntelligenceEntry.
|
|
353
|
+
*/
|
|
354
|
+
private rowToEntry(row: Record<string, unknown>): IntelligenceEntry {
|
|
355
|
+
return {
|
|
356
|
+
id: row.id as string,
|
|
357
|
+
type: row.type as IntelligenceEntry['type'],
|
|
358
|
+
domain: row.domain as string,
|
|
359
|
+
title: row.title as string,
|
|
360
|
+
severity: row.severity as IntelligenceEntry['severity'],
|
|
361
|
+
description: row.description as string,
|
|
362
|
+
context: (row.context as string) ?? undefined,
|
|
363
|
+
example: (row.example as string) ?? undefined,
|
|
364
|
+
counterExample: (row.counter_example as string) ?? undefined,
|
|
365
|
+
why: (row.why as string) ?? undefined,
|
|
366
|
+
tags: JSON.parse((row.tags as string) || '[]'),
|
|
367
|
+
appliesTo: JSON.parse((row.applies_to as string) || '[]'),
|
|
368
|
+
validFrom: (row.valid_from as number) ?? undefined,
|
|
369
|
+
validUntil: (row.valid_until as number) ?? undefined,
|
|
370
|
+
};
|
|
371
|
+
}
|
|
372
|
+
}
|
|
373
|
+
|
|
374
|
+
// ─── Module-level helpers ─────────────────────────────────────────
|
|
375
|
+
|
|
376
|
+
function rowToQueueItem(row: Record<string, unknown>): SyncQueueItem {
|
|
377
|
+
return {
|
|
378
|
+
id: row.id as number,
|
|
379
|
+
op: row.op as SyncOp,
|
|
380
|
+
entryId: row.entry_id as string,
|
|
381
|
+
dataset: row.dataset as string,
|
|
382
|
+
contentHash: (row.content_hash as string) ?? null,
|
|
383
|
+
status: row.status as SyncStatus,
|
|
384
|
+
attempts: row.attempts as number,
|
|
385
|
+
error: (row.error as string) ?? null,
|
|
386
|
+
createdAt: row.created_at as number,
|
|
387
|
+
processedAt: (row.processed_at as number) ?? null,
|
|
388
|
+
};
|
|
389
|
+
}
|