@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.
- package/.ai/memory/2026-03-04.md +2 -0
- package/.ai/tasks.json +7 -0
- package/LICENSE +21 -0
- package/dist/__tests__/config.test.d.ts +2 -0
- package/dist/__tests__/config.test.d.ts.map +1 -0
- package/dist/__tests__/config.test.js +128 -0
- package/dist/__tests__/config.test.js.map +1 -0
- package/dist/__tests__/e2e.test.d.ts +2 -0
- package/dist/__tests__/e2e.test.d.ts.map +1 -0
- package/dist/__tests__/e2e.test.js +211 -0
- package/dist/__tests__/e2e.test.js.map +1 -0
- package/dist/__tests__/engine.test.d.ts +2 -0
- package/dist/__tests__/engine.test.d.ts.map +1 -0
- package/dist/__tests__/engine.test.js +271 -0
- package/dist/__tests__/engine.test.js.map +1 -0
- package/dist/__tests__/format.test.d.ts +2 -0
- package/dist/__tests__/format.test.d.ts.map +1 -0
- package/dist/__tests__/format.test.js +200 -0
- package/dist/__tests__/format.test.js.map +1 -0
- package/dist/__tests__/labels.test.d.ts +2 -0
- package/dist/__tests__/labels.test.d.ts.map +1 -0
- package/dist/__tests__/labels.test.js +82 -0
- package/dist/__tests__/labels.test.js.map +1 -0
- package/dist/__tests__/loader.test.d.ts +2 -0
- package/dist/__tests__/loader.test.d.ts.map +1 -0
- package/dist/__tests__/loader.test.js +161 -0
- package/dist/__tests__/loader.test.js.map +1 -0
- package/dist/__tests__/logger.test.d.ts +2 -0
- package/dist/__tests__/logger.test.d.ts.map +1 -0
- package/dist/__tests__/logger.test.js +95 -0
- package/dist/__tests__/logger.test.js.map +1 -0
- package/dist/__tests__/nodes.test.d.ts +2 -0
- package/dist/__tests__/nodes.test.d.ts.map +1 -0
- package/dist/__tests__/nodes.test.js +83 -0
- package/dist/__tests__/nodes.test.js.map +1 -0
- package/dist/config.d.ts +29 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +141 -0
- package/dist/config.js.map +1 -0
- package/dist/contracts.d.ts +56 -0
- package/dist/contracts.d.ts.map +1 -0
- package/dist/contracts.js +2 -0
- package/dist/contracts.js.map +1 -0
- package/dist/engine.d.ts +38 -0
- package/dist/engine.d.ts.map +1 -0
- package/dist/engine.js +88 -0
- package/dist/engine.js.map +1 -0
- package/dist/format.d.ts +18 -0
- package/dist/format.d.ts.map +1 -0
- package/dist/format.js +89 -0
- package/dist/format.js.map +1 -0
- package/dist/index.d.ts +21 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +22 -0
- package/dist/index.js.map +1 -0
- package/dist/labels.d.ts +13 -0
- package/dist/labels.d.ts.map +1 -0
- package/dist/labels.js +36 -0
- package/dist/labels.js.map +1 -0
- package/dist/loader.d.ts +26 -0
- package/dist/loader.d.ts.map +1 -0
- package/dist/loader.js +120 -0
- package/dist/loader.js.map +1 -0
- package/dist/logger.d.ts +29 -0
- package/dist/logger.d.ts.map +1 -0
- package/dist/logger.js +29 -0
- package/dist/logger.js.map +1 -0
- package/dist/nodes.d.ts +15 -0
- package/dist/nodes.d.ts.map +1 -0
- package/dist/nodes.js +46 -0
- package/dist/nodes.js.map +1 -0
- package/dist/types.d.ts +111 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/package.json +23 -0
- package/src/__tests__/config.test.ts +166 -0
- package/src/__tests__/e2e.test.ts +257 -0
- package/src/__tests__/engine.test.ts +305 -0
- package/src/__tests__/format.test.ts +247 -0
- package/src/__tests__/labels.test.ts +96 -0
- package/src/__tests__/loader.test.ts +191 -0
- package/src/__tests__/logger.test.ts +113 -0
- package/src/__tests__/nodes.test.ts +103 -0
- package/src/config.ts +178 -0
- package/src/contracts.ts +71 -0
- package/src/engine.ts +145 -0
- package/src/format.ts +113 -0
- package/src/index.ts +63 -0
- package/src/labels.ts +40 -0
- package/src/loader.ts +152 -0
- package/src/logger.ts +49 -0
- package/src/nodes.ts +46 -0
- package/src/types.ts +123 -0
- package/tsconfig.json +23 -0
- 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"}
|
package/dist/types.d.ts
ADDED
|
@@ -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 @@
|
|
|
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
|
+
});
|