opencode-dux 1.0.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 (302) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +452 -0
  3. package/dist/agents/descriptions.d.ts +6 -0
  4. package/dist/agents/designer.d.ts +2 -0
  5. package/dist/agents/explorer.d.ts +2 -0
  6. package/dist/agents/fixer.d.ts +2 -0
  7. package/dist/agents/index.d.ts +22 -0
  8. package/dist/agents/interpreter.d.ts +2 -0
  9. package/dist/agents/librarian.d.ts +2 -0
  10. package/dist/agents/oracle.d.ts +2 -0
  11. package/dist/agents/orchestrator.d.ts +27 -0
  12. package/dist/agents/overrides.d.ts +18 -0
  13. package/dist/agents/prompt-blocks.d.ts +97 -0
  14. package/dist/agents/steward.d.ts +3 -0
  15. package/dist/cli/config-io.d.ts +24 -0
  16. package/dist/cli/config-manager.d.ts +4 -0
  17. package/dist/cli/index.d.ts +2 -0
  18. package/dist/cli/index.js +1006 -0
  19. package/dist/cli/install.d.ts +2 -0
  20. package/dist/cli/mcps.d.ts +13 -0
  21. package/dist/cli/model-key-normalization.d.ts +1 -0
  22. package/dist/cli/paths.d.ts +35 -0
  23. package/dist/cli/providers.d.ts +137 -0
  24. package/dist/cli/skills.d.ts +22 -0
  25. package/dist/cli/system.d.ts +5 -0
  26. package/dist/cli/types.d.ts +38 -0
  27. package/dist/config/constants.d.ts +12 -0
  28. package/dist/config/index.d.ts +4 -0
  29. package/dist/config/loader.d.ts +40 -0
  30. package/dist/config/runtime-preset.d.ts +12 -0
  31. package/dist/config/schema.d.ts +281 -0
  32. package/dist/config/utils.d.ts +10 -0
  33. package/dist/discovery/local/types.d.ts +79 -0
  34. package/dist/discovery/local.d.ts +73 -0
  35. package/dist/discovery/mcp-servers.d.ts +88 -0
  36. package/dist/discovery/skills.d.ts +94 -0
  37. package/dist/hooks/apply-patch/codec.d.ts +7 -0
  38. package/dist/hooks/apply-patch/errors.d.ts +25 -0
  39. package/dist/hooks/apply-patch/execution-context.d.ts +27 -0
  40. package/dist/hooks/apply-patch/index.d.ts +15 -0
  41. package/dist/hooks/apply-patch/matching.d.ts +26 -0
  42. package/dist/hooks/apply-patch/operations.d.ts +3 -0
  43. package/dist/hooks/apply-patch/patch.d.ts +2 -0
  44. package/dist/hooks/apply-patch/prepared-changes.d.ts +17 -0
  45. package/dist/hooks/apply-patch/resolution.d.ts +19 -0
  46. package/dist/hooks/apply-patch/rewrite.d.ts +7 -0
  47. package/dist/hooks/apply-patch/test-helpers.d.ts +6 -0
  48. package/dist/hooks/apply-patch/types.d.ts +80 -0
  49. package/dist/hooks/auto-update-checker/cache.d.ts +11 -0
  50. package/dist/hooks/auto-update-checker/checker.d.ts +32 -0
  51. package/dist/hooks/auto-update-checker/constants.d.ts +11 -0
  52. package/dist/hooks/auto-update-checker/index.d.ts +18 -0
  53. package/dist/hooks/auto-update-checker/types.d.ts +22 -0
  54. package/dist/hooks/chat-headers.d.ts +16 -0
  55. package/dist/hooks/context-pressure-reminder/index.d.ts +33 -0
  56. package/dist/hooks/delegate-task-retry/guidance.d.ts +2 -0
  57. package/dist/hooks/delegate-task-retry/hook.d.ts +8 -0
  58. package/dist/hooks/delegate-task-retry/index.d.ts +4 -0
  59. package/dist/hooks/delegate-task-retry/patterns.d.ts +11 -0
  60. package/dist/hooks/filter-available-skills/index.d.ts +32 -0
  61. package/dist/hooks/foreground-fallback/index.d.ts +72 -0
  62. package/dist/hooks/image-hook.d.ts +5 -0
  63. package/dist/hooks/index.d.ts +14 -0
  64. package/dist/hooks/json-error-recovery/hook.d.ts +18 -0
  65. package/dist/hooks/json-error-recovery/index.d.ts +1 -0
  66. package/dist/hooks/phase-reminder/index.d.ts +26 -0
  67. package/dist/hooks/post-file-tool-nudge/index.d.ts +19 -0
  68. package/dist/hooks/task-session-manager/index.d.ts +52 -0
  69. package/dist/hooks/todo-continuation/index.d.ts +53 -0
  70. package/dist/hooks/todo-continuation/todo-hygiene.d.ts +35 -0
  71. package/dist/index.d.ts +5 -0
  72. package/dist/index.js +31782 -0
  73. package/dist/mcp/context7.d.ts +6 -0
  74. package/dist/mcp/grep-app.d.ts +6 -0
  75. package/dist/mcp/index.d.ts +13 -0
  76. package/dist/mcp/types.d.ts +12 -0
  77. package/dist/mcp/websearch.d.ts +9 -0
  78. package/dist/skills/registry.d.ts +29 -0
  79. package/dist/subscriptions/accounts-store.d.ts +57 -0
  80. package/dist/subscriptions/index.d.ts +13 -0
  81. package/dist/subscriptions/neuralwatt-scraper.d.ts +14 -0
  82. package/dist/subscriptions/opencode-go-scraper.d.ts +27 -0
  83. package/dist/subscriptions/types.d.ts +115 -0
  84. package/dist/subscriptions/usage-service.d.ts +74 -0
  85. package/dist/tools/ast-grep/cli.d.ts +15 -0
  86. package/dist/tools/ast-grep/constants.d.ts +25 -0
  87. package/dist/tools/ast-grep/downloader.d.ts +5 -0
  88. package/dist/tools/ast-grep/index.d.ts +10 -0
  89. package/dist/tools/ast-grep/tools.d.ts +3 -0
  90. package/dist/tools/ast-grep/types.d.ts +30 -0
  91. package/dist/tools/ast-grep/utils.d.ts +4 -0
  92. package/dist/tools/delegate.d.ts +14 -0
  93. package/dist/tools/index.d.ts +5 -0
  94. package/dist/tools/preset-manager.d.ts +27 -0
  95. package/dist/tools/smartfetch/binary.d.ts +3 -0
  96. package/dist/tools/smartfetch/cache.d.ts +6 -0
  97. package/dist/tools/smartfetch/constants.d.ts +12 -0
  98. package/dist/tools/smartfetch/index.d.ts +3 -0
  99. package/dist/tools/smartfetch/network.d.ts +38 -0
  100. package/dist/tools/smartfetch/secondary-model.d.ts +28 -0
  101. package/dist/tools/smartfetch/tool.d.ts +3 -0
  102. package/dist/tools/smartfetch/types.d.ts +122 -0
  103. package/dist/tools/smartfetch/utils.d.ts +18 -0
  104. package/dist/tui-state.d.ts +168 -0
  105. package/dist/tui.d.ts +37 -0
  106. package/dist/tui.js +1896 -0
  107. package/dist/utils/agent-variant.d.ts +63 -0
  108. package/dist/utils/compat.d.ts +30 -0
  109. package/dist/utils/env.d.ts +1 -0
  110. package/dist/utils/index.d.ts +9 -0
  111. package/dist/utils/internal-initiator.d.ts +6 -0
  112. package/dist/utils/logger.d.ts +8 -0
  113. package/dist/utils/polling.d.ts +21 -0
  114. package/dist/utils/session-manager.d.ts +55 -0
  115. package/dist/utils/session.d.ts +90 -0
  116. package/dist/utils/subagent-depth.d.ts +35 -0
  117. package/dist/utils/system-collapse.d.ts +6 -0
  118. package/dist/utils/task.d.ts +4 -0
  119. package/dist/utils/zip-extractor.d.ts +1 -0
  120. package/index.ts +1 -0
  121. package/opencode-dux.schema.json +634 -0
  122. package/package.json +103 -0
  123. package/src/agents/descriptions.ts +55 -0
  124. package/src/agents/designer.test.ts +86 -0
  125. package/src/agents/designer.ts +154 -0
  126. package/src/agents/display-name.test.ts +186 -0
  127. package/src/agents/explorer.test.ts +79 -0
  128. package/src/agents/explorer.ts +144 -0
  129. package/src/agents/fixer.test.ts +79 -0
  130. package/src/agents/fixer.ts +145 -0
  131. package/src/agents/index.test.ts +472 -0
  132. package/src/agents/index.ts +248 -0
  133. package/src/agents/interpreter.ts +136 -0
  134. package/src/agents/librarian.test.ts +80 -0
  135. package/src/agents/librarian.ts +145 -0
  136. package/src/agents/oracle.test.ts +89 -0
  137. package/src/agents/oracle.ts +184 -0
  138. package/src/agents/orchestrator.test.ts +116 -0
  139. package/src/agents/orchestrator.ts +574 -0
  140. package/src/agents/overrides.ts +95 -0
  141. package/src/agents/prompt-blocks.test.ts +114 -0
  142. package/src/agents/prompt-blocks.ts +640 -0
  143. package/src/agents/steward.ts +146 -0
  144. package/src/cli/config-io.test.ts +536 -0
  145. package/src/cli/config-io.ts +473 -0
  146. package/src/cli/config-manager.test.ts +141 -0
  147. package/src/cli/config-manager.ts +4 -0
  148. package/src/cli/index.ts +88 -0
  149. package/src/cli/install.ts +282 -0
  150. package/src/cli/mcps.test.ts +62 -0
  151. package/src/cli/mcps.ts +39 -0
  152. package/src/cli/model-key-normalization.test.ts +21 -0
  153. package/src/cli/model-key-normalization.ts +60 -0
  154. package/src/cli/paths.test.ts +167 -0
  155. package/src/cli/paths.ts +144 -0
  156. package/src/cli/providers.test.ts +118 -0
  157. package/src/cli/providers.ts +141 -0
  158. package/src/cli/skills.test.ts +111 -0
  159. package/src/cli/skills.ts +103 -0
  160. package/src/cli/system.test.ts +91 -0
  161. package/src/cli/system.ts +180 -0
  162. package/src/cli/types.ts +43 -0
  163. package/src/config/constants.ts +58 -0
  164. package/src/config/index.ts +4 -0
  165. package/src/config/loader.test.ts +1194 -0
  166. package/src/config/loader.ts +269 -0
  167. package/src/config/model-resolution.test.ts +176 -0
  168. package/src/config/runtime-preset.test.ts +61 -0
  169. package/src/config/runtime-preset.ts +37 -0
  170. package/src/config/schema.ts +248 -0
  171. package/src/config/utils.test.ts +41 -0
  172. package/src/config/utils.ts +23 -0
  173. package/src/discovery/local/types.ts +85 -0
  174. package/src/discovery/local.ts +322 -0
  175. package/src/discovery/mcp-servers.ts +804 -0
  176. package/src/discovery/skills.ts +959 -0
  177. package/src/hooks/apply-patch/codec.test.ts +184 -0
  178. package/src/hooks/apply-patch/codec.ts +352 -0
  179. package/src/hooks/apply-patch/errors.ts +117 -0
  180. package/src/hooks/apply-patch/execution-context.ts +432 -0
  181. package/src/hooks/apply-patch/hook.test.ts +768 -0
  182. package/src/hooks/apply-patch/index.ts +126 -0
  183. package/src/hooks/apply-patch/matching.test.ts +215 -0
  184. package/src/hooks/apply-patch/matching.ts +586 -0
  185. package/src/hooks/apply-patch/operations.test.ts +1535 -0
  186. package/src/hooks/apply-patch/operations.ts +3 -0
  187. package/src/hooks/apply-patch/patch.ts +9 -0
  188. package/src/hooks/apply-patch/prepared-changes.ts +400 -0
  189. package/src/hooks/apply-patch/resolution.test.ts +420 -0
  190. package/src/hooks/apply-patch/resolution.ts +437 -0
  191. package/src/hooks/apply-patch/rewrite.ts +496 -0
  192. package/src/hooks/apply-patch/test-helpers.ts +52 -0
  193. package/src/hooks/apply-patch/types.ts +111 -0
  194. package/src/hooks/auto-update-checker/cache.test.ts +179 -0
  195. package/src/hooks/auto-update-checker/cache.ts +188 -0
  196. package/src/hooks/auto-update-checker/checker.test.ts +159 -0
  197. package/src/hooks/auto-update-checker/checker.ts +308 -0
  198. package/src/hooks/auto-update-checker/constants.ts +33 -0
  199. package/src/hooks/auto-update-checker/index.test.ts +282 -0
  200. package/src/hooks/auto-update-checker/index.ts +225 -0
  201. package/src/hooks/auto-update-checker/types.ts +26 -0
  202. package/src/hooks/chat-headers.test.ts +236 -0
  203. package/src/hooks/chat-headers.ts +97 -0
  204. package/src/hooks/context-pressure-reminder/index.test.ts +179 -0
  205. package/src/hooks/context-pressure-reminder/index.ts +137 -0
  206. package/src/hooks/delegate-task-retry/guidance.ts +41 -0
  207. package/src/hooks/delegate-task-retry/hook.ts +23 -0
  208. package/src/hooks/delegate-task-retry/index.test.ts +38 -0
  209. package/src/hooks/delegate-task-retry/index.ts +7 -0
  210. package/src/hooks/delegate-task-retry/patterns.ts +79 -0
  211. package/src/hooks/filter-available-skills/index.test.ts +297 -0
  212. package/src/hooks/filter-available-skills/index.ts +160 -0
  213. package/src/hooks/foreground-fallback/index.test.ts +624 -0
  214. package/src/hooks/foreground-fallback/index.ts +374 -0
  215. package/src/hooks/image-hook.ts +6 -0
  216. package/src/hooks/index.ts +17 -0
  217. package/src/hooks/json-error-recovery/hook.ts +73 -0
  218. package/src/hooks/json-error-recovery/index.test.ts +111 -0
  219. package/src/hooks/json-error-recovery/index.ts +6 -0
  220. package/src/hooks/phase-reminder/index.test.ts +74 -0
  221. package/src/hooks/phase-reminder/index.ts +85 -0
  222. package/src/hooks/post-file-tool-nudge/index.test.ts +94 -0
  223. package/src/hooks/post-file-tool-nudge/index.ts +63 -0
  224. package/src/hooks/task-session-manager/index.test.ts +833 -0
  225. package/src/hooks/task-session-manager/index.ts +434 -0
  226. package/src/hooks/todo-continuation/index.test.ts +3026 -0
  227. package/src/hooks/todo-continuation/index.ts +878 -0
  228. package/src/hooks/todo-continuation/todo-hygiene.test.ts +204 -0
  229. package/src/hooks/todo-continuation/todo-hygiene.ts +207 -0
  230. package/src/index.ts +1672 -0
  231. package/src/mcp/context7.ts +14 -0
  232. package/src/mcp/grep-app.ts +11 -0
  233. package/src/mcp/index.test.ts +96 -0
  234. package/src/mcp/index.ts +66 -0
  235. package/src/mcp/types.ts +16 -0
  236. package/src/mcp/websearch.ts +47 -0
  237. package/src/skills/codemap/README.md +60 -0
  238. package/src/skills/codemap/SKILL.md +174 -0
  239. package/src/skills/codemap/scripts/codemap.mjs +483 -0
  240. package/src/skills/codemap/scripts/codemap.test.ts +129 -0
  241. package/src/skills/registry.ts +218 -0
  242. package/src/skills/simplify/README.md +19 -0
  243. package/src/skills/simplify/SKILL.md +138 -0
  244. package/src/subscriptions/accounts-store.test.ts +236 -0
  245. package/src/subscriptions/accounts-store.ts +184 -0
  246. package/src/subscriptions/index.ts +30 -0
  247. package/src/subscriptions/neuralwatt-scraper.ts +108 -0
  248. package/src/subscriptions/opencode-go-scraper.ts +301 -0
  249. package/src/subscriptions/types.ts +145 -0
  250. package/src/subscriptions/usage-service.test.ts +202 -0
  251. package/src/subscriptions/usage-service.ts +651 -0
  252. package/src/tools/ast-grep/cli.ts +257 -0
  253. package/src/tools/ast-grep/constants.ts +214 -0
  254. package/src/tools/ast-grep/downloader.ts +131 -0
  255. package/src/tools/ast-grep/index.ts +24 -0
  256. package/src/tools/ast-grep/tools.ts +117 -0
  257. package/src/tools/ast-grep/types.ts +51 -0
  258. package/src/tools/ast-grep/utils.ts +126 -0
  259. package/src/tools/delegate-handoff.test.ts +18 -0
  260. package/src/tools/delegate.ts +508 -0
  261. package/src/tools/index.ts +8 -0
  262. package/src/tools/preset-manager.test.ts +795 -0
  263. package/src/tools/preset-manager.ts +332 -0
  264. package/src/tools/smartfetch/binary.ts +58 -0
  265. package/src/tools/smartfetch/cache.test.ts +34 -0
  266. package/src/tools/smartfetch/cache.ts +112 -0
  267. package/src/tools/smartfetch/constants.ts +29 -0
  268. package/src/tools/smartfetch/index.ts +8 -0
  269. package/src/tools/smartfetch/network.test.ts +178 -0
  270. package/src/tools/smartfetch/network.ts +614 -0
  271. package/src/tools/smartfetch/secondary-model.test.ts +85 -0
  272. package/src/tools/smartfetch/secondary-model.ts +276 -0
  273. package/src/tools/smartfetch/tool.test.ts +60 -0
  274. package/src/tools/smartfetch/tool.ts +832 -0
  275. package/src/tools/smartfetch/types.ts +135 -0
  276. package/src/tools/smartfetch/utils.test.ts +24 -0
  277. package/src/tools/smartfetch/utils.ts +456 -0
  278. package/src/tui-state.test.ts +867 -0
  279. package/src/tui-state.ts +1255 -0
  280. package/src/tui.test.ts +336 -0
  281. package/src/tui.ts +1539 -0
  282. package/src/utils/agent-variant.test.ts +244 -0
  283. package/src/utils/agent-variant.ts +187 -0
  284. package/src/utils/compat.ts +91 -0
  285. package/src/utils/env.ts +12 -0
  286. package/src/utils/index.ts +9 -0
  287. package/src/utils/internal-initiator.ts +28 -0
  288. package/src/utils/logger.test.ts +220 -0
  289. package/src/utils/logger.ts +136 -0
  290. package/src/utils/polling.test.ts +191 -0
  291. package/src/utils/polling.ts +67 -0
  292. package/src/utils/session-manager.test.ts +173 -0
  293. package/src/utils/session-manager.ts +356 -0
  294. package/src/utils/session.test.ts +110 -0
  295. package/src/utils/session.ts +389 -0
  296. package/src/utils/subagent-depth.test.ts +170 -0
  297. package/src/utils/subagent-depth.ts +75 -0
  298. package/src/utils/system-collapse.test.ts +86 -0
  299. package/src/utils/system-collapse.ts +24 -0
  300. package/src/utils/task.test.ts +24 -0
  301. package/src/utils/task.ts +20 -0
  302. package/src/utils/zip-extractor.ts +102 -0
