@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.
Files changed (245) hide show
  1. package/README.md +4 -16
  2. package/package.json +1 -2
  3. package/packages/cli/dist/agents.js +1 -1
  4. package/packages/cli/dist/agents.js.map +1 -1
  5. package/packages/cli/dist/cli.d.ts.map +1 -1
  6. package/packages/cli/dist/cli.js +2 -30
  7. package/packages/cli/dist/cli.js.map +1 -1
  8. package/packages/cli/dist/hooks.d.ts +6 -4
  9. package/packages/cli/dist/hooks.d.ts.map +1 -1
  10. package/packages/cli/dist/hooks.js +250 -98
  11. package/packages/cli/dist/hooks.js.map +1 -1
  12. package/packages/cli/dist/tests/hooks.test.js +245 -33
  13. package/packages/cli/dist/tests/hooks.test.js.map +1 -1
  14. package/references/conventions.md +1 -10
  15. package/references/workflow.md +1 -1
  16. package/servers/dashboard/README.md +55 -1
  17. package/servers/dashboard/dist/adapters/agy.d.ts +19 -0
  18. package/servers/dashboard/dist/adapters/agy.d.ts.map +1 -0
  19. package/servers/dashboard/dist/adapters/agy.js +108 -0
  20. package/servers/dashboard/dist/adapters/agy.js.map +1 -0
  21. package/servers/dashboard/dist/adapters/codex.d.ts.map +1 -1
  22. package/servers/dashboard/dist/adapters/codex.js +2 -0
  23. package/servers/dashboard/dist/adapters/codex.js.map +1 -1
  24. package/servers/dashboard/dist/adapters/kimi.d.ts +1 -1
  25. package/servers/dashboard/dist/adapters/kimi.d.ts.map +1 -1
  26. package/servers/dashboard/dist/adapters/kimi.js +9 -0
  27. package/servers/dashboard/dist/adapters/kimi.js.map +1 -1
  28. package/servers/dashboard/dist/adapters/shim.d.ts +1 -1
  29. package/servers/dashboard/dist/adapters/shim.d.ts.map +1 -1
  30. package/servers/dashboard/dist/adapters/shim.js +32 -11
  31. package/servers/dashboard/dist/adapters/shim.js.map +1 -1
  32. package/servers/dashboard/dist/adapters/shim.test.js +46 -4
  33. package/servers/dashboard/dist/adapters/shim.test.js.map +1 -1
  34. package/servers/dashboard/dist/lib/constants.d.ts +5 -0
  35. package/servers/dashboard/dist/lib/constants.d.ts.map +1 -0
  36. package/servers/dashboard/dist/lib/constants.js +46 -0
  37. package/servers/dashboard/dist/lib/constants.js.map +1 -0
  38. package/servers/dashboard/dist/lib/format.d.ts +6 -0
  39. package/servers/dashboard/dist/lib/format.d.ts.map +1 -0
  40. package/servers/dashboard/dist/lib/format.js +52 -0
  41. package/servers/dashboard/dist/lib/format.js.map +1 -0
  42. package/servers/dashboard/dist/lib/graph.d.ts +22 -0
  43. package/servers/dashboard/dist/lib/graph.d.ts.map +1 -0
  44. package/servers/dashboard/dist/lib/graph.js +45 -0
  45. package/servers/dashboard/dist/lib/graph.js.map +1 -0
  46. package/servers/dashboard/dist/lib/invocations.d.ts +32 -0
  47. package/servers/dashboard/dist/lib/invocations.d.ts.map +1 -0
  48. package/servers/dashboard/dist/lib/invocations.js +135 -0
  49. package/servers/dashboard/dist/lib/invocations.js.map +1 -0
  50. package/servers/dashboard/dist/lib/invocations.test.d.ts +2 -0
  51. package/servers/dashboard/dist/lib/invocations.test.d.ts.map +1 -0
  52. package/servers/dashboard/dist/lib/invocations.test.js +68 -0
  53. package/servers/dashboard/dist/lib/invocations.test.js.map +1 -0
  54. package/servers/dashboard/dist/lib/paths.d.ts +2 -0
  55. package/servers/dashboard/dist/lib/paths.d.ts.map +1 -0
  56. package/servers/dashboard/dist/lib/paths.js +40 -0
  57. package/servers/dashboard/dist/lib/paths.js.map +1 -0
  58. package/servers/dashboard/dist/presenter.d.ts.map +1 -1
  59. package/servers/dashboard/dist/presenter.js +2 -0
  60. package/servers/dashboard/dist/presenter.js.map +1 -1
  61. package/servers/dashboard/dist/public/assets/index-DjmMKbPN.css +1 -0
  62. package/servers/dashboard/dist/public/assets/index-DzOqMleZ.js +5323 -0
  63. package/servers/dashboard/dist/public/assets/index-DzOqMleZ.js.map +1 -0
  64. package/servers/dashboard/dist/public/index.html +16 -0
  65. package/servers/dashboard/dist/server.d.ts.map +1 -1
  66. package/servers/dashboard/dist/server.js +5 -1
  67. package/servers/dashboard/dist/server.js.map +1 -1
  68. package/servers/dashboard/dist/skills/infer.d.ts.map +1 -1
  69. package/servers/dashboard/dist/skills/infer.js +0 -6
  70. package/servers/dashboard/dist/skills/infer.js.map +1 -1
  71. package/servers/dashboard/dist/skills/infer.test.js +10 -3
  72. package/servers/dashboard/dist/skills/infer.test.js.map +1 -1
  73. package/servers/dashboard/dist/skills/mapping.d.ts +0 -3
  74. package/servers/dashboard/dist/skills/mapping.d.ts.map +1 -1
  75. package/servers/dashboard/dist/skills/mapping.js +0 -18
  76. package/servers/dashboard/dist/skills/mapping.js.map +1 -1
  77. package/servers/dashboard/dist/skills/registry.d.ts.map +1 -1
  78. package/servers/dashboard/dist/skills/registry.js +0 -1
  79. package/servers/dashboard/dist/skills/registry.js.map +1 -1
  80. package/servers/dashboard/dist/tests/adapters.test.d.ts +2 -0
  81. package/servers/dashboard/dist/tests/adapters.test.d.ts.map +1 -0
  82. package/servers/dashboard/dist/tests/adapters.test.js +180 -0
  83. package/servers/dashboard/dist/tests/adapters.test.js.map +1 -0
  84. package/servers/dashboard/dist/tests/lib-helpers.test.d.ts +2 -0
  85. package/servers/dashboard/dist/tests/lib-helpers.test.d.ts.map +1 -0
  86. package/servers/dashboard/dist/tests/lib-helpers.test.js +123 -0
  87. package/servers/dashboard/dist/tests/lib-helpers.test.js.map +1 -0
  88. package/servers/dashboard/dist/tests/shim.test.js +88 -2
  89. package/servers/dashboard/dist/tests/shim.test.js.map +1 -1
  90. package/servers/dashboard/dist/types.d.ts +5 -2
  91. package/servers/dashboard/dist/types.d.ts.map +1 -1
  92. package/servers/dashboard/package.json +22 -5
  93. package/servers/dashboard/src/adapters/agy.ts +136 -0
  94. package/servers/dashboard/src/adapters/codex.ts +2 -0
  95. package/servers/dashboard/src/adapters/kimi.ts +11 -1
  96. package/servers/dashboard/src/adapters/shim.test.ts +57 -4
  97. package/servers/dashboard/src/adapters/shim.ts +31 -11
  98. package/servers/dashboard/src/lib/constants.ts +44 -0
  99. package/servers/dashboard/src/lib/format.ts +44 -0
  100. package/servers/dashboard/src/lib/graph.ts +69 -0
  101. package/servers/dashboard/src/lib/invocations.test.ts +70 -0
  102. package/servers/dashboard/src/lib/invocations.ts +172 -0
  103. package/servers/dashboard/src/lib/paths.ts +35 -0
  104. package/servers/dashboard/src/presenter.ts +2 -0
  105. package/servers/dashboard/src/server.ts +5 -1
  106. package/servers/dashboard/src/skills/infer.test.ts +11 -3
  107. package/servers/dashboard/src/skills/infer.ts +1 -8
  108. package/servers/dashboard/src/skills/mapping.ts +0 -20
  109. package/servers/dashboard/src/skills/registry.ts +0 -1
  110. package/servers/dashboard/src/tests/adapters.test.ts +198 -0
  111. package/servers/dashboard/src/tests/lib-helpers.test.ts +133 -0
  112. package/servers/dashboard/src/tests/shim.test.ts +110 -2
  113. package/servers/dashboard/src/types.ts +5 -3
  114. package/servers/dashboard/ui/index.html +15 -0
  115. package/servers/dashboard/ui/postcss.config.js +6 -0
  116. package/servers/dashboard/ui/src/App.tsx +360 -0
  117. package/servers/dashboard/ui/src/components/ActiveSkillPanel.tsx +69 -0
  118. package/servers/dashboard/ui/src/components/ActivityGraph.tsx +74 -0
  119. package/servers/dashboard/ui/src/components/CommandPalette.tsx +200 -0
  120. package/servers/dashboard/ui/src/components/FileActivity.tsx +20 -0
  121. package/servers/dashboard/ui/src/components/FileDiff.tsx +68 -0
  122. package/servers/dashboard/ui/src/components/FileList.tsx +64 -0
  123. package/servers/dashboard/ui/src/components/FilterBar.tsx +208 -0
  124. package/servers/dashboard/ui/src/components/Network3D.tsx +178 -0
  125. package/servers/dashboard/ui/src/components/SessionSelector.tsx +95 -0
  126. package/servers/dashboard/ui/src/components/Sidebar.tsx +110 -0
  127. package/servers/dashboard/ui/src/components/TelemetryPanel.tsx +57 -0
  128. package/servers/dashboard/ui/src/components/Timeline.tsx +57 -0
  129. package/servers/dashboard/ui/src/components/TimelineRow.tsx +112 -0
  130. package/servers/dashboard/ui/src/components/TopBar.tsx +116 -0
  131. package/servers/dashboard/ui/src/components/ViewHeader.tsx +19 -0
  132. package/servers/dashboard/ui/src/components/ui/Icon.tsx +105 -0
  133. package/servers/dashboard/ui/src/components/ui/StatusBadge.tsx +19 -0
  134. package/servers/dashboard/ui/src/components/views/FilesView.tsx +23 -0
  135. package/servers/dashboard/ui/src/components/views/NetworkView.tsx +20 -0
  136. package/servers/dashboard/ui/src/components/views/Overview.tsx +135 -0
  137. package/servers/dashboard/ui/src/components/views/SessionsView.tsx +84 -0
  138. package/servers/dashboard/ui/src/components/views/SettingsView.tsx +138 -0
  139. package/servers/dashboard/ui/src/components/views/SkillsView.tsx +92 -0
  140. package/servers/dashboard/ui/src/components/views/TimelineView.tsx +46 -0
  141. package/servers/dashboard/ui/src/contexts/FilterContext.tsx +41 -0
  142. package/servers/dashboard/ui/src/contexts/PinnedSessionsContext.tsx +80 -0
  143. package/servers/dashboard/ui/src/contexts/SettingsContext.tsx +60 -0
  144. package/servers/dashboard/ui/src/hooks/useCommandPalette.ts +36 -0
  145. package/servers/dashboard/ui/src/hooks/useKeyboardShortcut.ts +38 -0
  146. package/servers/dashboard/ui/src/hooks/useNow.ts +12 -0
  147. package/servers/dashboard/ui/src/hooks/useReducedMotion.ts +15 -0
  148. package/servers/dashboard/ui/src/hooks/useSessions.ts +64 -0
  149. package/servers/dashboard/ui/src/hooks/useTheme.ts +30 -0
  150. package/servers/dashboard/ui/src/hooks/useViewport.ts +19 -0
  151. package/servers/dashboard/ui/src/hooks/useWebSocket.ts +118 -0
  152. package/servers/dashboard/ui/src/lib/export.test.ts +33 -0
  153. package/servers/dashboard/ui/src/lib/export.ts +39 -0
  154. package/servers/dashboard/ui/src/lib/filter.test.ts +95 -0
  155. package/servers/dashboard/ui/src/lib/filter.ts +178 -0
  156. package/servers/dashboard/ui/src/lib/format.test.ts +25 -0
  157. package/servers/dashboard/ui/src/lib/search.test.ts +52 -0
  158. package/servers/dashboard/ui/src/lib/search.ts +60 -0
  159. package/servers/dashboard/ui/src/lib/settings.test.ts +50 -0
  160. package/servers/dashboard/ui/src/lib/settings.ts +56 -0
  161. package/servers/dashboard/ui/src/lib/types.ts +124 -0
  162. package/servers/dashboard/ui/src/main.tsx +19 -0
  163. package/servers/dashboard/ui/src/styles/index.css +155 -0
  164. package/servers/dashboard/ui/tailwind.config.js +45 -0
  165. package/servers/dashboard/ui/tsconfig.json +33 -0
  166. package/servers/dashboard/ui/tsconfig.node.json +10 -0
  167. package/servers/dashboard/ui/vite.config.ts +37 -0
  168. package/servers/dashboard/ui/vitest.config.ts +8 -0
  169. package/skills/accessibility-auditor/SKILL.md +0 -20
  170. package/skills/architect/SKILL.md +0 -45
  171. package/skills/designer/SKILL.md +0 -30
  172. package/skills/docs-writer/SKILL.md +0 -13
  173. package/skills/engineer/SKILL.md +0 -30
  174. package/skills/maintainer/SKILL.md +0 -20
  175. package/skills/orchestrator/SKILL.md +0 -13
  176. package/skills/product-manager/SKILL.md +0 -20
  177. package/skills/researcher/SKILL.md +0 -20
  178. package/skills/reviewer/SKILL.md +0 -30
  179. package/skills/security-guard/SKILL.md +0 -20
  180. package/skills/shipper/SKILL.md +0 -33
  181. package/skills/tester/SKILL.md +0 -20
  182. package/packages/cli/dist/mcp.d.ts +0 -28
  183. package/packages/cli/dist/mcp.d.ts.map +0 -1
  184. package/packages/cli/dist/mcp.js +0 -148
  185. package/packages/cli/dist/mcp.js.map +0 -1
  186. package/packages/cli/dist/tests/mcp.test.d.ts +0 -2
  187. package/packages/cli/dist/tests/mcp.test.d.ts.map +0 -1
  188. package/packages/cli/dist/tests/mcp.test.js +0 -232
  189. package/packages/cli/dist/tests/mcp.test.js.map +0 -1
  190. package/references/obsidian-mcp-usage.md +0 -190
  191. package/servers/dashboard/public/app.js +0 -516
  192. package/servers/dashboard/public/index.html +0 -96
  193. package/servers/dashboard/public/styles.css +0 -819
  194. package/servers/obsidian-mcp/README.md +0 -82
  195. package/servers/obsidian-mcp/pyproject.toml +0 -32
  196. package/servers/obsidian-mcp/src/obsidian_mcp/__init__.py +0 -0
  197. package/servers/obsidian-mcp/src/obsidian_mcp/config.py +0 -47
  198. package/servers/obsidian-mcp/src/obsidian_mcp/indexer/__init__.py +0 -0
  199. package/servers/obsidian-mcp/src/obsidian_mcp/indexer/embeddings.py +0 -105
  200. package/servers/obsidian-mcp/src/obsidian_mcp/indexer/indexer.py +0 -79
  201. package/servers/obsidian-mcp/src/obsidian_mcp/indexer/store.py +0 -141
  202. package/servers/obsidian-mcp/src/obsidian_mcp/indexer/sync.py +0 -37
  203. package/servers/obsidian-mcp/src/obsidian_mcp/learning/__init__.py +0 -0
  204. package/servers/obsidian-mcp/src/obsidian_mcp/learning/detector.py +0 -66
  205. package/servers/obsidian-mcp/src/obsidian_mcp/learning/note_generator.py +0 -40
  206. package/servers/obsidian-mcp/src/obsidian_mcp/main.py +0 -4
  207. package/servers/obsidian-mcp/src/obsidian_mcp/models.py +0 -42
  208. package/servers/obsidian-mcp/src/obsidian_mcp/privacy/__init__.py +0 -0
  209. package/servers/obsidian-mcp/src/obsidian_mcp/privacy/filter.py +0 -68
  210. package/servers/obsidian-mcp/src/obsidian_mcp/rag/__init__.py +0 -0
  211. package/servers/obsidian-mcp/src/obsidian_mcp/rag/engine.py +0 -50
  212. package/servers/obsidian-mcp/src/obsidian_mcp/rag/graph_search.py +0 -55
  213. package/servers/obsidian-mcp/src/obsidian_mcp/rag/text_search.py +0 -37
  214. package/servers/obsidian-mcp/src/obsidian_mcp/rag/vector_search.py +0 -118
  215. package/servers/obsidian-mcp/src/obsidian_mcp/server.py +0 -61
  216. package/servers/obsidian-mcp/src/obsidian_mcp/tools/__init__.py +0 -0
  217. package/servers/obsidian-mcp/src/obsidian_mcp/tools/create.py +0 -43
  218. package/servers/obsidian-mcp/src/obsidian_mcp/tools/delete.py +0 -16
  219. package/servers/obsidian-mcp/src/obsidian_mcp/tools/learn.py +0 -42
  220. package/servers/obsidian-mcp/src/obsidian_mcp/tools/list.py +0 -16
  221. package/servers/obsidian-mcp/src/obsidian_mcp/tools/read.py +0 -15
  222. package/servers/obsidian-mcp/src/obsidian_mcp/tools/registry.py +0 -130
  223. package/servers/obsidian-mcp/src/obsidian_mcp/tools/related.py +0 -20
  224. package/servers/obsidian-mcp/src/obsidian_mcp/tools/search.py +0 -26
  225. package/servers/obsidian-mcp/src/obsidian_mcp/tools/sync.py +0 -22
  226. package/servers/obsidian-mcp/src/obsidian_mcp/tools/update.py +0 -34
  227. package/servers/obsidian-mcp/src/obsidian_mcp/vault/__init__.py +0 -0
  228. package/servers/obsidian-mcp/src/obsidian_mcp/vault/parser.py +0 -82
  229. package/servers/obsidian-mcp/src/obsidian_mcp/vault/repository.py +0 -68
  230. package/servers/obsidian-mcp/src/obsidian_mcp/vault/writer.py +0 -61
  231. package/servers/obsidian-mcp/tests/conftest.py +0 -39
  232. package/servers/obsidian-mcp/tests/test_async_tools.py +0 -87
  233. package/servers/obsidian-mcp/tests/test_edge_cases.py +0 -59
  234. package/servers/obsidian-mcp/tests/test_indexer.py +0 -27
  235. package/servers/obsidian-mcp/tests/test_integration.py +0 -90
  236. package/servers/obsidian-mcp/tests/test_learning.py +0 -34
  237. package/servers/obsidian-mcp/tests/test_privacy.py +0 -31
  238. package/servers/obsidian-mcp/tests/test_privacy_config.py +0 -44
  239. package/servers/obsidian-mcp/tests/test_rag.py +0 -64
  240. package/servers/obsidian-mcp/tests/test_read_raw.py +0 -37
  241. package/servers/obsidian-mcp/tests/test_tfidf_fallback.py +0 -54
  242. package/servers/obsidian-mcp/tests/test_tools.py +0 -108
  243. package/servers/obsidian-mcp/tests/test_vault.py +0 -103
  244. package/servers/obsidian-mcp/tests/test_writer.py +0 -139
  245. 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>'), '&lt;div&gt;&quot;x&quot; &amp; &#039;y&#039;&lt;/div&gt;');
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(args: string[], input?: string): Promise<{ exitCode: number | null; stderr: string }> {
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>
@@ -0,0 +1,6 @@
1
+ module.exports = {
2
+ plugins: {
3
+ tailwindcss: {},
4
+ autoprefixer: {},
5
+ },
6
+ };