aismemory 0.1.0 → 0.3.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.
Files changed (100) hide show
  1. package/README.md +1 -0
  2. package/dist/__tests__/auto-handoff.test.d.ts +1 -0
  3. package/dist/__tests__/auto-handoff.test.js +72 -0
  4. package/dist/__tests__/auto-handoff.test.js.map +1 -0
  5. package/dist/__tests__/cli-review.test.d.ts +1 -0
  6. package/dist/__tests__/cli-review.test.js +26 -0
  7. package/dist/__tests__/cli-review.test.js.map +1 -0
  8. package/dist/__tests__/config.test.d.ts +1 -0
  9. package/dist/__tests__/config.test.js +54 -0
  10. package/dist/__tests__/config.test.js.map +1 -0
  11. package/dist/__tests__/hydration.test.d.ts +1 -0
  12. package/dist/__tests__/hydration.test.js +111 -0
  13. package/dist/__tests__/hydration.test.js.map +1 -0
  14. package/dist/__tests__/integration.test.d.ts +1 -0
  15. package/dist/__tests__/integration.test.js +36 -0
  16. package/dist/__tests__/integration.test.js.map +1 -0
  17. package/dist/__tests__/local-claude-source.test.d.ts +1 -0
  18. package/dist/__tests__/local-claude-source.test.js +97 -0
  19. package/dist/__tests__/local-claude-source.test.js.map +1 -0
  20. package/dist/__tests__/local-mirror.test.d.ts +1 -0
  21. package/dist/__tests__/local-mirror.test.js +95 -0
  22. package/dist/__tests__/local-mirror.test.js.map +1 -0
  23. package/dist/__tests__/pipeline-ingestion.test.d.ts +1 -0
  24. package/dist/__tests__/pipeline-ingestion.test.js +161 -0
  25. package/dist/__tests__/pipeline-ingestion.test.js.map +1 -0
  26. package/dist/__tests__/pipeline-scope-resolver.test.d.ts +1 -0
  27. package/dist/__tests__/pipeline-scope-resolver.test.js +51 -0
  28. package/dist/__tests__/pipeline-scope-resolver.test.js.map +1 -0
  29. package/dist/__tests__/pipeline-trust-tagger.test.d.ts +1 -0
  30. package/dist/__tests__/pipeline-trust-tagger.test.js +24 -0
  31. package/dist/__tests__/pipeline-trust-tagger.test.js.map +1 -0
  32. package/dist/__tests__/prompt-resource.test.d.ts +1 -0
  33. package/dist/__tests__/prompt-resource.test.js +218 -0
  34. package/dist/__tests__/prompt-resource.test.js.map +1 -0
  35. package/dist/__tests__/read-claude-memory-tree.test.d.ts +1 -0
  36. package/dist/__tests__/read-claude-memory-tree.test.js +37 -0
  37. package/dist/__tests__/read-claude-memory-tree.test.js.map +1 -0
  38. package/dist/__tests__/refresh.test.d.ts +1 -0
  39. package/dist/__tests__/refresh.test.js +73 -0
  40. package/dist/__tests__/refresh.test.js.map +1 -0
  41. package/dist/__tests__/trust-ledger.test.d.ts +1 -0
  42. package/dist/__tests__/trust-ledger.test.js +55 -0
  43. package/dist/__tests__/trust-ledger.test.js.map +1 -0
  44. package/dist/auto-handoff.d.ts +15 -0
  45. package/dist/auto-handoff.js +53 -0
  46. package/dist/auto-handoff.js.map +1 -0
  47. package/dist/cli/read-claude-memory-tree.d.ts +7 -0
  48. package/dist/cli/read-claude-memory-tree.js +65 -0
  49. package/dist/cli/read-claude-memory-tree.js.map +1 -0
  50. package/dist/cli/sync-memory.d.ts +2 -0
  51. package/dist/cli/sync-memory.js +155 -0
  52. package/dist/cli/sync-memory.js.map +1 -0
  53. package/dist/config.d.ts +11 -0
  54. package/dist/config.js +51 -0
  55. package/dist/config.js.map +1 -0
  56. package/dist/hydration.d.ts +2 -0
  57. package/dist/hydration.js +62 -0
  58. package/dist/hydration.js.map +1 -0
  59. package/dist/index.js +173 -11
  60. package/dist/index.js.map +1 -1
  61. package/dist/local-mirror.d.ts +15 -0
  62. package/dist/local-mirror.js +56 -0
  63. package/dist/local-mirror.js.map +1 -0
  64. package/dist/pipeline/dedupe.d.ts +11 -0
  65. package/dist/pipeline/dedupe.js +21 -0
  66. package/dist/pipeline/dedupe.js.map +1 -0
  67. package/dist/pipeline/ingestion.d.ts +56 -0
  68. package/dist/pipeline/ingestion.js +86 -0
  69. package/dist/pipeline/ingestion.js.map +1 -0
  70. package/dist/pipeline/scope-resolver.d.ts +15 -0
  71. package/dist/pipeline/scope-resolver.js +28 -0
  72. package/dist/pipeline/scope-resolver.js.map +1 -0
  73. package/dist/pipeline/trust-tagger.d.ts +10 -0
  74. package/dist/pipeline/trust-tagger.js +12 -0
  75. package/dist/pipeline/trust-tagger.js.map +1 -0
  76. package/dist/pipeline/types.d.ts +41 -0
  77. package/dist/pipeline/types.js +6 -0
  78. package/dist/pipeline/types.js.map +1 -0
  79. package/dist/prompt-resource.d.ts +2 -0
  80. package/dist/prompt-resource.js +77 -0
  81. package/dist/prompt-resource.js.map +1 -0
  82. package/dist/refresh.d.ts +3 -0
  83. package/dist/refresh.js +25 -0
  84. package/dist/refresh.js.map +1 -0
  85. package/dist/review/cli-review.d.ts +10 -0
  86. package/dist/review/cli-review.js +47 -0
  87. package/dist/review/cli-review.js.map +1 -0
  88. package/dist/sources/local-claude.d.ts +11 -0
  89. package/dist/sources/local-claude.js +137 -0
  90. package/dist/sources/local-claude.js.map +1 -0
  91. package/dist/sources/types.d.ts +53 -0
  92. package/dist/sources/types.js +9 -0
  93. package/dist/sources/types.js.map +1 -0
  94. package/dist/trust-ledger.d.ts +8 -0
  95. package/dist/trust-ledger.js +61 -0
  96. package/dist/trust-ledger.js.map +1 -0
  97. package/dist/types.d.ts +45 -0
  98. package/dist/types.js +2 -0
  99. package/dist/types.js.map +1 -0
  100. package/package.json +5 -3