@@ -0,0 +1,297 @@
1
+ import { describe, expect, test } from 'bun:test';
2
+ import type { PluginInput } from '@opencode-ai/plugin';
3
+ import type { PluginConfig } from '../../config';
4
+ import {
5
+ createFilterAvailableSkillsHook,
6
+ filterAvailableSkillsText,
7
+ } from './index';
8
+
9
+ const mockCtx = {} as PluginInput;
10
+
11
+ function skillBlock(name: string): string {
12
+ return `<skill>
13
+ <name>${name}</name>
14
+ <description>${name} description</description>
15
+ <location>file:///tmp/${name}</location>
16
+ </skill>`;
17
+ }
18
+
19
+ function availableSkillsBlock(...names: string[]): string {
20
+ return `<available_skills>
21
+ ${names.map((name) => skillBlock(name)).join('\n')}
22
+ </available_skills>`;
23
+ }
24
+
25
+ describe('filterAvailableSkillsText', () => {
26
+ test('keeps only allowed skills using exact skill names', () => {
27
+ const text = availableSkillsBlock('skill1', 'skill2', 'skill3');
28
+ const result = filterAvailableSkillsText(text, {
29
+ '*': 'deny',
30
+ skill1: 'allow',
31
+ skill3: 'allow',
32
+ });
33
+
34
+ expect(result).toContain('<name>skill1</name>');
35
+ expect(result).not.toContain('<name>skill2</name>');
36
+ expect(result).toContain('<name>skill3</name>');
37
+ });
38
+
39
+ test('renders No skills available when nothing is allowed', () => {
40
+ const result = filterAvailableSkillsText(availableSkillsBlock('skill1'), {
41
+ '*': 'deny',
42
+ });
43
+
44
+ expect(result).toContain('No skills available.');
45
+ expect(result).not.toContain('<name>skill1</name>');
46
+ });
47
+ });
48
+
49
+ describe('createFilterAvailableSkillsHook', () => {
50
+ test('filters system prompt skill blocks for explicit agent skills', async () => {
51
+ const config: PluginConfig = {
52
+ agents: {
53
+ explorer: {
54
+ skills: ['skill1', 'skill3'],
55
+ },
56
+ },
57
+ };
58
+
59
+ const hook = createFilterAvailableSkillsHook(mockCtx, config);
60
+ const output = {
61
+ messages: [
62
+ {
63
+ info: { role: 'system' },
64
+ parts: [
65
+ {
66
+ type: 'text',
67
+ text: availableSkillsBlock('skill1', 'skill2', 'skill3'),
68
+ },
69
+ ],
70
+ },
71
+ {
72
+ info: { role: 'user', agent: 'explorer' },
73
+ parts: [{ type: 'text', text: 'check skills' }],
74
+ },
75
+ ],
76
+ };
77
+
78
+ await hook['experimental.chat.messages.transform']({}, output);
79
+
80
+ const resultText = output.messages[0].parts[0].text;
81
+ expect(resultText).toContain('<name>skill1</name>');
82
+ expect(resultText).not.toContain('<name>skill2</name>');
83
+ expect(resultText).toContain('<name>skill3</name>');
84
+ });
85
+
86
+ test('shows no skills for agents configured with an empty skills list', async () => {
87
+ const config: PluginConfig = {
88
+ agents: {
89
+ fixer: {
90
+ skills: [],
91
+ },
92
+ },
93
+ };
94
+
95
+ const hook = createFilterAvailableSkillsHook(mockCtx, config);
96
+ const output = {
97
+ messages: [
98
+ {
99
+ info: { role: 'system' },
100
+ parts: [{ type: 'text', text: availableSkillsBlock('skill1') }],
101
+ },
102
+ {
103
+ info: { role: 'user', agent: 'fixer' },
104
+ parts: [{ type: 'text', text: 'check skills' }],
105
+ },
106
+ ],
107
+ };
108
+
109
+ await hook['experimental.chat.messages.transform']({}, output);
110
+
111
+ const resultText = output.messages[0].parts[0].text;
112
+ expect(resultText).toContain('No skills available.');
113
+ expect(resultText).not.toContain('<name>skill1</name>');
114
+ });
115
+
116
+ test('denies all skills when no config is provided', async () => {
117
+ const hook = createFilterAvailableSkillsHook(mockCtx, {});
118
+ const output = {
119
+ messages: [
120
+ {
121
+ info: { role: 'system' },
122
+ parts: [
123
+ { type: 'text', text: availableSkillsBlock('skill1', 'skill2') },
124
+ ],
125
+ },
126
+ {
127
+ info: { role: 'user', agent: 'orchestrator' },
128
+ parts: [{ type: 'text', text: 'check skills' }],
129
+ },
130
+ ],
131
+ };
132
+
133
+ await hook['experimental.chat.messages.transform']({}, output);
134
+
135
+ const resultText = output.messages[0].parts[0].text;
136
+ expect(resultText).toContain('No skills available.');
137
+ expect(resultText).not.toContain('<name>skill1</name>');
138
+ });
139
+
140
+ test('supports wildcard allow with explicit exclusions', async () => {
141
+ const config: PluginConfig = {
142
+ agents: {
143
+ designer: {
144
+ skills: ['*', '!skill2'],
145
+ },
146
+ },
147
+ };
148
+
149
+ const hook = createFilterAvailableSkillsHook(mockCtx, config);
150
+ const output = {
151
+ messages: [
152
+ {
153
+ info: { role: 'system' },
154
+ parts: [
155
+ { type: 'text', text: availableSkillsBlock('skill1', 'skill2') },
156
+ ],
157
+ },
158
+ {
159
+ info: { role: 'user', agent: 'designer' },
160
+ parts: [{ type: 'text', text: 'check skills' }],
161
+ },
162
+ ],
163
+ };
164
+
165
+ await hook['experimental.chat.messages.transform']({}, output);
166
+
167
+ const resultText = output.messages[0].parts[0].text;
168
+ expect(resultText).toContain('<name>skill1</name>');
169
+ expect(resultText).not.toContain('<name>skill2</name>');
170
+ });
171
+
172
+ test('defaults to orchestrator when no agent is present', async () => {
173
+ const hook = createFilterAvailableSkillsHook(mockCtx, {});
174
+ const output = {
175
+ messages: [
176
+ {
177
+ info: { role: 'system' },
178
+ parts: [{ type: 'text', text: availableSkillsBlock('skill1') }],
179
+ },
180
+ {
181
+ info: { role: 'user' },
182
+ parts: [{ type: 'text', text: 'check skills' }],
183
+ },
184
+ ],
185
+ };
186
+
187
+ await hook['experimental.chat.messages.transform']({}, output);
188
+
189
+ // Without explicit config, all skills are denied
190
+ expect(output.messages[0].parts[0].text).toContain('No skills available.');
191
+ });
192
+
193
+ test('filters multiple skill blocks across messages', async () => {
194
+ const config: PluginConfig = {
195
+ agents: {
196
+ explorer: {
197
+ skills: ['skill1'],
198
+ },
199
+ },
200
+ };
201
+
202
+ const hook = createFilterAvailableSkillsHook(mockCtx, config);
203
+ const output = {
204
+ messages: [
205
+ {
206
+ info: { role: 'system' },
207
+ parts: [
208
+ {
209
+ type: 'text',
210
+ text: `Intro\n${availableSkillsBlock('skill1', 'skill2')}`,
211
+ },
212
+ ],
213
+ },
214
+ {
215
+ info: { role: 'developer' },
216
+ parts: [
217
+ { type: 'text', text: availableSkillsBlock('skill2', 'skill3') },
218
+ ],
219
+ },
220
+ {
221
+ info: { role: 'user', agent: 'explorer' },
222
+ parts: [{ type: 'text', text: 'check skills' }],
223
+ },
224
+ ],
225
+ };
226
+
227
+ await hook['experimental.chat.messages.transform']({}, output);
228
+
229
+ expect(output.messages[0].parts[0].text).toContain('<name>skill1</name>');
230
+ expect(output.messages[0].parts[0].text).not.toContain(
231
+ '<name>skill2</name>',
232
+ );
233
+ expect(output.messages[1].parts[0].text).toContain('No skills available.');
234
+ });
235
+
236
+ test('reuses permission rules without caching the final skills block text', async () => {
237
+ const config: PluginConfig = {
238
+ agents: {
239
+ explorer: {
240
+ skills: ['skill1', 'skill3'],
241
+ },
242
+ },
243
+ };
244
+
245
+ const hook = createFilterAvailableSkillsHook(mockCtx, config);
246
+ const firstOutput = {
247
+ messages: [
248
+ {
249
+ info: { role: 'system' },
250
+ parts: [
251
+ {
252
+ type: 'text',
253
+ text: availableSkillsBlock('skill1', 'skill2'),
254
+ },
255
+ ],
256
+ },
257
+ {
258
+ info: { role: 'user', agent: 'explorer' },
259
+ parts: [{ type: 'text', text: 'check skills' }],
260
+ },
261
+ ],
262
+ };
263
+ const secondOutput = {
264
+ messages: [
265
+ {
266
+ info: { role: 'system' },
267
+ parts: [
268
+ {
269
+ type: 'text',
270
+ text: availableSkillsBlock('skill2', 'skill3'),
271
+ },
272
+ ],
273
+ },
274
+ {
275
+ info: { role: 'user', agent: 'explorer' },
276
+ parts: [{ type: 'text', text: 'check skills' }],
277
+ },
278
+ ],
279
+ };
280
+
281
+ await hook['experimental.chat.messages.transform']({}, firstOutput);
282
+ await hook['experimental.chat.messages.transform']({}, secondOutput);
283
+
284
+ expect(firstOutput.messages[0].parts[0].text).toContain(
285
+ '<name>skill1</name>',
286
+ );
287
+ expect(firstOutput.messages[0].parts[0].text).not.toContain(
288
+ '<name>skill3</name>',
289
+ );
290
+ expect(secondOutput.messages[0].parts[0].text).not.toContain(
291
+ '<name>skill1</name>',
292
+ );
293
+ expect(secondOutput.messages[0].parts[0].text).toContain(
294
+ '<name>skill3</name>',
295
+ );
296
+ });
297
+ });
@@ -0,0 +1,160 @@
1
+ /**
2
+ * Filter available_skills block based on the current agent's permission.skill rules.
3
+ * OpenCode core injects `<available_skills>` globally, so this hook rewrites that
4
+ * block before the prompt is sent.
5
+ */
6
+ import type { PluginInput } from '@opencode-ai/plugin';
7
+ import { getSkillPermissionsForAgent } from '../../cli/skills';
8
+ import { getAgentOverride, type PluginConfig } from '../../config';
9
+
10
+ interface MessageInfo {
11
+ role: string;
12
+ agent?: string;
13
+ }
14
+
15
+ interface MessagePart {
16
+ type: string;
17
+ text?: string;
18
+ [key: string]: unknown;
19
+ }
20
+
21
+ interface MessageWithParts {
22
+ info: MessageInfo;
23
+ parts: MessagePart[];
24
+ }
25
+
26
+ const AVAILABLE_SKILLS_BLOCK_REGEX =
27
+ /<available_skills>\s*([\s\S]*?)\s*<\/available_skills>/g;
28
+ const SKILL_NAME_REGEX = /<name>([^<]+)<\/name>/;
29
+
30
+ type SkillRule = 'allow' | 'ask' | 'deny';
31
+
32
+ interface SkillEntry {
33
+ name: string;
34
+ block: string;
35
+ }
36
+
37
+ function getCurrentAgent(messages: MessageWithParts[]): string {
38
+ for (let index = messages.length - 1; index >= 0; index -= 1) {
39
+ const message = messages[index];
40
+ if (message.info.role === 'user') {
41
+ return message.info.agent ?? 'orchestrator';
42
+ }
43
+ }
44
+
45
+ return 'orchestrator';
46
+ }
47
+
48
+ function extractSkillEntries(blockContent: string): SkillEntry[] {
49
+ const entries: SkillEntry[] = [];
50
+ const skillEntryRegex = /<skill>\s*([\s\S]*?)\s*<\/skill>/g;
51
+
52
+ for (const match of blockContent.matchAll(skillEntryRegex)) {
53
+ const block = match[0];
54
+ const nameMatch = block.match(SKILL_NAME_REGEX);
55
+ if (!nameMatch) {
56
+ continue;
57
+ }
58
+
59
+ entries.push({
60
+ name: nameMatch[1].trim(),
61
+ block,
62
+ });
63
+ }
64
+
65
+ return entries;
66
+ }
67
+
68
+ function isSkillAllowed(
69
+ skillName: string,
70
+ permissionRules: Record<string, SkillRule>,
71
+ ): boolean {
72
+ const specificRule = permissionRules[skillName];
73
+ if (specificRule !== undefined) {
74
+ return specificRule === 'allow';
75
+ }
76
+
77
+ return permissionRules['*'] === 'allow';
78
+ }
79
+
80
+ function filterAvailableSkillsText(
81
+ text: string,
82
+ permissionRules: Record<string, SkillRule>,
83
+ ): string {
84
+ return text.replace(
85
+ AVAILABLE_SKILLS_BLOCK_REGEX,
86
+ (_fullMatch, blockContent: string) => {
87
+ const allowedEntries = extractSkillEntries(blockContent).filter((entry) =>
88
+ isSkillAllowed(entry.name, permissionRules),
89
+ );
90
+
91
+ if (allowedEntries.length === 0) {
92
+ return '<available_skills>\nNo skills available.\n</available_skills>';
93
+ }
94
+
95
+ return `<available_skills>\n${allowedEntries
96
+ .map((entry) => entry.block)
97
+ .join('\n')}\n</available_skills>`;
98
+ },
99
+ );
100
+ }
101
+
102
+ /**
103
+ * Creates the experimental.chat.messages.transform hook for filtering available skills.
104
+ * This hook runs right before sending to API, so it doesn't affect UI display.
105
+ */
106
+ export function createFilterAvailableSkillsHook(
107
+ ctx: PluginInput,
108
+ config: PluginConfig,
109
+ ) {
110
+ const permissionRulesByAgent = new Map<string, Record<string, SkillRule>>();
111
+
112
+ const getPermissionRules = async (
113
+ agentName: string,
114
+ ): Promise<Record<string, SkillRule>> => {
115
+ const cached = permissionRulesByAgent.get(agentName);
116
+ if (cached) {
117
+ return cached;
118
+ }
119
+
120
+ const configuredSkills = getAgentOverride(config, agentName)?.skills;
121
+ const permissionRules = await getSkillPermissionsForAgent(
122
+ agentName,
123
+ configuredSkills,
124
+ ctx,
125
+ );
126
+ permissionRulesByAgent.set(agentName, permissionRules);
127
+ return permissionRules;
128
+ };
129
+
130
+ return {
131
+ 'experimental.chat.messages.transform': async (
132
+ _input: Record<string, never>,
133
+ output: { messages: MessageWithParts[] },
134
+ ): Promise<void> => {
135
+ const { messages } = output;
136
+ if (messages.length === 0) {
137
+ return;
138
+ }
139
+
140
+ const agentName = getCurrentAgent(messages);
141
+ const permissionRules = await getPermissionRules(agentName);
142
+
143
+ for (const message of messages) {
144
+ for (const part of message.parts) {
145
+ if (
146
+ part.type !== 'text' ||
147
+ !part.text ||
148
+ !part.text.includes('<available_skills>')
149
+ ) {
150
+ continue;
151
+ }
152
+
153
+ part.text = filterAvailableSkillsText(part.text, permissionRules);
154
+ }
155
+ }
156
+ },
157
+ };
158
+ }
159
+
160
+ export { filterAvailableSkillsText };