@dot-ai/core 0.5.2

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 (96) hide show
  1. package/.ai/memory/2026-03-04.md +2 -0
  2. package/.ai/tasks.json +7 -0
  3. package/LICENSE +21 -0
  4. package/dist/__tests__/config.test.d.ts +2 -0
  5. package/dist/__tests__/config.test.d.ts.map +1 -0
  6. package/dist/__tests__/config.test.js +128 -0
  7. package/dist/__tests__/config.test.js.map +1 -0
  8. package/dist/__tests__/e2e.test.d.ts +2 -0
  9. package/dist/__tests__/e2e.test.d.ts.map +1 -0
  10. package/dist/__tests__/e2e.test.js +211 -0
  11. package/dist/__tests__/e2e.test.js.map +1 -0
  12. package/dist/__tests__/engine.test.d.ts +2 -0
  13. package/dist/__tests__/engine.test.d.ts.map +1 -0
  14. package/dist/__tests__/engine.test.js +271 -0
  15. package/dist/__tests__/engine.test.js.map +1 -0
  16. package/dist/__tests__/format.test.d.ts +2 -0
  17. package/dist/__tests__/format.test.d.ts.map +1 -0
  18. package/dist/__tests__/format.test.js +200 -0
  19. package/dist/__tests__/format.test.js.map +1 -0
  20. package/dist/__tests__/labels.test.d.ts +2 -0
  21. package/dist/__tests__/labels.test.d.ts.map +1 -0
  22. package/dist/__tests__/labels.test.js +82 -0
  23. package/dist/__tests__/labels.test.js.map +1 -0
  24. package/dist/__tests__/loader.test.d.ts +2 -0
  25. package/dist/__tests__/loader.test.d.ts.map +1 -0
  26. package/dist/__tests__/loader.test.js +161 -0
  27. package/dist/__tests__/loader.test.js.map +1 -0
  28. package/dist/__tests__/logger.test.d.ts +2 -0
  29. package/dist/__tests__/logger.test.d.ts.map +1 -0
  30. package/dist/__tests__/logger.test.js +95 -0
  31. package/dist/__tests__/logger.test.js.map +1 -0
  32. package/dist/__tests__/nodes.test.d.ts +2 -0
  33. package/dist/__tests__/nodes.test.d.ts.map +1 -0
  34. package/dist/__tests__/nodes.test.js +83 -0
  35. package/dist/__tests__/nodes.test.js.map +1 -0
  36. package/dist/config.d.ts +29 -0
  37. package/dist/config.d.ts.map +1 -0
  38. package/dist/config.js +141 -0
  39. package/dist/config.js.map +1 -0
  40. package/dist/contracts.d.ts +56 -0
  41. package/dist/contracts.d.ts.map +1 -0
  42. package/dist/contracts.js +2 -0
  43. package/dist/contracts.js.map +1 -0
  44. package/dist/engine.d.ts +38 -0
  45. package/dist/engine.d.ts.map +1 -0
  46. package/dist/engine.js +88 -0
  47. package/dist/engine.js.map +1 -0
  48. package/dist/format.d.ts +18 -0
  49. package/dist/format.d.ts.map +1 -0
  50. package/dist/format.js +89 -0
  51. package/dist/format.js.map +1 -0
  52. package/dist/index.d.ts +21 -0
  53. package/dist/index.d.ts.map +1 -0
  54. package/dist/index.js +22 -0
  55. package/dist/index.js.map +1 -0
  56. package/dist/labels.d.ts +13 -0
  57. package/dist/labels.d.ts.map +1 -0
  58. package/dist/labels.js +36 -0
  59. package/dist/labels.js.map +1 -0
  60. package/dist/loader.d.ts +26 -0
  61. package/dist/loader.d.ts.map +1 -0
  62. package/dist/loader.js +120 -0
  63. package/dist/loader.js.map +1 -0
  64. package/dist/logger.d.ts +29 -0
  65. package/dist/logger.d.ts.map +1 -0
  66. package/dist/logger.js +29 -0
  67. package/dist/logger.js.map +1 -0
  68. package/dist/nodes.d.ts +15 -0
  69. package/dist/nodes.d.ts.map +1 -0
  70. package/dist/nodes.js +46 -0
  71. package/dist/nodes.js.map +1 -0
  72. package/dist/types.d.ts +111 -0
  73. package/dist/types.d.ts.map +1 -0
  74. package/dist/types.js +2 -0
  75. package/dist/types.js.map +1 -0
  76. package/package.json +23 -0
  77. package/src/__tests__/config.test.ts +166 -0
  78. package/src/__tests__/e2e.test.ts +257 -0
  79. package/src/__tests__/engine.test.ts +305 -0
  80. package/src/__tests__/format.test.ts +247 -0
  81. package/src/__tests__/labels.test.ts +96 -0
  82. package/src/__tests__/loader.test.ts +191 -0
  83. package/src/__tests__/logger.test.ts +113 -0
  84. package/src/__tests__/nodes.test.ts +103 -0
  85. package/src/config.ts +178 -0
  86. package/src/contracts.ts +71 -0
  87. package/src/engine.ts +145 -0
  88. package/src/format.ts +113 -0
  89. package/src/index.ts +63 -0
  90. package/src/labels.ts +40 -0
  91. package/src/loader.ts +152 -0
  92. package/src/logger.ts +49 -0
  93. package/src/nodes.ts +46 -0
  94. package/src/types.ts +123 -0
  95. package/tsconfig.json +23 -0
  96. package/tsconfig.tsbuildinfo +1 -0
