@build-astron-co/nimbus 0.2.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 (313) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +628 -0
  3. package/bin/nimbus +38 -0
  4. package/package.json +80 -0
  5. package/src/__tests__/app.test.ts +76 -0
  6. package/src/__tests__/audit.test.ts +877 -0
  7. package/src/__tests__/circuit-breaker.test.ts +116 -0
  8. package/src/__tests__/cli-run.test.ts +115 -0
  9. package/src/__tests__/context-manager.test.ts +502 -0
  10. package/src/__tests__/context.test.ts +242 -0
  11. package/src/__tests__/enterprise.test.ts +401 -0
  12. package/src/__tests__/generator.test.ts +433 -0
  13. package/src/__tests__/hooks.test.ts +582 -0
  14. package/src/__tests__/init.test.ts +436 -0
  15. package/src/__tests__/intent-parser.test.ts +229 -0
  16. package/src/__tests__/llm-router.test.ts +209 -0
  17. package/src/__tests__/lsp.test.ts +293 -0
  18. package/src/__tests__/modes.test.ts +336 -0
  19. package/src/__tests__/permissions.test.ts +338 -0
  20. package/src/__tests__/serve.test.ts +275 -0
  21. package/src/__tests__/sessions.test.ts +227 -0
  22. package/src/__tests__/sharing.test.ts +288 -0
  23. package/src/__tests__/snapshots.test.ts +581 -0
  24. package/src/__tests__/state-db.test.ts +334 -0
  25. package/src/__tests__/stream-with-tools.test.ts +732 -0
  26. package/src/__tests__/subagents.test.ts +176 -0
  27. package/src/__tests__/system-prompt.test.ts +169 -0
  28. package/src/__tests__/tool-converter.test.ts +256 -0
  29. package/src/__tests__/tool-schemas.test.ts +397 -0
  30. package/src/__tests__/tools.test.ts +143 -0
  31. package/src/__tests__/version.test.ts +49 -0
  32. package/src/agent/compaction-agent.ts +227 -0
  33. package/src/agent/context-manager.ts +435 -0
  34. package/src/agent/context.ts +427 -0
  35. package/src/agent/deploy-preview.ts +426 -0
  36. package/src/agent/index.ts +68 -0
  37. package/src/agent/loop.ts +717 -0
  38. package/src/agent/modes.ts +429 -0
  39. package/src/agent/permissions.ts +466 -0
  40. package/src/agent/subagents/base.ts +116 -0
  41. package/src/agent/subagents/cost.ts +51 -0
  42. package/src/agent/subagents/explore.ts +42 -0
  43. package/src/agent/subagents/general.ts +54 -0
  44. package/src/agent/subagents/index.ts +102 -0
  45. package/src/agent/subagents/infra.ts +59 -0
  46. package/src/agent/subagents/security.ts +69 -0
  47. package/src/agent/system-prompt.ts +436 -0
  48. package/src/app.ts +122 -0
  49. package/src/audit/activity-log.ts +290 -0
  50. package/src/audit/compliance-checker.ts +540 -0
  51. package/src/audit/cost-tracker.ts +318 -0
  52. package/src/audit/index.ts +23 -0
  53. package/src/audit/security-scanner.ts +596 -0
  54. package/src/auth/guard.ts +75 -0
  55. package/src/auth/index.ts +56 -0
  56. package/src/auth/oauth.ts +455 -0
  57. package/src/auth/providers.ts +470 -0
  58. package/src/auth/sso.ts +113 -0
  59. package/src/auth/store.ts +505 -0
  60. package/src/auth/types.ts +187 -0
  61. package/src/build.ts +141 -0
  62. package/src/cli/index.ts +16 -0
  63. package/src/cli/init.ts +854 -0
  64. package/src/cli/openapi-spec.ts +356 -0
  65. package/src/cli/run.ts +237 -0
  66. package/src/cli/serve-auth.ts +80 -0
  67. package/src/cli/serve.ts +462 -0
  68. package/src/cli/web.ts +67 -0
  69. package/src/cli.ts +1417 -0
  70. package/src/clients/core-engine-client.ts +227 -0
  71. package/src/clients/enterprise-client.ts +334 -0
  72. package/src/clients/generator-client.ts +351 -0
  73. package/src/clients/git-client.ts +627 -0
  74. package/src/clients/github-client.ts +410 -0
  75. package/src/clients/helm-client.ts +504 -0
  76. package/src/clients/index.ts +80 -0
  77. package/src/clients/k8s-client.ts +497 -0
  78. package/src/clients/llm-client.ts +161 -0
  79. package/src/clients/rest-client.ts +130 -0
  80. package/src/clients/service-discovery.ts +33 -0
  81. package/src/clients/terraform-client.ts +482 -0
  82. package/src/clients/tools-client.ts +1843 -0
  83. package/src/clients/ws-client.ts +115 -0
  84. package/src/commands/analyze/index.ts +352 -0
  85. package/src/commands/apply/helm.ts +473 -0
  86. package/src/commands/apply/index.ts +213 -0
  87. package/src/commands/apply/k8s.ts +454 -0
  88. package/src/commands/apply/terraform.ts +582 -0
  89. package/src/commands/ask.ts +167 -0
  90. package/src/commands/audit/index.ts +238 -0
  91. package/src/commands/auth-cloud.ts +294 -0
  92. package/src/commands/auth-list.ts +134 -0
  93. package/src/commands/auth-profile.ts +121 -0
  94. package/src/commands/auth-status.ts +141 -0
  95. package/src/commands/aws/ec2.ts +501 -0
  96. package/src/commands/aws/iam.ts +397 -0
  97. package/src/commands/aws/index.ts +133 -0
  98. package/src/commands/aws/lambda.ts +396 -0
  99. package/src/commands/aws/rds.ts +439 -0
  100. package/src/commands/aws/s3.ts +439 -0
  101. package/src/commands/aws/vpc.ts +393 -0
  102. package/src/commands/aws-discover.ts +649 -0
  103. package/src/commands/aws-terraform.ts +805 -0
  104. package/src/commands/azure/aks.ts +376 -0
  105. package/src/commands/azure/functions.ts +253 -0
  106. package/src/commands/azure/index.ts +116 -0
  107. package/src/commands/azure/storage.ts +478 -0
  108. package/src/commands/azure/vm.ts +355 -0
  109. package/src/commands/billing/index.ts +256 -0
  110. package/src/commands/chat.ts +314 -0
  111. package/src/commands/config.ts +346 -0
  112. package/src/commands/cost/cloud-cost-estimator.ts +266 -0
  113. package/src/commands/cost/estimator.ts +79 -0
  114. package/src/commands/cost/index.ts +594 -0
  115. package/src/commands/cost/parsers/terraform.ts +273 -0
  116. package/src/commands/cost/parsers/types.ts +25 -0
  117. package/src/commands/cost/pricing/aws.ts +544 -0
  118. package/src/commands/cost/pricing/azure.ts +499 -0
  119. package/src/commands/cost/pricing/gcp.ts +396 -0
  120. package/src/commands/cost/pricing/index.ts +40 -0
  121. package/src/commands/demo.ts +250 -0
  122. package/src/commands/doctor.ts +794 -0
  123. package/src/commands/drift/index.ts +439 -0
  124. package/src/commands/explain.ts +277 -0
  125. package/src/commands/feedback.ts +389 -0
  126. package/src/commands/fix.ts +324 -0
  127. package/src/commands/fs/index.ts +402 -0
  128. package/src/commands/gcp/compute.ts +325 -0
  129. package/src/commands/gcp/functions.ts +271 -0
  130. package/src/commands/gcp/gke.ts +438 -0
  131. package/src/commands/gcp/iam.ts +344 -0
  132. package/src/commands/gcp/index.ts +129 -0
  133. package/src/commands/gcp/storage.ts +284 -0
  134. package/src/commands/generate-helm.ts +1249 -0
  135. package/src/commands/generate-k8s.ts +1560 -0
  136. package/src/commands/generate-terraform.ts +1460 -0
  137. package/src/commands/gh/index.ts +863 -0
  138. package/src/commands/git/index.ts +1343 -0
  139. package/src/commands/helm/index.ts +1126 -0
  140. package/src/commands/help.ts +539 -0
  141. package/src/commands/history.ts +142 -0
  142. package/src/commands/import.ts +868 -0
  143. package/src/commands/index.ts +367 -0
  144. package/src/commands/init.ts +1046 -0
  145. package/src/commands/k8s/index.ts +1137 -0
  146. package/src/commands/login.ts +631 -0
  147. package/src/commands/logout.ts +83 -0
  148. package/src/commands/onboarding.ts +228 -0
  149. package/src/commands/plan/display.ts +279 -0
  150. package/src/commands/plan/index.ts +599 -0
  151. package/src/commands/preview.ts +452 -0
  152. package/src/commands/questionnaire.ts +1270 -0
  153. package/src/commands/resume.ts +55 -0
  154. package/src/commands/team/index.ts +346 -0
  155. package/src/commands/template.ts +232 -0
  156. package/src/commands/tf/index.ts +1034 -0
  157. package/src/commands/upgrade.ts +550 -0
  158. package/src/commands/usage/index.ts +134 -0
  159. package/src/commands/version.ts +170 -0
  160. package/src/compat/index.ts +2 -0
  161. package/src/compat/runtime.ts +12 -0
  162. package/src/compat/sqlite.ts +107 -0
  163. package/src/config/index.ts +17 -0
  164. package/src/config/manager.ts +530 -0
  165. package/src/config/safety-policy.ts +358 -0
  166. package/src/config/schema.ts +125 -0
  167. package/src/config/types.ts +527 -0
  168. package/src/context/context-db.ts +199 -0
  169. package/src/demo/index.ts +349 -0
  170. package/src/demo/scenarios/full-journey.ts +229 -0
  171. package/src/demo/scenarios/getting-started.ts +127 -0
  172. package/src/demo/scenarios/helm-release.ts +341 -0
  173. package/src/demo/scenarios/k8s-deployment.ts +194 -0
  174. package/src/demo/scenarios/terraform-vpc.ts +170 -0
  175. package/src/demo/types.ts +92 -0
  176. package/src/engine/cost-estimator.ts +438 -0
  177. package/src/engine/diagram-generator.ts +256 -0
  178. package/src/engine/drift-detector.ts +902 -0
  179. package/src/engine/executor.ts +1035 -0
  180. package/src/engine/index.ts +76 -0
  181. package/src/engine/orchestrator.ts +636 -0
  182. package/src/engine/planner.ts +720 -0
  183. package/src/engine/safety.ts +743 -0
  184. package/src/engine/verifier.ts +770 -0
  185. package/src/enterprise/audit.ts +348 -0
  186. package/src/enterprise/auth.ts +270 -0
  187. package/src/enterprise/billing.ts +822 -0
  188. package/src/enterprise/index.ts +17 -0
  189. package/src/enterprise/teams.ts +443 -0
  190. package/src/generator/best-practices.ts +1608 -0
  191. package/src/generator/helm.ts +630 -0
  192. package/src/generator/index.ts +37 -0
  193. package/src/generator/intent-parser.ts +514 -0
  194. package/src/generator/kubernetes.ts +976 -0
  195. package/src/generator/terraform.ts +1867 -0
  196. package/src/history/index.ts +8 -0
  197. package/src/history/manager.ts +322 -0
  198. package/src/history/types.ts +34 -0
  199. package/src/hooks/config.ts +432 -0
  200. package/src/hooks/engine.ts +391 -0
  201. package/src/hooks/index.ts +4 -0
  202. package/src/llm/auth-bridge.ts +198 -0
  203. package/src/llm/circuit-breaker.ts +140 -0
  204. package/src/llm/config-loader.ts +201 -0
  205. package/src/llm/cost-calculator.ts +171 -0
  206. package/src/llm/index.ts +8 -0
  207. package/src/llm/model-aliases.ts +115 -0
  208. package/src/llm/provider-registry.ts +63 -0
  209. package/src/llm/providers/anthropic.ts +433 -0
  210. package/src/llm/providers/bedrock.ts +477 -0
  211. package/src/llm/providers/google.ts +405 -0
  212. package/src/llm/providers/ollama.ts +767 -0
  213. package/src/llm/providers/openai-compatible.ts +340 -0
  214. package/src/llm/providers/openai.ts +328 -0
  215. package/src/llm/providers/openrouter.ts +338 -0
  216. package/src/llm/router.ts +1035 -0
  217. package/src/llm/types.ts +232 -0
  218. package/src/lsp/client.ts +298 -0
  219. package/src/lsp/languages.ts +116 -0
  220. package/src/lsp/manager.ts +278 -0
  221. package/src/mcp/client.ts +402 -0
  222. package/src/mcp/index.ts +5 -0
  223. package/src/mcp/manager.ts +133 -0
  224. package/src/nimbus.ts +214 -0
  225. package/src/plugins/index.ts +27 -0
  226. package/src/plugins/loader.ts +334 -0
  227. package/src/plugins/manager.ts +376 -0
  228. package/src/plugins/types.ts +284 -0
  229. package/src/scanners/cicd-scanner.ts +258 -0
  230. package/src/scanners/cloud-scanner.ts +466 -0
  231. package/src/scanners/framework-scanner.ts +469 -0
  232. package/src/scanners/iac-scanner.ts +388 -0
  233. package/src/scanners/index.ts +539 -0
  234. package/src/scanners/language-scanner.ts +276 -0
  235. package/src/scanners/package-manager-scanner.ts +277 -0
  236. package/src/scanners/types.ts +172 -0
  237. package/src/sessions/manager.ts +365 -0
  238. package/src/sessions/types.ts +44 -0
  239. package/src/sharing/sync.ts +296 -0
  240. package/src/sharing/viewer.ts +97 -0
  241. package/src/snapshots/index.ts +2 -0
  242. package/src/snapshots/manager.ts +530 -0
  243. package/src/state/artifacts.ts +147 -0
  244. package/src/state/audit.ts +137 -0
  245. package/src/state/billing.ts +240 -0
  246. package/src/state/checkpoints.ts +117 -0
  247. package/src/state/config.ts +67 -0
  248. package/src/state/conversations.ts +14 -0
  249. package/src/state/credentials.ts +154 -0
  250. package/src/state/db.ts +58 -0
  251. package/src/state/index.ts +26 -0
  252. package/src/state/messages.ts +115 -0
  253. package/src/state/projects.ts +123 -0
  254. package/src/state/schema.ts +236 -0
  255. package/src/state/sessions.ts +147 -0
  256. package/src/state/teams.ts +200 -0
  257. package/src/telemetry.ts +108 -0
  258. package/src/tools/aws-ops.ts +952 -0
  259. package/src/tools/azure-ops.ts +579 -0
  260. package/src/tools/file-ops.ts +593 -0
  261. package/src/tools/gcp-ops.ts +625 -0
  262. package/src/tools/git-ops.ts +773 -0
  263. package/src/tools/github-ops.ts +799 -0
  264. package/src/tools/helm-ops.ts +943 -0
  265. package/src/tools/index.ts +17 -0
  266. package/src/tools/k8s-ops.ts +819 -0
  267. package/src/tools/schemas/converter.ts +184 -0
  268. package/src/tools/schemas/devops.ts +612 -0
  269. package/src/tools/schemas/index.ts +73 -0
  270. package/src/tools/schemas/standard.ts +1144 -0
  271. package/src/tools/schemas/types.ts +705 -0
  272. package/src/tools/terraform-ops.ts +862 -0
  273. package/src/types/ambient.d.ts +193 -0
  274. package/src/types/config.ts +83 -0
  275. package/src/types/drift.ts +116 -0
  276. package/src/types/enterprise.ts +335 -0
  277. package/src/types/index.ts +20 -0
  278. package/src/types/plan.ts +44 -0
  279. package/src/types/request.ts +65 -0
  280. package/src/types/response.ts +54 -0
  281. package/src/types/service.ts +51 -0
  282. package/src/ui/App.tsx +997 -0
  283. package/src/ui/DeployPreview.tsx +169 -0
  284. package/src/ui/Header.tsx +68 -0
  285. package/src/ui/InputBox.tsx +350 -0
  286. package/src/ui/MessageList.tsx +585 -0
  287. package/src/ui/PermissionPrompt.tsx +151 -0
  288. package/src/ui/StatusBar.tsx +158 -0
  289. package/src/ui/ToolCallDisplay.tsx +409 -0
  290. package/src/ui/chat-ui.ts +853 -0
  291. package/src/ui/index.ts +33 -0
  292. package/src/ui/ink/index.ts +711 -0
  293. package/src/ui/streaming.ts +176 -0
  294. package/src/ui/types.ts +57 -0
  295. package/src/utils/analytics.ts +72 -0
  296. package/src/utils/cost-warning.ts +27 -0
  297. package/src/utils/env.ts +46 -0
  298. package/src/utils/errors.ts +69 -0
  299. package/src/utils/event-bus.ts +38 -0
  300. package/src/utils/index.ts +24 -0
  301. package/src/utils/logger.ts +171 -0
  302. package/src/utils/rate-limiter.ts +121 -0
  303. package/src/utils/service-auth.ts +49 -0
  304. package/src/utils/validation.ts +53 -0
  305. package/src/version.ts +4 -0
  306. package/src/watcher/index.ts +163 -0
  307. package/src/wizard/approval.ts +383 -0
  308. package/src/wizard/index.ts +25 -0
  309. package/src/wizard/prompts.ts +338 -0
  310. package/src/wizard/types.ts +171 -0
  311. package/src/wizard/ui.ts +556 -0
  312. package/src/wizard/wizard.ts +304 -0
  313. package/tsconfig.json +24 -0