@@ -0,0 +1,55 @@
1
+ import { describe, it, expect, beforeEach, afterEach } from 'vitest';
2
+ import { mkdirSync, mkdtempSync, readdirSync, rmSync, writeFileSync } from 'node:fs';
3
+ import { tmpdir } from 'node:os';
4
+ import { join } from 'node:path';
5
+ import { TrustLedger } from '../trust-ledger.js';
6
+ describe('TrustLedger', () => {
7
+ let tmpHome;
8
+ beforeEach(() => {
9
+ tmpHome = mkdtempSync(join(tmpdir(), 'aismem-trust-'));
10
+ });
11
+ afterEach(() => rmSync(tmpHome, { recursive: true, force: true }));
12
+ it('creates a ledger entry on first sync and marks as untrusted', () => {
13
+ const ledger = new TrustLedger(tmpHome);
14
+ expect(ledger.isTrusted('agent1', 'claude-local', 'project:corbot')).toBe(false);
15
+ ledger.recordSync('agent1', 'claude-local', 'project:corbot');
16
+ expect(ledger.isTrusted('agent1', 'claude-local', 'project:corbot')).toBe(false);
17
+ });
18
+ it('marks as trusted on second sync', () => {
19
+ const ledger = new TrustLedger(tmpHome);
20
+ ledger.recordSync('agent1', 'claude-local', 'project:corbot');
21
+ ledger.recordSync('agent1', 'claude-local', 'project:corbot');
22
+ expect(ledger.isTrusted('agent1', 'claude-local', 'project:corbot')).toBe(true);
23
+ });
24
+ it('keeps separate trust per (agent, source, scope)', () => {
25
+ const ledger = new TrustLedger(tmpHome);
26
+ ledger.recordSync('agent1', 'claude-local', 'project:a');
27
+ ledger.recordSync('agent1', 'claude-local', 'project:a');
28
+ expect(ledger.isTrusted('agent1', 'claude-local', 'project:a')).toBe(true);
29
+ expect(ledger.isTrusted('agent1', 'claude-local', 'project:b')).toBe(false);
30
+ expect(ledger.isTrusted('agent2', 'claude-local', 'project:a')).toBe(false);
31
+ });
32
+ it('survives reload from disk', () => {
33
+ const ledger1 = new TrustLedger(tmpHome);
34
+ ledger1.recordSync('agent1', 'claude-local', 'project:corbot');
35
+ ledger1.recordSync('agent1', 'claude-local', 'project:corbot');
36
+ const ledger2 = new TrustLedger(tmpHome);
37
+ expect(ledger2.isTrusted('agent1', 'claude-local', 'project:corbot')).toBe(true);
38
+ });
39
+ it('recovers gracefully from a corrupted ledger file', () => {
40
+ // Pre-populate the .aismemory dir with an invalid JSON file to simulate corruption
41
+ const ledgerDir = join(tmpHome, '.aismemory');
42
+ mkdirSync(ledgerDir, { recursive: true });
43
+ writeFileSync(join(ledgerDir, 'trust-ledger.json'), '{not valid json', 'utf8');
44
+ // Constructor must not throw
45
+ let ledger;
46
+ expect(() => { ledger = new TrustLedger(tmpHome); }).not.toThrow();
47
+ // Should behave as fresh — no trust
48
+ expect(ledger.isTrusted('agent1', 'claude-local', 'project:corbot')).toBe(false);
49
+ // A .corrupt-* backup file must exist in the .aismemory directory
50
+ const files = readdirSync(ledgerDir);
51
+ const corruptBackup = files.find(f => f.startsWith('trust-ledger.json.corrupt-'));
52
+ expect(corruptBackup).toBeDefined();
53
+ });
54
+ });
55
+ //# sourceMappingURL=trust-ledger.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"trust-ledger.test.js","sourceRoot":"","sources":["../../src/__tests__/trust-ledger.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,15 @@
1
+ export declare class ActivityTracker {
2
+ private toolCounts;
3
+ private storedMemories;
4
+ recordToolUse(toolName: string): void;
5
+ recordMemoryStored(content: string): void;
6
+ getToolCounts(): Record<string, number>;
7
+ getStoredMemories(): string[];
8
+ hasActivity(): boolean;
9
+ }
10
+ export interface HandoffPayload {
11
+ summary: string;
12
+ keyLearnings: string[];
13
+ }
14
+ export declare function buildHandoffPayload(tracker: ActivityTracker): HandoffPayload;
15
+ export declare function setupAutoHandoff(aisUrl: string, agentId: string, token: string, tracker: ActivityTracker, onBeforeExit?: () => void): void;
@@ -0,0 +1,53 @@
1
+ export class ActivityTracker {
2
+ toolCounts = {};
3
+ storedMemories = [];
4
+ recordToolUse(toolName) {
5
+ this.toolCounts[toolName] = (this.toolCounts[toolName] ?? 0) + 1;
6
+ }
7
+ recordMemoryStored(content) {
8
+ this.storedMemories.push(content);
9
+ }
10
+ getToolCounts() {
11
+ return { ...this.toolCounts };
12
+ }
13
+ getStoredMemories() {
14
+ return [...this.storedMemories];
15
+ }
16
+ hasActivity() {
17
+ return Object.keys(this.toolCounts).length > 0;
18
+ }
19
+ }
20
+ export function buildHandoffPayload(tracker) {
21
+ if (!tracker.hasActivity()) {
22
+ return { summary: 'No significant activity this session.', keyLearnings: [] };
23
+ }
24
+ const toolParts = Object.entries(tracker.getToolCounts())
25
+ .map(([name, count]) => `${name} (${count})`)
26
+ .join(', ');
27
+ const memories = tracker.getStoredMemories();
28
+ const summary = `Used tools: ${toolParts}.${memories.length > 0 ? ` Stored ${memories.length} memories.` : ''}`;
29
+ return { summary, keyLearnings: memories };
30
+ }
31
+ export function setupAutoHandoff(aisUrl, agentId, token, tracker, onBeforeExit) {
32
+ const doHandoff = () => {
33
+ if (!tracker.hasActivity())
34
+ return;
35
+ const payload = buildHandoffPayload(tracker);
36
+ fetch(`${aisUrl}/v1/agents/${agentId}/handoff`, {
37
+ method: 'POST',
38
+ headers: { 'Content-Type': 'application/json', Authorization: `Bearer ${token}` },
39
+ body: JSON.stringify(payload),
40
+ }).catch(() => { }); // Best-effort
41
+ if (onBeforeExit)
42
+ onBeforeExit();
43
+ };
44
+ process.on('SIGTERM', () => {
45
+ doHandoff();
46
+ process.exit(0);
47
+ });
48
+ process.on('SIGINT', () => {
49
+ doHandoff();
50
+ process.exit(0);
51
+ });
52
+ }
53
+ //# sourceMappingURL=auto-handoff.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auto-handoff.js","sourceRoot":"","sources":["../src/auto-handoff.ts"],"names":[],"mappings":"AAAA,MAAM,OAAO,eAAe;IAClB,UAAU,GAA2B,EAAE,CAAC;IACxC,cAAc,GAAa,EAAE,CAAC;IAEtC,aAAa,CAAC,QAAgB;QAC5B,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC;IACnE,CAAC;IAED,kBAAkB,CAAC,OAAe;QAChC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACpC,CAAC;IAED,aAAa;QACX,OAAO,EAAE,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;IAChC,CAAC;IAED,iBAAiB;QACf,OAAO,CAAC,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC;IAClC,CAAC;IAED,WAAW;QACT,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,MAAM,GAAG,CAAC,CAAC;IACjD,CAAC;CACF;AAOD,MAAM,UAAU,mBAAmB,CAAC,OAAwB;IAC1D,IAAI,CAAC,OAAO,CAAC,WAAW,EAAE,EAAE,CAAC;QAC3B,OAAO,EAAE,OAAO,EAAE,uCAAuC,EAAE,YAAY,EAAE,EAAE,EAAE,CAAC;IAChF,CAAC;IACD,MAAM,SAAS,GAAG,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,aAAa,EAAE,CAAC;SACtD,GAAG,CAAC,CAAC,CAAC,IAAI,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC,GAAG,IAAI,KAAK,KAAK,GAAG,CAAC;SAC5C,IAAI,CAAC,IAAI,CAAC,CAAC;IACd,MAAM,QAAQ,GAAG,OAAO,CAAC,iBAAiB,EAAE,CAAC;IAC7C,MAAM,OAAO,GAAG,eAAe,SAAS,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,WAAW,QAAQ,CAAC,MAAM,YAAY,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC;IAChH,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,QAAQ,EAAE,CAAC;AAC7C,CAAC;AAED,MAAM,UAAU,gBAAgB,CAC9B,MAAc,EACd,OAAe,EACf,KAAa,EACb,OAAwB,EACxB,YAAyB;IAEzB,MAAM,SAAS,GAAG,GAAS,EAAE;QAC3B,IAAI,CAAC,OAAO,CAAC,WAAW,EAAE;YAAE,OAAO;QACnC,MAAM,OAAO,GAAG,mBAAmB,CAAC,OAAO,CAAC,CAAC;QAC7C,KAAK,CAAC,GAAG,MAAM,cAAc,OAAO,UAAU,EAAE;YAC9C,MAAM,EAAE,MAAM;YACd,OAAO,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,aAAa,EAAE,UAAU,KAAK,EAAE,EAAE;YACjF,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC;SAC9B,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC,CAAC,cAAc;QAClC,IAAI,YAAY;YAAE,YAAY,EAAE,CAAC;IACnC,CAAC,CAAC;IACF,OAAO,CAAC,EAAE,CAAC,SAAS,EAAE,GAAG,EAAE;QACzB,SAAS,EAAE,CAAC;QACZ,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC,CAAC,CAAC;IACH,OAAO,CAAC,EAAE,CAAC,QAAQ,EAAE,GAAG,EAAE;QACxB,SAAS,EAAE,CAAC;QACZ,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,7 @@
1
+ import type { FilesystemResult } from '../sources/types.js';
2
+ /**
3
+ * Walks ~/.claude/projects/<project>/memory/ subtrees only. Project dirs
4
+ * without a memory/ subdir are skipped. Anything outside a memory/ subdir
5
+ * (handoffs, scratch notes, etc.) is NOT read.
6
+ */
7
+ export declare function readClaudeMemoryTree(projectsRoot: string): FilesystemResult;
@@ -0,0 +1,65 @@
1
+ import { readFileSync, readdirSync, statSync } from 'node:fs';
2
+ import { join } from 'node:path';
3
+ /**
4
+ * Walks ~/.claude/projects/<project>/memory/ subtrees only. Project dirs
5
+ * without a memory/ subdir are skipped. Anything outside a memory/ subdir
6
+ * (handoffs, scratch notes, etc.) is NOT read.
7
+ */
8
+ export function readClaudeMemoryTree(projectsRoot) {
9
+ const files = [];
10
+ let rootStat;
11
+ try {
12
+ rootStat = statSync(projectsRoot);
13
+ }
14
+ catch {
15
+ return { rootPath: projectsRoot, files };
16
+ }
17
+ if (!rootStat.isDirectory()) {
18
+ return { rootPath: projectsRoot, files };
19
+ }
20
+ for (const projectName of readdirSync(projectsRoot)) {
21
+ const projectDir = join(projectsRoot, projectName);
22
+ let projectStat;
23
+ try {
24
+ projectStat = statSync(projectDir);
25
+ }
26
+ catch {
27
+ continue;
28
+ }
29
+ if (!projectStat.isDirectory())
30
+ continue;
31
+ const memoryDir = join(projectDir, 'memory');
32
+ let memoryStat;
33
+ try {
34
+ memoryStat = statSync(memoryDir);
35
+ }
36
+ catch {
37
+ continue;
38
+ }
39
+ if (!memoryStat.isDirectory())
40
+ continue;
41
+ // Walk the memory/ subtree only.
42
+ const stack = [memoryDir];
43
+ while (stack.length > 0) {
44
+ const dir = stack.pop();
45
+ for (const name of readdirSync(dir)) {
46
+ const full = join(dir, name);
47
+ let st;
48
+ try {
49
+ st = statSync(full);
50
+ }
51
+ catch {
52
+ continue;
53
+ }
54
+ if (st.isDirectory()) {
55
+ stack.push(full);
56
+ }
57
+ else if (name.endsWith('.md')) {
58
+ files.push({ path: full, content: readFileSync(full, 'utf8') });
59
+ }
60
+ }
61
+ }
62
+ }
63
+ return { rootPath: projectsRoot, files };
64
+ }
65
+ //# sourceMappingURL=read-claude-memory-tree.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"read-claude-memory-tree.js","sourceRoot":"","sources":["../../src/cli/read-claude-memory-tree.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAC9D,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAGjC;;;;GAIG;AACH,MAAM,UAAU,oBAAoB,CAAC,YAAoB;IACvD,MAAM,KAAK,GAA8B,EAAE,CAAC;IAC5C,IAAI,QAAQ,CAAC;IACb,IAAI,CAAC;QACH,QAAQ,GAAG,QAAQ,CAAC,YAAY,CAAC,CAAC;IACpC,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,QAAQ,EAAE,YAAY,EAAE,KAAK,EAAE,CAAC;IAC3C,CAAC;IACD,IAAI,CAAC,QAAQ,CAAC,WAAW,EAAE,EAAE,CAAC;QAC5B,OAAO,EAAE,QAAQ,EAAE,YAAY,EAAE,KAAK,EAAE,CAAC;IAC3C,CAAC;IACD,KAAK,MAAM,WAAW,IAAI,WAAW,CAAC,YAAY,CAAC,EAAE,CAAC;QACpD,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,EAAE,WAAW,CAAC,CAAC;QACnD,IAAI,WAAW,CAAC;QAChB,IAAI,CAAC;YACH,WAAW,GAAG,QAAQ,CAAC,UAAU,CAAC,CAAC;QACrC,CAAC;QAAC,MAAM,CAAC;YACP,SAAS;QACX,CAAC;QACD,IAAI,CAAC,WAAW,CAAC,WAAW,EAAE;YAAE,SAAS;QACzC,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;QAC7C,IAAI,UAAU,CAAC;QACf,IAAI,CAAC;YACH,UAAU,GAAG,QAAQ,CAAC,SAAS,CAAC,CAAC;QACnC,CAAC;QAAC,MAAM,CAAC;YACP,SAAS;QACX,CAAC;QACD,IAAI,CAAC,UAAU,CAAC,WAAW,EAAE;YAAE,SAAS;QACxC,iCAAiC;QACjC,MAAM,KAAK,GAAa,CAAC,SAAS,CAAC,CAAC;QACpC,OAAO,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACxB,MAAM,GAAG,GAAG,KAAK,CAAC,GAAG,EAAG,CAAC;YACzB,KAAK,MAAM,IAAI,IAAI,WAAW,CAAC,GAAG,CAAC,EAAE,CAAC;gBACpC,MAAM,IAAI,GAAG,IAAI,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;gBAC7B,IAAI,EAAE,CAAC;gBACP,IAAI,CAAC;oBACH,EAAE,GAAG,QAAQ,CAAC,IAAI,CAAC,CAAC;gBACtB,CAAC;gBAAC,MAAM,CAAC;oBACP,SAAS;gBACX,CAAC;gBACD,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC;oBACrB,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBACnB,CAAC;qBAAM,IAAI,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC;oBAChC,KAAK,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,OAAO,EAAE,YAAY,CAAC,IAAI,EAAE,MAAM,CAAC,EAAE,CAAC,CAAC;gBAClE,CAAC;YACH,CAAC;QACH,CAAC;IACH,CAAC;IACD,OAAO,EAAE,QAAQ,EAAE,YAAY,EAAE,KAAK,EAAE,CAAC;AAC3C,CAAC"}
@@ -0,0 +1,2 @@
1
+ #!/usr/bin/env node
2
+ export {};
@@ -0,0 +1,155 @@
1
+ #!/usr/bin/env node
2
+ import { join, resolve } from 'node:path';
3
+ import { homedir } from 'node:os';
4
+ import { createInterface } from 'node:readline/promises';
5
+ import { runIngestion } from '../pipeline/ingestion.js';
6
+ import { LocalClaudeSource } from '../sources/local-claude.js';
7
+ import { SyncConfig } from '../config.js';
8
+ import { TrustLedger } from '../trust-ledger.js';
9
+ import { LocalMirror } from '../local-mirror.js';
10
+ import { interactiveReview, bulkSelectAll } from '../review/cli-review.js';
11
+ import { resolveScope } from '../pipeline/scope-resolver.js';
12
+ import { readClaudeMemoryTree } from './read-claude-memory-tree.js';
13
+ function parseArgs(argv) {
14
+ const args = { source: 'claude-local', dryRun: false, saveScope: false };
15
+ for (let i = 0; i < argv.length; i++) {
16
+ if (argv[i] === '--source' && argv[i + 1]) {
17
+ args.source = argv[++i];
18
+ }
19
+ else if (argv[i] === '--scope' && argv[i + 1]) {
20
+ args.scope = argv[++i];
21
+ }
22
+ else if (argv[i] === '--dry-run') {
23
+ args.dryRun = true;
24
+ }
25
+ else if (argv[i] === '--save-scope') {
26
+ args.saveScope = true;
27
+ }
28
+ }
29
+ return args;
30
+ }
31
+ function getAdapter(id) {
32
+ if (id === 'claude-local')
33
+ return new LocalClaudeSource();
34
+ throw new Error(`Unknown source: ${id}. Phase 1 supports only 'claude-local'.`);
35
+ }
36
+ async function bulkStoreViaAis(agentId, drafts) {
37
+ const apiBase = process.env['AIS_DOMAIN']
38
+ ? `https://${process.env['AIS_DOMAIN']}`
39
+ : 'https://ais.agentsandswarms.ai';
40
+ const apiKey = process.env['AIS_SERVICE_KEY'] ?? process.env['AIS_API_KEY'];
41
+ const tenantId = process.env['AIS_TENANT_ID'];
42
+ if (!apiKey || !tenantId)
43
+ throw new Error('AIS_API_KEY and AIS_TENANT_ID must be set');
44
+ const body = {
45
+ memories: drafts.map((d) => ({
46
+ content: d.content,
47
+ type: d.type,
48
+ importance: d.importance,
49
+ metadata: { provenance: d.provenance, trustScoreHint: d.trustScore },
50
+ })),
51
+ };
52
+ const res = await fetch(`${apiBase}/v1/agents/${agentId}/memory/bulk`, {
53
+ method: 'POST',
54
+ headers: {
55
+ 'Content-Type': 'application/json',
56
+ 'x-api-key': apiKey,
57
+ 'x-tenant-id': tenantId,
58
+ },
59
+ body: JSON.stringify(body),
60
+ });
61
+ if (!res.ok) {
62
+ const bodyText = await res.text().catch(() => '');
63
+ throw new Error(`AIS bulk store failed: HTTP ${res.status} ${res.statusText}${bodyText ? ` — ${bodyText.slice(0, 500)}` : ''}`);
64
+ }
65
+ const json = (await res.json());
66
+ return { created: json.data?.created ?? [], failed: json.data?.failed ?? [] };
67
+ }
68
+ async function main() {
69
+ const args = parseArgs(process.argv.slice(2));
70
+ const agentId = process.env['AIS_AGENT_ID'];
71
+ if (!agentId)
72
+ throw new Error('AIS_AGENT_ID must be set (your bonded agent id)');
73
+ const adapter = getAdapter(args.source);
74
+ const home = homedir();
75
+ const config = new SyncConfig(home);
76
+ const ledger = new TrustLedger(home);
77
+ const mirror = new LocalMirror(home, config.retentionDays);
78
+ const claudeRoot = join(home, '.claude', 'projects');
79
+ const rawSourceData = readClaudeMemoryTree(claudeRoot);
80
+ const projectPath = resolve(process.cwd());
81
+ const scope = args.scope
82
+ ? { kind: 'user-named', label: args.scope }
83
+ : await resolveScope({
84
+ config,
85
+ projectPath,
86
+ activeGoal: null,
87
+ sessionTopic: null,
88
+ confirmer: args.saveScope && process.stdin.isTTY
89
+ ? async (p) => {
90
+ process.stdout.write(`Proposed scope: ${p.kind} / "${p.label}"${p.projectPath ? ` (path: ${p.projectPath})` : ''}\n`);
91
+ const rl = createInterface({ input: process.stdin, output: process.stdout });
92
+ try {
93
+ const answer = (await rl.question('Use this scope and save as default? [y/N] '))
94
+ .trim()
95
+ .toLowerCase();
96
+ if (!(answer === 'y' || answer === 'yes')) {
97
+ throw new Error('Scope not confirmed; aborting. Rerun without --save-scope or with --scope LABEL.');
98
+ }
99
+ return p;
100
+ }
101
+ finally {
102
+ rl.close();
103
+ }
104
+ }
105
+ : async (p) => p,
106
+ saveAsDefault: args.saveScope,
107
+ });
108
+ const scopeKey = [
109
+ scope.kind,
110
+ scope.goalId ?? '',
111
+ scope.projectPath ?? '',
112
+ scope.label,
113
+ ].join('|');
114
+ const firstSyncForScope = !ledger.isTrusted(agentId, adapter.id, scopeKey);
115
+ const review = async (drafts) => args.dryRun
116
+ ? bulkSelectAll(drafts)
117
+ : interactiveReview(drafts, {
118
+ sourceId: adapter.id,
119
+ scopeLabel: scope.label,
120
+ firstSyncForScope,
121
+ });
122
+ const bulkStore = args.dryRun
123
+ ? async (drafts) => ({
124
+ created: drafts.map((_, i) => ({ index: i, id: `dry-run-${i}` })),
125
+ failed: [],
126
+ })
127
+ : (drafts) => bulkStoreViaAis(agentId, drafts);
128
+ const result = await runIngestion({
129
+ adapter,
130
+ rawSourceData,
131
+ scope,
132
+ agentId,
133
+ config,
134
+ ledger,
135
+ mirror,
136
+ existingHashes: new Set(),
137
+ bulkStore,
138
+ review,
139
+ });
140
+ process.stdout.write(`\n=== Sync complete ===\n`);
141
+ process.stdout.write(`imported: ${result.imported}\n`);
142
+ process.stdout.write(`skippedExactDupes: ${result.skippedExactDupes}\n`);
143
+ process.stdout.write(`skippedByUser: ${result.skippedByUser}\n`);
144
+ process.stdout.write(`failed: ${result.failed.length}\n`);
145
+ if (result.warnings.length > 0) {
146
+ process.stdout.write(`\nwarnings:\n`);
147
+ for (const w of result.warnings)
148
+ process.stdout.write(` - ${w}\n`);
149
+ }
150
+ }
151
+ main().catch((err) => {
152
+ process.stderr.write(`sync failed: ${err instanceof Error ? err.message : String(err)}\n`);
153
+ process.exit(1);
154
+ });
155
+ //# sourceMappingURL=sync-memory.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sync-memory.js","sourceRoot":"","sources":["../../src/cli/sync-memory.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,11 @@
1
+ import type { SyncScope } from './pipeline/types.js';
2
+ export declare class SyncConfig {
3
+ private readonly filePath;
4
+ private data;
5
+ constructor(home?: string);
6
+ get retentionDays(): number;
7
+ autoCommit(sourceId: string): boolean;
8
+ scopePreference(projectPath: string): SyncScope | null;
9
+ saveScopePreference(projectPath: string, scope: SyncScope): void;
10
+ private persist;
11
+ }
package/dist/config.js ADDED
@@ -0,0 +1,51 @@
1
+ import { existsSync, mkdirSync, readFileSync, writeFileSync, renameSync } from 'node:fs';
2
+ import { join } from 'node:path';
3
+ import { homedir } from 'node:os';
4
+ const DEFAULT_RETENTION_DAYS = 30;
5
+ export class SyncConfig {
6
+ filePath;
7
+ data;
8
+ constructor(home = homedir()) {
9
+ const dir = join(home, '.aismemory');
10
+ this.filePath = join(dir, 'config.json');
11
+ if (!existsSync(dir)) {
12
+ mkdirSync(dir, { recursive: true, mode: 0o700 });
13
+ }
14
+ if (existsSync(this.filePath)) {
15
+ try {
16
+ this.data = JSON.parse(readFileSync(this.filePath, 'utf8'));
17
+ }
18
+ catch {
19
+ const backup = `${this.filePath}.corrupt-${Date.now()}`;
20
+ try {
21
+ renameSync(this.filePath, backup);
22
+ }
23
+ catch { /* ignore */ }
24
+ console.warn(`aismemory: corrupt config.json backed up to ${backup}, using defaults`);
25
+ this.data = {};
26
+ }
27
+ }
28
+ else {
29
+ this.data = {};
30
+ }
31
+ }
32
+ get retentionDays() {
33
+ return this.data.retentionDays ?? DEFAULT_RETENTION_DAYS;
34
+ }
35
+ autoCommit(sourceId) {
36
+ return this.data.autoCommit?.[sourceId] === true;
37
+ }
38
+ scopePreference(projectPath) {
39
+ return this.data.scopePreferences?.[projectPath] ?? null;
40
+ }
41
+ saveScopePreference(projectPath, scope) {
42
+ this.data.scopePreferences = { ...(this.data.scopePreferences ?? {}), [projectPath]: scope };
43
+ this.persist();
44
+ }
45
+ persist() {
46
+ const tmp = `${this.filePath}.tmp`;
47
+ writeFileSync(tmp, JSON.stringify(this.data, null, 2), { mode: 0o600 });
48
+ renameSync(tmp, this.filePath);
49
+ }
50
+ }
51
+ //# sourceMappingURL=config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.js","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,SAAS,EAAE,YAAY,EAAE,aAAa,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACzF,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AASlC,MAAM,sBAAsB,GAAG,EAAE,CAAC;AAElC,MAAM,OAAO,UAAU;IACJ,QAAQ,CAAS;IAC1B,IAAI,CAAa;IAEzB,YAAY,OAAe,OAAO,EAAE;QAClC,MAAM,GAAG,GAAG,IAAI,CAAC,IAAI,EAAE,YAAY,CAAC,CAAC;QACrC,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,GAAG,EAAE,aAAa,CAAC,CAAC;QACzC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;YACrB,SAAS,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;QACnD,CAAC;QACD,IAAI,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC9B,IAAI,CAAC;gBACH,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAe,CAAC;YAC5E,CAAC;YAAC,MAAM,CAAC;gBACP,MAAM,MAAM,GAAG,GAAG,IAAI,CAAC,QAAQ,YAAY,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC;gBACxD,IAAI,CAAC;oBAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;gBAAC,CAAC;gBAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC;gBACjE,OAAO,CAAC,IAAI,CAAC,+CAA+C,MAAM,kBAAkB,CAAC,CAAC;gBACtF,IAAI,CAAC,IAAI,GAAG,EAAE,CAAC;YACjB,CAAC;QACH,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,IAAI,GAAG,EAAE,CAAC;QACjB,CAAC;IACH,CAAC;IAED,IAAI,aAAa;QACf,OAAO,IAAI,CAAC,IAAI,CAAC,aAAa,IAAI,sBAAsB,CAAC;IAC3D,CAAC;IAED,UAAU,CAAC,QAAgB;QACzB,OAAO,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC,QAAQ,CAAC,KAAK,IAAI,CAAC;IACnD,CAAC;IAED,eAAe,CAAC,WAAmB;QACjC,OAAO,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC,WAAW,CAAC,IAAI,IAAI,CAAC;IAC3D,CAAC;IAED,mBAAmB,CAAC,WAAmB,EAAE,KAAgB;QACvD,IAAI,CAAC,IAAI,CAAC,gBAAgB,GAAG,EAAE,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,gBAAgB,IAAI,EAAE,CAAC,EAAE,CAAC,WAAW,CAAC,EAAE,KAAK,EAAE,CAAC;QAC7F,IAAI,CAAC,OAAO,EAAE,CAAC;IACjB,CAAC;IAEO,OAAO;QACb,MAAM,GAAG,GAAG,GAAG,IAAI,CAAC,QAAQ,MAAM,CAAC;QACnC,aAAa,CAAC,GAAG,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;QACxE,UAAU,CAAC,GAAG,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;IACjC,CAAC;CACF"}
@@ -0,0 +1,2 @@
1
+ import type { AgentContext } from './types.js';
2
+ export declare function hydrateIdentity(aisUrl: string, agentId: string, token: string): Promise<AgentContext | null>;
@@ -0,0 +1,62 @@
1
+ export async function hydrateIdentity(aisUrl, agentId, token) {
2
+ try {
3
+ const res = await fetch(`${aisUrl}/v1/agents/${agentId}/full-context`, {
4
+ headers: { Authorization: `Bearer ${token}` },
5
+ });
6
+ if (!res.ok)
7
+ return null;
8
+ const json = (await res.json());
9
+ if (!json.success || !json.data)
10
+ return null;
11
+ const d = json.data;
12
+ const identity = {
13
+ id: d.identity?.id ?? '',
14
+ did: d.identity?.did ?? '',
15
+ name: d.identity?.name ?? '',
16
+ description: d.identity?.description ?? '',
17
+ status: d.identity?.status ?? '',
18
+ };
19
+ const personality = d.personality != null
20
+ ? {
21
+ traits: d.personality.traits ?? [],
22
+ communicationStyle: d.personality.communicationStyle ?? '',
23
+ tonePreferences: d.personality.tonePreferences ?? [],
24
+ }
25
+ : null;
26
+ const goals = (d.goals ?? []).map((g) => ({
27
+ title: g.title ?? '',
28
+ progress: g.progress ?? 0,
29
+ status: g.status ?? '',
30
+ }));
31
+ const recentMemories = (d.recentMemories ?? []).map((m) => ({
32
+ id: m.id ?? '',
33
+ content: m.content ?? '',
34
+ type: m.type ?? '',
35
+ importance: m.importance ?? 0,
36
+ createdAt: m.createdAt ?? '',
37
+ }));
38
+ const skills = (d.skills ?? []).map((s) => ({ name: s.name ?? '' }));
39
+ const lastHandoff = d.lastHandoff != null
40
+ ? {
41
+ summary: d.lastHandoff.summary ?? '',
42
+ keyLearnings: d.lastHandoff.keyLearnings ?? [],
43
+ taskContext: d.lastHandoff.taskContext ?? '',
44
+ createdAt: d.lastHandoff.createdAt ?? '',
45
+ }
46
+ : null;
47
+ return {
48
+ identity,
49
+ personality,
50
+ goals,
51
+ recentMemories,
52
+ skills,
53
+ lastHandoff,
54
+ assembledAt: d.assembledAt ?? '',
55
+ totalTokens: d.context?.totalTokens ?? 0,
56
+ };
57
+ }
58
+ catch {
59
+ return null;
60
+ }
61
+ }
62
+ //# sourceMappingURL=hydration.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hydration.js","sourceRoot":"","sources":["../src/hydration.ts"],"names":[],"mappings":"AAEA,MAAM,CAAC,KAAK,UAAU,eAAe,CACnC,MAAc,EACd,OAAe,EACf,KAAa;IAEb,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,GAAG,MAAM,cAAc,OAAO,eAAe,EAAE;YACrE,OAAO,EAAE,EAAE,aAAa,EAAE,UAAU,KAAK,EAAE,EAAE;SAC9C,CAAC,CAAC;QACH,IAAI,CAAC,GAAG,CAAC,EAAE;YAAE,OAAO,IAAI,CAAC;QACzB,MAAM,IAAI,GAAG,CAAC,MAAM,GAAG,CAAC,IAAI,EAAE,CAqB7B,CAAC;QACF,IAAI,CAAC,IAAI,CAAC,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI;YAAE,OAAO,IAAI,CAAC;QAE7C,MAAM,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC;QAEpB,MAAM,QAAQ,GAAG;YACf,EAAE,EAAE,CAAC,CAAC,QAAQ,EAAE,EAAE,IAAI,EAAE;YACxB,GAAG,EAAE,CAAC,CAAC,QAAQ,EAAE,GAAG,IAAI,EAAE;YAC1B,IAAI,EAAE,CAAC,CAAC,QAAQ,EAAE,IAAI,IAAI,EAAE;YAC5B,WAAW,EAAE,CAAC,CAAC,QAAQ,EAAE,WAAW,IAAI,EAAE;YAC1C,MAAM,EAAE,CAAC,CAAC,QAAQ,EAAE,MAAM,IAAI,EAAE;SACjC,CAAC;QAEF,MAAM,WAAW,GACf,CAAC,CAAC,WAAW,IAAI,IAAI;YACnB,CAAC,CAAC;gBACE,MAAM,EAAE,CAAC,CAAC,WAAW,CAAC,MAAM,IAAI,EAAE;gBAClC,kBAAkB,EAAE,CAAC,CAAC,WAAW,CAAC,kBAAkB,IAAI,EAAE;gBAC1D,eAAe,EAAE,CAAC,CAAC,WAAW,CAAC,eAAe,IAAI,EAAE;aACrD;YACH,CAAC,CAAC,IAAI,CAAC;QAEX,MAAM,KAAK,GAAG,CAAC,CAAC,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YACxC,KAAK,EAAE,CAAC,CAAC,KAAK,IAAI,EAAE;YACpB,QAAQ,EAAE,CAAC,CAAC,QAAQ,IAAI,CAAC;YACzB,MAAM,EAAE,CAAC,CAAC,MAAM,IAAI,EAAE;SACvB,CAAC,CAAC,CAAC;QAEJ,MAAM,cAAc,GAAG,CAAC,CAAC,CAAC,cAAc,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YAC1D,EAAE,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE;YACd,OAAO,EAAE,CAAC,CAAC,OAAO,IAAI,EAAE;YACxB,IAAI,EAAE,CAAC,CAAC,IAAI,IAAI,EAAE;YAClB,UAAU,EAAE,CAAC,CAAC,UAAU,IAAI,CAAC;YAC7B,SAAS,EAAE,CAAC,CAAC,SAAS,IAAI,EAAE;SAC7B,CAAC,CAAC,CAAC;QAEJ,MAAM,MAAM,GAAG,CAAC,CAAC,CAAC,MAAM,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC;QAErE,MAAM,WAAW,GACf,CAAC,CAAC,WAAW,IAAI,IAAI;YACnB,CAAC,CAAC;gBACE,OAAO,EAAE,CAAC,CAAC,WAAW,CAAC,OAAO,IAAI,EAAE;gBACpC,YAAY,EAAE,CAAC,CAAC,WAAW,CAAC,YAAY,IAAI,EAAE;gBAC9C,WAAW,EAAE,CAAC,CAAC,WAAW,CAAC,WAAW,IAAI,EAAE;gBAC5C,SAAS,EAAE,CAAC,CAAC,WAAW,CAAC,SAAS,IAAI,EAAE;aACzC;YACH,CAAC,CAAC,IAAI,CAAC;QAEX,OAAO;YACL,QAAQ;YACR,WAAW;YACX,KAAK;YACL,cAAc;YACd,MAAM;YACN,WAAW;YACX,WAAW,EAAE,CAAC,CAAC,WAAW,IAAI,EAAE;YAChC,WAAW,EAAE,CAAC,CAAC,OAAO,EAAE,WAAW,IAAI,CAAC;SACzC,CAAC;IACJ,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC"}