@2amtech/hai 0.0.1

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 (239) hide show
  1. package/README.md +212 -0
  2. package/dist/ai/agent-info.d.ts +1 -0
  3. package/dist/ai/agent-info.js +42 -0
  4. package/dist/ai/agent-info.js.map +1 -0
  5. package/dist/ai/agents/architect.d.ts +3 -0
  6. package/dist/ai/agents/architect.js +107 -0
  7. package/dist/ai/agents/architect.js.map +1 -0
  8. package/dist/ai/agents/backend-dev.d.ts +3 -0
  9. package/dist/ai/agents/backend-dev.js +33 -0
  10. package/dist/ai/agents/backend-dev.js.map +1 -0
  11. package/dist/ai/agents/frontend-dev.d.ts +3 -0
  12. package/dist/ai/agents/frontend-dev.js +31 -0
  13. package/dist/ai/agents/frontend-dev.js.map +1 -0
  14. package/dist/ai/agents/index.d.ts +3 -0
  15. package/dist/ai/agents/index.js +17 -0
  16. package/dist/ai/agents/index.js.map +1 -0
  17. package/dist/ai/agents/refactorer.d.ts +3 -0
  18. package/dist/ai/agents/refactorer.js +38 -0
  19. package/dist/ai/agents/refactorer.js.map +1 -0
  20. package/dist/ai/agents/researcher.d.ts +3 -0
  21. package/dist/ai/agents/researcher.js +28 -0
  22. package/dist/ai/agents/researcher.js.map +1 -0
  23. package/dist/ai/agents/security-reviewer.d.ts +3 -0
  24. package/dist/ai/agents/security-reviewer.js +99 -0
  25. package/dist/ai/agents/security-reviewer.js.map +1 -0
  26. package/dist/ai/agents/types.d.ts +6 -0
  27. package/dist/ai/agents/types.js +2 -0
  28. package/dist/ai/agents/types.js.map +1 -0
  29. package/dist/ai/agents/verifier.d.ts +3 -0
  30. package/dist/ai/agents/verifier.js +28 -0
  31. package/dist/ai/agents/verifier.js.map +1 -0
  32. package/dist/ai/prompts/document.d.ts +3 -0
  33. package/dist/ai/prompts/document.js +57 -0
  34. package/dist/ai/prompts/document.js.map +1 -0
  35. package/dist/ai/prompts/implement.d.ts +3 -0
  36. package/dist/ai/prompts/implement.js +128 -0
  37. package/dist/ai/prompts/implement.js.map +1 -0
  38. package/dist/ai/prompts/index.d.ts +3 -0
  39. package/dist/ai/prompts/index.js +17 -0
  40. package/dist/ai/prompts/index.js.map +1 -0
  41. package/dist/ai/prompts/optimize-ai.d.ts +3 -0
  42. package/dist/ai/prompts/optimize-ai.js +104 -0
  43. package/dist/ai/prompts/optimize-ai.js.map +1 -0
  44. package/dist/ai/prompts/pull-specs.d.ts +3 -0
  45. package/dist/ai/prompts/pull-specs.js +10 -0
  46. package/dist/ai/prompts/pull-specs.js.map +1 -0
  47. package/dist/ai/prompts/pull-tickets.d.ts +3 -0
  48. package/dist/ai/prompts/pull-tickets.js +11 -0
  49. package/dist/ai/prompts/pull-tickets.js.map +1 -0
  50. package/dist/ai/prompts/pull.d.ts +3 -0
  51. package/dist/ai/prompts/pull.js +9 -0
  52. package/dist/ai/prompts/pull.js.map +1 -0
  53. package/dist/ai/prompts/review-changes.d.ts +3 -0
  54. package/dist/ai/prompts/review-changes.js +88 -0
  55. package/dist/ai/prompts/review-changes.js.map +1 -0
  56. package/dist/ai/prompts/types.d.ts +6 -0
  57. package/dist/ai/prompts/types.js +2 -0
  58. package/dist/ai/prompts/types.js.map +1 -0
  59. package/dist/app.d.ts +14 -0
  60. package/dist/app.js +40 -0
  61. package/dist/app.js.map +1 -0
  62. package/dist/cli.d.ts +2 -0
  63. package/dist/cli.js +79 -0
  64. package/dist/cli.js.map +1 -0
  65. package/dist/commands/get.d.ts +8 -0
  66. package/dist/commands/get.js +73 -0
  67. package/dist/commands/get.js.map +1 -0
  68. package/dist/commands/init.d.ts +3 -0
  69. package/dist/commands/init.js +113 -0
  70. package/dist/commands/init.js.map +1 -0
  71. package/dist/commands/mcp/helpers.d.ts +15 -0
  72. package/dist/commands/mcp/helpers.js +21 -0
  73. package/dist/commands/mcp/helpers.js.map +1 -0
  74. package/dist/commands/mcp/pull.d.ts +3 -0
  75. package/dist/commands/mcp/pull.js +85 -0
  76. package/dist/commands/mcp/pull.js.map +1 -0
  77. package/dist/commands/mcp/server.d.ts +2 -0
  78. package/dist/commands/mcp/server.js +80 -0
  79. package/dist/commands/mcp/server.js.map +1 -0
  80. package/dist/commands/mcp/specs.d.ts +2 -0
  81. package/dist/commands/mcp/specs.js +325 -0
  82. package/dist/commands/mcp/specs.js.map +1 -0
  83. package/dist/commands/mcp/tickets.d.ts +3 -0
  84. package/dist/commands/mcp/tickets.js +77 -0
  85. package/dist/commands/mcp/tickets.js.map +1 -0
  86. package/dist/commands/pull.d.ts +10 -0
  87. package/dist/commands/pull.js +61 -0
  88. package/dist/commands/pull.js.map +1 -0
  89. package/dist/commands/status.d.ts +2 -0
  90. package/dist/commands/status.js +102 -0
  91. package/dist/commands/status.js.map +1 -0
  92. package/dist/components/ticket-renderer.d.ts +5 -0
  93. package/dist/components/ticket-renderer.js +89 -0
  94. package/dist/components/ticket-renderer.js.map +1 -0
  95. package/dist/contexts/config-context.d.ts +17 -0
  96. package/dist/contexts/config-context.js +77 -0
  97. package/dist/contexts/config-context.js.map +1 -0
  98. package/dist/contexts/error-context.d.ts +12 -0
  99. package/dist/contexts/error-context.js +28 -0
  100. package/dist/contexts/error-context.js.map +1 -0
  101. package/dist/helpers/config.d.ts +4 -0
  102. package/dist/helpers/config.js +29 -0
  103. package/dist/helpers/config.js.map +1 -0
  104. package/dist/helpers/fs.d.ts +8 -0
  105. package/dist/helpers/fs.js +83 -0
  106. package/dist/helpers/fs.js.map +1 -0
  107. package/dist/helpers/init.d.ts +2 -0
  108. package/dist/helpers/init.js +18 -0
  109. package/dist/helpers/init.js.map +1 -0
  110. package/dist/helpers/preflight.d.ts +8 -0
  111. package/dist/helpers/preflight.js +77 -0
  112. package/dist/helpers/preflight.js.map +1 -0
  113. package/dist/helpers/pull.d.ts +9 -0
  114. package/dist/helpers/pull.js +79 -0
  115. package/dist/helpers/pull.js.map +1 -0
  116. package/dist/helpers/symlink.d.ts +9 -0
  117. package/dist/helpers/symlink.js +104 -0
  118. package/dist/helpers/symlink.js.map +1 -0
  119. package/dist/helpers/tickets.d.ts +4 -0
  120. package/dist/helpers/tickets.js +56 -0
  121. package/dist/helpers/tickets.js.map +1 -0
  122. package/dist/hooks/use-async.d.ts +6 -0
  123. package/dist/hooks/use-async.js +37 -0
  124. package/dist/hooks/use-async.js.map +1 -0
  125. package/dist/plugins/atlassian/adf-to-markdown.d.ts +11 -0
  126. package/dist/plugins/atlassian/adf-to-markdown.js +121 -0
  127. package/dist/plugins/atlassian/adf-to-markdown.js.map +1 -0
  128. package/dist/plugins/atlassian/auth-step.d.ts +10 -0
  129. package/dist/plugins/atlassian/auth-step.js +59 -0
  130. package/dist/plugins/atlassian/auth-step.js.map +1 -0
  131. package/dist/plugins/atlassian/confluence-client.d.ts +24 -0
  132. package/dist/plugins/atlassian/confluence-client.js +250 -0
  133. package/dist/plugins/atlassian/confluence-client.js.map +1 -0
  134. package/dist/plugins/atlassian/data.d.ts +18 -0
  135. package/dist/plugins/atlassian/data.js +12 -0
  136. package/dist/plugins/atlassian/data.js.map +1 -0
  137. package/dist/plugins/atlassian/index.d.ts +2 -0
  138. package/dist/plugins/atlassian/index.js +57 -0
  139. package/dist/plugins/atlassian/index.js.map +1 -0
  140. package/dist/plugins/atlassian/init-step.d.ts +11 -0
  141. package/dist/plugins/atlassian/init-step.js +54 -0
  142. package/dist/plugins/atlassian/init-step.js.map +1 -0
  143. package/dist/plugins/atlassian/jira-client.d.ts +38 -0
  144. package/dist/plugins/atlassian/jira-client.js +308 -0
  145. package/dist/plugins/atlassian/jira-client.js.map +1 -0
  146. package/dist/plugins/atlassian/specs-step.d.ts +13 -0
  147. package/dist/plugins/atlassian/specs-step.js +71 -0
  148. package/dist/plugins/atlassian/specs-step.js.map +1 -0
  149. package/dist/plugins/atlassian/tickets-step.d.ts +8 -0
  150. package/dist/plugins/atlassian/tickets-step.js +45 -0
  151. package/dist/plugins/atlassian/tickets-step.js.map +1 -0
  152. package/dist/plugins/claude-code/agents.d.ts +4 -0
  153. package/dist/plugins/claude-code/agents.js +56 -0
  154. package/dist/plugins/claude-code/agents.js.map +1 -0
  155. package/dist/plugins/claude-code/commands.d.ts +4 -0
  156. package/dist/plugins/claude-code/commands.js +54 -0
  157. package/dist/plugins/claude-code/commands.js.map +1 -0
  158. package/dist/plugins/claude-code/index.d.ts +2 -0
  159. package/dist/plugins/claude-code/index.js +57 -0
  160. package/dist/plugins/claude-code/index.js.map +1 -0
  161. package/dist/plugins/cursor/index.d.ts +2 -0
  162. package/dist/plugins/cursor/index.js +60 -0
  163. package/dist/plugins/cursor/index.js.map +1 -0
  164. package/dist/plugins/index.d.ts +1 -0
  165. package/dist/plugins/index.js +16 -0
  166. package/dist/plugins/index.js.map +1 -0
  167. package/dist/plugins/local/index.d.ts +2 -0
  168. package/dist/plugins/local/index.js +53 -0
  169. package/dist/plugins/local/index.js.map +1 -0
  170. package/dist/plugins/local/init-step.d.ts +9 -0
  171. package/dist/plugins/local/init-step.js +24 -0
  172. package/dist/plugins/local/init-step.js.map +1 -0
  173. package/dist/plugins/local/local-spec-provider.d.ts +9 -0
  174. package/dist/plugins/local/local-spec-provider.js +66 -0
  175. package/dist/plugins/local/local-spec-provider.js.map +1 -0
  176. package/dist/plugins/local/local-ticket-provider.d.ts +24 -0
  177. package/dist/plugins/local/local-ticket-provider.js +89 -0
  178. package/dist/plugins/local/local-ticket-provider.js.map +1 -0
  179. package/dist/plugins/none/index.d.ts +2 -0
  180. package/dist/plugins/none/index.js +78 -0
  181. package/dist/plugins/none/index.js.map +1 -0
  182. package/dist/plugins/other-ai/index.d.ts +2 -0
  183. package/dist/plugins/other-ai/index.js +48 -0
  184. package/dist/plugins/other-ai/index.js.map +1 -0
  185. package/dist/plugins/registry.d.ts +4 -0
  186. package/dist/plugins/registry.js +18 -0
  187. package/dist/plugins/registry.js.map +1 -0
  188. package/dist/plugins/resolve.d.ts +8 -0
  189. package/dist/plugins/resolve.js +23 -0
  190. package/dist/plugins/resolve.js.map +1 -0
  191. package/dist/plugins/types.d.ts +36 -0
  192. package/dist/plugins/types.js +2 -0
  193. package/dist/plugins/types.js.map +1 -0
  194. package/dist/plugins/vscode-copilot/agents.d.ts +4 -0
  195. package/dist/plugins/vscode-copilot/agents.js +51 -0
  196. package/dist/plugins/vscode-copilot/agents.js.map +1 -0
  197. package/dist/plugins/vscode-copilot/index.d.ts +2 -0
  198. package/dist/plugins/vscode-copilot/index.js +59 -0
  199. package/dist/plugins/vscode-copilot/index.js.map +1 -0
  200. package/dist/plugins/vscode-copilot/prompts.d.ts +4 -0
  201. package/dist/plugins/vscode-copilot/prompts.js +60 -0
  202. package/dist/plugins/vscode-copilot/prompts.js.map +1 -0
  203. package/dist/schemas/config.d.ts +24 -0
  204. package/dist/schemas/config.js +38 -0
  205. package/dist/schemas/config.js.map +1 -0
  206. package/dist/types.d.ts +84 -0
  207. package/dist/types.js +2 -0
  208. package/dist/types.js.map +1 -0
  209. package/dist/ui/confirm.d.ts +7 -0
  210. package/dist/ui/confirm.js +19 -0
  211. package/dist/ui/confirm.js.map +1 -0
  212. package/dist/ui/error-display.d.ts +6 -0
  213. package/dist/ui/error-display.js +23 -0
  214. package/dist/ui/error-display.js.map +1 -0
  215. package/dist/ui/masked-input.d.ts +9 -0
  216. package/dist/ui/masked-input.js +25 -0
  217. package/dist/ui/masked-input.js.map +1 -0
  218. package/dist/ui/multi-select.d.ts +13 -0
  219. package/dist/ui/multi-select.js +127 -0
  220. package/dist/ui/multi-select.js.map +1 -0
  221. package/dist/ui/select-input.d.ts +12 -0
  222. package/dist/ui/select-input.js +81 -0
  223. package/dist/ui/select-input.js.map +1 -0
  224. package/dist/ui/spinner.d.ts +6 -0
  225. package/dist/ui/spinner.js +18 -0
  226. package/dist/ui/spinner.js.map +1 -0
  227. package/dist/ui/text-input.d.ts +9 -0
  228. package/dist/ui/text-input.js +23 -0
  229. package/dist/ui/text-input.js.map +1 -0
  230. package/dist/wizard/passthrough-step.d.ts +6 -0
  231. package/dist/wizard/passthrough-step.js +8 -0
  232. package/dist/wizard/passthrough-step.js.map +1 -0
  233. package/dist/wizard/plugin-select-step.d.ts +10 -0
  234. package/dist/wizard/plugin-select-step.js +31 -0
  235. package/dist/wizard/plugin-select-step.js.map +1 -0
  236. package/dist/wizard/summary-step.d.ts +2 -0
  237. package/dist/wizard/summary-step.js +97 -0
  238. package/dist/wizard/summary-step.js.map +1 -0
  239. package/package.json +79 -0