@@ -0,0 +1,389 @@
1
+ /**
2
+ * Feedback Command
3
+ *
4
+ * Collect and submit user feedback via GitHub issues or browser
5
+ *
6
+ * Usage: nimbus feedback [options]
7
+ */
8
+
9
+ import { logger } from '../utils';
10
+ import { ui } from '../wizard';
11
+ import { input, select } from '../wizard/prompts';
12
+
13
+ /**
14
+ * Command options
15
+ */
16
+ export interface FeedbackOptions {
17
+ bug?: boolean;
18
+ feature?: boolean;
19
+ question?: boolean;
20
+ title?: string;
21
+ body?: string;
22
+ open?: boolean;
23
+ json?: boolean;
24
+ }
25
+
26
+ /**
27
+ * Feedback type configuration
28
+ */
29
+ interface FeedbackType {
30
+ label: string;
31
+ emoji: string;
32
+ template: string;
33
+ labels: string[];
34
+ }
35
+
36
+ const FEEDBACK_TYPES: Record<string, FeedbackType> = {
37
+ bug: {
38
+ label: 'Bug Report',
39
+ emoji: 'bug',
40
+ template: `## Bug Description
41
+ <!-- A clear and concise description of what the bug is -->
42
+
43
+ ## Steps to Reproduce
44
+ 1.
45
+ 2.
46
+ 3.
47
+
48
+ ## Expected Behavior
49
+ <!-- What you expected to happen -->
50
+
51
+ ## Actual Behavior
52
+ <!-- What actually happened -->
53
+
54
+ ## Environment
55
+ - Nimbus Version:
56
+ - OS:
57
+ - Node Version:
58
+
59
+ ## Additional Context
60
+ <!-- Any other relevant information -->
61
+ `,
62
+ labels: ['bug', 'triage'],
63
+ },
64
+ feature: {
65
+ label: 'Feature Request',
66
+ emoji: 'sparkles',
67
+ template: `## Feature Description
68
+ <!-- A clear and concise description of the feature -->
69
+
70
+ ## Use Case
71
+ <!-- Why do you need this feature? What problem does it solve? -->
72
+
73
+ ## Proposed Solution
74
+ <!-- How do you think this should work? -->
75
+
76
+ ## Alternatives Considered
77
+ <!-- Any alternative solutions or features you've considered -->
78
+
79
+ ## Additional Context
80
+ <!-- Any other relevant information -->
81
+ `,
82
+ labels: ['enhancement', 'feature-request'],
83
+ },
84
+ question: {
85
+ label: 'Question',
86
+ emoji: 'question',
87
+ template: `## Question
88
+ <!-- What would you like to know? -->
89
+
90
+ ## Context
91
+ <!-- What are you trying to accomplish? -->
92
+
93
+ ## What I've Tried
94
+ <!-- What documentation or approaches have you already tried? -->
95
+ `,
96
+ labels: ['question', 'help-wanted'],
97
+ },
98
+ };
99
+
100
+ const GITHUB_REPO = 'the-ai-project-co/nimbus';
101
+ const GITHUB_ISSUES_URL = `https://github.com/${GITHUB_REPO}/issues`;
102
+
103
+ /**
104
+ * Get system information for bug reports
105
+ */
106
+ async function getSystemInfo(): Promise<Record<string, string>> {
107
+ const os = await import('os');
108
+ const { execFileSync } = await import('child_process');
109
+
110
+ const info: Record<string, string> = {
111
+ OS: `${os.platform()} ${os.release()}`,
112
+ Architecture: os.arch(),
113
+ 'Node Version': process.version,
114
+ };
115
+
116
+ // Try to get Nimbus version
117
+ try {
118
+ const fs = await import('fs/promises');
119
+ const path = await import('path');
120
+ const packagePath = path.join(__dirname, '../../package.json');
121
+ const packageJson = JSON.parse(await fs.readFile(packagePath, 'utf-8'));
122
+ info['Nimbus Version'] = packageJson.version;
123
+ } catch {
124
+ info['Nimbus Version'] = 'unknown';
125
+ }
126
+
127
+ // Try to get Bun version
128
+ try {
129
+ const bunVersion = execFileSync('bun', ['--version'], { encoding: 'utf-8' }).trim();
130
+ info['Bun Version'] = bunVersion;
131
+ } catch {
132
+ // Bun not installed
133
+ }
134
+
135
+ return info;
136
+ }
137
+
138
+ /**
139
+ * Open URL in browser
140
+ */
141
+ async function openInBrowser(url: string): Promise<boolean> {
142
+ const { exec } = await import('child_process');
143
+ const { promisify } = await import('util');
144
+ const execAsync = promisify(exec);
145
+
146
+ try {
147
+ const platform = process.platform;
148
+ let command: string;
149
+
150
+ if (platform === 'darwin') {
151
+ command = `open "${url}"`;
152
+ } else if (platform === 'win32') {
153
+ command = `start "" "${url}"`;
154
+ } else {
155
+ command = `xdg-open "${url}"`;
156
+ }
157
+
158
+ await execAsync(command);
159
+ return true;
160
+ } catch {
161
+ return false;
162
+ }
163
+ }
164
+
165
+ /**
166
+ * Build GitHub issue URL with pre-filled content
167
+ */
168
+ function buildIssueUrl(type: string, title: string, body: string, labels: string[]): string {
169
+ const params = new URLSearchParams({
170
+ title,
171
+ body,
172
+ labels: labels.join(','),
173
+ });
174
+
175
+ return `${GITHUB_ISSUES_URL}/new?${params.toString()}`;
176
+ }
177
+
178
+ /**
179
+ * Interactive feedback collection
180
+ */
181
+ async function collectFeedbackInteractively(
182
+ options: FeedbackOptions
183
+ ): Promise<{ type: string; title: string; body: string } | null> {
184
+ // Determine feedback type
185
+ let feedbackType: string | undefined;
186
+
187
+ if (options.bug) {
188
+ feedbackType = 'bug';
189
+ } else if (options.feature) {
190
+ feedbackType = 'feature';
191
+ } else if (options.question) {
192
+ feedbackType = 'question';
193
+ } else {
194
+ // Ask user to select type
195
+ feedbackType = await select({
196
+ message: 'What type of feedback would you like to provide?',
197
+ options: [
198
+ { label: 'Bug Report', value: 'bug' },
199
+ { label: 'Feature Request', value: 'feature' },
200
+ { label: 'Question', value: 'question' },
201
+ ],
202
+ });
203
+
204
+ if (!feedbackType) {
205
+ return null;
206
+ }
207
+ }
208
+
209
+ const typeConfig = FEEDBACK_TYPES[feedbackType];
210
+
211
+ ui.newLine();
212
+ ui.print(ui.bold(`${typeConfig.label}`));
213
+ ui.newLine();
214
+
215
+ // Get title
216
+ let title = options.title;
217
+ if (!title) {
218
+ title = await input({
219
+ message: 'Brief summary (title):',
220
+ placeholder: `Enter a short description of your ${feedbackType === 'bug' ? 'issue' : feedbackType === 'feature' ? 'request' : 'question'}`,
221
+ });
222
+
223
+ if (!title || title.trim() === '') {
224
+ ui.warning('Feedback cancelled - no title provided');
225
+ return null;
226
+ }
227
+ }
228
+
229
+ // Get body/description
230
+ let body = options.body;
231
+ if (!body) {
232
+ ui.newLine();
233
+ ui.info('Please provide details (press Enter twice to finish):');
234
+ ui.dim('Tip: You can also edit the full template on GitHub');
235
+ ui.newLine();
236
+
237
+ body = await input({
238
+ message: 'Details (optional):',
239
+ placeholder: 'Describe the issue, feature, or question in detail...',
240
+ });
241
+ }
242
+
243
+ // Build full body with template
244
+ let fullBody = typeConfig.template;
245
+
246
+ if (body && body.trim()) {
247
+ // Replace first section placeholder with user's input
248
+ fullBody = fullBody.replace(/<!-- .+? -->/, body.trim());
249
+ }
250
+
251
+ // Add system info for bug reports
252
+ if (feedbackType === 'bug') {
253
+ const systemInfo = await getSystemInfo();
254
+ const envSection = Object.entries(systemInfo)
255
+ .map(([key, value]) => `- ${key}: ${value}`)
256
+ .join('\n');
257
+
258
+ fullBody = fullBody.replace(
259
+ '## Environment\n- Nimbus Version: \n- OS: \n- Node Version:',
260
+ `## Environment\n${envSection}`
261
+ );
262
+ }
263
+
264
+ return {
265
+ type: feedbackType,
266
+ title: title.trim(),
267
+ body: fullBody,
268
+ };
269
+ }
270
+
271
+ /**
272
+ * Run the feedback command
273
+ */
274
+ export async function feedbackCommand(options: FeedbackOptions = {}): Promise<void> {
275
+ logger.debug('Running feedback command', { options });
276
+
277
+ ui.header('Nimbus Feedback');
278
+ ui.info('Help us improve Nimbus by sharing your feedback!');
279
+ ui.newLine();
280
+
281
+ // Quick open mode - just open GitHub issues
282
+ if (options.open) {
283
+ ui.info('Opening GitHub issues page...');
284
+ const opened = await openInBrowser(GITHUB_ISSUES_URL);
285
+
286
+ if (opened) {
287
+ ui.success('Opened in browser');
288
+ } else {
289
+ ui.print(`Visit: ${GITHUB_ISSUES_URL}`);
290
+ }
291
+ return;
292
+ }
293
+
294
+ // Collect feedback interactively
295
+ const feedback = await collectFeedbackInteractively(options);
296
+
297
+ if (!feedback) {
298
+ return;
299
+ }
300
+
301
+ const typeConfig = FEEDBACK_TYPES[feedback.type];
302
+ const issueUrl = buildIssueUrl(feedback.type, feedback.title, feedback.body, typeConfig.labels);
303
+
304
+ // JSON output
305
+ if (options.json) {
306
+ console.log(
307
+ JSON.stringify(
308
+ {
309
+ type: feedback.type,
310
+ title: feedback.title,
311
+ url: issueUrl,
312
+ labels: typeConfig.labels,
313
+ },
314
+ null,
315
+ 2
316
+ )
317
+ );
318
+ return;
319
+ }
320
+
321
+ ui.newLine();
322
+ ui.print(ui.bold('Feedback Summary'));
323
+ ui.print(` Type: ${typeConfig.label}`);
324
+ ui.print(` Title: ${feedback.title}`);
325
+ ui.newLine();
326
+
327
+ // Open in browser
328
+ ui.info('Opening GitHub to submit your feedback...');
329
+
330
+ const opened = await openInBrowser(issueUrl);
331
+
332
+ if (opened) {
333
+ ui.success('Opened in browser - please review and submit the issue');
334
+ } else {
335
+ ui.warning('Could not open browser automatically');
336
+ ui.newLine();
337
+ ui.print('Please copy and paste this URL to submit your feedback:');
338
+ ui.print(issueUrl);
339
+ }
340
+
341
+ ui.newLine();
342
+ ui.info('Thank you for your feedback!');
343
+ }
344
+
345
+ /**
346
+ * Parse feedback command options from CLI args
347
+ */
348
+ export function parseFeedbackOptions(args: string[]): FeedbackOptions {
349
+ const options: FeedbackOptions = {};
350
+
351
+ for (let i = 0; i < args.length; i++) {
352
+ const arg = args[i];
353
+
354
+ switch (arg) {
355
+ case '--bug':
356
+ case '-b':
357
+ options.bug = true;
358
+ break;
359
+ case '--feature':
360
+ case '-f':
361
+ options.feature = true;
362
+ break;
363
+ case '--question':
364
+ case '-q':
365
+ options.question = true;
366
+ break;
367
+ case '--title':
368
+ case '-t':
369
+ options.title = args[++i];
370
+ break;
371
+ case '--body':
372
+ case '-m':
373
+ options.body = args[++i];
374
+ break;
375
+ case '--open':
376
+ case '-o':
377
+ options.open = true;
378
+ break;
379
+ case '--json':
380
+ options.json = true;
381
+ break;
382
+ }
383
+ }
384
+
385
+ return options;
386
+ }
387
+
388
+ // Export as default command
389
+ export default feedbackCommand;
@@ -0,0 +1,324 @@
1
+ /**
2
+ * Fix Command
3
+ *
4
+ * AI-assisted error fixing
5
+ *
6
+ * Usage: nimbus fix <error-or-file> [options]
7
+ */
8
+
9
+ import { logger } from '../utils';
10
+ import { ui, confirm } from '../wizard';
11
+ import { llmClient } from '../clients';
12
+
13
+ /**
14
+ * Command options
15
+ */
16
+ export interface FixOptions {
17
+ file?: string;
18
+ autoApply?: boolean;
19
+ dryRun?: boolean;
20
+ json?: boolean;
21
+ }
22
+
23
+ /**
24
+ * Fix suggestion from AI
25
+ */
26
+ interface FixSuggestion {
27
+ problem: string;
28
+ explanation: string;
29
+ fix: string;
30
+ originalCode?: string;
31
+ fixedCode?: string;
32
+ filePath?: string;
33
+ lineNumber?: number;
34
+ }
35
+
36
+ /**
37
+ * Parse fix response from AI
38
+ */
39
+ function parseFixResponse(response: string): FixSuggestion {
40
+ // Try to extract structured sections from the response
41
+ const problemMatch = response.match(
42
+ /(?:problem|issue|error):\s*(.+?)(?=\n(?:explanation|fix|solution)|$)/is
43
+ );
44
+ const explanationMatch = response.match(
45
+ /(?:explanation|cause|reason):\s*(.+?)(?=\n(?:fix|solution)|$)/is
46
+ );
47
+ const fixMatch = response.match(
48
+ /(?:fix|solution|resolution):\s*(.+?)(?=\n(?:original|fixed)|$)/is
49
+ );
50
+ const originalMatch = response.match(/(?:original|before)[^:]*:\s*```[\w]*\n([\s\S]*?)```/i);
51
+ const fixedMatch = response.match(/(?:fixed|after|corrected)[^:]*:\s*```[\w]*\n([\s\S]*?)```/i);
52
+
53
+ return {
54
+ problem: problemMatch?.[1]?.trim() || 'Unable to parse problem description',
55
+ explanation: explanationMatch?.[1]?.trim() || response.split('\n').slice(0, 3).join('\n'),
56
+ fix: fixMatch?.[1]?.trim() || 'See suggested code below',
57
+ originalCode: originalMatch?.[1]?.trim(),
58
+ fixedCode: fixedMatch?.[1]?.trim(),
59
+ };
60
+ }
61
+
62
+ /**
63
+ * Build the fix prompt
64
+ */
65
+ function buildFixPrompt(errorContent: string, fileContent?: string, filePath?: string): string {
66
+ let prompt = `Please help fix this error. Analyze the problem and provide a solution.
67
+
68
+ Error:
69
+ \`\`\`
70
+ ${errorContent}
71
+ \`\`\`
72
+ `;
73
+
74
+ if (fileContent && filePath) {
75
+ prompt += `
76
+ Source file (${filePath}):
77
+ \`\`\`
78
+ ${fileContent}
79
+ \`\`\`
80
+ `;
81
+ }
82
+
83
+ prompt += `
84
+ Please provide:
85
+ 1. **Problem**: A brief description of what's wrong
86
+ 2. **Explanation**: Why this error occurs
87
+ 3. **Fix**: How to fix it
88
+
89
+ If you can provide code changes, please show:
90
+ - **Original**: The problematic code
91
+ - **Fixed**: The corrected code
92
+
93
+ Format your response with clear section headers.`;
94
+
95
+ return prompt;
96
+ }
97
+
98
+ /**
99
+ * Display fix suggestion
100
+ */
101
+ function displayFixSuggestion(suggestion: FixSuggestion): void {
102
+ ui.newLine();
103
+
104
+ // Problem
105
+ ui.print(ui.color('Problem:', 'yellow'));
106
+ ui.print(` ${suggestion.problem}`);
107
+ ui.newLine();
108
+
109
+ // Explanation
110
+ ui.print(ui.color('Explanation:', 'blue'));
111
+ for (const line of suggestion.explanation.split('\n')) {
112
+ ui.print(` ${line}`);
113
+ }
114
+ ui.newLine();
115
+
116
+ // Fix
117
+ ui.print(ui.color('Suggested Fix:', 'green'));
118
+ for (const line of suggestion.fix.split('\n')) {
119
+ ui.print(` ${line}`);
120
+ }
121
+
122
+ // Show diff if we have original and fixed code
123
+ if (suggestion.originalCode && suggestion.fixedCode) {
124
+ ui.newLine();
125
+ ui.print(ui.color('Code Changes:', 'cyan'));
126
+ ui.newLine();
127
+
128
+ // Show original
129
+ ui.print(ui.color('Before:', 'red'));
130
+ ui.print('```');
131
+ for (const line of suggestion.originalCode.split('\n')) {
132
+ ui.print(ui.color(`- ${line}`, 'red'));
133
+ }
134
+ ui.print('```');
135
+
136
+ ui.newLine();
137
+
138
+ // Show fixed
139
+ ui.print(ui.color('After:', 'green'));
140
+ ui.print('```');
141
+ for (const line of suggestion.fixedCode.split('\n')) {
142
+ ui.print(ui.color(`+ ${line}`, 'green'));
143
+ }
144
+ ui.print('```');
145
+ }
146
+ }
147
+
148
+ /**
149
+ * Apply the fix to a file
150
+ */
151
+ async function applyFix(suggestion: FixSuggestion, filePath: string): Promise<boolean> {
152
+ if (!suggestion.originalCode || !suggestion.fixedCode) {
153
+ ui.warning('Cannot auto-apply: No code diff provided');
154
+ return false;
155
+ }
156
+
157
+ try {
158
+ const fs = await import('fs/promises');
159
+ const content = await fs.readFile(filePath, 'utf-8');
160
+
161
+ // Try to find and replace the original code
162
+ if (content.includes(suggestion.originalCode)) {
163
+ const newContent = content.replace(suggestion.originalCode, suggestion.fixedCode);
164
+ await fs.writeFile(filePath, newContent, 'utf-8');
165
+ return true;
166
+ } else {
167
+ ui.warning('Could not find the original code in the file');
168
+ ui.info('The file may have been modified since the analysis');
169
+ return false;
170
+ }
171
+ } catch (error: any) {
172
+ ui.error(`Failed to apply fix: ${error.message}`);
173
+ return false;
174
+ }
175
+ }
176
+
177
+ /**
178
+ * Run the fix command
179
+ */
180
+ export async function fixCommand(errorOrFile: string, options: FixOptions = {}): Promise<void> {
181
+ logger.info('Running fix command', { errorOrFile, options });
182
+
183
+ let errorContent: string;
184
+ let fileContent: string | undefined;
185
+ let filePath: string | undefined;
186
+
187
+ // Determine what we're fixing
188
+ if (options.file) {
189
+ // Error content with explicit file
190
+ errorContent = errorOrFile;
191
+ filePath = options.file;
192
+
193
+ try {
194
+ const fs = await import('fs/promises');
195
+ fileContent = await fs.readFile(filePath, 'utf-8');
196
+ } catch (error: any) {
197
+ ui.warning(`Could not read file ${filePath}: ${error.message}`);
198
+ }
199
+ } else if (errorOrFile) {
200
+ // Check if it's a file path
201
+ try {
202
+ const fs = await import('fs/promises');
203
+ const stat = await fs.stat(errorOrFile);
204
+
205
+ if (stat.isFile()) {
206
+ filePath = errorOrFile;
207
+ fileContent = await fs.readFile(errorOrFile, 'utf-8');
208
+
209
+ // For a file without explicit error, we'll analyze the whole file
210
+ errorContent = `Please analyze this file for potential issues and errors:\n${errorOrFile}`;
211
+ } else {
212
+ errorContent = errorOrFile;
213
+ }
214
+ } catch {
215
+ // Not a file, treat as error message
216
+ errorContent = errorOrFile;
217
+ }
218
+ } else {
219
+ ui.error('Please provide an error message or file to fix');
220
+ ui.newLine();
221
+ ui.print('Usage: nimbus fix <error-or-file> [options]');
222
+ ui.newLine();
223
+ ui.print('Examples:');
224
+ ui.print(' nimbus fix "Error: undefined variable"');
225
+ ui.print(' nimbus fix ./broken.tf');
226
+ ui.print(' nimbus fix "Error: invalid syntax" --file ./app.py');
227
+ ui.print(' nimbus fix ./config.yaml --auto-apply');
228
+ process.exit(1);
229
+ }
230
+
231
+ // Display header
232
+ ui.header('Nimbus Fix');
233
+ if (filePath) {
234
+ ui.info(`File: ${filePath}`);
235
+ }
236
+ ui.info(`Error: ${errorContent.slice(0, 100)}${errorContent.length > 100 ? '...' : ''}`);
237
+ ui.newLine();
238
+
239
+ // Check if LLM is available
240
+ const llmAvailable = await llmClient.isAvailable();
241
+
242
+ if (!llmAvailable) {
243
+ ui.error('LLM service is not available');
244
+ ui.info('Make sure you have configured an LLM provider with "nimbus login"');
245
+ process.exit(1);
246
+ }
247
+
248
+ // Build prompt
249
+ const prompt = buildFixPrompt(errorContent, fileContent, filePath);
250
+
251
+ ui.startSpinner({ message: 'Analyzing error...' });
252
+
253
+ try {
254
+ let response = '';
255
+
256
+ for await (const chunk of llmClient.chat(prompt, [])) {
257
+ if (chunk.type === 'content' && chunk.content) {
258
+ response += chunk.content;
259
+ } else if (chunk.type === 'error') {
260
+ ui.stopSpinnerFail('Error');
261
+ ui.error(chunk.message || chunk.error || 'Unknown error');
262
+ process.exit(1);
263
+ }
264
+ }
265
+
266
+ ui.stopSpinnerSuccess('Analysis complete');
267
+
268
+ // Parse and display the suggestion
269
+ const suggestion = parseFixResponse(response);
270
+ suggestion.filePath = filePath;
271
+
272
+ displayFixSuggestion(suggestion);
273
+
274
+ // JSON output mode
275
+ if (options.json) {
276
+ console.log(JSON.stringify(suggestion, null, 2));
277
+ return;
278
+ }
279
+
280
+ // Dry run - don't apply
281
+ if (options.dryRun) {
282
+ ui.newLine();
283
+ ui.info('Dry run mode - no changes applied');
284
+ return;
285
+ }
286
+
287
+ // Apply the fix if requested and possible
288
+ if (filePath && suggestion.originalCode && suggestion.fixedCode) {
289
+ ui.newLine();
290
+
291
+ const shouldApply =
292
+ options.autoApply ||
293
+ (await confirm({
294
+ message: 'Apply this fix?',
295
+ defaultValue: false,
296
+ }));
297
+
298
+ if (shouldApply) {
299
+ ui.startSpinner({ message: 'Applying fix...' });
300
+
301
+ const applied = await applyFix(suggestion, filePath);
302
+
303
+ if (applied) {
304
+ ui.stopSpinnerSuccess('Fix applied successfully!');
305
+ ui.newLine();
306
+ ui.info(`File updated: ${filePath}`);
307
+ ui.info('Please review the changes and test your code');
308
+ } else {
309
+ ui.stopSpinnerFail('Could not apply fix automatically');
310
+ ui.info('Please apply the suggested changes manually');
311
+ }
312
+ } else {
313
+ ui.info('Fix not applied');
314
+ }
315
+ }
316
+ } catch (error: any) {
317
+ ui.stopSpinnerFail('Failed');
318
+ ui.error(error.message);
319
+ process.exit(1);
320
+ }
321
+ }
322
+
323
+ // Export as default
324
+ export default fixCommand;