aismemory 0.2.0 → 0.4.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/__tests__/cli-review.test.d.ts +1 -0
- package/dist/__tests__/cli-review.test.js +26 -0
- package/dist/__tests__/cli-review.test.js.map +1 -0
- package/dist/__tests__/config.test.d.ts +1 -0
- package/dist/__tests__/config.test.js +54 -0
- package/dist/__tests__/config.test.js.map +1 -0
- package/dist/__tests__/local-claude-source.test.d.ts +1 -0
- package/dist/__tests__/local-claude-source.test.js +97 -0
- package/dist/__tests__/local-claude-source.test.js.map +1 -0
- package/dist/__tests__/local-mirror.test.d.ts +1 -0
- package/dist/__tests__/local-mirror.test.js +95 -0
- package/dist/__tests__/local-mirror.test.js.map +1 -0
- package/dist/__tests__/mcp-init-no-auth.test.d.ts +1 -0
- package/dist/__tests__/mcp-init-no-auth.test.js +75 -0
- package/dist/__tests__/mcp-init-no-auth.test.js.map +1 -0
- package/dist/__tests__/pipeline-ingestion.test.d.ts +1 -0
- package/dist/__tests__/pipeline-ingestion.test.js +161 -0
- package/dist/__tests__/pipeline-ingestion.test.js.map +1 -0
- package/dist/__tests__/pipeline-scope-resolver.test.d.ts +1 -0
- package/dist/__tests__/pipeline-scope-resolver.test.js +51 -0
- package/dist/__tests__/pipeline-scope-resolver.test.js.map +1 -0
- package/dist/__tests__/pipeline-trust-tagger.test.d.ts +1 -0
- package/dist/__tests__/pipeline-trust-tagger.test.js +24 -0
- package/dist/__tests__/pipeline-trust-tagger.test.js.map +1 -0
- package/dist/__tests__/read-claude-memory-tree.test.d.ts +1 -0
- package/dist/__tests__/read-claude-memory-tree.test.js +37 -0
- package/dist/__tests__/read-claude-memory-tree.test.js.map +1 -0
- package/dist/__tests__/trust-ledger.test.d.ts +1 -0
- package/dist/__tests__/trust-ledger.test.js +55 -0
- package/dist/__tests__/trust-ledger.test.js.map +1 -0
- package/dist/cli/read-claude-memory-tree.d.ts +7 -0
- package/dist/cli/read-claude-memory-tree.js +65 -0
- package/dist/cli/read-claude-memory-tree.js.map +1 -0
- package/dist/cli/sync-memory.d.ts +2 -0
- package/dist/cli/sync-memory.js +155 -0
- package/dist/cli/sync-memory.js.map +1 -0
- package/dist/config.d.ts +11 -0
- package/dist/config.js +51 -0
- package/dist/config.js.map +1 -0
- package/dist/index.js +154 -54
- package/dist/index.js.map +1 -1
- package/dist/local-mirror.d.ts +15 -0
- package/dist/local-mirror.js +56 -0
- package/dist/local-mirror.js.map +1 -0
- package/dist/pipeline/dedupe.d.ts +11 -0
- package/dist/pipeline/dedupe.js +21 -0
- package/dist/pipeline/dedupe.js.map +1 -0
- package/dist/pipeline/ingestion.d.ts +56 -0
- package/dist/pipeline/ingestion.js +86 -0
- package/dist/pipeline/ingestion.js.map +1 -0
- package/dist/pipeline/scope-resolver.d.ts +15 -0
- package/dist/pipeline/scope-resolver.js +28 -0
- package/dist/pipeline/scope-resolver.js.map +1 -0
- package/dist/pipeline/trust-tagger.d.ts +10 -0
- package/dist/pipeline/trust-tagger.js +12 -0
- package/dist/pipeline/trust-tagger.js.map +1 -0
- package/dist/pipeline/types.d.ts +41 -0
- package/dist/pipeline/types.js +6 -0
- package/dist/pipeline/types.js.map +1 -0
- package/dist/review/cli-review.d.ts +10 -0
- package/dist/review/cli-review.js +47 -0
- package/dist/review/cli-review.js.map +1 -0
- package/dist/sources/local-claude.d.ts +11 -0
- package/dist/sources/local-claude.js +137 -0
- package/dist/sources/local-claude.js.map +1 -0
- package/dist/sources/types.d.ts +53 -0
- package/dist/sources/types.js +9 -0
- package/dist/sources/types.js.map +1 -0
- package/dist/trust-ledger.d.ts +8 -0
- package/dist/trust-ledger.js +61 -0
- package/dist/trust-ledger.js.map +1 -0
- package/package.json +1 -1
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import { mkdtempSync, rmSync } from 'node:fs';
|
|
3
|
+
import { tmpdir } from 'node:os';
|
|
4
|
+
import { join } from 'node:path';
|
|
5
|
+
import { resolveScope } from '../pipeline/scope-resolver.js';
|
|
6
|
+
import { SyncConfig } from '../config.js';
|
|
7
|
+
describe('resolveScope', () => {
|
|
8
|
+
let tmpHome;
|
|
9
|
+
beforeEach(() => {
|
|
10
|
+
tmpHome = mkdtempSync(join(tmpdir(), 'aismem-resolve-'));
|
|
11
|
+
});
|
|
12
|
+
afterEach(() => rmSync(tmpHome, { recursive: true, force: true }));
|
|
13
|
+
it('returns saved preference when present', async () => {
|
|
14
|
+
const cfg = new SyncConfig(tmpHome);
|
|
15
|
+
cfg.saveScopePreference('/tmp/myproj', { kind: 'project', label: 'myproj', projectPath: '/tmp/myproj' });
|
|
16
|
+
const scope = await resolveScope({
|
|
17
|
+
config: cfg,
|
|
18
|
+
projectPath: '/tmp/myproj',
|
|
19
|
+
activeGoal: null,
|
|
20
|
+
sessionTopic: null,
|
|
21
|
+
confirmer: async () => { throw new Error('should not prompt'); },
|
|
22
|
+
});
|
|
23
|
+
expect(scope.kind).toBe('project');
|
|
24
|
+
expect(scope.label).toBe('myproj');
|
|
25
|
+
});
|
|
26
|
+
it('proposes project scope when no preference but projectPath set', async () => {
|
|
27
|
+
const cfg = new SyncConfig(tmpHome);
|
|
28
|
+
const scope = await resolveScope({
|
|
29
|
+
config: cfg,
|
|
30
|
+
projectPath: '/tmp/newproj',
|
|
31
|
+
activeGoal: null,
|
|
32
|
+
sessionTopic: null,
|
|
33
|
+
confirmer: async (proposed) => proposed,
|
|
34
|
+
});
|
|
35
|
+
expect(scope.kind).toBe('project');
|
|
36
|
+
expect(scope.projectPath).toBe('/tmp/newproj');
|
|
37
|
+
});
|
|
38
|
+
it('prefers active goal over project', async () => {
|
|
39
|
+
const cfg = new SyncConfig(tmpHome);
|
|
40
|
+
const scope = await resolveScope({
|
|
41
|
+
config: cfg,
|
|
42
|
+
projectPath: '/tmp/proj',
|
|
43
|
+
activeGoal: { id: 'g1', description: 'ship sync' },
|
|
44
|
+
sessionTopic: null,
|
|
45
|
+
confirmer: async (proposed) => proposed,
|
|
46
|
+
});
|
|
47
|
+
expect(scope.kind).toBe('goal');
|
|
48
|
+
expect(scope.goalId).toBe('g1');
|
|
49
|
+
});
|
|
50
|
+
});
|
|
51
|
+
//# sourceMappingURL=pipeline-scope-resolver.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pipeline-scope-resolver.test.js","sourceRoot":"","sources":["../../src/__tests__/pipeline-scope-resolver.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { applyTrustTags } from '../pipeline/trust-tagger.js';
|
|
3
|
+
function draft(trustScore) {
|
|
4
|
+
const prov = {
|
|
5
|
+
source: 'claude-local',
|
|
6
|
+
sourceHash: 'h',
|
|
7
|
+
importedAt: new Date().toISOString(),
|
|
8
|
+
importedBy: 'a1',
|
|
9
|
+
sourceScope: { kind: 'project', label: 'p' },
|
|
10
|
+
sourcePrompt: '',
|
|
11
|
+
};
|
|
12
|
+
return { content: 'c', type: 'fact', importance: 0.5, trustScore, provenance: prov };
|
|
13
|
+
}
|
|
14
|
+
describe('applyTrustTags', () => {
|
|
15
|
+
it('sets default trustScore=0.5 for drafts that came in at 0', () => {
|
|
16
|
+
const result = applyTrustTags([draft(0)], { defaultTrustScore: 0.5 });
|
|
17
|
+
expect(result[0]?.trustScore).toBe(0.5);
|
|
18
|
+
});
|
|
19
|
+
it('preserves non-zero trustScore from adapter', () => {
|
|
20
|
+
const result = applyTrustTags([draft(0.8)], { defaultTrustScore: 0.5 });
|
|
21
|
+
expect(result[0]?.trustScore).toBe(0.8);
|
|
22
|
+
});
|
|
23
|
+
});
|
|
24
|
+
//# sourceMappingURL=pipeline-trust-tagger.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pipeline-trust-tagger.test.js","sourceRoot":"","sources":["../../src/__tests__/pipeline-trust-tagger.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EAAE,cAAc,EAAE,MAAM,6BAA6B,CAAC;AAG7D,SAAS,KAAK,CAAC,UAAkB;IAC/B,MAAM,IAAI,GAAqB;QAC7B,MAAM,EAAE,cAAc;QACtB,UAAU,EAAE,GAAG;QACf,UAAU,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;QACpC,UAAU,EAAE,IAAI;QAChB,WAAW,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,KAAK,EAAE,GAAG,EAAe;QACzD,YAAY,EAAE,EAAE;KACjB,CAAC;IACF,OAAO,EAAE,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,MAAM,EAAE,UAAU,EAAE,GAAG,EAAE,UAAU,EAAE,UAAU,EAAE,IAAI,EAAE,CAAC;AACvF,CAAC;AAED,QAAQ,CAAC,gBAAgB,EAAE,GAAG,EAAE;IAC9B,EAAE,CAAC,0DAA0D,EAAE,GAAG,EAAE;QAClE,MAAM,MAAM,GAAG,cAAc,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,EAAE,iBAAiB,EAAE,GAAG,EAAE,CAAC,CAAC;QACtE,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,UAAU,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAC1C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,4CAA4C,EAAE,GAAG,EAAE;QACpD,MAAM,MAAM,GAAG,cAAc,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,EAAE,EAAE,iBAAiB,EAAE,GAAG,EAAE,CAAC,CAAC;QACxE,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,UAAU,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAC1C,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import { mkdtempSync, mkdirSync, writeFileSync, rmSync } from 'node:fs';
|
|
3
|
+
import { tmpdir } from 'node:os';
|
|
4
|
+
import { join } from 'node:path';
|
|
5
|
+
import { readClaudeMemoryTree } from '../cli/read-claude-memory-tree.js';
|
|
6
|
+
describe('readClaudeMemoryTree', () => {
|
|
7
|
+
let tmpRoot;
|
|
8
|
+
beforeEach(() => {
|
|
9
|
+
tmpRoot = mkdtempSync(join(tmpdir(), 'aismem-walker-'));
|
|
10
|
+
});
|
|
11
|
+
afterEach(() => rmSync(tmpRoot, { recursive: true, force: true }));
|
|
12
|
+
it('only reads .md files from project/memory subtrees', () => {
|
|
13
|
+
// Project A with memory/ dir and .md files
|
|
14
|
+
const projectA = join(tmpRoot, 'projectA');
|
|
15
|
+
const memA = join(projectA, 'memory');
|
|
16
|
+
mkdirSync(memA, { recursive: true });
|
|
17
|
+
writeFileSync(join(memA, 'MEMORY.md'), '# A');
|
|
18
|
+
writeFileSync(join(memA, 'feedback_x.md'), '# fx');
|
|
19
|
+
// Also a .md OUTSIDE memory/ — should NOT be included
|
|
20
|
+
writeFileSync(join(projectA, 'README.md'), '# readme');
|
|
21
|
+
// Project B without memory/ dir — should be skipped entirely
|
|
22
|
+
const projectB = join(tmpRoot, 'projectB');
|
|
23
|
+
mkdirSync(projectB, { recursive: true });
|
|
24
|
+
writeFileSync(join(projectB, 'CLAUDE.md'), '# claude');
|
|
25
|
+
const result = readClaudeMemoryTree(tmpRoot);
|
|
26
|
+
const paths = result.files.map((f) => f.path);
|
|
27
|
+
expect(paths).toContain(join(memA, 'MEMORY.md'));
|
|
28
|
+
expect(paths).toContain(join(memA, 'feedback_x.md'));
|
|
29
|
+
expect(paths).not.toContain(join(projectA, 'README.md'));
|
|
30
|
+
expect(paths).not.toContain(join(projectB, 'CLAUDE.md'));
|
|
31
|
+
});
|
|
32
|
+
it('returns empty when root does not exist or is not a directory', () => {
|
|
33
|
+
const result = readClaudeMemoryTree(join(tmpRoot, 'does-not-exist'));
|
|
34
|
+
expect(result.files).toEqual([]);
|
|
35
|
+
});
|
|
36
|
+
});
|
|
37
|
+
//# sourceMappingURL=read-claude-memory-tree.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"read-claude-memory-tree.test.js","sourceRoot":"","sources":["../../src/__tests__/read-claude-memory-tree.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AACrE,OAAO,EAAE,WAAW,EAAE,SAAS,EAAE,aAAa,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACxE,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,oBAAoB,EAAE,MAAM,mCAAmC,CAAC;AAEzE,QAAQ,CAAC,sBAAsB,EAAE,GAAG,EAAE;IACpC,IAAI,OAAe,CAAC;IAEpB,UAAU,CAAC,GAAG,EAAE;QACd,OAAO,GAAG,WAAW,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,gBAAgB,CAAC,CAAC,CAAC;IAC1D,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;IAEnE,EAAE,CAAC,mDAAmD,EAAE,GAAG,EAAE;QAC3D,2CAA2C;QAC3C,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;QAC3C,MAAM,IAAI,GAAG,IAAI,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;QACtC,SAAS,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QACrC,aAAa,CAAC,IAAI,CAAC,IAAI,EAAE,WAAW,CAAC,EAAE,KAAK,CAAC,CAAC;QAC9C,aAAa,CAAC,IAAI,CAAC,IAAI,EAAE,eAAe,CAAC,EAAE,MAAM,CAAC,CAAC;QACnD,sDAAsD;QACtD,aAAa,CAAC,IAAI,CAAC,QAAQ,EAAE,WAAW,CAAC,EAAE,UAAU,CAAC,CAAC;QAEvD,6DAA6D;QAC7D,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;QAC3C,SAAS,CAAC,QAAQ,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QACzC,aAAa,CAAC,IAAI,CAAC,QAAQ,EAAE,WAAW,CAAC,EAAE,UAAU,CAAC,CAAC;QAEvD,MAAM,MAAM,GAAG,oBAAoB,CAAC,OAAO,CAAC,CAAC;QAC7C,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QAC9C,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,EAAE,WAAW,CAAC,CAAC,CAAC;QACjD,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,EAAE,eAAe,CAAC,CAAC,CAAC;QACrD,MAAM,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,EAAE,WAAW,CAAC,CAAC,CAAC;QACzD,MAAM,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,EAAE,WAAW,CAAC,CAAC,CAAC;IAC3D,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8DAA8D,EAAE,GAAG,EAAE;QACtE,MAAM,MAAM,GAAG,oBAAoB,CAAC,IAAI,CAAC,OAAO,EAAE,gBAAgB,CAAC,CAAC,CAAC;QACrE,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;IACnC,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -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,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,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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"}
|
package/dist/config.d.ts
ADDED
|
@@ -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"}
|