@@ -0,0 +1,57 @@
1
+ import { writeFile, mkdir } from 'node:fs/promises';
2
+ import { join } from 'node:path';
3
+ import { readJsonFileSafe } from '../../helpers/fs.js';
4
+ import { PassthroughStep } from '../../wizard/passthrough-step.js';
5
+ import { generateAgentInfo } from '../../ai/agent-info.js';
6
+ import { getClaudeAgents } from './agents.js';
7
+ import { getClaudeCommands } from './commands.js';
8
+ export const claudeCodePlugin = {
9
+ id: 'claude-code',
10
+ name: 'Claude Code',
11
+ getCapabilities() {
12
+ return ['ai'];
13
+ },
14
+ needsConfiguration() {
15
+ return false;
16
+ },
17
+ InitStep: PassthroughStep,
18
+ createProvider(capability, _data) {
19
+ if (capability !== 'ai') {
20
+ throw new Error(`Claude Code plugin does not support capability: ${capability}`);
21
+ }
22
+ const provider = {
23
+ name: 'Claude Code',
24
+ async install(cwd) {
25
+ // Register MCP server in .mcp.json
26
+ const configPath = join(cwd, '.mcp.json');
27
+ const existing = await readJsonFileSafe(configPath);
28
+ const mcpServers = (existing['mcpServers'] ?? {});
29
+ mcpServers['hai'] = {
30
+ command: 'npx',
31
+ args: ['hai', 'mcp', '.'],
32
+ };
33
+ existing['mcpServers'] = mcpServers;
34
+ await writeFile(configPath, JSON.stringify(existing, null, 2) + '\n', 'utf-8');
35
+ // Install prompt commands
36
+ const commandsDir = join(cwd, '.claude', 'commands');
37
+ await mkdir(commandsDir, { recursive: true });
38
+ for (const command of getClaudeCommands()) {
39
+ await writeFile(join(commandsDir, command.filename), command.content, 'utf-8');
40
+ }
41
+ // Install agent definitions
42
+ const agentsDir = join(cwd, '.claude', 'agents');
43
+ await mkdir(agentsDir, { recursive: true });
44
+ for (const agent of getClaudeAgents()) {
45
+ await writeFile(join(agentsDir, agent.filename), agent.content, 'utf-8');
46
+ }
47
+ // Generate agent info in .ai/ directory
48
+ await writeFile(join(cwd, '.ai', 'AGENTS.md'), generateAgentInfo(), 'utf-8');
49
+ },
50
+ };
51
+ return provider;
52
+ },
53
+ summarize(_capability, _data) {
54
+ return '';
55
+ },
56
+ };
57
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../source/plugins/claude-code/index.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import type { HaiPlugin } from '../types.js';
2
+ export declare const cursorPlugin: HaiPlugin;
@@ -0,0 +1,60 @@
1
+ import { writeFile, mkdir } from 'node:fs/promises';
2
+ import { join } from 'node:path';
3
+ import { PassthroughStep } from '../../wizard/passthrough-step.js';
4
+ import { readJsonFileSafe } from '../../helpers/fs.js';
5
+ import { generateAgentInfo } from '../../ai/agent-info.js';
6
+ import { PROMPTS } from '../../ai/prompts/index.js';
7
+ import { AGENTS } from '../../ai/agents/index.js';
8
+ export const cursorPlugin = {
9
+ id: 'cursor',
10
+ name: 'Cursor',
11
+ getCapabilities() {
12
+ return ['ai'];
13
+ },
14
+ needsConfiguration() {
15
+ return false;
16
+ },
17
+ InitStep: PassthroughStep,
18
+ createProvider(capability, _data) {
19
+ if (capability !== 'ai') {
20
+ throw new Error(`Cursor plugin does not support capability: ${capability}`);
21
+ }
22
+ const provider = {
23
+ name: 'Cursor',
24
+ async install(cwd) {
25
+ // Register MCP server in .cursor/mcp.json
26
+ const dir = join(cwd, '.cursor');
27
+ await mkdir(dir, { recursive: true });
28
+ const configPath = join(dir, 'mcp.json');
29
+ const existing = await readJsonFileSafe(configPath);
30
+ const mcpServers = (existing['mcpServers'] ?? {});
31
+ mcpServers['hai'] = {
32
+ command: 'npx',
33
+ args: ['hai', 'mcp', '.'],
34
+ };
35
+ existing['mcpServers'] = mcpServers;
36
+ await writeFile(configPath, JSON.stringify(existing, null, 2) + '\n', 'utf-8');
37
+ // Install prompt commands to .cursor/commands/
38
+ const commandsDir = join(cwd, '.cursor', 'commands');
39
+ await mkdir(commandsDir, { recursive: true });
40
+ for (const [name, prompt] of Object.entries(PROMPTS)) {
41
+ await writeFile(join(commandsDir, `${name}.md`), prompt.body + '\n', 'utf-8');
42
+ }
43
+ // Install agent definitions to .cursor/agents/
44
+ const agentsDir = join(cwd, '.cursor', 'agents');
45
+ await mkdir(agentsDir, { recursive: true });
46
+ for (const [key, agent] of Object.entries(AGENTS)) {
47
+ const content = `# ${agent.name}\n\n${agent.description}\n\n${agent.body}\n`;
48
+ await writeFile(join(agentsDir, `${key}.md`), content, 'utf-8');
49
+ }
50
+ // Generate agent info in .ai/ directory
51
+ await writeFile(join(cwd, '.ai', 'AGENTS.md'), generateAgentInfo(), 'utf-8');
52
+ },
53
+ };
54
+ return provider;
55
+ },
56
+ summarize(_capability, _data) {
57
+ return 'MCP: .cursor/mcp.json, Commands: .cursor/commands/, Agents: .cursor/agents/';
58
+ },
59
+ };
60
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../source/plugins/cursor/index.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,16 @@
1
+ import { registerPlugin } from './registry.js';
2
+ import { atlassianPlugin } from './atlassian/index.js';
3
+ import { claudeCodePlugin } from './claude-code/index.js';
4
+ import { vscodeCopilotPlugin } from './vscode-copilot/index.js';
5
+ import { cursorPlugin } from './cursor/index.js';
6
+ import { otherAiPlugin } from './other-ai/index.js';
7
+ import { localPlugin } from './local/index.js';
8
+ import { nonePlugin } from './none/index.js';
9
+ registerPlugin(atlassianPlugin);
10
+ registerPlugin(claudeCodePlugin);
11
+ registerPlugin(vscodeCopilotPlugin);
12
+ registerPlugin(cursorPlugin);
13
+ registerPlugin(otherAiPlugin);
14
+ registerPlugin(localPlugin);
15
+ registerPlugin(nonePlugin);
16
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../source/plugins/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,cAAc,EAAC,MAAM,eAAe,CAAC;AAC7C,OAAO,EAAC,eAAe,EAAC,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAC,gBAAgB,EAAC,MAAM,wBAAwB,CAAC;AACxD,OAAO,EAAC,mBAAmB,EAAC,MAAM,2BAA2B,CAAC;AAC9D,OAAO,EAAC,YAAY,EAAC,MAAM,mBAAmB,CAAC;AAC/C,OAAO,EAAC,aAAa,EAAC,MAAM,qBAAqB,CAAC;AAClD,OAAO,EAAC,WAAW,EAAC,MAAM,kBAAkB,CAAC;AAC7C,OAAO,EAAC,UAAU,EAAC,MAAM,iBAAiB,CAAC;AAE3C,cAAc,CAAC,eAAe,CAAC,CAAC;AAChC,cAAc,CAAC,gBAAgB,CAAC,CAAC;AACjC,cAAc,CAAC,mBAAmB,CAAC,CAAC;AACpC,cAAc,CAAC,YAAY,CAAC,CAAC;AAC7B,cAAc,CAAC,aAAa,CAAC,CAAC;AAC9B,cAAc,CAAC,WAAW,CAAC,CAAC;AAC5B,cAAc,CAAC,UAAU,CAAC,CAAC"}
@@ -0,0 +1,2 @@
1
+ import type { HaiPlugin } from '../types.js';
2
+ export declare const localPlugin: HaiPlugin;
@@ -0,0 +1,53 @@
1
+ import { join } from 'node:path';
2
+ import LocalInitStep from './init-step.js';
3
+ import { LocalTicketProvider } from './local-ticket-provider.js';
4
+ import { LocalSpecProvider } from './local-spec-provider.js';
5
+ export const localPlugin = {
6
+ id: 'local',
7
+ name: 'Local',
8
+ getCapabilities() {
9
+ return ['tickets', 'specs'];
10
+ },
11
+ needsConfiguration() {
12
+ return true;
13
+ },
14
+ InitStep: LocalInitStep,
15
+ createProvider(capability, data) {
16
+ const pluginData = data;
17
+ const cwd = process.cwd();
18
+ if (capability === 'tickets') {
19
+ const ticketPath = pluginData.tickets?.path;
20
+ if (!ticketPath) {
21
+ throw new Error('Local plugin: no tickets path configured');
22
+ }
23
+ return new LocalTicketProvider(join(cwd, ticketPath));
24
+ }
25
+ if (capability === 'specs') {
26
+ const specPath = pluginData.specs?.path;
27
+ if (!specPath) {
28
+ throw new Error('Local plugin: no specs path configured');
29
+ }
30
+ return new LocalSpecProvider(join(cwd, specPath));
31
+ }
32
+ throw new Error(`Local plugin does not support capability: ${capability}`);
33
+ },
34
+ getTicketProjects(data) {
35
+ const pluginData = data;
36
+ return pluginData.tickets?.path ? ['local'] : [];
37
+ },
38
+ getSpecConfig(data) {
39
+ const pluginData = data;
40
+ return { spaces: pluginData.specs?.path ? ['local'] : [] };
41
+ },
42
+ summarize(capability, data) {
43
+ const pluginData = data;
44
+ if (capability === 'tickets') {
45
+ return `Local folder: ${pluginData.tickets?.path ?? '(not set)'}`;
46
+ }
47
+ if (capability === 'specs') {
48
+ return `Local folder: ${pluginData.specs?.path ?? '(not set)'}`;
49
+ }
50
+ return '';
51
+ },
52
+ };
53
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../source/plugins/local/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,IAAI,EAAC,MAAM,WAAW,CAAC;AAO/B,OAAO,aAAa,MAAM,gBAAgB,CAAC;AAC3C,OAAO,EAAC,mBAAmB,EAAC,MAAM,4BAA4B,CAAC;AAC/D,OAAO,EAAC,iBAAiB,EAAC,MAAM,0BAA0B,CAAC;AAO3D,MAAM,CAAC,MAAM,WAAW,GAAc;IACrC,EAAE,EAAE,OAAO;IACX,IAAI,EAAE,OAAO;IAEb,eAAe;QACd,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;IAC7B,CAAC;IAED,kBAAkB;QACjB,OAAO,IAAI,CAAC;IACb,CAAC;IAED,QAAQ,EAAE,aAAa;IAEvB,cAAc,CACb,UAAa,EACb,IAAgB;QAEhB,MAAM,UAAU,GAAG,IAAuB,CAAC;QAC3C,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;QAE1B,IAAI,UAAU,KAAK,SAAS,EAAE,CAAC;YAC9B,MAAM,UAAU,GAAG,UAAU,CAAC,OAAO,EAAE,IAAI,CAAC;YAC5C,IAAI,CAAC,UAAU,EAAE,CAAC;gBACjB,MAAM,IAAI,KAAK,CAAC,0CAA0C,CAAC,CAAC;YAC7D,CAAC;YAED,OAAO,IAAI,mBAAmB,CAC7B,IAAI,CAAC,GAAG,EAAE,UAAU,CAAC,CACQ,CAAC;QAChC,CAAC;QAED,IAAI,UAAU,KAAK,OAAO,EAAE,CAAC;YAC5B,MAAM,QAAQ,GAAG,UAAU,CAAC,KAAK,EAAE,IAAI,CAAC;YACxC,IAAI,CAAC,QAAQ,EAAE,CAAC;gBACf,MAAM,IAAI,KAAK,CAAC,wCAAwC,CAAC,CAAC;YAC3D,CAAC;YAED,OAAO,IAAI,iBAAiB,CAC3B,IAAI,CAAC,GAAG,EAAE,QAAQ,CAAC,CACU,CAAC;QAChC,CAAC;QAED,MAAM,IAAI,KAAK,CACd,6CAA6C,UAAU,EAAE,CACzD,CAAC;IACH,CAAC;IAED,iBAAiB,CAAC,IAAgB;QACjC,MAAM,UAAU,GAAG,IAAuB,CAAC;QAC3C,OAAO,UAAU,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;IAClD,CAAC;IAED,aAAa,CAAC,IAAgB;QAC7B,MAAM,UAAU,GAAG,IAAuB,CAAC;QAC3C,OAAO,EAAC,MAAM,EAAE,UAAU,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,EAAC,CAAC;IAC1D,CAAC;IAED,SAAS,CAAC,UAA0B,EAAE,IAAgB;QACrD,MAAM,UAAU,GAAG,IAAuB,CAAC;QAC3C,IAAI,UAAU,KAAK,SAAS,EAAE,CAAC;YAC9B,OAAO,iBAAiB,UAAU,CAAC,OAAO,EAAE,IAAI,IAAI,WAAW,EAAE,CAAC;QACnE,CAAC;QAED,IAAI,UAAU,KAAK,OAAO,EAAE,CAAC;YAC5B,OAAO,iBAAiB,UAAU,CAAC,KAAK,EAAE,IAAI,IAAI,WAAW,EAAE,CAAC;QACjE,CAAC;QAED,OAAO,EAAE,CAAC;IACX,CAAC;CACD,CAAC"}
@@ -0,0 +1,9 @@
1
+ import React from 'react';
2
+ import type { CapabilityType, PluginData } from '../types.js';
3
+ type Props = {
4
+ capability: CapabilityType;
5
+ existingData?: PluginData;
6
+ onComplete: (data: PluginData) => void;
7
+ };
8
+ export default function LocalInitStep({ capability, existingData, onComplete, }: Props): React.JSX.Element | null;
9
+ export {};
@@ -0,0 +1,24 @@
1
+ import React from 'react';
2
+ import TextInput from '../../ui/text-input.js';
3
+ export default function LocalInitStep({ capability, existingData, onComplete, }) {
4
+ if (capability === 'tickets') {
5
+ const existing = existingData?.['tickets'];
6
+ return (React.createElement(TextInput, { label: "Tickets folder path (relative to project root)", placeholder: "e.g. docs/tickets", defaultValue: existing?.path, onSubmit: value => {
7
+ onComplete({
8
+ ...existingData,
9
+ tickets: { path: value },
10
+ });
11
+ } }));
12
+ }
13
+ if (capability === 'specs') {
14
+ const existing = existingData?.['specs'];
15
+ return (React.createElement(TextInput, { label: "Specs folder path (relative to project root)", placeholder: "e.g. docs/specs", defaultValue: existing?.path, onSubmit: value => {
16
+ onComplete({
17
+ ...existingData,
18
+ specs: { path: value },
19
+ });
20
+ } }));
21
+ }
22
+ return null;
23
+ }
24
+ //# sourceMappingURL=init-step.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"init-step.js","sourceRoot":"","sources":["../../../source/plugins/local/init-step.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,SAAS,MAAM,wBAAwB,CAAC;AAS/C,MAAM,CAAC,OAAO,UAAU,aAAa,CAAC,EACrC,UAAU,EACV,YAAY,EACZ,UAAU,GACH;IACP,IAAI,UAAU,KAAK,SAAS,EAAE,CAAC;QAC9B,MAAM,QAAQ,GAAG,YAAY,EAAE,CAAC,SAAS,CAE7B,CAAC;QACb,OAAO,CACN,oBAAC,SAAS,IACT,KAAK,EAAC,gDAAgD,EACtD,WAAW,EAAC,mBAAmB,EAC/B,YAAY,EAAE,QAAQ,EAAE,IAAI,EAC5B,QAAQ,EAAE,KAAK,CAAC,EAAE;gBACjB,UAAU,CAAC;oBACV,GAAG,YAAY;oBACf,OAAO,EAAE,EAAC,IAAI,EAAE,KAAK,EAAC;iBACtB,CAAC,CAAC;YACJ,CAAC,GACA,CACF,CAAC;IACH,CAAC;IAED,IAAI,UAAU,KAAK,OAAO,EAAE,CAAC;QAC5B,MAAM,QAAQ,GAAG,YAAY,EAAE,CAAC,OAAO,CAA+B,CAAC;QACvE,OAAO,CACN,oBAAC,SAAS,IACT,KAAK,EAAC,8CAA8C,EACpD,WAAW,EAAC,iBAAiB,EAC7B,YAAY,EAAE,QAAQ,EAAE,IAAI,EAC5B,QAAQ,EAAE,KAAK,CAAC,EAAE;gBACjB,UAAU,CAAC;oBACV,GAAG,YAAY;oBACf,KAAK,EAAE,EAAC,IAAI,EAAE,KAAK,EAAC;iBACpB,CAAC,CAAC;YACJ,CAAC,GACA,CACF,CAAC;IACH,CAAC;IAED,OAAO,IAAI,CAAC;AACb,CAAC"}
@@ -0,0 +1,9 @@
1
+ import type { SpecPage, SpecProvider, Space, SyncOptions, SyncResult } from '../../types.js';
2
+ export declare class LocalSpecProvider implements SpecProvider {
3
+ private readonly basePath;
4
+ constructor(basePath: string);
5
+ sync(destPath: string, _options?: SyncOptions): Promise<SyncResult>;
6
+ fetchSpaces(): Promise<Space[]>;
7
+ fetchPages(_spaceKey: string, _ancestorTitle?: string): Promise<SpecPage[]>;
8
+ private readAllPages;
9
+ }
@@ -0,0 +1,66 @@
1
+ import { readFile, readdir, stat } from 'node:fs/promises';
2
+ import { join, relative } from 'node:path';
3
+ import { linkOrCopy } from '../../helpers/symlink.js';
4
+ export class LocalSpecProvider {
5
+ constructor(basePath) {
6
+ Object.defineProperty(this, "basePath", {
7
+ enumerable: true,
8
+ configurable: true,
9
+ writable: true,
10
+ value: void 0
11
+ });
12
+ this.basePath = basePath;
13
+ }
14
+ async sync(destPath, _options) {
15
+ await linkOrCopy(destPath, this.basePath);
16
+ return { downloaded: [], updated: [], removed: [] };
17
+ }
18
+ async fetchSpaces() {
19
+ return [{ id: 'local', key: 'local', name: 'Local' }];
20
+ }
21
+ async fetchPages(_spaceKey, _ancestorTitle) {
22
+ return this.readAllPages(this.basePath, []);
23
+ }
24
+ async readAllPages(dir, ancestors) {
25
+ let entries;
26
+ try {
27
+ entries = await readdir(dir);
28
+ }
29
+ catch {
30
+ return [];
31
+ }
32
+ const pages = [];
33
+ for (const entry of entries) {
34
+ const fullPath = join(dir, entry);
35
+ const info = await stat(fullPath);
36
+ if (info.isDirectory()) {
37
+ const nested = await this.readAllPages(fullPath, [
38
+ ...ancestors,
39
+ entry,
40
+ ]);
41
+ pages.push(...nested);
42
+ continue;
43
+ }
44
+ if (!entry.endsWith('.md'))
45
+ continue;
46
+ try {
47
+ const content = await readFile(fullPath, 'utf-8');
48
+ const title = entry.slice(0, -3);
49
+ const relPath = relative(this.basePath, fullPath);
50
+ pages.push({
51
+ id: `local:${relPath}`,
52
+ title,
53
+ updated: info.mtime.toISOString(),
54
+ body: content,
55
+ source: `local:${relPath}`,
56
+ ancestors,
57
+ });
58
+ }
59
+ catch {
60
+ // Skip unreadable files
61
+ }
62
+ }
63
+ return pages;
64
+ }
65
+ }
66
+ //# sourceMappingURL=local-spec-provider.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"local-spec-provider.js","sourceRoot":"","sources":["../../../source/plugins/local/local-spec-provider.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,QAAQ,EAAE,OAAO,EAAE,IAAI,EAAC,MAAM,kBAAkB,CAAC;AACzD,OAAO,EAAC,IAAI,EAAE,QAAQ,EAAC,MAAM,WAAW,CAAC;AAQzC,OAAO,EAAC,UAAU,EAAC,MAAM,0BAA0B,CAAC;AAEpD,MAAM,OAAO,iBAAiB;IAG7B,YAAY,QAAgB;QAFX;;;;;WAAiB;QAGjC,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;IAC1B,CAAC;IAED,KAAK,CAAC,IAAI,CAAC,QAAgB,EAAE,QAAsB;QAClD,MAAM,UAAU,CAAC,QAAQ,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC1C,OAAO,EAAC,UAAU,EAAE,EAAE,EAAE,OAAO,EAAE,EAAE,EAAE,OAAO,EAAE,EAAE,EAAC,CAAC;IACnD,CAAC;IAED,KAAK,CAAC,WAAW;QAChB,OAAO,CAAC,EAAC,EAAE,EAAE,OAAO,EAAE,GAAG,EAAE,OAAO,EAAE,IAAI,EAAE,OAAO,EAAC,CAAC,CAAC;IACrD,CAAC;IAED,KAAK,CAAC,UAAU,CACf,SAAiB,EACjB,cAAuB;QAEvB,OAAO,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;IAC7C,CAAC;IAEO,KAAK,CAAC,YAAY,CACzB,GAAW,EACX,SAAmB;QAEnB,IAAI,OAAiB,CAAC;QACtB,IAAI,CAAC;YACJ,OAAO,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,CAAC;QAC9B,CAAC;QAAC,MAAM,CAAC;YACR,OAAO,EAAE,CAAC;QACX,CAAC;QAED,MAAM,KAAK,GAAe,EAAE,CAAC;QAE7B,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;YAC7B,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;YAClC,MAAM,IAAI,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,CAAC;YAElC,IAAI,IAAI,CAAC,WAAW,EAAE,EAAE,CAAC;gBACxB,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE;oBAChD,GAAG,SAAS;oBACZ,KAAK;iBACL,CAAC,CAAC;gBACH,KAAK,CAAC,IAAI,CAAC,GAAG,MAAM,CAAC,CAAC;gBACtB,SAAS;YACV,CAAC;YAED,IAAI,CAAC,KAAK,CAAC,QAAQ,CAAC,KAAK,CAAC;gBAAE,SAAS;YAErC,IAAI,CAAC;gBACJ,MAAM,OAAO,GAAG,MAAM,QAAQ,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;gBAClD,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;gBACjC,MAAM,OAAO,GAAG,QAAQ,CAAC,IAAI,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;gBAElD,KAAK,CAAC,IAAI,CAAC;oBACV,EAAE,EAAE,SAAS,OAAO,EAAE;oBACtB,KAAK;oBACL,OAAO,EAAE,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE;oBACjC,IAAI,EAAE,OAAO;oBACb,MAAM,EAAE,SAAS,OAAO,EAAE;oBAC1B,SAAS;iBACT,CAAC,CAAC;YACJ,CAAC;YAAC,MAAM,CAAC;gBACR,wBAAwB;YACzB,CAAC;QACF,CAAC;QAED,OAAO,KAAK,CAAC;IACd,CAAC;CACD"}
@@ -0,0 +1,24 @@
1
+ import type { Ticket, TicketProvider, Project, SyncOptions, SyncResult } from '../../types.js';
2
+ export declare class LocalTicketProvider implements TicketProvider {
3
+ private readonly basePath;
4
+ constructor(basePath: string);
5
+ fetchProjects(): Promise<Project[]>;
6
+ fetchIssue(key: string): Promise<{
7
+ ticket: Ticket;
8
+ linkedKeys: string[];
9
+ }>;
10
+ sync(destPath: string, _options?: SyncOptions): Promise<SyncResult>;
11
+ fetchWorkingTickets(_projects: string[]): Promise<{
12
+ tickets: Ticket[];
13
+ linkedKeys: string[];
14
+ }>;
15
+ fetchDoneTickets(_projects: string[]): Promise<{
16
+ tickets: Ticket[];
17
+ linkedKeys: string[];
18
+ }>;
19
+ searchByText(query: string): Promise<{
20
+ tickets: Ticket[];
21
+ linkedKeys: string[];
22
+ }>;
23
+ private readAllTickets;
24
+ }
@@ -0,0 +1,89 @@
1
+ import { readFile, readdir } from 'node:fs/promises';
2
+ import { join } from 'node:path';
3
+ import { linkOrCopy } from '../../helpers/symlink.js';
4
+ function parseTicketFromMarkdown(key, content) {
5
+ const lines = content.split('\n');
6
+ const firstLine = lines[0] ?? '';
7
+ const summary = firstLine.startsWith('# ')
8
+ ? firstLine.slice(2).trim()
9
+ : key;
10
+ return {
11
+ key,
12
+ summary,
13
+ status: 'Open',
14
+ statusCategory: 'To Do',
15
+ assignee: null,
16
+ priority: 'Medium',
17
+ created: new Date().toISOString(),
18
+ updated: new Date().toISOString(),
19
+ description: content,
20
+ comments: [],
21
+ links: [],
22
+ attachments: [],
23
+ };
24
+ }
25
+ export class LocalTicketProvider {
26
+ constructor(basePath) {
27
+ Object.defineProperty(this, "basePath", {
28
+ enumerable: true,
29
+ configurable: true,
30
+ writable: true,
31
+ value: void 0
32
+ });
33
+ this.basePath = basePath;
34
+ }
35
+ async fetchProjects() {
36
+ return [{ key: 'local', name: 'Local' }];
37
+ }
38
+ async fetchIssue(key) {
39
+ const filePath = join(this.basePath, `${key}.md`);
40
+ const content = await readFile(filePath, 'utf-8');
41
+ return {
42
+ ticket: parseTicketFromMarkdown(key, content),
43
+ linkedKeys: [],
44
+ };
45
+ }
46
+ async sync(destPath, _options) {
47
+ await linkOrCopy(destPath, this.basePath);
48
+ return { downloaded: [], updated: [], removed: [] };
49
+ }
50
+ async fetchWorkingTickets(_projects) {
51
+ const tickets = await this.readAllTickets();
52
+ return { tickets, linkedKeys: [] };
53
+ }
54
+ async fetchDoneTickets(_projects) {
55
+ return { tickets: [], linkedKeys: [] };
56
+ }
57
+ async searchByText(query) {
58
+ const all = await this.readAllTickets();
59
+ const lower = query.toLowerCase();
60
+ const tickets = all.filter(t => t.key.toLowerCase().includes(lower) ||
61
+ t.summary.toLowerCase().includes(lower) ||
62
+ t.description.toLowerCase().includes(lower));
63
+ return { tickets, linkedKeys: [] };
64
+ }
65
+ async readAllTickets() {
66
+ let entries;
67
+ try {
68
+ entries = await readdir(this.basePath);
69
+ }
70
+ catch {
71
+ return [];
72
+ }
73
+ const tickets = [];
74
+ for (const entry of entries) {
75
+ if (!entry.endsWith('.md'))
76
+ continue;
77
+ const key = entry.slice(0, -3);
78
+ try {
79
+ const content = await readFile(join(this.basePath, entry), 'utf-8');
80
+ tickets.push(parseTicketFromMarkdown(key, content));
81
+ }
82
+ catch {
83
+ // Skip unreadable files
84
+ }
85
+ }
86
+ return tickets;
87
+ }
88
+ }
89
+ //# sourceMappingURL=local-ticket-provider.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"local-ticket-provider.js","sourceRoot":"","sources":["../../../source/plugins/local/local-ticket-provider.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import type { HaiPlugin } from '../types.js';
2
+ export declare const nonePlugin: HaiPlugin;
@@ -0,0 +1,78 @@
1
+ import { PassthroughStep } from '../../wizard/passthrough-step.js';
2
+ const noneTicketProvider = {
3
+ async fetchProjects() {
4
+ return [];
5
+ },
6
+ async fetchIssue(_key) {
7
+ return {
8
+ ticket: {
9
+ key: '',
10
+ summary: '',
11
+ status: '',
12
+ statusCategory: '',
13
+ assignee: null,
14
+ priority: '',
15
+ created: '',
16
+ updated: '',
17
+ description: '',
18
+ comments: [],
19
+ links: [],
20
+ attachments: [],
21
+ },
22
+ linkedKeys: [],
23
+ };
24
+ },
25
+ async fetchWorkingTickets() {
26
+ return { tickets: [], linkedKeys: [] };
27
+ },
28
+ async fetchDoneTickets() {
29
+ return { tickets: [], linkedKeys: [] };
30
+ },
31
+ async searchByText() {
32
+ return { tickets: [], linkedKeys: [] };
33
+ },
34
+ async sync() {
35
+ return { downloaded: [], updated: [], removed: [] };
36
+ },
37
+ };
38
+ const noneSpecProvider = {
39
+ async fetchSpaces() {
40
+ return [];
41
+ },
42
+ async fetchPages() {
43
+ return [];
44
+ },
45
+ async sync() {
46
+ return { downloaded: [], updated: [], removed: [] };
47
+ },
48
+ };
49
+ export const nonePlugin = {
50
+ id: 'none',
51
+ name: 'None',
52
+ getCapabilities() {
53
+ return ['tickets', 'specs'];
54
+ },
55
+ needsConfiguration() {
56
+ return false;
57
+ },
58
+ InitStep: PassthroughStep,
59
+ createProvider(capability, _data) {
60
+ if (capability === 'tickets') {
61
+ return noneTicketProvider;
62
+ }
63
+ if (capability === 'specs') {
64
+ return noneSpecProvider;
65
+ }
66
+ throw new Error(`None plugin does not support capability: ${capability}`);
67
+ },
68
+ getTicketProjects() {
69
+ return [];
70
+ },
71
+ getSpecConfig() {
72
+ return { spaces: [] };
73
+ },
74
+ summarize() {
75
+ return 'Disabled';
76
+ },
77
+ };
78
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../source/plugins/none/index.ts"],"names":[],"mappings":"AAOA,OAAO,EAAC,eAAe,EAAC,MAAM,kCAAkC,CAAC;AAEjE,MAAM,kBAAkB,GAAmB;IAC1C,KAAK,CAAC,aAAa;QAClB,OAAO,EAAE,CAAC;IACX,CAAC;IACD,KAAK,CAAC,UAAU,CAAC,IAAI;QACpB,OAAO;YACN,MAAM,EAAE;gBACP,GAAG,EAAE,EAAE;gBACP,OAAO,EAAE,EAAE;gBACX,MAAM,EAAE,EAAE;gBACV,cAAc,EAAE,EAAE;gBAClB,QAAQ,EAAE,IAAI;gBACd,QAAQ,EAAE,EAAE;gBACZ,OAAO,EAAE,EAAE;gBACX,OAAO,EAAE,EAAE;gBACX,WAAW,EAAE,EAAE;gBACf,QAAQ,EAAE,EAAE;gBACZ,KAAK,EAAE,EAAE;gBACT,WAAW,EAAE,EAAE;aACf;YACD,UAAU,EAAE,EAAE;SACd,CAAC;IACH,CAAC;IACD,KAAK,CAAC,mBAAmB;QACxB,OAAO,EAAC,OAAO,EAAE,EAAE,EAAE,UAAU,EAAE,EAAE,EAAC,CAAC;IACtC,CAAC;IACD,KAAK,CAAC,gBAAgB;QACrB,OAAO,EAAC,OAAO,EAAE,EAAE,EAAE,UAAU,EAAE,EAAE,EAAC,CAAC;IACtC,CAAC;IACD,KAAK,CAAC,YAAY;QACjB,OAAO,EAAC,OAAO,EAAE,EAAE,EAAE,UAAU,EAAE,EAAE,EAAC,CAAC;IACtC,CAAC;IACD,KAAK,CAAC,IAAI;QACT,OAAO,EAAC,UAAU,EAAE,EAAE,EAAE,OAAO,EAAE,EAAE,EAAE,OAAO,EAAE,EAAE,EAAC,CAAC;IACnD,CAAC;CACD,CAAC;AAEF,MAAM,gBAAgB,GAAiB;IACtC,KAAK,CAAC,WAAW;QAChB,OAAO,EAAE,CAAC;IACX,CAAC;IACD,KAAK,CAAC,UAAU;QACf,OAAO,EAAE,CAAC;IACX,CAAC;IACD,KAAK,CAAC,IAAI;QACT,OAAO,EAAC,UAAU,EAAE,EAAE,EAAE,OAAO,EAAE,EAAE,EAAE,OAAO,EAAE,EAAE,EAAC,CAAC;IACnD,CAAC;CACD,CAAC;AAEF,MAAM,CAAC,MAAM,UAAU,GAAc;IACpC,EAAE,EAAE,MAAM;IACV,IAAI,EAAE,MAAM;IAEZ,eAAe;QACd,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;IAC7B,CAAC;IAED,kBAAkB;QACjB,OAAO,KAAK,CAAC;IACd,CAAC;IAED,QAAQ,EAAE,eAAe;IAEzB,cAAc,CACb,UAAa,EACb,KAAiB;QAEjB,IAAI,UAAU,KAAK,SAAS,EAAE,CAAC;YAC9B,OAAO,kBAA+C,CAAC;QACxD,CAAC;QAED,IAAI,UAAU,KAAK,OAAO,EAAE,CAAC;YAC5B,OAAO,gBAA6C,CAAC;QACtD,CAAC;QAED,MAAM,IAAI,KAAK,CACd,4CAA4C,UAAU,EAAE,CACxD,CAAC;IACH,CAAC;IAED,iBAAiB;QAChB,OAAO,EAAE,CAAC;IACX,CAAC;IAED,aAAa;QACZ,OAAO,EAAC,MAAM,EAAE,EAAE,EAAC,CAAC;IACrB,CAAC;IAED,SAAS;QACR,OAAO,UAAU,CAAC;IACnB,CAAC;CACD,CAAC"}
@@ -0,0 +1,2 @@
1
+ import type { HaiPlugin } from '../types.js';
2
+ export declare const otherAiPlugin: HaiPlugin;
@@ -0,0 +1,48 @@
1
+ import { writeFile, mkdir } from 'node:fs/promises';
2
+ import { join } from 'node:path';
3
+ import { PassthroughStep } from '../../wizard/passthrough-step.js';
4
+ import { generateAgentInfo } from '../../ai/agent-info.js';
5
+ import { PROMPTS } from '../../ai/prompts/index.js';
6
+ import { AGENTS } from '../../ai/agents/index.js';
7
+ export const otherAiPlugin = {
8
+ id: 'other',
9
+ name: 'Other',
10
+ getCapabilities() {
11
+ return ['ai'];
12
+ },
13
+ needsConfiguration() {
14
+ return false;
15
+ },
16
+ InitStep: PassthroughStep,
17
+ createProvider(capability, _data) {
18
+ if (capability !== 'ai') {
19
+ throw new Error(`Other AI plugin does not support capability: ${capability}`);
20
+ }
21
+ const provider = {
22
+ name: 'Other',
23
+ async install(cwd) {
24
+ // For "other" providers, MCP registration is manual.
25
+ // Just install prompts to .ai/prompts/
26
+ const dir = join(cwd, '.ai', 'prompts');
27
+ await mkdir(dir, { recursive: true });
28
+ for (const [name, prompt] of Object.entries(PROMPTS)) {
29
+ const filename = `${name}.md`;
30
+ await writeFile(join(dir, filename), prompt.body + '\n', 'utf-8');
31
+ }
32
+ const agentsDir = join(cwd, '.ai', 'agents');
33
+ await mkdir(agentsDir, { recursive: true });
34
+ for (const [key, agent] of Object.entries(AGENTS)) {
35
+ const content = `# ${agent.name}\n\n${agent.description}\n\n${agent.body}\n`;
36
+ await writeFile(join(agentsDir, `${key}.md`), content, 'utf-8');
37
+ }
38
+ // Generate agent info in .ai/ directory
39
+ await writeFile(join(cwd, '.ai', 'AGENTS.md'), generateAgentInfo(), 'utf-8');
40
+ },
41
+ };
42
+ return provider;
43
+ },
44
+ summarize(_capability, _data) {
45
+ return 'Agents: .ai/agents; Prompts: .ai/prompts';
46
+ },
47
+ };
48
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../source/plugins/other-ai/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,SAAS,EAAE,KAAK,EAAC,MAAM,kBAAkB,CAAC;AAClD,OAAO,EAAC,IAAI,EAAC,MAAM,WAAW,CAAC;AAQ/B,OAAO,EAAC,eAAe,EAAC,MAAM,kCAAkC,CAAC;AACjE,OAAO,EAAC,iBAAiB,EAAC,MAAM,wBAAwB,CAAC;AACzD,OAAO,EAAC,OAAO,EAAC,MAAM,2BAA2B,CAAC;AAClD,OAAO,EAAC,MAAM,EAAC,MAAM,0BAA0B,CAAC;AAEhD,MAAM,CAAC,MAAM,aAAa,GAAc;IACvC,EAAE,EAAE,OAAO;IACX,IAAI,EAAE,OAAO;IAEb,eAAe;QACd,OAAO,CAAC,IAAI,CAAC,CAAC;IACf,CAAC;IAED,kBAAkB;QACjB,OAAO,KAAK,CAAC;IACd,CAAC;IAED,QAAQ,EAAE,eAAe;IAEzB,cAAc,CACb,UAAa,EACb,KAAiB;QAEjB,IAAI,UAAU,KAAK,IAAI,EAAE,CAAC;YACzB,MAAM,IAAI,KAAK,CACd,gDAAgD,UAAU,EAAE,CAC5D,CAAC;QACH,CAAC;QAED,MAAM,QAAQ,GAAe;YAC5B,IAAI,EAAE,OAAO;YAEb,KAAK,CAAC,OAAO,CAAC,GAAW;gBACxB,qDAAqD;gBACrD,uCAAuC;gBACvC,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,KAAK,EAAE,SAAS,CAAC,CAAC;gBACxC,MAAM,KAAK,CAAC,GAAG,EAAE,EAAC,SAAS,EAAE,IAAI,EAAC,CAAC,CAAC;gBACpC,KAAK,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;oBACtD,MAAM,QAAQ,GAAG,GAAG,IAAI,KAAK,CAAC;oBAC9B,MAAM,SAAS,CACd,IAAI,CAAC,GAAG,EAAE,QAAQ,CAAC,EACnB,MAAM,CAAC,IAAI,GAAG,IAAI,EAClB,OAAO,CACP,CAAC;gBACH,CAAC;gBAED,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,KAAK,EAAE,QAAQ,CAAC,CAAC;gBAC7C,MAAM,KAAK,CAAC,SAAS,EAAE,EAAC,SAAS,EAAE,IAAI,EAAC,CAAC,CAAC;gBAC1C,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;oBACnD,MAAM,OAAO,GAAG,KAAK,KAAK,CAAC,IAAI,OAAO,KAAK,CAAC,WAAW,OAAO,KAAK,CAAC,IAAI,IAAI,CAAC;oBAC7E,MAAM,SAAS,CACd,IAAI,CAAC,SAAS,EAAE,GAAG,GAAG,KAAK,CAAC,EAC5B,OAAO,EACP,OAAO,CACP,CAAC;gBACH,CAAC;gBAED,wCAAwC;gBACxC,MAAM,SAAS,CACd,IAAI,CAAC,GAAG,EAAE,KAAK,EAAE,WAAW,CAAC,EAC7B,iBAAiB,EAAE,EACnB,OAAO,CACP,CAAC;YACH,CAAC;SACD,CAAC;QAEF,OAAO,QAA0B,CAAC;IACnC,CAAC;IAED,SAAS,CAAC,WAA2B,EAAE,KAAiB;QACvD,OAAO,0CAA0C,CAAC;IACnD,CAAC;CACD,CAAC"}
@@ -0,0 +1,4 @@
1
+ import type { CapabilityType, HaiPlugin } from './types.js';
2
+ export declare function registerPlugin(plugin: HaiPlugin): void;
3
+ export declare function getPlugin(id: string): HaiPlugin;
4
+ export declare function getPluginsForCapability(cap: CapabilityType): HaiPlugin[];
@@ -0,0 +1,18 @@
1
+ const plugins = new Map();
2
+ export function registerPlugin(plugin) {
3
+ if (plugins.has(plugin.id)) {
4
+ throw new Error(`Plugin "${plugin.id}" is already registered`);
5
+ }
6
+ plugins.set(plugin.id, plugin);
7
+ }
8
+ export function getPlugin(id) {
9
+ const plugin = plugins.get(id);
10
+ if (!plugin) {
11
+ throw new Error(`Unknown plugin: "${id}". Available: ${[...plugins.keys()].join(', ')}`);
12
+ }
13
+ return plugin;
14
+ }
15
+ export function getPluginsForCapability(cap) {
16
+ return [...plugins.values()].filter(p => p.getCapabilities().includes(cap));
17
+ }
18
+ //# sourceMappingURL=registry.js.map