package/dist/nodes.js ADDED
@@ -0,0 +1,46 @@
1
+ import { readdirSync } from 'node:fs';
2
+ import { join } from 'node:path';
3
+ /**
4
+ * Discover all .ai/ directories in a workspace.
5
+ * Always includes root. Scans configurable directories for sub-nodes.
6
+ *
7
+ * @param root - workspace root (absolute path)
8
+ * @param scanDirs - directories to scan for sub-nodes (default: ["projects"])
9
+ */
10
+ export function discoverNodes(root, scanDirs = ['projects']) {
11
+ const nodes = [
12
+ { name: 'root', path: join(root, '.ai'), root: true },
13
+ ];
14
+ for (const dir of scanDirs) {
15
+ const scanPath = join(root, dir);
16
+ try {
17
+ const entries = readdirSync(scanPath, { withFileTypes: true });
18
+ for (const entry of entries) {
19
+ if (!entry.isDirectory())
20
+ continue;
21
+ const aiPath = join(scanPath, entry.name, '.ai');
22
+ try {
23
+ readdirSync(aiPath); // existence check
24
+ nodes.push({ name: entry.name, path: aiPath, root: false });
25
+ }
26
+ catch {
27
+ // No .ai/ in this directory
28
+ }
29
+ }
30
+ }
31
+ catch {
32
+ // Scan directory doesn't exist
33
+ }
34
+ }
35
+ return nodes;
36
+ }
37
+ /**
38
+ * Parse scanDirs from a config string value.
39
+ * Returns empty array if not configured.
40
+ */
41
+ export function parseScanDirs(value) {
42
+ if (!value || typeof value !== 'string')
43
+ return [];
44
+ return value.split(',').map(s => s.trim()).filter(Boolean);
45
+ }
46
+ //# sourceMappingURL=nodes.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"nodes.js","sourceRoot":"","sources":["../src/nodes.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AACtC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAGjC;;;;;;GAMG;AACH,MAAM,UAAU,aAAa,CAAC,IAAY,EAAE,WAAqB,CAAC,UAAU,CAAC;IAC3E,MAAM,KAAK,GAAW;QACpB,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE;KACtD,CAAC;IAEF,KAAK,MAAM,GAAG,IAAI,QAAQ,EAAE,CAAC;QAC3B,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;QACjC,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,WAAW,CAAC,QAAQ,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;YAC/D,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;gBAC5B,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE;oBAAE,SAAS;gBACnC,MAAM,MAAM,GAAG,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;gBACjD,IAAI,CAAC;oBACH,WAAW,CAAC,MAAM,CAAC,CAAC,CAAC,kBAAkB;oBACvC,KAAK,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,KAAK,CAAC,IAAI,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;gBAC9D,CAAC;gBAAC,MAAM,CAAC;oBACP,4BAA4B;gBAC9B,CAAC;YACH,CAAC;QACH,CAAC;QAAC,MAAM,CAAC;YACP,+BAA+B;QACjC,CAAC;IACH,CAAC;IAED,OAAO,KAAK,CAAC;AACf,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,aAAa,CAAC,KAAc;IAC1C,IAAI,CAAC,KAAK,IAAI,OAAO,KAAK,KAAK,QAAQ;QAAE,OAAO,EAAE,CAAC;IACnD,OAAO,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;AAC7D,CAAC"}
@@ -0,0 +1,111 @@
1
+ /**
2
+ * A label is a boolean tag matched against the prompt.
3
+ * No scores — matched or not.
4
+ */
5
+ export interface Label {
6
+ name: string;
7
+ source: string;
8
+ }
9
+ /**
10
+ * A workspace node — a directory containing .ai/ context.
11
+ * Root node is always included. Sub-nodes are discovered via scanDirs.
12
+ */
13
+ export interface Node {
14
+ name: string;
15
+ path: string;
16
+ root: boolean;
17
+ }
18
+ export interface MemoryEntry {
19
+ content: string;
20
+ type: string;
21
+ source: string;
22
+ date?: string;
23
+ labels?: string[];
24
+ node?: string;
25
+ }
26
+ export interface Skill {
27
+ name: string;
28
+ description: string;
29
+ labels: string[];
30
+ triggers?: string[];
31
+ path?: string;
32
+ content?: string;
33
+ dependsOn?: string[];
34
+ requiresTools?: string[];
35
+ enabled?: boolean;
36
+ node?: string;
37
+ }
38
+ export interface Identity {
39
+ type: string;
40
+ content: string;
41
+ source: string;
42
+ priority: number;
43
+ node?: string;
44
+ }
45
+ export interface Task {
46
+ id: string;
47
+ text: string;
48
+ status: string;
49
+ priority?: string;
50
+ project?: string;
51
+ tags?: string[];
52
+ metadata?: Record<string, unknown>;
53
+ }
54
+ export interface Tool {
55
+ name: string;
56
+ description: string;
57
+ labels: string[];
58
+ config: Record<string, unknown>;
59
+ source: string;
60
+ node?: string;
61
+ }
62
+ export interface RoutingResult {
63
+ model: string;
64
+ reason: string;
65
+ fallback?: string;
66
+ }
67
+ /**
68
+ * The output of the enrich() call.
69
+ * This is what adapters consume to inject into the agent.
70
+ */
71
+ export interface EnrichedContext {
72
+ prompt: string;
73
+ labels: Label[];
74
+ identities: Identity[];
75
+ memories: MemoryEntry[];
76
+ skills: Skill[];
77
+ tools: Tool[];
78
+ routing: RoutingResult;
79
+ }
80
+ /**
81
+ * Filter for task queries
82
+ */
83
+ export interface TaskFilter {
84
+ status?: string;
85
+ project?: string;
86
+ tags?: string[];
87
+ }
88
+ /**
89
+ * Configuration from dot-ai.yml
90
+ */
91
+ export interface DebugConfig {
92
+ logPath?: string;
93
+ }
94
+ export interface WorkspaceConfig {
95
+ scanDirs?: string;
96
+ }
97
+ export interface DotAiConfig {
98
+ memory?: ProviderConfig;
99
+ skills?: ProviderConfig;
100
+ identity?: ProviderConfig;
101
+ routing?: ProviderConfig;
102
+ tasks?: ProviderConfig;
103
+ tools?: ProviderConfig;
104
+ debug?: DebugConfig;
105
+ workspace?: WorkspaceConfig;
106
+ }
107
+ export interface ProviderConfig {
108
+ use: string;
109
+ with?: Record<string, unknown>;
110
+ }
111
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"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"}
package/dist/types.js ADDED
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":""}
package/package.json ADDED
@@ -0,0 +1,23 @@
1
+ {
2
+ "name": "@dot-ai/core",
3
+ "version": "0.5.2",
4
+ "repository": {
5
+ "type": "git",
6
+ "url": "https://github.com/jogelin/dot-ai",
7
+ "directory": "packages/core"
8
+ },
9
+ "type": "module",
10
+ "main": "dist/index.js",
11
+ "types": "dist/index.d.ts",
12
+ "devDependencies": {
13
+ "@types/node": "^22.0.0",
14
+ "typescript": "^5.9.3"
15
+ },
16
+ "publishConfig": {
17
+ "access": "public"
18
+ },
19
+ "scripts": {
20
+ "build": "tsc",
21
+ "test": "vitest run"
22
+ }
23
+ }
@@ -0,0 +1,166 @@
1
+ import { describe, it, expect, beforeEach } from 'vitest';
2
+ import { mkdtemp, writeFile, mkdir } from 'node:fs/promises';
3
+ import { tmpdir } from 'node:os';
4
+ import { join } from 'node:path';
5
+ import { loadConfig, resolveConfig } from '../config.js';
6
+
7
+ let testDir: string;
8
+
9
+ beforeEach(async () => {
10
+ testDir = await mkdtemp(join(tmpdir(), 'dot-ai-test-'));
11
+ await mkdir(join(testDir, '.ai'), { recursive: true });
12
+ });
13
+
14
+ describe('loadConfig', () => {
15
+ it('returns empty config when no file exists', async () => {
16
+ const config = await loadConfig(testDir);
17
+ expect(config).toEqual({});
18
+ });
19
+
20
+ it('returns empty config when .ai directory has no dot-ai.yml', async () => {
21
+ const config = await loadConfig('/nonexistent/path/to/workspace');
22
+ expect(config).toEqual({});
23
+ });
24
+
25
+ it('parses a simple memory section', async () => {
26
+ await writeFile(
27
+ join(testDir, '.ai', 'dot-ai.yml'),
28
+ `memory:\n use: @dot-ai/provider-file-memory\n`,
29
+ 'utf-8',
30
+ );
31
+ const config = await loadConfig(testDir);
32
+ expect(config.memory).toEqual({ use: '@dot-ai/provider-file-memory' });
33
+ });
34
+
35
+ it('parses multiple provider sections', async () => {
36
+ await writeFile(
37
+ join(testDir, '.ai', 'dot-ai.yml'),
38
+ [
39
+ 'memory:',
40
+ ' use: @dot-ai/provider-file-memory',
41
+ 'skills:',
42
+ ' use: @dot-ai/provider-file-skills',
43
+ 'routing:',
44
+ ' use: @dot-ai/provider-rules-routing',
45
+ ].join('\n'),
46
+ 'utf-8',
47
+ );
48
+ const config = await loadConfig(testDir);
49
+ expect(config.memory?.use).toBe('@dot-ai/provider-file-memory');
50
+ expect(config.skills?.use).toBe('@dot-ai/provider-file-skills');
51
+ expect(config.routing?.use).toBe('@dot-ai/provider-rules-routing');
52
+ });
53
+
54
+ it('parses nested with block', async () => {
55
+ await writeFile(
56
+ join(testDir, '.ai', 'dot-ai.yml'),
57
+ [
58
+ 'memory:',
59
+ ' use: @dot-ai/cockpit-memory',
60
+ ' url: http://localhost:3010',
61
+ ].join('\n'),
62
+ 'utf-8',
63
+ );
64
+ const config = await loadConfig(testDir);
65
+ expect(config.memory?.use).toBe('@dot-ai/cockpit-memory');
66
+ expect(config.memory?.with?.['url']).toBe('http://localhost:3010');
67
+ });
68
+
69
+ it('resolves ${ENV_VAR} references', async () => {
70
+ process.env['DOT_AI_TEST_URL'] = 'http://test-server:9999';
71
+ await writeFile(
72
+ join(testDir, '.ai', 'dot-ai.yml'),
73
+ [
74
+ 'memory:',
75
+ ' use: @dot-ai/cockpit-memory',
76
+ ' url: ${DOT_AI_TEST_URL}',
77
+ ].join('\n'),
78
+ 'utf-8',
79
+ );
80
+ const config = await loadConfig(testDir);
81
+ expect(config.memory?.with?.['url']).toBe('http://test-server:9999');
82
+ delete process.env['DOT_AI_TEST_URL'];
83
+ });
84
+
85
+ it('replaces undefined ENV_VAR with empty string', async () => {
86
+ delete process.env['MISSING_ENV_VAR'];
87
+ await writeFile(
88
+ join(testDir, '.ai', 'dot-ai.yml'),
89
+ [
90
+ 'memory:',
91
+ ' use: @dot-ai/provider-file-memory',
92
+ ' key: ${MISSING_ENV_VAR}',
93
+ ].join('\n'),
94
+ 'utf-8',
95
+ );
96
+ const config = await loadConfig(testDir);
97
+ expect(config.memory?.with?.['key']).toBe('');
98
+ });
99
+
100
+ it('skips comment lines (# prefix)', async () => {
101
+ await writeFile(
102
+ join(testDir, '.ai', 'dot-ai.yml'),
103
+ [
104
+ '# This is a comment',
105
+ 'memory:',
106
+ ' # nested comment',
107
+ ' use: @dot-ai/provider-file-memory',
108
+ ].join('\n'),
109
+ 'utf-8',
110
+ );
111
+ const config = await loadConfig(testDir);
112
+ expect(config.memory?.use).toBe('@dot-ai/provider-file-memory');
113
+ });
114
+ });
115
+
116
+ describe('resolveConfig', () => {
117
+ it('fills all defaults for empty config', () => {
118
+ const resolved = resolveConfig({});
119
+ expect(resolved.memory.use).toBe('@dot-ai/provider-file-memory');
120
+ expect(resolved.skills.use).toBe('@dot-ai/provider-file-skills');
121
+ expect(resolved.identity.use).toBe('@dot-ai/provider-file-identity');
122
+ expect(resolved.routing.use).toBe('@dot-ai/provider-rules-routing');
123
+ expect(resolved.tasks.use).toBe('@dot-ai/provider-file-tasks');
124
+ expect(resolved.tools.use).toBe('@dot-ai/provider-file-tools');
125
+ });
126
+
127
+ it('preserves existing memory config', () => {
128
+ const resolved = resolveConfig({ memory: { use: '@dot-ai/cockpit-memory', with: { url: 'http://x' } } });
129
+ expect(resolved.memory.use).toBe('@dot-ai/cockpit-memory');
130
+ expect(resolved.memory.with?.['url']).toBe('http://x');
131
+ });
132
+
133
+ it('fills defaults only for missing providers', () => {
134
+ const resolved = resolveConfig({ memory: { use: '@dot-ai/cockpit-memory' } });
135
+ expect(resolved.memory.use).toBe('@dot-ai/cockpit-memory');
136
+ expect(resolved.skills.use).toBe('@dot-ai/provider-file-skills');
137
+ expect(resolved.routing.use).toBe('@dot-ai/provider-rules-routing');
138
+ });
139
+
140
+ it('returns a Required<DotAiConfig> with all keys present', () => {
141
+ const resolved = resolveConfig({});
142
+ const keys = ['memory', 'skills', 'identity', 'routing', 'tasks', 'tools'] as const;
143
+ for (const key of keys) {
144
+ expect(resolved[key]).toBeDefined();
145
+ expect(typeof resolved[key].use).toBe('string');
146
+ }
147
+ });
148
+
149
+ it('preserves all six provided providers', () => {
150
+ const full = {
151
+ memory: { use: 'mem' },
152
+ skills: { use: 'ski' },
153
+ identity: { use: 'idn' },
154
+ routing: { use: 'rte' },
155
+ tasks: { use: 'tsk' },
156
+ tools: { use: 'tls' },
157
+ };
158
+ const resolved = resolveConfig(full);
159
+ expect(resolved.memory.use).toBe('mem');
160
+ expect(resolved.skills.use).toBe('ski');
161
+ expect(resolved.identity.use).toBe('idn');
162
+ expect(resolved.routing.use).toBe('rte');
163
+ expect(resolved.tasks.use).toBe('tsk');
164
+ expect(resolved.tools.use).toBe('tls');
165
+ });
166
+ });
@@ -0,0 +1,257 @@
1
+ import { describe, it, expect, beforeEach } from 'vitest';
2
+ import { mkdtemp, mkdir, writeFile } from 'node:fs/promises';
3
+ import { tmpdir } from 'node:os';
4
+ import { join } from 'node:path';
5
+ import {
6
+ loadConfig,
7
+ resolveConfig,
8
+ registerDefaults,
9
+ clearProviders,
10
+ createProviders,
11
+ boot,
12
+ enrich,
13
+ learn,
14
+ } from '../index.js';
15
+
16
+ describe('E2E: full pipeline', () => {
17
+ let root: string;
18
+
19
+ beforeEach(async () => {
20
+ clearProviders();
21
+ registerDefaults();
22
+
23
+ root = await mkdtemp(join(tmpdir(), 'dot-ai-e2e-'));
24
+ const ai = join(root, '.ai');
25
+
26
+ // Create .ai/ structure
27
+ await mkdir(ai, { recursive: true });
28
+ await mkdir(join(ai, 'memory'), { recursive: true });
29
+ await mkdir(join(ai, 'skills', 'ts-standards'), { recursive: true });
30
+ await mkdir(join(ai, 'skills', 'code-review'), { recursive: true });
31
+ await mkdir(join(ai, 'tools'), { recursive: true });
32
+
33
+ // Identity files
34
+ await writeFile(join(ai, 'AGENTS.md'), '# AGENTS.md\n\nYou are Kiwi, a helpful assistant.\n\n## Rules\n- Always be concise\n- Use TypeScript');
35
+ await writeFile(join(ai, 'SOUL.md'), '# SOUL.md\n\nBe genuine. Skip the fluff.');
36
+ await writeFile(join(ai, 'USER.md'), '# USER.md\n\nJo, developer, Belgium.');
37
+ await writeFile(join(ai, 'IDENTITY.md'), '# IDENTITY.md\n\nName: Kiwi\nEmoji: 🥝');
38
+
39
+ // Skills with frontmatter
40
+ await writeFile(join(ai, 'skills', 'ts-standards', 'SKILL.md'), [
41
+ '---',
42
+ 'description: TypeScript coding standards',
43
+ 'labels: [typescript, code, standards]',
44
+ 'triggers: [auto]',
45
+ '---',
46
+ '',
47
+ '## TypeScript Standards',
48
+ '',
49
+ '- Use strict mode',
50
+ '- Prefer const over let',
51
+ '- Add type annotations',
52
+ ].join('\n'));
53
+
54
+ await writeFile(join(ai, 'skills', 'code-review', 'SKILL.md'), [
55
+ '---',
56
+ 'description: Code review guidelines',
57
+ 'labels: [review, code-fix, bug]',
58
+ 'triggers: [auto]',
59
+ '---',
60
+ '',
61
+ '## Code Review',
62
+ '',
63
+ '- Check for edge cases',
64
+ '- Verify error handling',
65
+ ].join('\n'));
66
+
67
+ // Tools
68
+ await writeFile(join(ai, 'tools', 'eslint.yaml'), [
69
+ 'name: eslint',
70
+ 'description: TypeScript linter',
71
+ 'labels: [typescript, lint, code]',
72
+ ].join('\n'));
73
+
74
+ // Memory (some past entries)
75
+ await writeFile(join(ai, 'memory', '2026-03-01.md'), [
76
+ '- Fixed auth middleware N+1 query bug',
77
+ '- Decided to use JWT for auth tokens',
78
+ '- Rate limiting added to auth endpoints',
79
+ ].join('\n'));
80
+
81
+ await writeFile(join(ai, 'memory', '2026-03-02.md'), [
82
+ '- Refactored database connection pooling',
83
+ '- Updated React test suite to use vitest',
84
+ ].join('\n'));
85
+
86
+ // Config — uses 4-space indent for 'with' block (parsed as options)
87
+ // The parser maps 4-space keys directly into section.with
88
+ await writeFile(join(ai, 'dot-ai.yml'), [
89
+ '# dot-ai config',
90
+ 'memory:',
91
+ ' use: @dot-ai/provider-file-memory',
92
+ ` root: ${root}`,
93
+ 'skills:',
94
+ ' use: @dot-ai/provider-file-skills',
95
+ ` root: ${root}`,
96
+ 'identity:',
97
+ ' use: @dot-ai/provider-file-identity',
98
+ ` root: ${root}`,
99
+ 'routing:',
100
+ ' use: @dot-ai/provider-rules-routing',
101
+ 'tasks:',
102
+ ' use: @dot-ai/provider-file-tasks',
103
+ ` root: ${root}`,
104
+ 'tools:',
105
+ ' use: @dot-ai/provider-file-tools',
106
+ ` root: ${root}`,
107
+ ].join('\n'));
108
+ });
109
+
110
+ it('runs the complete pipeline: config → providers → boot → enrich', async () => {
111
+ // 1. Load config
112
+ const config = await loadConfig(root);
113
+ expect(config).toBeDefined();
114
+
115
+ // 2. Create providers
116
+ const providers = await createProviders(config);
117
+ expect(providers).toBeDefined();
118
+ expect(providers.memory).toBeDefined();
119
+ expect(providers.skills).toBeDefined();
120
+ expect(providers.identity).toBeDefined();
121
+ expect(providers.routing).toBeDefined();
122
+ expect(providers.tasks).toBeDefined();
123
+ expect(providers.tools).toBeDefined();
124
+
125
+ // 3. Boot
126
+ const cache = await boot(providers);
127
+ expect(cache.identities.length).toBe(4); // AGENTS, SOUL, USER, IDENTITY
128
+ expect(cache.skills.length).toBe(2); // ts-standards, code-review
129
+ expect(cache.vocabulary.length).toBeGreaterThan(0);
130
+ expect(cache.vocabulary).toContain('typescript');
131
+ expect(cache.vocabulary).toContain('code');
132
+
133
+ // 4. Enrich a prompt
134
+ const ctx = await enrich('Fix the TypeScript bug in the auth module', providers, cache);
135
+
136
+ // Check labels extracted
137
+ expect(ctx.labels.length).toBeGreaterThan(0);
138
+ expect(ctx.labels.some(l => l.name === 'typescript')).toBe(true);
139
+
140
+ // Check identities loaded
141
+ expect(ctx.identities.length).toBe(4);
142
+ expect(ctx.identities.find(i => i.type === 'agents')?.content).toContain('Kiwi');
143
+
144
+ // Check memories found (should match "auth" and/or "bug")
145
+ expect(ctx.memories.length).toBeGreaterThan(0);
146
+ expect(ctx.memories.some(m => m.content.includes('auth'))).toBe(true);
147
+
148
+ // Check skills matched (typescript label should match ts-standards)
149
+ expect(ctx.skills.length).toBeGreaterThan(0);
150
+ expect(ctx.skills.some(s => s.name === 'ts-standards')).toBe(true);
151
+
152
+ // Check tools matched (typescript label should match eslint)
153
+ expect(ctx.tools.length).toBeGreaterThan(0);
154
+ expect(ctx.tools.some(t => t.name === 'eslint')).toBe(true);
155
+
156
+ // Check routing (code-fix → sonnet)
157
+ expect(ctx.routing.model).toBeDefined();
158
+ });
159
+
160
+ it('enriches differently for different prompts', async () => {
161
+ const config = await loadConfig(root);
162
+ const providers = await createProviders(config);
163
+ const cache = await boot(providers);
164
+
165
+ // A code-related prompt should match code skills and tools
166
+ const codeFix = await enrich('Fix the TypeScript bug', providers, cache);
167
+ expect(codeFix.skills.some(s => s.name === 'ts-standards')).toBe(true);
168
+ expect(codeFix.tools.some(t => t.name === 'eslint')).toBe(true);
169
+
170
+ // A review-related prompt should match code-review skill
171
+ const review = await enrich('Review this code for bugs', providers, cache);
172
+ expect(review.skills.some(s => s.name === 'code-review')).toBe(true);
173
+ });
174
+
175
+ it('learns from agent responses and retrieves later', async () => {
176
+ const config = await loadConfig(root);
177
+ const providers = await createProviders(config);
178
+ const cache = await boot(providers);
179
+
180
+ // Learn something
181
+ await learn('Discovered that the connection pool was configured wrong', providers);
182
+
183
+ // Should be retrievable via memory search
184
+ const ctx = await enrich('What about the connection pool?', providers, cache);
185
+ expect(ctx.memories.some(m => m.content.includes('connection pool'))).toBe(true);
186
+ });
187
+
188
+ it('works with empty .ai/ directory (minimal config)', async () => {
189
+ const emptyRoot = await mkdtemp(join(tmpdir(), 'dot-ai-empty-'));
190
+ await mkdir(join(emptyRoot, '.ai'), { recursive: true });
191
+ await writeFile(join(emptyRoot, '.ai', 'dot-ai.yml'), '# empty config');
192
+
193
+ const config = await loadConfig(emptyRoot);
194
+ // Inject root so providers read from the empty dir (not process.cwd())
195
+ const resolvedConfig = {
196
+ memory: { use: '@dot-ai/provider-file-memory', with: { root: emptyRoot } },
197
+ skills: { use: '@dot-ai/provider-file-skills', with: { root: emptyRoot } },
198
+ identity: { use: '@dot-ai/provider-file-identity', with: { root: emptyRoot } },
199
+ routing: { use: '@dot-ai/provider-rules-routing' },
200
+ tasks: { use: '@dot-ai/provider-file-tasks', with: { root: emptyRoot } },
201
+ tools: { use: '@dot-ai/provider-file-tools', with: { root: emptyRoot } },
202
+ ...config,
203
+ };
204
+ const providers = await createProviders(resolvedConfig);
205
+ const cache = await boot(providers);
206
+
207
+ expect(cache.identities).toEqual([]);
208
+ expect(cache.skills).toEqual([]);
209
+ expect(cache.vocabulary).toEqual([]);
210
+
211
+ const ctx = await enrich('Hello', providers, cache);
212
+ expect(ctx.identities).toEqual([]);
213
+ expect(ctx.memories).toEqual([]);
214
+ expect(ctx.skills).toEqual([]);
215
+ });
216
+
217
+ it('unregistered provider falls back to noop gracefully', async () => {
218
+ // Auto-discovery: when a provider name is not in the registry and cannot be
219
+ // dynamically imported (e.g. package doesn't exist), createProviders must
220
+ // return a working noop instead of throwing.
221
+ clearProviders(); // no defaults registered
222
+ const providers = await createProviders({
223
+ memory: { use: '@dot-ai/nonexistent-memory-provider' },
224
+ });
225
+ // Should not throw — noop memory is returned
226
+ const memories = await providers.memory.search('anything');
227
+ expect(memories).toEqual([]);
228
+ await expect(
229
+ providers.memory.store({ content: 'x', type: 'log' }),
230
+ ).resolves.toBeUndefined();
231
+ });
232
+
233
+ it('boot caches skills and vocabulary for reuse across prompts', async () => {
234
+ const config = await loadConfig(root);
235
+ const providers = await createProviders(config);
236
+ const cache = await boot(providers);
237
+
238
+ // Run enrich multiple times — should all use same cache
239
+ const ctx1 = await enrich('Fix typescript', providers, cache);
240
+ const ctx2 = await enrich('Review code', providers, cache);
241
+ const ctx3 = await enrich('Something unrelated', providers, cache);
242
+
243
+ // All should have same identities from cache
244
+ expect(ctx1.identities).toEqual(ctx2.identities);
245
+ expect(ctx2.identities).toEqual(ctx3.identities);
246
+ });
247
+
248
+ it('resolveConfig fills defaults for missing providers', () => {
249
+ const resolved = resolveConfig({});
250
+ expect(resolved.memory.use).toBe('@dot-ai/provider-file-memory');
251
+ expect(resolved.skills.use).toBe('@dot-ai/provider-file-skills');
252
+ expect(resolved.identity.use).toBe('@dot-ai/provider-file-identity');
253
+ expect(resolved.routing.use).toBe('@dot-ai/provider-rules-routing');
254
+ expect(resolved.tasks.use).toBe('@dot-ai/provider-file-tasks');
255
+ expect(resolved.tools.use).toBe('@dot-ai/provider-file-tools');
256
+ });
257
+ });