@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.
- package/LICENSE +21 -0
- package/README.md +628 -0
- package/bin/nimbus +38 -0
- package/package.json +80 -0
- package/src/__tests__/app.test.ts +76 -0
- package/src/__tests__/audit.test.ts +877 -0
- package/src/__tests__/circuit-breaker.test.ts +116 -0
- package/src/__tests__/cli-run.test.ts +115 -0
- package/src/__tests__/context-manager.test.ts +502 -0
- package/src/__tests__/context.test.ts +242 -0
- package/src/__tests__/enterprise.test.ts +401 -0
- package/src/__tests__/generator.test.ts +433 -0
- package/src/__tests__/hooks.test.ts +582 -0
- package/src/__tests__/init.test.ts +436 -0
- package/src/__tests__/intent-parser.test.ts +229 -0
- package/src/__tests__/llm-router.test.ts +209 -0
- package/src/__tests__/lsp.test.ts +293 -0
- package/src/__tests__/modes.test.ts +336 -0
- package/src/__tests__/permissions.test.ts +338 -0
- package/src/__tests__/serve.test.ts +275 -0
- package/src/__tests__/sessions.test.ts +227 -0
- package/src/__tests__/sharing.test.ts +288 -0
- package/src/__tests__/snapshots.test.ts +581 -0
- package/src/__tests__/state-db.test.ts +334 -0
- package/src/__tests__/stream-with-tools.test.ts +732 -0
- package/src/__tests__/subagents.test.ts +176 -0
- package/src/__tests__/system-prompt.test.ts +169 -0
- package/src/__tests__/tool-converter.test.ts +256 -0
- package/src/__tests__/tool-schemas.test.ts +397 -0
- package/src/__tests__/tools.test.ts +143 -0
- package/src/__tests__/version.test.ts +49 -0
- package/src/agent/compaction-agent.ts +227 -0
- package/src/agent/context-manager.ts +435 -0
- package/src/agent/context.ts +427 -0
- package/src/agent/deploy-preview.ts +426 -0
- package/src/agent/index.ts +68 -0
- package/src/agent/loop.ts +717 -0
- package/src/agent/modes.ts +429 -0
- package/src/agent/permissions.ts +466 -0
- package/src/agent/subagents/base.ts +116 -0
- package/src/agent/subagents/cost.ts +51 -0
- package/src/agent/subagents/explore.ts +42 -0
- package/src/agent/subagents/general.ts +54 -0
- package/src/agent/subagents/index.ts +102 -0
- package/src/agent/subagents/infra.ts +59 -0
- package/src/agent/subagents/security.ts +69 -0
- package/src/agent/system-prompt.ts +436 -0
- package/src/app.ts +122 -0
- package/src/audit/activity-log.ts +290 -0
- package/src/audit/compliance-checker.ts +540 -0
- package/src/audit/cost-tracker.ts +318 -0
- package/src/audit/index.ts +23 -0
- package/src/audit/security-scanner.ts +596 -0
- package/src/auth/guard.ts +75 -0
- package/src/auth/index.ts +56 -0
- package/src/auth/oauth.ts +455 -0
- package/src/auth/providers.ts +470 -0
- package/src/auth/sso.ts +113 -0
- package/src/auth/store.ts +505 -0
- package/src/auth/types.ts +187 -0
- package/src/build.ts +141 -0
- package/src/cli/index.ts +16 -0
- package/src/cli/init.ts +854 -0
- package/src/cli/openapi-spec.ts +356 -0
- package/src/cli/run.ts +237 -0
- package/src/cli/serve-auth.ts +80 -0
- package/src/cli/serve.ts +462 -0
- package/src/cli/web.ts +67 -0
- package/src/cli.ts +1417 -0
- package/src/clients/core-engine-client.ts +227 -0
- package/src/clients/enterprise-client.ts +334 -0
- package/src/clients/generator-client.ts +351 -0
- package/src/clients/git-client.ts +627 -0
- package/src/clients/github-client.ts +410 -0
- package/src/clients/helm-client.ts +504 -0
- package/src/clients/index.ts +80 -0
- package/src/clients/k8s-client.ts +497 -0
- package/src/clients/llm-client.ts +161 -0
- package/src/clients/rest-client.ts +130 -0
- package/src/clients/service-discovery.ts +33 -0
- package/src/clients/terraform-client.ts +482 -0
- package/src/clients/tools-client.ts +1843 -0
- package/src/clients/ws-client.ts +115 -0
- package/src/commands/analyze/index.ts +352 -0
- package/src/commands/apply/helm.ts +473 -0
- package/src/commands/apply/index.ts +213 -0
- package/src/commands/apply/k8s.ts +454 -0
- package/src/commands/apply/terraform.ts +582 -0
- package/src/commands/ask.ts +167 -0
- package/src/commands/audit/index.ts +238 -0
- package/src/commands/auth-cloud.ts +294 -0
- package/src/commands/auth-list.ts +134 -0
- package/src/commands/auth-profile.ts +121 -0
- package/src/commands/auth-status.ts +141 -0
- package/src/commands/aws/ec2.ts +501 -0
- package/src/commands/aws/iam.ts +397 -0
- package/src/commands/aws/index.ts +133 -0
- package/src/commands/aws/lambda.ts +396 -0
- package/src/commands/aws/rds.ts +439 -0
- package/src/commands/aws/s3.ts +439 -0
- package/src/commands/aws/vpc.ts +393 -0
- package/src/commands/aws-discover.ts +649 -0
- package/src/commands/aws-terraform.ts +805 -0
- package/src/commands/azure/aks.ts +376 -0
- package/src/commands/azure/functions.ts +253 -0
- package/src/commands/azure/index.ts +116 -0
- package/src/commands/azure/storage.ts +478 -0
- package/src/commands/azure/vm.ts +355 -0
- package/src/commands/billing/index.ts +256 -0
- package/src/commands/chat.ts +314 -0
- package/src/commands/config.ts +346 -0
- package/src/commands/cost/cloud-cost-estimator.ts +266 -0
- package/src/commands/cost/estimator.ts +79 -0
- package/src/commands/cost/index.ts +594 -0
- package/src/commands/cost/parsers/terraform.ts +273 -0
- package/src/commands/cost/parsers/types.ts +25 -0
- package/src/commands/cost/pricing/aws.ts +544 -0
- package/src/commands/cost/pricing/azure.ts +499 -0
- package/src/commands/cost/pricing/gcp.ts +396 -0
- package/src/commands/cost/pricing/index.ts +40 -0
- package/src/commands/demo.ts +250 -0
- package/src/commands/doctor.ts +794 -0
- package/src/commands/drift/index.ts +439 -0
- package/src/commands/explain.ts +277 -0
- package/src/commands/feedback.ts +389 -0
- package/src/commands/fix.ts +324 -0
- package/src/commands/fs/index.ts +402 -0
- package/src/commands/gcp/compute.ts +325 -0
- package/src/commands/gcp/functions.ts +271 -0
- package/src/commands/gcp/gke.ts +438 -0
- package/src/commands/gcp/iam.ts +344 -0
- package/src/commands/gcp/index.ts +129 -0
- package/src/commands/gcp/storage.ts +284 -0
- package/src/commands/generate-helm.ts +1249 -0
- package/src/commands/generate-k8s.ts +1560 -0
- package/src/commands/generate-terraform.ts +1460 -0
- package/src/commands/gh/index.ts +863 -0
- package/src/commands/git/index.ts +1343 -0
- package/src/commands/helm/index.ts +1126 -0
- package/src/commands/help.ts +539 -0
- package/src/commands/history.ts +142 -0
- package/src/commands/import.ts +868 -0
- package/src/commands/index.ts +367 -0
- package/src/commands/init.ts +1046 -0
- package/src/commands/k8s/index.ts +1137 -0
- package/src/commands/login.ts +631 -0
- package/src/commands/logout.ts +83 -0
- package/src/commands/onboarding.ts +228 -0
- package/src/commands/plan/display.ts +279 -0
- package/src/commands/plan/index.ts +599 -0
- package/src/commands/preview.ts +452 -0
- package/src/commands/questionnaire.ts +1270 -0
- package/src/commands/resume.ts +55 -0
- package/src/commands/team/index.ts +346 -0
- package/src/commands/template.ts +232 -0
- package/src/commands/tf/index.ts +1034 -0
- package/src/commands/upgrade.ts +550 -0
- package/src/commands/usage/index.ts +134 -0
- package/src/commands/version.ts +170 -0
- package/src/compat/index.ts +2 -0
- package/src/compat/runtime.ts +12 -0
- package/src/compat/sqlite.ts +107 -0
- package/src/config/index.ts +17 -0
- package/src/config/manager.ts +530 -0
- package/src/config/safety-policy.ts +358 -0
- package/src/config/schema.ts +125 -0
- package/src/config/types.ts +527 -0
- package/src/context/context-db.ts +199 -0
- package/src/demo/index.ts +349 -0
- package/src/demo/scenarios/full-journey.ts +229 -0
- package/src/demo/scenarios/getting-started.ts +127 -0
- package/src/demo/scenarios/helm-release.ts +341 -0
- package/src/demo/scenarios/k8s-deployment.ts +194 -0
- package/src/demo/scenarios/terraform-vpc.ts +170 -0
- package/src/demo/types.ts +92 -0
- package/src/engine/cost-estimator.ts +438 -0
- package/src/engine/diagram-generator.ts +256 -0
- package/src/engine/drift-detector.ts +902 -0
- package/src/engine/executor.ts +1035 -0
- package/src/engine/index.ts +76 -0
- package/src/engine/orchestrator.ts +636 -0
- package/src/engine/planner.ts +720 -0
- package/src/engine/safety.ts +743 -0
- package/src/engine/verifier.ts +770 -0
- package/src/enterprise/audit.ts +348 -0
- package/src/enterprise/auth.ts +270 -0
- package/src/enterprise/billing.ts +822 -0
- package/src/enterprise/index.ts +17 -0
- package/src/enterprise/teams.ts +443 -0
- package/src/generator/best-practices.ts +1608 -0
- package/src/generator/helm.ts +630 -0
- package/src/generator/index.ts +37 -0
- package/src/generator/intent-parser.ts +514 -0
- package/src/generator/kubernetes.ts +976 -0
- package/src/generator/terraform.ts +1867 -0
- package/src/history/index.ts +8 -0
- package/src/history/manager.ts +322 -0
- package/src/history/types.ts +34 -0
- package/src/hooks/config.ts +432 -0
- package/src/hooks/engine.ts +391 -0
- package/src/hooks/index.ts +4 -0
- package/src/llm/auth-bridge.ts +198 -0
- package/src/llm/circuit-breaker.ts +140 -0
- package/src/llm/config-loader.ts +201 -0
- package/src/llm/cost-calculator.ts +171 -0
- package/src/llm/index.ts +8 -0
- package/src/llm/model-aliases.ts +115 -0
- package/src/llm/provider-registry.ts +63 -0
- package/src/llm/providers/anthropic.ts +433 -0
- package/src/llm/providers/bedrock.ts +477 -0
- package/src/llm/providers/google.ts +405 -0
- package/src/llm/providers/ollama.ts +767 -0
- package/src/llm/providers/openai-compatible.ts +340 -0
- package/src/llm/providers/openai.ts +328 -0
- package/src/llm/providers/openrouter.ts +338 -0
- package/src/llm/router.ts +1035 -0
- package/src/llm/types.ts +232 -0
- package/src/lsp/client.ts +298 -0
- package/src/lsp/languages.ts +116 -0
- package/src/lsp/manager.ts +278 -0
- package/src/mcp/client.ts +402 -0
- package/src/mcp/index.ts +5 -0
- package/src/mcp/manager.ts +133 -0
- package/src/nimbus.ts +214 -0
- package/src/plugins/index.ts +27 -0
- package/src/plugins/loader.ts +334 -0
- package/src/plugins/manager.ts +376 -0
- package/src/plugins/types.ts +284 -0
- package/src/scanners/cicd-scanner.ts +258 -0
- package/src/scanners/cloud-scanner.ts +466 -0
- package/src/scanners/framework-scanner.ts +469 -0
- package/src/scanners/iac-scanner.ts +388 -0
- package/src/scanners/index.ts +539 -0
- package/src/scanners/language-scanner.ts +276 -0
- package/src/scanners/package-manager-scanner.ts +277 -0
- package/src/scanners/types.ts +172 -0
- package/src/sessions/manager.ts +365 -0
- package/src/sessions/types.ts +44 -0
- package/src/sharing/sync.ts +296 -0
- package/src/sharing/viewer.ts +97 -0
- package/src/snapshots/index.ts +2 -0
- package/src/snapshots/manager.ts +530 -0
- package/src/state/artifacts.ts +147 -0
- package/src/state/audit.ts +137 -0
- package/src/state/billing.ts +240 -0
- package/src/state/checkpoints.ts +117 -0
- package/src/state/config.ts +67 -0
- package/src/state/conversations.ts +14 -0
- package/src/state/credentials.ts +154 -0
- package/src/state/db.ts +58 -0
- package/src/state/index.ts +26 -0
- package/src/state/messages.ts +115 -0
- package/src/state/projects.ts +123 -0
- package/src/state/schema.ts +236 -0
- package/src/state/sessions.ts +147 -0
- package/src/state/teams.ts +200 -0
- package/src/telemetry.ts +108 -0
- package/src/tools/aws-ops.ts +952 -0
- package/src/tools/azure-ops.ts +579 -0
- package/src/tools/file-ops.ts +593 -0
- package/src/tools/gcp-ops.ts +625 -0
- package/src/tools/git-ops.ts +773 -0
- package/src/tools/github-ops.ts +799 -0
- package/src/tools/helm-ops.ts +943 -0
- package/src/tools/index.ts +17 -0
- package/src/tools/k8s-ops.ts +819 -0
- package/src/tools/schemas/converter.ts +184 -0
- package/src/tools/schemas/devops.ts +612 -0
- package/src/tools/schemas/index.ts +73 -0
- package/src/tools/schemas/standard.ts +1144 -0
- package/src/tools/schemas/types.ts +705 -0
- package/src/tools/terraform-ops.ts +862 -0
- package/src/types/ambient.d.ts +193 -0
- package/src/types/config.ts +83 -0
- package/src/types/drift.ts +116 -0
- package/src/types/enterprise.ts +335 -0
- package/src/types/index.ts +20 -0
- package/src/types/plan.ts +44 -0
- package/src/types/request.ts +65 -0
- package/src/types/response.ts +54 -0
- package/src/types/service.ts +51 -0
- package/src/ui/App.tsx +997 -0
- package/src/ui/DeployPreview.tsx +169 -0
- package/src/ui/Header.tsx +68 -0
- package/src/ui/InputBox.tsx +350 -0
- package/src/ui/MessageList.tsx +585 -0
- package/src/ui/PermissionPrompt.tsx +151 -0
- package/src/ui/StatusBar.tsx +158 -0
- package/src/ui/ToolCallDisplay.tsx +409 -0
- package/src/ui/chat-ui.ts +853 -0
- package/src/ui/index.ts +33 -0
- package/src/ui/ink/index.ts +711 -0
- package/src/ui/streaming.ts +176 -0
- package/src/ui/types.ts +57 -0
- package/src/utils/analytics.ts +72 -0
- package/src/utils/cost-warning.ts +27 -0
- package/src/utils/env.ts +46 -0
- package/src/utils/errors.ts +69 -0
- package/src/utils/event-bus.ts +38 -0
- package/src/utils/index.ts +24 -0
- package/src/utils/logger.ts +171 -0
- package/src/utils/rate-limiter.ts +121 -0
- package/src/utils/service-auth.ts +49 -0
- package/src/utils/validation.ts +53 -0
- package/src/version.ts +4 -0
- package/src/watcher/index.ts +163 -0
- package/src/wizard/approval.ts +383 -0
- package/src/wizard/index.ts +25 -0
- package/src/wizard/prompts.ts +338 -0
- package/src/wizard/types.ts +171 -0
- package/src/wizard/ui.ts +556 -0
- package/src/wizard/wizard.ts +304 -0
- 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;
|