@hasna/terminal 2.2.0 → 2.3.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/dist/cli.js +29 -12
- package/package.json +1 -1
- package/src/ai.ts +50 -36
- package/src/cli.tsx +29 -12
- package/src/context-hints.ts +89 -0
- package/src/discover.ts +238 -0
- package/src/economy.ts +53 -0
- package/src/output-store.ts +65 -0
- package/src/providers/index.ts +4 -4
- package/src/sessions-db.ts +81 -0
- package/temp/rtk/.claude/agents/code-reviewer.md +221 -0
- package/temp/rtk/.claude/agents/debugger.md +519 -0
- package/temp/rtk/.claude/agents/rtk-testing-specialist.md +461 -0
- package/temp/rtk/.claude/agents/rust-rtk.md +511 -0
- package/temp/rtk/.claude/agents/technical-writer.md +355 -0
- package/temp/rtk/.claude/commands/diagnose.md +352 -0
- package/temp/rtk/.claude/commands/test-routing.md +362 -0
- package/temp/rtk/.claude/hooks/bash/pre-commit-format.sh +16 -0
- package/temp/rtk/.claude/hooks/rtk-rewrite.sh +70 -0
- package/temp/rtk/.claude/hooks/rtk-suggest.sh +152 -0
- package/temp/rtk/.claude/rules/cli-testing.md +526 -0
- package/temp/rtk/.claude/skills/issue-triage/SKILL.md +348 -0
- package/temp/rtk/.claude/skills/issue-triage/templates/issue-comment.md +134 -0
- package/temp/rtk/.claude/skills/performance.md +435 -0
- package/temp/rtk/.claude/skills/pr-triage/SKILL.md +315 -0
- package/temp/rtk/.claude/skills/pr-triage/templates/review-comment.md +71 -0
- package/temp/rtk/.claude/skills/repo-recap.md +206 -0
- package/temp/rtk/.claude/skills/rtk-tdd/SKILL.md +78 -0
- package/temp/rtk/.claude/skills/rtk-tdd/references/testing-patterns.md +124 -0
- package/temp/rtk/.claude/skills/security-guardian.md +503 -0
- package/temp/rtk/.claude/skills/ship.md +404 -0
- package/temp/rtk/.github/workflows/benchmark.yml +34 -0
- package/temp/rtk/.github/workflows/dco-check.yaml +12 -0
- package/temp/rtk/.github/workflows/release-please.yml +51 -0
- package/temp/rtk/.github/workflows/release.yml +343 -0
- package/temp/rtk/.github/workflows/security-check.yml +135 -0
- package/temp/rtk/.github/workflows/validate-docs.yml +78 -0
- package/temp/rtk/.release-please-manifest.json +3 -0
- package/temp/rtk/ARCHITECTURE.md +1491 -0
- package/temp/rtk/CHANGELOG.md +640 -0
- package/temp/rtk/CLAUDE.md +605 -0
- package/temp/rtk/CONTRIBUTING.md +199 -0
- package/temp/rtk/Cargo.lock +1668 -0
- package/temp/rtk/Cargo.toml +64 -0
- package/temp/rtk/Formula/rtk.rb +43 -0
- package/temp/rtk/INSTALL.md +390 -0
- package/temp/rtk/LICENSE +21 -0
- package/temp/rtk/README.md +386 -0
- package/temp/rtk/README_es.md +159 -0
- package/temp/rtk/README_fr.md +197 -0
- package/temp/rtk/README_ja.md +159 -0
- package/temp/rtk/README_ko.md +159 -0
- package/temp/rtk/README_zh.md +167 -0
- package/temp/rtk/ROADMAP.md +15 -0
- package/temp/rtk/SECURITY.md +217 -0
- package/temp/rtk/TEST_EXEC_TIME.md +102 -0
- package/temp/rtk/build.rs +57 -0
- package/temp/rtk/docs/AUDIT_GUIDE.md +432 -0
- package/temp/rtk/docs/FEATURES.md +1410 -0
- package/temp/rtk/docs/TROUBLESHOOTING.md +309 -0
- package/temp/rtk/docs/filter-workflow.md +102 -0
- package/temp/rtk/docs/images/gain-dashboard.jpg +0 -0
- package/temp/rtk/docs/tracking.md +583 -0
- package/temp/rtk/hooks/opencode-rtk.ts +39 -0
- package/temp/rtk/hooks/rtk-awareness.md +29 -0
- package/temp/rtk/hooks/rtk-rewrite.sh +61 -0
- package/temp/rtk/hooks/test-rtk-rewrite.sh +442 -0
- package/temp/rtk/install.sh +124 -0
- package/temp/rtk/release-please-config.json +10 -0
- package/temp/rtk/scripts/benchmark.sh +592 -0
- package/temp/rtk/scripts/check-installation.sh +162 -0
- package/temp/rtk/scripts/install-local.sh +37 -0
- package/temp/rtk/scripts/rtk-economics.sh +137 -0
- package/temp/rtk/scripts/test-all.sh +561 -0
- package/temp/rtk/scripts/test-aristote.sh +227 -0
- package/temp/rtk/scripts/test-tracking.sh +79 -0
- package/temp/rtk/scripts/update-readme-metrics.sh +32 -0
- package/temp/rtk/scripts/validate-docs.sh +73 -0
- package/temp/rtk/src/aws_cmd.rs +880 -0
- package/temp/rtk/src/binlog.rs +1645 -0
- package/temp/rtk/src/cargo_cmd.rs +1727 -0
- package/temp/rtk/src/cc_economics.rs +1157 -0
- package/temp/rtk/src/ccusage.rs +340 -0
- package/temp/rtk/src/config.rs +187 -0
- package/temp/rtk/src/container.rs +855 -0
- package/temp/rtk/src/curl_cmd.rs +134 -0
- package/temp/rtk/src/deps.rs +268 -0
- package/temp/rtk/src/diff_cmd.rs +367 -0
- package/temp/rtk/src/discover/mod.rs +274 -0
- package/temp/rtk/src/discover/provider.rs +388 -0
- package/temp/rtk/src/discover/registry.rs +2022 -0
- package/temp/rtk/src/discover/report.rs +202 -0
- package/temp/rtk/src/discover/rules.rs +667 -0
- package/temp/rtk/src/display_helpers.rs +402 -0
- package/temp/rtk/src/dotnet_cmd.rs +1771 -0
- package/temp/rtk/src/dotnet_format_report.rs +133 -0
- package/temp/rtk/src/dotnet_trx.rs +593 -0
- package/temp/rtk/src/env_cmd.rs +204 -0
- package/temp/rtk/src/filter.rs +462 -0
- package/temp/rtk/src/filters/README.md +52 -0
- package/temp/rtk/src/filters/ansible-playbook.toml +34 -0
- package/temp/rtk/src/filters/basedpyright.toml +47 -0
- package/temp/rtk/src/filters/biome.toml +45 -0
- package/temp/rtk/src/filters/brew-install.toml +37 -0
- package/temp/rtk/src/filters/composer-install.toml +40 -0
- package/temp/rtk/src/filters/df.toml +16 -0
- package/temp/rtk/src/filters/dotnet-build.toml +64 -0
- package/temp/rtk/src/filters/du.toml +16 -0
- package/temp/rtk/src/filters/fail2ban-client.toml +15 -0
- package/temp/rtk/src/filters/gcc.toml +49 -0
- package/temp/rtk/src/filters/gcloud.toml +22 -0
- package/temp/rtk/src/filters/hadolint.toml +24 -0
- package/temp/rtk/src/filters/helm.toml +29 -0
- package/temp/rtk/src/filters/iptables.toml +27 -0
- package/temp/rtk/src/filters/jj.toml +28 -0
- package/temp/rtk/src/filters/jq.toml +24 -0
- package/temp/rtk/src/filters/make.toml +41 -0
- package/temp/rtk/src/filters/markdownlint.toml +24 -0
- package/temp/rtk/src/filters/mix-compile.toml +27 -0
- package/temp/rtk/src/filters/mix-format.toml +15 -0
- package/temp/rtk/src/filters/mvn-build.toml +44 -0
- package/temp/rtk/src/filters/oxlint.toml +43 -0
- package/temp/rtk/src/filters/ping.toml +63 -0
- package/temp/rtk/src/filters/pio-run.toml +40 -0
- package/temp/rtk/src/filters/poetry-install.toml +50 -0
- package/temp/rtk/src/filters/pre-commit.toml +35 -0
- package/temp/rtk/src/filters/ps.toml +16 -0
- package/temp/rtk/src/filters/quarto-render.toml +41 -0
- package/temp/rtk/src/filters/rsync.toml +48 -0
- package/temp/rtk/src/filters/shellcheck.toml +27 -0
- package/temp/rtk/src/filters/shopify-theme.toml +29 -0
- package/temp/rtk/src/filters/skopeo.toml +45 -0
- package/temp/rtk/src/filters/sops.toml +16 -0
- package/temp/rtk/src/filters/ssh.toml +44 -0
- package/temp/rtk/src/filters/stat.toml +34 -0
- package/temp/rtk/src/filters/swift-build.toml +41 -0
- package/temp/rtk/src/filters/systemctl-status.toml +33 -0
- package/temp/rtk/src/filters/terraform-plan.toml +35 -0
- package/temp/rtk/src/filters/tofu-fmt.toml +16 -0
- package/temp/rtk/src/filters/tofu-init.toml +38 -0
- package/temp/rtk/src/filters/tofu-plan.toml +35 -0
- package/temp/rtk/src/filters/tofu-validate.toml +17 -0
- package/temp/rtk/src/filters/trunk-build.toml +39 -0
- package/temp/rtk/src/filters/ty.toml +50 -0
- package/temp/rtk/src/filters/uv-sync.toml +37 -0
- package/temp/rtk/src/filters/xcodebuild.toml +99 -0
- package/temp/rtk/src/filters/yamllint.toml +25 -0
- package/temp/rtk/src/find_cmd.rs +598 -0
- package/temp/rtk/src/format_cmd.rs +386 -0
- package/temp/rtk/src/gain.rs +723 -0
- package/temp/rtk/src/gh_cmd.rs +1651 -0
- package/temp/rtk/src/git.rs +2012 -0
- package/temp/rtk/src/go_cmd.rs +592 -0
- package/temp/rtk/src/golangci_cmd.rs +254 -0
- package/temp/rtk/src/grep_cmd.rs +288 -0
- package/temp/rtk/src/gt_cmd.rs +810 -0
- package/temp/rtk/src/hook_audit_cmd.rs +283 -0
- package/temp/rtk/src/hook_check.rs +171 -0
- package/temp/rtk/src/init.rs +1859 -0
- package/temp/rtk/src/integrity.rs +537 -0
- package/temp/rtk/src/json_cmd.rs +231 -0
- package/temp/rtk/src/learn/detector.rs +628 -0
- package/temp/rtk/src/learn/mod.rs +119 -0
- package/temp/rtk/src/learn/report.rs +184 -0
- package/temp/rtk/src/lint_cmd.rs +694 -0
- package/temp/rtk/src/local_llm.rs +316 -0
- package/temp/rtk/src/log_cmd.rs +248 -0
- package/temp/rtk/src/ls.rs +324 -0
- package/temp/rtk/src/main.rs +2482 -0
- package/temp/rtk/src/mypy_cmd.rs +389 -0
- package/temp/rtk/src/next_cmd.rs +241 -0
- package/temp/rtk/src/npm_cmd.rs +236 -0
- package/temp/rtk/src/parser/README.md +267 -0
- package/temp/rtk/src/parser/error.rs +46 -0
- package/temp/rtk/src/parser/formatter.rs +336 -0
- package/temp/rtk/src/parser/mod.rs +311 -0
- package/temp/rtk/src/parser/types.rs +119 -0
- package/temp/rtk/src/pip_cmd.rs +302 -0
- package/temp/rtk/src/playwright_cmd.rs +479 -0
- package/temp/rtk/src/pnpm_cmd.rs +573 -0
- package/temp/rtk/src/prettier_cmd.rs +221 -0
- package/temp/rtk/src/prisma_cmd.rs +482 -0
- package/temp/rtk/src/psql_cmd.rs +382 -0
- package/temp/rtk/src/pytest_cmd.rs +384 -0
- package/temp/rtk/src/read.rs +217 -0
- package/temp/rtk/src/rewrite_cmd.rs +50 -0
- package/temp/rtk/src/ruff_cmd.rs +402 -0
- package/temp/rtk/src/runner.rs +271 -0
- package/temp/rtk/src/summary.rs +297 -0
- package/temp/rtk/src/tee.rs +405 -0
- package/temp/rtk/src/telemetry.rs +248 -0
- package/temp/rtk/src/toml_filter.rs +1655 -0
- package/temp/rtk/src/tracking.rs +1416 -0
- package/temp/rtk/src/tree.rs +209 -0
- package/temp/rtk/src/tsc_cmd.rs +259 -0
- package/temp/rtk/src/utils.rs +432 -0
- package/temp/rtk/src/verify_cmd.rs +47 -0
- package/temp/rtk/src/vitest_cmd.rs +385 -0
- package/temp/rtk/src/wc_cmd.rs +401 -0
- package/temp/rtk/src/wget_cmd.rs +260 -0
- package/temp/rtk/tests/fixtures/dotnet/build_failed.txt +11 -0
- package/temp/rtk/tests/fixtures/dotnet/format_changes.json +31 -0
- package/temp/rtk/tests/fixtures/dotnet/format_empty.json +1 -0
- package/temp/rtk/tests/fixtures/dotnet/format_success.json +12 -0
- package/temp/rtk/tests/fixtures/dotnet/test_failed.txt +18 -0
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
pub mod detector;
|
|
2
|
+
pub mod report;
|
|
3
|
+
|
|
4
|
+
use crate::discover::provider::{ClaudeProvider, SessionProvider};
|
|
5
|
+
use anyhow::Result;
|
|
6
|
+
use detector::{deduplicate_corrections, find_corrections, CommandExecution};
|
|
7
|
+
use report::{format_console_report, write_rules_file};
|
|
8
|
+
|
|
9
|
+
pub fn run(
|
|
10
|
+
project: Option<String>,
|
|
11
|
+
all: bool,
|
|
12
|
+
since: u64,
|
|
13
|
+
format: String,
|
|
14
|
+
write_rules: bool,
|
|
15
|
+
min_confidence: f64,
|
|
16
|
+
min_occurrences: usize,
|
|
17
|
+
) -> Result<()> {
|
|
18
|
+
let provider = ClaudeProvider;
|
|
19
|
+
|
|
20
|
+
// Determine project filter (same logic as discover)
|
|
21
|
+
let project_filter = if all {
|
|
22
|
+
None
|
|
23
|
+
} else if let Some(p) = project {
|
|
24
|
+
Some(p)
|
|
25
|
+
} else {
|
|
26
|
+
// Default: current working directory
|
|
27
|
+
let cwd = std::env::current_dir()?;
|
|
28
|
+
let cwd_str = cwd.to_string_lossy().to_string();
|
|
29
|
+
let encoded = ClaudeProvider::encode_project_path(&cwd_str);
|
|
30
|
+
Some(encoded)
|
|
31
|
+
};
|
|
32
|
+
|
|
33
|
+
// Discover sessions
|
|
34
|
+
let sessions = provider.discover_sessions(project_filter.as_deref(), Some(since))?;
|
|
35
|
+
|
|
36
|
+
if sessions.is_empty() {
|
|
37
|
+
println!("No Claude Code sessions found in the last {} days.", since);
|
|
38
|
+
return Ok(());
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
// Extract commands from all sessions
|
|
42
|
+
let mut all_commands: Vec<CommandExecution> = Vec::new();
|
|
43
|
+
|
|
44
|
+
for session_path in &sessions {
|
|
45
|
+
let extracted = match provider.extract_commands(session_path) {
|
|
46
|
+
Ok(cmds) => cmds,
|
|
47
|
+
Err(_) => continue, // Skip malformed sessions
|
|
48
|
+
};
|
|
49
|
+
|
|
50
|
+
for ext_cmd in extracted {
|
|
51
|
+
// Only process commands with output content
|
|
52
|
+
if let Some(output) = ext_cmd.output_content {
|
|
53
|
+
all_commands.push(CommandExecution {
|
|
54
|
+
command: ext_cmd.command,
|
|
55
|
+
is_error: ext_cmd.is_error,
|
|
56
|
+
output,
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
// Sort by sequence index to maintain chronological order
|
|
63
|
+
// (already sorted by extraction order within each session)
|
|
64
|
+
|
|
65
|
+
// Find corrections
|
|
66
|
+
let corrections = find_corrections(&all_commands);
|
|
67
|
+
|
|
68
|
+
if corrections.is_empty() {
|
|
69
|
+
println!(
|
|
70
|
+
"No CLI corrections detected in {} sessions.",
|
|
71
|
+
sessions.len()
|
|
72
|
+
);
|
|
73
|
+
return Ok(());
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
// Filter by confidence
|
|
77
|
+
let filtered: Vec<_> = corrections
|
|
78
|
+
.into_iter()
|
|
79
|
+
.filter(|c| c.confidence >= min_confidence)
|
|
80
|
+
.collect();
|
|
81
|
+
|
|
82
|
+
// Deduplicate
|
|
83
|
+
let mut rules = deduplicate_corrections(filtered.clone());
|
|
84
|
+
|
|
85
|
+
// Filter by occurrences
|
|
86
|
+
rules.retain(|r| r.occurrences >= min_occurrences);
|
|
87
|
+
|
|
88
|
+
// Output
|
|
89
|
+
match format.as_str() {
|
|
90
|
+
"json" => {
|
|
91
|
+
// JSON output
|
|
92
|
+
let json = serde_json::json!({
|
|
93
|
+
"sessions_scanned": sessions.len(),
|
|
94
|
+
"total_corrections": filtered.len(),
|
|
95
|
+
"rules": rules.iter().map(|r| serde_json::json!({
|
|
96
|
+
"wrong": r.wrong_pattern,
|
|
97
|
+
"right": r.right_pattern,
|
|
98
|
+
"error_type": r.error_type.as_str(),
|
|
99
|
+
"occurrences": r.occurrences,
|
|
100
|
+
"base_command": r.base_command,
|
|
101
|
+
})).collect::<Vec<_>>(),
|
|
102
|
+
});
|
|
103
|
+
println!("{}", serde_json::to_string_pretty(&json)?);
|
|
104
|
+
}
|
|
105
|
+
_ => {
|
|
106
|
+
// Text output
|
|
107
|
+
let report = format_console_report(&rules, filtered.len(), sessions.len(), since);
|
|
108
|
+
print!("{}", report);
|
|
109
|
+
|
|
110
|
+
if write_rules && !rules.is_empty() {
|
|
111
|
+
let rules_path = ".claude/rules/cli-corrections.md";
|
|
112
|
+
write_rules_file(&rules, rules_path)?;
|
|
113
|
+
println!("\nWritten to: {}", rules_path);
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
Ok(())
|
|
119
|
+
}
|
|
@@ -0,0 +1,184 @@
|
|
|
1
|
+
use crate::learn::detector::CorrectionRule;
|
|
2
|
+
use anyhow::Result;
|
|
3
|
+
use std::collections::HashMap;
|
|
4
|
+
use std::fs;
|
|
5
|
+
use std::path::Path;
|
|
6
|
+
|
|
7
|
+
pub fn format_console_report(
|
|
8
|
+
rules: &[CorrectionRule],
|
|
9
|
+
total_corrections: usize,
|
|
10
|
+
sessions: usize,
|
|
11
|
+
days: u64,
|
|
12
|
+
) -> String {
|
|
13
|
+
let mut output = String::new();
|
|
14
|
+
|
|
15
|
+
output.push_str(&format!(
|
|
16
|
+
"RTK Learn -- {} rules from {} corrections ({} sessions, {} days)\n",
|
|
17
|
+
rules.len(),
|
|
18
|
+
total_corrections,
|
|
19
|
+
sessions,
|
|
20
|
+
days
|
|
21
|
+
));
|
|
22
|
+
|
|
23
|
+
if rules.is_empty() {
|
|
24
|
+
output.push_str("\nNo CLI corrections detected.\n");
|
|
25
|
+
return output;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
output.push('\n');
|
|
29
|
+
|
|
30
|
+
for rule in rules {
|
|
31
|
+
let count_marker = if rule.occurrences > 1 {
|
|
32
|
+
format!("[{}x] ", rule.occurrences)
|
|
33
|
+
} else {
|
|
34
|
+
" ".to_string()
|
|
35
|
+
};
|
|
36
|
+
|
|
37
|
+
output.push_str(&format!(
|
|
38
|
+
"{}{} → {}\n",
|
|
39
|
+
count_marker, rule.wrong_pattern, rule.right_pattern
|
|
40
|
+
));
|
|
41
|
+
|
|
42
|
+
// Show error snippet (first line only)
|
|
43
|
+
let error_line = rule.example_error.lines().next().unwrap_or("").trim();
|
|
44
|
+
if !error_line.is_empty() {
|
|
45
|
+
output.push_str(&format!(" Error: {}\n", error_line));
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
output
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
pub fn write_rules_file(rules: &[CorrectionRule], path: &str) -> Result<()> {
|
|
53
|
+
let path_obj = Path::new(path);
|
|
54
|
+
|
|
55
|
+
// Create parent directory if it doesn't exist
|
|
56
|
+
if let Some(parent) = path_obj.parent() {
|
|
57
|
+
fs::create_dir_all(parent)?;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
let mut content = String::new();
|
|
61
|
+
content.push_str("# CLI Corrections (auto-generated by rtk learn)\n");
|
|
62
|
+
content.push_str("# Run `rtk learn --write-rules` to update\n\n");
|
|
63
|
+
|
|
64
|
+
if rules.is_empty() {
|
|
65
|
+
content.push_str("No CLI corrections detected yet.\n");
|
|
66
|
+
fs::write(path, content)?;
|
|
67
|
+
return Ok(());
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
// Group by base command
|
|
71
|
+
let mut grouped: HashMap<String, Vec<&CorrectionRule>> = HashMap::new();
|
|
72
|
+
for rule in rules {
|
|
73
|
+
grouped
|
|
74
|
+
.entry(rule.base_command.clone())
|
|
75
|
+
.or_default()
|
|
76
|
+
.push(rule);
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
// Sort base commands alphabetically
|
|
80
|
+
let mut base_commands: Vec<String> = grouped.keys().cloned().collect();
|
|
81
|
+
base_commands.sort();
|
|
82
|
+
|
|
83
|
+
for base_cmd in base_commands {
|
|
84
|
+
let rules_for_cmd = grouped.get(&base_cmd).unwrap();
|
|
85
|
+
|
|
86
|
+
// Capitalize first letter for section header
|
|
87
|
+
let section_header = capitalize_first(&base_cmd);
|
|
88
|
+
content.push_str(&format!("## {}\n", section_header));
|
|
89
|
+
|
|
90
|
+
for rule in rules_for_cmd {
|
|
91
|
+
let occurrence_note = if rule.occurrences > 1 {
|
|
92
|
+
format!(" (seen {}x)", rule.occurrences)
|
|
93
|
+
} else {
|
|
94
|
+
String::new()
|
|
95
|
+
};
|
|
96
|
+
|
|
97
|
+
content.push_str(&format!(
|
|
98
|
+
"- Use `{}` not `{}`{}\n",
|
|
99
|
+
rule.right_pattern, rule.wrong_pattern, occurrence_note
|
|
100
|
+
));
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
content.push('\n');
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
fs::write(path, content)?;
|
|
107
|
+
Ok(())
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
fn capitalize_first(s: &str) -> String {
|
|
111
|
+
let mut chars = s.chars();
|
|
112
|
+
match chars.next() {
|
|
113
|
+
None => String::new(),
|
|
114
|
+
Some(first) => first.to_uppercase().collect::<String>() + chars.as_str(),
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
#[cfg(test)]
|
|
119
|
+
mod tests {
|
|
120
|
+
use super::*;
|
|
121
|
+
use crate::learn::detector::ErrorType;
|
|
122
|
+
|
|
123
|
+
#[test]
|
|
124
|
+
fn test_format_console_report_empty() {
|
|
125
|
+
let report = format_console_report(&[], 0, 0, 30);
|
|
126
|
+
assert!(report.contains("0 rules"));
|
|
127
|
+
assert!(report.contains("0 corrections"));
|
|
128
|
+
assert!(report.contains("No CLI corrections detected"));
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
#[test]
|
|
132
|
+
fn test_format_console_report_with_rules() {
|
|
133
|
+
let rules = vec![
|
|
134
|
+
CorrectionRule {
|
|
135
|
+
wrong_pattern: "git commit --ammend".to_string(),
|
|
136
|
+
right_pattern: "git commit --amend".to_string(),
|
|
137
|
+
error_type: ErrorType::UnknownFlag,
|
|
138
|
+
occurrences: 3,
|
|
139
|
+
base_command: "git commit".to_string(),
|
|
140
|
+
example_error: "error: unexpected argument '--ammend'".to_string(),
|
|
141
|
+
},
|
|
142
|
+
CorrectionRule {
|
|
143
|
+
wrong_pattern: "gh pr edit -t".to_string(),
|
|
144
|
+
right_pattern: "gh pr edit --title".to_string(),
|
|
145
|
+
error_type: ErrorType::UnknownFlag,
|
|
146
|
+
occurrences: 1,
|
|
147
|
+
base_command: "gh pr".to_string(),
|
|
148
|
+
example_error: "unknown flag: -t".to_string(),
|
|
149
|
+
},
|
|
150
|
+
];
|
|
151
|
+
|
|
152
|
+
let report = format_console_report(&rules, 4, 10, 30);
|
|
153
|
+
assert!(report.contains("2 rules"));
|
|
154
|
+
assert!(report.contains("4 corrections"));
|
|
155
|
+
assert!(report.contains("[3x]"));
|
|
156
|
+
assert!(report.contains("--ammend"));
|
|
157
|
+
assert!(report.contains("--amend"));
|
|
158
|
+
assert!(report.contains("Error: error: unexpected argument"));
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
#[test]
|
|
162
|
+
fn test_write_rules_file_markdown() {
|
|
163
|
+
let rules = vec![CorrectionRule {
|
|
164
|
+
wrong_pattern: "git commit --ammend".to_string(),
|
|
165
|
+
right_pattern: "git commit --amend".to_string(),
|
|
166
|
+
error_type: ErrorType::UnknownFlag,
|
|
167
|
+
occurrences: 3,
|
|
168
|
+
base_command: "git commit".to_string(),
|
|
169
|
+
example_error: "error: unexpected argument '--ammend'".to_string(),
|
|
170
|
+
}];
|
|
171
|
+
|
|
172
|
+
let temp_dir = tempfile::tempdir().unwrap();
|
|
173
|
+
let path = temp_dir.path().join("cli-corrections.md");
|
|
174
|
+
let path_str = path.to_str().unwrap();
|
|
175
|
+
|
|
176
|
+
write_rules_file(&rules, path_str).unwrap();
|
|
177
|
+
|
|
178
|
+
let content = fs::read_to_string(&path).unwrap();
|
|
179
|
+
assert!(content.contains("# CLI Corrections"));
|
|
180
|
+
assert!(content.contains("## Git commit"));
|
|
181
|
+
assert!(content.contains("Use `git commit --amend` not `git commit --ammend`"));
|
|
182
|
+
assert!(content.contains("(seen 3x)"));
|
|
183
|
+
}
|
|
184
|
+
}
|