@archznn/crewloop-skills 0.6.0 → 0.7.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/README.md +4 -16
- package/package.json +1 -2
- package/packages/cli/dist/agents.js +1 -1
- package/packages/cli/dist/agents.js.map +1 -1
- package/packages/cli/dist/cli.d.ts.map +1 -1
- package/packages/cli/dist/cli.js +2 -30
- package/packages/cli/dist/cli.js.map +1 -1
- package/packages/cli/dist/hooks.d.ts +6 -4
- package/packages/cli/dist/hooks.d.ts.map +1 -1
- package/packages/cli/dist/hooks.js +250 -98
- package/packages/cli/dist/hooks.js.map +1 -1
- package/packages/cli/dist/tests/hooks.test.js +245 -33
- package/packages/cli/dist/tests/hooks.test.js.map +1 -1
- package/references/conventions.md +1 -10
- package/references/workflow.md +1 -1
- package/servers/dashboard/README.md +55 -1
- package/servers/dashboard/dist/adapters/agy.d.ts +19 -0
- package/servers/dashboard/dist/adapters/agy.d.ts.map +1 -0
- package/servers/dashboard/dist/adapters/agy.js +108 -0
- package/servers/dashboard/dist/adapters/agy.js.map +1 -0
- package/servers/dashboard/dist/adapters/codex.d.ts.map +1 -1
- package/servers/dashboard/dist/adapters/codex.js +2 -0
- package/servers/dashboard/dist/adapters/codex.js.map +1 -1
- package/servers/dashboard/dist/adapters/kimi.d.ts +1 -1
- package/servers/dashboard/dist/adapters/kimi.d.ts.map +1 -1
- package/servers/dashboard/dist/adapters/kimi.js +9 -0
- package/servers/dashboard/dist/adapters/kimi.js.map +1 -1
- package/servers/dashboard/dist/adapters/shim.d.ts +1 -1
- package/servers/dashboard/dist/adapters/shim.d.ts.map +1 -1
- package/servers/dashboard/dist/adapters/shim.js +32 -11
- package/servers/dashboard/dist/adapters/shim.js.map +1 -1
- package/servers/dashboard/dist/adapters/shim.test.js +46 -4
- package/servers/dashboard/dist/adapters/shim.test.js.map +1 -1
- package/servers/dashboard/dist/lib/constants.d.ts +5 -0
- package/servers/dashboard/dist/lib/constants.d.ts.map +1 -0
- package/servers/dashboard/dist/lib/constants.js +46 -0
- package/servers/dashboard/dist/lib/constants.js.map +1 -0
- package/servers/dashboard/dist/lib/format.d.ts +6 -0
- package/servers/dashboard/dist/lib/format.d.ts.map +1 -0
- package/servers/dashboard/dist/lib/format.js +52 -0
- package/servers/dashboard/dist/lib/format.js.map +1 -0
- package/servers/dashboard/dist/lib/graph.d.ts +22 -0
- package/servers/dashboard/dist/lib/graph.d.ts.map +1 -0
- package/servers/dashboard/dist/lib/graph.js +45 -0
- package/servers/dashboard/dist/lib/graph.js.map +1 -0
- package/servers/dashboard/dist/lib/invocations.d.ts +32 -0
- package/servers/dashboard/dist/lib/invocations.d.ts.map +1 -0
- package/servers/dashboard/dist/lib/invocations.js +135 -0
- package/servers/dashboard/dist/lib/invocations.js.map +1 -0
- package/servers/dashboard/dist/lib/invocations.test.d.ts +2 -0
- package/servers/dashboard/dist/lib/invocations.test.d.ts.map +1 -0
- package/servers/dashboard/dist/lib/invocations.test.js +68 -0
- package/servers/dashboard/dist/lib/invocations.test.js.map +1 -0
- package/servers/dashboard/dist/lib/paths.d.ts +2 -0
- package/servers/dashboard/dist/lib/paths.d.ts.map +1 -0
- package/servers/dashboard/dist/lib/paths.js +40 -0
- package/servers/dashboard/dist/lib/paths.js.map +1 -0
- package/servers/dashboard/dist/presenter.d.ts.map +1 -1
- package/servers/dashboard/dist/presenter.js +2 -0
- package/servers/dashboard/dist/presenter.js.map +1 -1
- package/servers/dashboard/dist/public/assets/index-DjmMKbPN.css +1 -0
- package/servers/dashboard/dist/public/assets/index-DzOqMleZ.js +5323 -0
- package/servers/dashboard/dist/public/assets/index-DzOqMleZ.js.map +1 -0
- package/servers/dashboard/dist/public/index.html +16 -0
- package/servers/dashboard/dist/server.d.ts.map +1 -1
- package/servers/dashboard/dist/server.js +5 -1
- package/servers/dashboard/dist/server.js.map +1 -1
- package/servers/dashboard/dist/skills/infer.d.ts.map +1 -1
- package/servers/dashboard/dist/skills/infer.js +0 -6
- package/servers/dashboard/dist/skills/infer.js.map +1 -1
- package/servers/dashboard/dist/skills/infer.test.js +10 -3
- package/servers/dashboard/dist/skills/infer.test.js.map +1 -1
- package/servers/dashboard/dist/skills/mapping.d.ts +0 -3
- package/servers/dashboard/dist/skills/mapping.d.ts.map +1 -1
- package/servers/dashboard/dist/skills/mapping.js +0 -18
- package/servers/dashboard/dist/skills/mapping.js.map +1 -1
- package/servers/dashboard/dist/skills/registry.d.ts.map +1 -1
- package/servers/dashboard/dist/skills/registry.js +0 -1
- package/servers/dashboard/dist/skills/registry.js.map +1 -1
- package/servers/dashboard/dist/tests/adapters.test.d.ts +2 -0
- package/servers/dashboard/dist/tests/adapters.test.d.ts.map +1 -0
- package/servers/dashboard/dist/tests/adapters.test.js +180 -0
- package/servers/dashboard/dist/tests/adapters.test.js.map +1 -0
- package/servers/dashboard/dist/tests/lib-helpers.test.d.ts +2 -0
- package/servers/dashboard/dist/tests/lib-helpers.test.d.ts.map +1 -0
- package/servers/dashboard/dist/tests/lib-helpers.test.js +123 -0
- package/servers/dashboard/dist/tests/lib-helpers.test.js.map +1 -0
- package/servers/dashboard/dist/tests/shim.test.js +88 -2
- package/servers/dashboard/dist/tests/shim.test.js.map +1 -1
- package/servers/dashboard/dist/types.d.ts +5 -2
- package/servers/dashboard/dist/types.d.ts.map +1 -1
- package/servers/dashboard/package.json +22 -5
- package/servers/dashboard/src/adapters/agy.ts +136 -0
- package/servers/dashboard/src/adapters/codex.ts +2 -0
- package/servers/dashboard/src/adapters/kimi.ts +11 -1
- package/servers/dashboard/src/adapters/shim.test.ts +57 -4
- package/servers/dashboard/src/adapters/shim.ts +31 -11
- package/servers/dashboard/src/lib/constants.ts +44 -0
- package/servers/dashboard/src/lib/format.ts +44 -0
- package/servers/dashboard/src/lib/graph.ts +69 -0
- package/servers/dashboard/src/lib/invocations.test.ts +70 -0
- package/servers/dashboard/src/lib/invocations.ts +172 -0
- package/servers/dashboard/src/lib/paths.ts +35 -0
- package/servers/dashboard/src/presenter.ts +2 -0
- package/servers/dashboard/src/server.ts +5 -1
- package/servers/dashboard/src/skills/infer.test.ts +11 -3
- package/servers/dashboard/src/skills/infer.ts +1 -8
- package/servers/dashboard/src/skills/mapping.ts +0 -20
- package/servers/dashboard/src/skills/registry.ts +0 -1
- package/servers/dashboard/src/tests/adapters.test.ts +198 -0
- package/servers/dashboard/src/tests/lib-helpers.test.ts +133 -0
- package/servers/dashboard/src/tests/shim.test.ts +110 -2
- package/servers/dashboard/src/types.ts +5 -3
- package/servers/dashboard/ui/index.html +15 -0
- package/servers/dashboard/ui/postcss.config.js +6 -0
- package/servers/dashboard/ui/src/App.tsx +360 -0
- package/servers/dashboard/ui/src/components/ActiveSkillPanel.tsx +69 -0
- package/servers/dashboard/ui/src/components/ActivityGraph.tsx +74 -0
- package/servers/dashboard/ui/src/components/CommandPalette.tsx +200 -0
- package/servers/dashboard/ui/src/components/FileActivity.tsx +20 -0
- package/servers/dashboard/ui/src/components/FileDiff.tsx +68 -0
- package/servers/dashboard/ui/src/components/FileList.tsx +64 -0
- package/servers/dashboard/ui/src/components/FilterBar.tsx +208 -0
- package/servers/dashboard/ui/src/components/Network3D.tsx +178 -0
- package/servers/dashboard/ui/src/components/SessionSelector.tsx +95 -0
- package/servers/dashboard/ui/src/components/Sidebar.tsx +110 -0
- package/servers/dashboard/ui/src/components/TelemetryPanel.tsx +57 -0
- package/servers/dashboard/ui/src/components/Timeline.tsx +57 -0
- package/servers/dashboard/ui/src/components/TimelineRow.tsx +112 -0
- package/servers/dashboard/ui/src/components/TopBar.tsx +116 -0
- package/servers/dashboard/ui/src/components/ViewHeader.tsx +19 -0
- package/servers/dashboard/ui/src/components/ui/Icon.tsx +105 -0
- package/servers/dashboard/ui/src/components/ui/StatusBadge.tsx +19 -0
- package/servers/dashboard/ui/src/components/views/FilesView.tsx +23 -0
- package/servers/dashboard/ui/src/components/views/NetworkView.tsx +20 -0
- package/servers/dashboard/ui/src/components/views/Overview.tsx +135 -0
- package/servers/dashboard/ui/src/components/views/SessionsView.tsx +84 -0
- package/servers/dashboard/ui/src/components/views/SettingsView.tsx +138 -0
- package/servers/dashboard/ui/src/components/views/SkillsView.tsx +92 -0
- package/servers/dashboard/ui/src/components/views/TimelineView.tsx +46 -0
- package/servers/dashboard/ui/src/contexts/FilterContext.tsx +41 -0
- package/servers/dashboard/ui/src/contexts/PinnedSessionsContext.tsx +80 -0
- package/servers/dashboard/ui/src/contexts/SettingsContext.tsx +60 -0
- package/servers/dashboard/ui/src/hooks/useCommandPalette.ts +36 -0
- package/servers/dashboard/ui/src/hooks/useKeyboardShortcut.ts +38 -0
- package/servers/dashboard/ui/src/hooks/useNow.ts +12 -0
- package/servers/dashboard/ui/src/hooks/useReducedMotion.ts +15 -0
- package/servers/dashboard/ui/src/hooks/useSessions.ts +64 -0
- package/servers/dashboard/ui/src/hooks/useTheme.ts +30 -0
- package/servers/dashboard/ui/src/hooks/useViewport.ts +19 -0
- package/servers/dashboard/ui/src/hooks/useWebSocket.ts +118 -0
- package/servers/dashboard/ui/src/lib/export.test.ts +33 -0
- package/servers/dashboard/ui/src/lib/export.ts +39 -0
- package/servers/dashboard/ui/src/lib/filter.test.ts +95 -0
- package/servers/dashboard/ui/src/lib/filter.ts +178 -0
- package/servers/dashboard/ui/src/lib/format.test.ts +25 -0
- package/servers/dashboard/ui/src/lib/search.test.ts +52 -0
- package/servers/dashboard/ui/src/lib/search.ts +60 -0
- package/servers/dashboard/ui/src/lib/settings.test.ts +50 -0
- package/servers/dashboard/ui/src/lib/settings.ts +56 -0
- package/servers/dashboard/ui/src/lib/types.ts +124 -0
- package/servers/dashboard/ui/src/main.tsx +19 -0
- package/servers/dashboard/ui/src/styles/index.css +155 -0
- package/servers/dashboard/ui/tailwind.config.js +45 -0
- package/servers/dashboard/ui/tsconfig.json +33 -0
- package/servers/dashboard/ui/tsconfig.node.json +10 -0
- package/servers/dashboard/ui/vite.config.ts +37 -0
- package/servers/dashboard/ui/vitest.config.ts +8 -0
- package/skills/accessibility-auditor/SKILL.md +0 -20
- package/skills/architect/SKILL.md +0 -45
- package/skills/designer/SKILL.md +0 -30
- package/skills/docs-writer/SKILL.md +0 -13
- package/skills/engineer/SKILL.md +0 -30
- package/skills/maintainer/SKILL.md +0 -20
- package/skills/orchestrator/SKILL.md +0 -13
- package/skills/product-manager/SKILL.md +0 -20
- package/skills/researcher/SKILL.md +0 -20
- package/skills/reviewer/SKILL.md +0 -30
- package/skills/security-guard/SKILL.md +0 -20
- package/skills/shipper/SKILL.md +0 -33
- package/skills/tester/SKILL.md +0 -20
- package/packages/cli/dist/mcp.d.ts +0 -28
- package/packages/cli/dist/mcp.d.ts.map +0 -1
- package/packages/cli/dist/mcp.js +0 -148
- package/packages/cli/dist/mcp.js.map +0 -1
- package/packages/cli/dist/tests/mcp.test.d.ts +0 -2
- package/packages/cli/dist/tests/mcp.test.d.ts.map +0 -1
- package/packages/cli/dist/tests/mcp.test.js +0 -232
- package/packages/cli/dist/tests/mcp.test.js.map +0 -1
- package/references/obsidian-mcp-usage.md +0 -190
- package/servers/dashboard/public/app.js +0 -516
- package/servers/dashboard/public/index.html +0 -96
- package/servers/dashboard/public/styles.css +0 -819
- package/servers/obsidian-mcp/README.md +0 -82
- package/servers/obsidian-mcp/pyproject.toml +0 -32
- package/servers/obsidian-mcp/src/obsidian_mcp/__init__.py +0 -0
- package/servers/obsidian-mcp/src/obsidian_mcp/config.py +0 -47
- package/servers/obsidian-mcp/src/obsidian_mcp/indexer/__init__.py +0 -0
- package/servers/obsidian-mcp/src/obsidian_mcp/indexer/embeddings.py +0 -105
- package/servers/obsidian-mcp/src/obsidian_mcp/indexer/indexer.py +0 -79
- package/servers/obsidian-mcp/src/obsidian_mcp/indexer/store.py +0 -141
- package/servers/obsidian-mcp/src/obsidian_mcp/indexer/sync.py +0 -37
- package/servers/obsidian-mcp/src/obsidian_mcp/learning/__init__.py +0 -0
- package/servers/obsidian-mcp/src/obsidian_mcp/learning/detector.py +0 -66
- package/servers/obsidian-mcp/src/obsidian_mcp/learning/note_generator.py +0 -40
- package/servers/obsidian-mcp/src/obsidian_mcp/main.py +0 -4
- package/servers/obsidian-mcp/src/obsidian_mcp/models.py +0 -42
- package/servers/obsidian-mcp/src/obsidian_mcp/privacy/__init__.py +0 -0
- package/servers/obsidian-mcp/src/obsidian_mcp/privacy/filter.py +0 -68
- package/servers/obsidian-mcp/src/obsidian_mcp/rag/__init__.py +0 -0
- package/servers/obsidian-mcp/src/obsidian_mcp/rag/engine.py +0 -50
- package/servers/obsidian-mcp/src/obsidian_mcp/rag/graph_search.py +0 -55
- package/servers/obsidian-mcp/src/obsidian_mcp/rag/text_search.py +0 -37
- package/servers/obsidian-mcp/src/obsidian_mcp/rag/vector_search.py +0 -118
- package/servers/obsidian-mcp/src/obsidian_mcp/server.py +0 -61
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/__init__.py +0 -0
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/create.py +0 -43
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/delete.py +0 -16
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/learn.py +0 -42
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/list.py +0 -16
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/read.py +0 -15
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/registry.py +0 -130
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/related.py +0 -20
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/search.py +0 -26
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/sync.py +0 -22
- package/servers/obsidian-mcp/src/obsidian_mcp/tools/update.py +0 -34
- package/servers/obsidian-mcp/src/obsidian_mcp/vault/__init__.py +0 -0
- package/servers/obsidian-mcp/src/obsidian_mcp/vault/parser.py +0 -82
- package/servers/obsidian-mcp/src/obsidian_mcp/vault/repository.py +0 -68
- package/servers/obsidian-mcp/src/obsidian_mcp/vault/writer.py +0 -61
- package/servers/obsidian-mcp/tests/conftest.py +0 -39
- package/servers/obsidian-mcp/tests/test_async_tools.py +0 -87
- package/servers/obsidian-mcp/tests/test_edge_cases.py +0 -59
- package/servers/obsidian-mcp/tests/test_indexer.py +0 -27
- package/servers/obsidian-mcp/tests/test_integration.py +0 -90
- package/servers/obsidian-mcp/tests/test_learning.py +0 -34
- package/servers/obsidian-mcp/tests/test_privacy.py +0 -31
- package/servers/obsidian-mcp/tests/test_privacy_config.py +0 -44
- package/servers/obsidian-mcp/tests/test_rag.py +0 -64
- package/servers/obsidian-mcp/tests/test_read_raw.py +0 -37
- package/servers/obsidian-mcp/tests/test_tfidf_fallback.py +0 -54
- package/servers/obsidian-mcp/tests/test_tools.py +0 -108
- package/servers/obsidian-mcp/tests/test_vault.py +0 -103
- package/servers/obsidian-mcp/tests/test_writer.py +0 -139
- package/skills/obsidian-second-brain/SKILL.md +0 -298
|
@@ -0,0 +1,198 @@
|
|
|
1
|
+
import { describe, it } from 'node:test';
|
|
2
|
+
import assert from 'node:assert';
|
|
3
|
+
import { normalizeKimi } from '../adapters/kimi';
|
|
4
|
+
import { normalizeCodex } from '../adapters/codex';
|
|
5
|
+
import { normalizeAgy } from '../adapters/agy';
|
|
6
|
+
|
|
7
|
+
describe('normalizeKimi', () => {
|
|
8
|
+
it('forwards tool_input as input', () => {
|
|
9
|
+
const event = normalizeKimi({
|
|
10
|
+
hook_event_name: 'PreToolUse',
|
|
11
|
+
session_id: 'session-1',
|
|
12
|
+
cwd: '/tmp',
|
|
13
|
+
tool_name: 'Bash',
|
|
14
|
+
tool_input: { command: 'echo hello' },
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
assert.ok(event);
|
|
18
|
+
assert.strictEqual(event!.event_type, 'tool_start');
|
|
19
|
+
assert.deepStrictEqual(event!.input, { command: 'echo hello' });
|
|
20
|
+
assert.strictEqual(event!.output, undefined);
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
it('forwards tool_output object as output', () => {
|
|
24
|
+
const event = normalizeKimi({
|
|
25
|
+
hook_event_name: 'PostToolUse',
|
|
26
|
+
session_id: 'session-1',
|
|
27
|
+
cwd: '/tmp',
|
|
28
|
+
tool_name: 'Bash',
|
|
29
|
+
tool_input: { command: 'echo hello' },
|
|
30
|
+
tool_output: { stdout: 'hello\n' },
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
assert.ok(event);
|
|
34
|
+
assert.strictEqual(event!.event_type, 'tool_end');
|
|
35
|
+
assert.deepStrictEqual(event!.input, { command: 'echo hello' });
|
|
36
|
+
assert.deepStrictEqual(event!.output, { stdout: 'hello\n' });
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
it('wraps string tool_output in an object', () => {
|
|
40
|
+
const event = normalizeKimi({
|
|
41
|
+
hook_event_name: 'PostToolUse',
|
|
42
|
+
session_id: 'session-1',
|
|
43
|
+
cwd: '/tmp',
|
|
44
|
+
tool_name: 'Bash',
|
|
45
|
+
tool_input: { command: 'echo hello' },
|
|
46
|
+
tool_output: 'hello\n',
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
assert.ok(event);
|
|
50
|
+
assert.deepStrictEqual(event!.output, { output: 'hello\n' });
|
|
51
|
+
});
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
describe('normalizeCodex', () => {
|
|
55
|
+
it('forwards toolInput as input and toolResponse as output', () => {
|
|
56
|
+
const event = normalizeCodex({
|
|
57
|
+
hook_event_name: 'PostToolUse',
|
|
58
|
+
sessionId: 'session-2',
|
|
59
|
+
toolName: 'ReadFile',
|
|
60
|
+
toolInput: { path: '/tmp/foo.txt' },
|
|
61
|
+
toolResponse: { content: 'bar' },
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
assert.ok(event);
|
|
65
|
+
assert.strictEqual(event!.event_type, 'tool_end');
|
|
66
|
+
assert.deepStrictEqual(event!.input, { path: '/tmp/foo.txt' });
|
|
67
|
+
assert.deepStrictEqual(event!.output, { content: 'bar' });
|
|
68
|
+
});
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
describe('normalizeAgy', () => {
|
|
72
|
+
it('normalizes PreToolUse with tool mapping and detail extraction', () => {
|
|
73
|
+
const event = normalizeAgy({
|
|
74
|
+
hook_event_name: 'PreToolUse',
|
|
75
|
+
conversationId: 'conv-1',
|
|
76
|
+
stepIdx: 3,
|
|
77
|
+
toolCall: {
|
|
78
|
+
name: 'run_command',
|
|
79
|
+
args: { CommandLine: 'git status', Cwd: '/tmp' },
|
|
80
|
+
},
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
assert.ok(event);
|
|
84
|
+
assert.strictEqual(event!.source, 'agy');
|
|
85
|
+
assert.strictEqual(event!.event_type, 'tool_start');
|
|
86
|
+
assert.strictEqual(event!.session_id, 'conv-1');
|
|
87
|
+
assert.strictEqual(event!.tool, 'Bash');
|
|
88
|
+
assert.strictEqual(event!.detail, 'git status');
|
|
89
|
+
assert.strictEqual(event!.id, 'agy:conv-1:3');
|
|
90
|
+
assert.deepStrictEqual(event!.input, { CommandLine: 'git status', Cwd: '/tmp' });
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
it('normalizes PostToolUse without tool name and wraps error', () => {
|
|
94
|
+
const event = normalizeAgy({
|
|
95
|
+
hook_event_name: 'PostToolUse',
|
|
96
|
+
conversationId: 'conv-1',
|
|
97
|
+
stepIdx: 3,
|
|
98
|
+
error: 'exit status 1',
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
assert.ok(event);
|
|
102
|
+
assert.strictEqual(event!.event_type, 'tool_end');
|
|
103
|
+
assert.strictEqual(event!.session_id, 'conv-1');
|
|
104
|
+
assert.strictEqual(event!.tool, undefined);
|
|
105
|
+
assert.strictEqual(event!.id, 'agy:conv-1:3');
|
|
106
|
+
assert.deepStrictEqual(event!.output, { error: 'exit status 1' });
|
|
107
|
+
});
|
|
108
|
+
|
|
109
|
+
it('maps AGY view_file to Read and extracts AbsolutePath', () => {
|
|
110
|
+
const event = normalizeAgy({
|
|
111
|
+
hook_event_name: 'PreToolUse',
|
|
112
|
+
conversationId: 'conv-2',
|
|
113
|
+
stepIdx: 0,
|
|
114
|
+
toolCall: {
|
|
115
|
+
name: 'view_file',
|
|
116
|
+
args: { AbsolutePath: '/home/user/README.md' },
|
|
117
|
+
},
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
assert.ok(event);
|
|
121
|
+
assert.strictEqual(event!.tool, 'Read');
|
|
122
|
+
assert.strictEqual(event!.detail, '/home/user/README.md');
|
|
123
|
+
});
|
|
124
|
+
|
|
125
|
+
it('ignores unsupported hook event names', () => {
|
|
126
|
+
const event = normalizeAgy({
|
|
127
|
+
hook_event_name: 'PreInvocation',
|
|
128
|
+
conversationId: 'conv-3',
|
|
129
|
+
stepIdx: 0,
|
|
130
|
+
});
|
|
131
|
+
|
|
132
|
+
assert.strictEqual(event, undefined);
|
|
133
|
+
});
|
|
134
|
+
|
|
135
|
+
it('falls back to sessionId and random id when conversationId is missing', () => {
|
|
136
|
+
const event = normalizeAgy({
|
|
137
|
+
hook_event_name: 'PreToolUse',
|
|
138
|
+
sessionId: 'sess-fallback',
|
|
139
|
+
toolCall: { name: 'list_dir', args: { DirectoryPath: '/tmp' } },
|
|
140
|
+
});
|
|
141
|
+
|
|
142
|
+
assert.ok(event);
|
|
143
|
+
assert.strictEqual(event!.session_id, 'sess-fallback');
|
|
144
|
+
assert.ok(event!.id.startsWith('agy:sess-fallback:'));
|
|
145
|
+
assert.strictEqual(event!.tool, 'Glob');
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
it('infers skill from AGY skill file read', () => {
|
|
149
|
+
const event = normalizeAgy({
|
|
150
|
+
hook_event_name: 'PreToolUse',
|
|
151
|
+
conversationId: 'conv-skill',
|
|
152
|
+
stepIdx: 0,
|
|
153
|
+
toolCall: {
|
|
154
|
+
name: 'view_file',
|
|
155
|
+
args: {
|
|
156
|
+
AbsolutePath: '/home/arch/.agents/skills/orchestrator/SKILL.md',
|
|
157
|
+
IsSkillFile: true,
|
|
158
|
+
toolSummary: 'Orchestrator skill',
|
|
159
|
+
},
|
|
160
|
+
},
|
|
161
|
+
});
|
|
162
|
+
|
|
163
|
+
assert.ok(event);
|
|
164
|
+
assert.strictEqual(event!.tool, 'Read');
|
|
165
|
+
assert.strictEqual(event!.skill, 'orchestrator');
|
|
166
|
+
});
|
|
167
|
+
|
|
168
|
+
it('does not infer skill for ordinary AGY file reads', () => {
|
|
169
|
+
const event = normalizeAgy({
|
|
170
|
+
hook_event_name: 'PreToolUse',
|
|
171
|
+
conversationId: 'conv-file',
|
|
172
|
+
stepIdx: 0,
|
|
173
|
+
toolCall: {
|
|
174
|
+
name: 'view_file',
|
|
175
|
+
args: { AbsolutePath: '/home/arch/README.md' },
|
|
176
|
+
},
|
|
177
|
+
});
|
|
178
|
+
|
|
179
|
+
assert.ok(event);
|
|
180
|
+
assert.strictEqual(event!.tool, 'Read');
|
|
181
|
+
assert.strictEqual(event!.skill, undefined);
|
|
182
|
+
});
|
|
183
|
+
|
|
184
|
+
it('infers skill from Windows-style skill file path', () => {
|
|
185
|
+
const event = normalizeAgy({
|
|
186
|
+
hook_event_name: 'PreToolUse',
|
|
187
|
+
conversationId: 'conv-win',
|
|
188
|
+
stepIdx: 0,
|
|
189
|
+
toolCall: {
|
|
190
|
+
name: 'view_file',
|
|
191
|
+
args: { AbsolutePath: 'C:\\Users\\arch\\.agents\\skills\\engineer\\SKILL.md' },
|
|
192
|
+
},
|
|
193
|
+
});
|
|
194
|
+
|
|
195
|
+
assert.ok(event);
|
|
196
|
+
assert.strictEqual(event!.skill, 'engineer');
|
|
197
|
+
});
|
|
198
|
+
});
|
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
import { describe, it } from 'node:test';
|
|
2
|
+
import assert from 'node:assert/strict';
|
|
3
|
+
import { formatDuration, formatTime, truncate, escapeHtml, prettyJson } from '../lib/format';
|
|
4
|
+
import { resolvePath } from '../lib/paths';
|
|
5
|
+
import { projectInvocations, buildFileActivity, operationType } from '../lib/invocations';
|
|
6
|
+
import { buildGraph3D } from '../lib/graph';
|
|
7
|
+
import type { ClientEvent, ClientSession } from '../types';
|
|
8
|
+
|
|
9
|
+
describe('format', () => {
|
|
10
|
+
it('formats duration', () => {
|
|
11
|
+
assert.equal(formatDuration(0), '00:00');
|
|
12
|
+
assert.equal(formatDuration(61000), '01:01');
|
|
13
|
+
assert.equal(formatDuration(3661000), '1:01:01');
|
|
14
|
+
assert.equal(formatDuration(undefined), '00:00');
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
it('formats time', () => {
|
|
18
|
+
const ts = new Date('2026-06-26T14:30:45').getTime();
|
|
19
|
+
assert.equal(formatTime(ts), '14:30:45');
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
it('truncates strings', () => {
|
|
23
|
+
assert.equal(truncate('hello', 10), 'hello');
|
|
24
|
+
assert.equal(truncate('hello world', 6), 'hello…');
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
it('escapes html', () => {
|
|
28
|
+
assert.equal(escapeHtml('<div>"x" & \'y\'</div>'), '<div>"x" & 'y'</div>');
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
it('pretty prints json', () => {
|
|
32
|
+
assert.equal(prettyJson({ a: 1 }), '{\n "a": 1\n}');
|
|
33
|
+
});
|
|
34
|
+
});
|
|
35
|
+
|
|
36
|
+
describe('resolvePath', () => {
|
|
37
|
+
it('resolves input.path', () => {
|
|
38
|
+
assert.equal(resolvePath({ path: 'a.txt' }), 'a.txt');
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
it('prefers input.path over output.path', () => {
|
|
42
|
+
assert.equal(resolvePath({ path: 'in.txt' }, { path: 'out.txt' }), 'in.txt');
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
it('resolves operations[].path', () => {
|
|
46
|
+
assert.equal(resolvePath({ operations: [{ path: 'op.txt' }] }), 'op.txt');
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
it('resolves camelCase filePath', () => {
|
|
50
|
+
assert.equal(resolvePath({ filePath: 'camel.txt' }), 'camel.txt');
|
|
51
|
+
});
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
describe('projectInvocations', () => {
|
|
55
|
+
it('pairs tool_start with tool_end (server newest-first order)', () => {
|
|
56
|
+
const events: ClientEvent[] = [
|
|
57
|
+
{ id: '2', timestamp: 1100, event_type: 'tool_end', tool: 'Read', status: 'success', duration_ms: 100, output: { content: 'hi' } },
|
|
58
|
+
{ id: '1', timestamp: 1000, event_type: 'tool_start', tool: 'Read', input: { path: 'a.txt' } },
|
|
59
|
+
];
|
|
60
|
+
const invs = projectInvocations(events);
|
|
61
|
+
assert.equal(invs.length, 1);
|
|
62
|
+
assert.equal(invs[0].status, 'success');
|
|
63
|
+
assert.equal(invs[0].durationMs, 100);
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
it('keeps running invocations when end is missing', () => {
|
|
67
|
+
const events: ClientEvent[] = [
|
|
68
|
+
{ id: '1', timestamp: 1000, event_type: 'tool_start', tool: 'Bash', input: { command: 'ls' } },
|
|
69
|
+
];
|
|
70
|
+
const invs = projectInvocations(events);
|
|
71
|
+
assert.equal(invs.length, 1);
|
|
72
|
+
assert.equal(invs[0].status, 'running');
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
it('renders newest invocations at the top', () => {
|
|
76
|
+
const events: ClientEvent[] = [
|
|
77
|
+
{ id: '4', timestamp: 4000, event_type: 'tool_end', tool: 'Read', status: 'success' },
|
|
78
|
+
{ id: '3', timestamp: 3000, event_type: 'tool_start', tool: 'Read' },
|
|
79
|
+
{ id: '2', timestamp: 2000, event_type: 'tool_end', tool: 'Write', status: 'success' },
|
|
80
|
+
{ id: '1', timestamp: 1000, event_type: 'tool_start', tool: 'Write' },
|
|
81
|
+
];
|
|
82
|
+
const invs = projectInvocations(events);
|
|
83
|
+
assert.equal(invs.length, 2);
|
|
84
|
+
assert.equal(invs[0].tool, 'Read');
|
|
85
|
+
assert.equal(invs[1].tool, 'Write');
|
|
86
|
+
});
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
describe('buildFileActivity', () => {
|
|
90
|
+
it('groups operations by path', () => {
|
|
91
|
+
const invs = [
|
|
92
|
+
{ id: '1', tool: 'Write', eventType: 'tool_end', startTime: 1000, status: 'success', input: { path: 'a.txt' }, output: { diff: '+x' } },
|
|
93
|
+
{ id: '2', tool: 'Read', eventType: 'tool_end', startTime: 1100, status: 'success', input: { path: 'a.txt' } },
|
|
94
|
+
];
|
|
95
|
+
const activity = buildFileActivity(invs, resolvePath);
|
|
96
|
+
assert.equal(activity.length, 1);
|
|
97
|
+
assert.equal(activity[0].path, 'a.txt');
|
|
98
|
+
assert.equal(activity[0].snippet, '+x');
|
|
99
|
+
});
|
|
100
|
+
});
|
|
101
|
+
|
|
102
|
+
describe('operationType', () => {
|
|
103
|
+
it('classifies tools', () => {
|
|
104
|
+
assert.equal(operationType('Read'), 'read');
|
|
105
|
+
assert.equal(operationType('Write'), 'edit');
|
|
106
|
+
assert.equal(operationType('EditFile'), 'edit');
|
|
107
|
+
assert.equal(operationType('Bash'), 'other');
|
|
108
|
+
});
|
|
109
|
+
});
|
|
110
|
+
|
|
111
|
+
describe('buildGraph3D', () => {
|
|
112
|
+
it('builds skill-tool-file graph', () => {
|
|
113
|
+
const session: ClientSession = {
|
|
114
|
+
id: 's1',
|
|
115
|
+
source: 'kimi',
|
|
116
|
+
activeSkill: { name: 'engineer', confidence: 'explicit' },
|
|
117
|
+
lifecycle: 'running',
|
|
118
|
+
events: [],
|
|
119
|
+
startTime: 0,
|
|
120
|
+
lastActivity: 0,
|
|
121
|
+
toolCounts: {},
|
|
122
|
+
};
|
|
123
|
+
const invs = [
|
|
124
|
+
{ id: '1', tool: 'Read', eventType: 'tool_end', startTime: 1000, status: 'success', input: { path: 'a.txt' }, output: {} },
|
|
125
|
+
];
|
|
126
|
+
const graph = buildGraph3D(session, invs);
|
|
127
|
+
assert.equal(graph.nodes.length, 3);
|
|
128
|
+
assert.ok(graph.nodes.some((n) => n.id === 'skill:engineer'));
|
|
129
|
+
assert.ok(graph.nodes.some((n) => n.id === 'tool:Read'));
|
|
130
|
+
assert.ok(graph.nodes.some((n) => n.id === 'file:a.txt'));
|
|
131
|
+
assert.equal(graph.links.length, 2);
|
|
132
|
+
});
|
|
133
|
+
});
|
|
@@ -2,13 +2,20 @@ import { describe, it } from 'node:test';
|
|
|
2
2
|
import assert from 'node:assert';
|
|
3
3
|
import path from 'node:path';
|
|
4
4
|
import { spawn } from 'node:child_process';
|
|
5
|
+
import http from 'node:http';
|
|
6
|
+
import type { AddressInfo } from 'node:net';
|
|
5
7
|
import { runShim } from '../adapters/shim';
|
|
6
8
|
|
|
7
|
-
function runBin(
|
|
9
|
+
function runBin(
|
|
10
|
+
args: string[],
|
|
11
|
+
input?: string,
|
|
12
|
+
env?: Record<string, string>
|
|
13
|
+
): Promise<{ exitCode: number | null; stderr: string }> {
|
|
8
14
|
const binPath = path.join(__dirname, '..', '..', 'bin', 'crewloop-shim.js');
|
|
9
15
|
return new Promise((resolve, reject) => {
|
|
10
16
|
const child = spawn(process.execPath, [binPath, ...args], {
|
|
11
17
|
stdio: ['pipe', 'pipe', 'pipe'],
|
|
18
|
+
env: env ? { ...process.env, ...env } : process.env,
|
|
12
19
|
});
|
|
13
20
|
let stderr = '';
|
|
14
21
|
child.stderr!.on('data', (chunk) => {
|
|
@@ -40,6 +47,107 @@ describe('shim binary', () => {
|
|
|
40
47
|
const { exitCode, stderr } = await runBin(['unknown']);
|
|
41
48
|
assert.strictEqual(exitCode, 1);
|
|
42
49
|
assert.ok(stderr.includes('unknown source'));
|
|
43
|
-
assert.ok(stderr.includes('crewloop-shim <kimi|codex>'));
|
|
50
|
+
assert.ok(stderr.includes('crewloop-shim <kimi|codex|agy>'));
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
it('exits 0 for agy source with empty stdin', async () => {
|
|
54
|
+
const { exitCode, stderr } = await runBin(['agy']);
|
|
55
|
+
assert.strictEqual(exitCode, 0);
|
|
56
|
+
assert.strictEqual(stderr, '');
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
it('delivers a POST /event before exiting', async () => {
|
|
60
|
+
let receivedBody: string | undefined;
|
|
61
|
+
const server = http.createServer((req, res) => {
|
|
62
|
+
if (req.method === 'POST' && req.url === '/event') {
|
|
63
|
+
let body = '';
|
|
64
|
+
req.on('data', (chunk) => {
|
|
65
|
+
body += chunk.toString();
|
|
66
|
+
});
|
|
67
|
+
req.on('end', () => {
|
|
68
|
+
receivedBody = body;
|
|
69
|
+
res.writeHead(200, { 'Content-Type': 'application/json' });
|
|
70
|
+
res.end(JSON.stringify({ ok: true }));
|
|
71
|
+
});
|
|
72
|
+
}
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
await new Promise<void>((resolve) => server.listen(0, resolve));
|
|
76
|
+
const { port } = server.address() as AddressInfo;
|
|
77
|
+
|
|
78
|
+
const input = JSON.stringify({
|
|
79
|
+
hook_event_name: 'PostToolUse',
|
|
80
|
+
session_id: 'test-session',
|
|
81
|
+
cwd: '/tmp',
|
|
82
|
+
tool_name: 'TestTool',
|
|
83
|
+
tool_input: { command: 'echo hello' },
|
|
84
|
+
tool_output: 'hello\n',
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
try {
|
|
88
|
+
const { exitCode } = await runBin(
|
|
89
|
+
['kimi', '--default-skill', 'orchestrator'],
|
|
90
|
+
input,
|
|
91
|
+
{ CREWLOOP_DASHBOARD_URL: `http://127.0.0.1:${port}` }
|
|
92
|
+
);
|
|
93
|
+
|
|
94
|
+
assert.strictEqual(exitCode, 0);
|
|
95
|
+
assert.ok(receivedBody, 'shim should have sent a request body');
|
|
96
|
+
const parsed = JSON.parse(receivedBody!);
|
|
97
|
+
assert.strictEqual(parsed.event_type, 'tool_end');
|
|
98
|
+
assert.strictEqual(parsed.source, 'kimi');
|
|
99
|
+
assert.strictEqual(parsed.session_id, 'test-session');
|
|
100
|
+
assert.deepStrictEqual(parsed.input, { command: 'echo hello' });
|
|
101
|
+
assert.deepStrictEqual(parsed.output, { output: 'hello\n' });
|
|
102
|
+
} finally {
|
|
103
|
+
server.close();
|
|
104
|
+
}
|
|
105
|
+
});
|
|
106
|
+
|
|
107
|
+
it('delivers AGY event with default skill fallback', async () => {
|
|
108
|
+
let receivedBody: string | undefined;
|
|
109
|
+
const server = http.createServer((req, res) => {
|
|
110
|
+
if (req.method === 'POST' && req.url === '/event') {
|
|
111
|
+
let body = '';
|
|
112
|
+
req.on('data', (chunk) => {
|
|
113
|
+
body += chunk.toString();
|
|
114
|
+
});
|
|
115
|
+
req.on('end', () => {
|
|
116
|
+
receivedBody = body;
|
|
117
|
+
res.writeHead(200, { 'Content-Type': 'application/json' });
|
|
118
|
+
res.end(JSON.stringify({ ok: true }));
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
});
|
|
122
|
+
|
|
123
|
+
await new Promise<void>((resolve) => server.listen(0, resolve));
|
|
124
|
+
const { port } = server.address() as AddressInfo;
|
|
125
|
+
|
|
126
|
+
const input = JSON.stringify({
|
|
127
|
+
hook_event_name: 'PreToolUse',
|
|
128
|
+
conversationId: 'agy-session',
|
|
129
|
+
stepIdx: 0,
|
|
130
|
+
toolCall: {
|
|
131
|
+
name: 'run_command',
|
|
132
|
+
args: { CommandLine: 'git status' },
|
|
133
|
+
},
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
try {
|
|
137
|
+
const { exitCode } = await runBin(
|
|
138
|
+
['agy', '--default-skill', 'orchestrator'],
|
|
139
|
+
input,
|
|
140
|
+
{ CREWLOOP_DASHBOARD_URL: `http://127.0.0.1:${port}` }
|
|
141
|
+
);
|
|
142
|
+
|
|
143
|
+
assert.strictEqual(exitCode, 0);
|
|
144
|
+
assert.ok(receivedBody, 'shim should have sent a request body');
|
|
145
|
+
const parsed = JSON.parse(receivedBody!);
|
|
146
|
+
assert.strictEqual(parsed.event_type, 'tool_start');
|
|
147
|
+
assert.strictEqual(parsed.source, 'agy');
|
|
148
|
+
assert.strictEqual(parsed.skill, 'orchestrator');
|
|
149
|
+
} finally {
|
|
150
|
+
server.close();
|
|
151
|
+
}
|
|
44
152
|
});
|
|
45
153
|
});
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
export type AgentSource = 'kimi' | 'codex' | 'opencode' | 'log-watcher';
|
|
1
|
+
export type AgentSource = 'kimi' | 'codex' | 'opencode' | 'log-watcher' | 'agy';
|
|
2
2
|
|
|
3
3
|
export type EventType =
|
|
4
4
|
| 'session_start'
|
|
@@ -20,6 +20,8 @@ export interface DashboardEvent {
|
|
|
20
20
|
detail?: string;
|
|
21
21
|
status?: EventStatus;
|
|
22
22
|
duration_ms?: number;
|
|
23
|
+
input?: Record<string, unknown>;
|
|
24
|
+
output?: Record<string, unknown>;
|
|
23
25
|
}
|
|
24
26
|
|
|
25
27
|
export interface Session {
|
|
@@ -54,6 +56,8 @@ export interface ClientEvent {
|
|
|
54
56
|
status?: EventStatus;
|
|
55
57
|
duration_ms?: number;
|
|
56
58
|
skill?: string;
|
|
59
|
+
input?: Record<string, unknown>;
|
|
60
|
+
output?: Record<string, unknown>;
|
|
57
61
|
}
|
|
58
62
|
|
|
59
63
|
export interface ClientSession {
|
|
@@ -96,8 +100,6 @@ export interface SkillMeta {
|
|
|
96
100
|
icon: string;
|
|
97
101
|
}
|
|
98
102
|
|
|
99
|
-
export type ToolToSkillMap = Record<string, string | undefined>;
|
|
100
|
-
|
|
101
103
|
export interface SkillInferenceResult {
|
|
102
104
|
skill: string | undefined;
|
|
103
105
|
confidence: 'explicit' | 'heuristic' | 'unknown';
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
<!DOCTYPE html>
|
|
2
|
+
<html lang="en">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="UTF-8" />
|
|
5
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
|
6
|
+
<title>CrewLoop Dashboard</title>
|
|
7
|
+
<link rel="preconnect" href="https://fonts.googleapis.com" />
|
|
8
|
+
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
|
|
9
|
+
<link href="https://fonts.googleapis.com/css2?family=Teko:wght@400;500;600&family=JetBrains+Mono:wght@400;500;600&display=swap" rel="stylesheet" />
|
|
10
|
+
</head>
|
|
11
|
+
<body>
|
|
12
|
+
<div id="root"></div>
|
|
13
|
+
<script type="module" src="/src/main.tsx"></script>
|
|
14
|
+
</body>
|
|
15
|
+
</html>
|