@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.
Files changed (205) hide show
  1. package/dist/cli.js +29 -12
  2. package/package.json +1 -1
  3. package/src/ai.ts +50 -36
  4. package/src/cli.tsx +29 -12
  5. package/src/context-hints.ts +89 -0
  6. package/src/discover.ts +238 -0
  7. package/src/economy.ts +53 -0
  8. package/src/output-store.ts +65 -0
  9. package/src/providers/index.ts +4 -4
  10. package/src/sessions-db.ts +81 -0
  11. package/temp/rtk/.claude/agents/code-reviewer.md +221 -0
  12. package/temp/rtk/.claude/agents/debugger.md +519 -0
  13. package/temp/rtk/.claude/agents/rtk-testing-specialist.md +461 -0
  14. package/temp/rtk/.claude/agents/rust-rtk.md +511 -0
  15. package/temp/rtk/.claude/agents/technical-writer.md +355 -0
  16. package/temp/rtk/.claude/commands/diagnose.md +352 -0
  17. package/temp/rtk/.claude/commands/test-routing.md +362 -0
  18. package/temp/rtk/.claude/hooks/bash/pre-commit-format.sh +16 -0
  19. package/temp/rtk/.claude/hooks/rtk-rewrite.sh +70 -0
  20. package/temp/rtk/.claude/hooks/rtk-suggest.sh +152 -0
  21. package/temp/rtk/.claude/rules/cli-testing.md +526 -0
  22. package/temp/rtk/.claude/skills/issue-triage/SKILL.md +348 -0
  23. package/temp/rtk/.claude/skills/issue-triage/templates/issue-comment.md +134 -0
  24. package/temp/rtk/.claude/skills/performance.md +435 -0
  25. package/temp/rtk/.claude/skills/pr-triage/SKILL.md +315 -0
  26. package/temp/rtk/.claude/skills/pr-triage/templates/review-comment.md +71 -0
  27. package/temp/rtk/.claude/skills/repo-recap.md +206 -0
  28. package/temp/rtk/.claude/skills/rtk-tdd/SKILL.md +78 -0
  29. package/temp/rtk/.claude/skills/rtk-tdd/references/testing-patterns.md +124 -0
  30. package/temp/rtk/.claude/skills/security-guardian.md +503 -0
  31. package/temp/rtk/.claude/skills/ship.md +404 -0
  32. package/temp/rtk/.github/workflows/benchmark.yml +34 -0
  33. package/temp/rtk/.github/workflows/dco-check.yaml +12 -0
  34. package/temp/rtk/.github/workflows/release-please.yml +51 -0
  35. package/temp/rtk/.github/workflows/release.yml +343 -0
  36. package/temp/rtk/.github/workflows/security-check.yml +135 -0
  37. package/temp/rtk/.github/workflows/validate-docs.yml +78 -0
  38. package/temp/rtk/.release-please-manifest.json +3 -0
  39. package/temp/rtk/ARCHITECTURE.md +1491 -0
  40. package/temp/rtk/CHANGELOG.md +640 -0
  41. package/temp/rtk/CLAUDE.md +605 -0
  42. package/temp/rtk/CONTRIBUTING.md +199 -0
  43. package/temp/rtk/Cargo.lock +1668 -0
  44. package/temp/rtk/Cargo.toml +64 -0
  45. package/temp/rtk/Formula/rtk.rb +43 -0
  46. package/temp/rtk/INSTALL.md +390 -0
  47. package/temp/rtk/LICENSE +21 -0
  48. package/temp/rtk/README.md +386 -0
  49. package/temp/rtk/README_es.md +159 -0
  50. package/temp/rtk/README_fr.md +197 -0
  51. package/temp/rtk/README_ja.md +159 -0
  52. package/temp/rtk/README_ko.md +159 -0
  53. package/temp/rtk/README_zh.md +167 -0
  54. package/temp/rtk/ROADMAP.md +15 -0
  55. package/temp/rtk/SECURITY.md +217 -0
  56. package/temp/rtk/TEST_EXEC_TIME.md +102 -0
  57. package/temp/rtk/build.rs +57 -0
  58. package/temp/rtk/docs/AUDIT_GUIDE.md +432 -0
  59. package/temp/rtk/docs/FEATURES.md +1410 -0
  60. package/temp/rtk/docs/TROUBLESHOOTING.md +309 -0
  61. package/temp/rtk/docs/filter-workflow.md +102 -0
  62. package/temp/rtk/docs/images/gain-dashboard.jpg +0 -0
  63. package/temp/rtk/docs/tracking.md +583 -0
  64. package/temp/rtk/hooks/opencode-rtk.ts +39 -0
  65. package/temp/rtk/hooks/rtk-awareness.md +29 -0
  66. package/temp/rtk/hooks/rtk-rewrite.sh +61 -0
  67. package/temp/rtk/hooks/test-rtk-rewrite.sh +442 -0
  68. package/temp/rtk/install.sh +124 -0
  69. package/temp/rtk/release-please-config.json +10 -0
  70. package/temp/rtk/scripts/benchmark.sh +592 -0
  71. package/temp/rtk/scripts/check-installation.sh +162 -0
  72. package/temp/rtk/scripts/install-local.sh +37 -0
  73. package/temp/rtk/scripts/rtk-economics.sh +137 -0
  74. package/temp/rtk/scripts/test-all.sh +561 -0
  75. package/temp/rtk/scripts/test-aristote.sh +227 -0
  76. package/temp/rtk/scripts/test-tracking.sh +79 -0
  77. package/temp/rtk/scripts/update-readme-metrics.sh +32 -0
  78. package/temp/rtk/scripts/validate-docs.sh +73 -0
  79. package/temp/rtk/src/aws_cmd.rs +880 -0
  80. package/temp/rtk/src/binlog.rs +1645 -0
  81. package/temp/rtk/src/cargo_cmd.rs +1727 -0
  82. package/temp/rtk/src/cc_economics.rs +1157 -0
  83. package/temp/rtk/src/ccusage.rs +340 -0
  84. package/temp/rtk/src/config.rs +187 -0
  85. package/temp/rtk/src/container.rs +855 -0
  86. package/temp/rtk/src/curl_cmd.rs +134 -0
  87. package/temp/rtk/src/deps.rs +268 -0
  88. package/temp/rtk/src/diff_cmd.rs +367 -0
  89. package/temp/rtk/src/discover/mod.rs +274 -0
  90. package/temp/rtk/src/discover/provider.rs +388 -0
  91. package/temp/rtk/src/discover/registry.rs +2022 -0
  92. package/temp/rtk/src/discover/report.rs +202 -0
  93. package/temp/rtk/src/discover/rules.rs +667 -0
  94. package/temp/rtk/src/display_helpers.rs +402 -0
  95. package/temp/rtk/src/dotnet_cmd.rs +1771 -0
  96. package/temp/rtk/src/dotnet_format_report.rs +133 -0
  97. package/temp/rtk/src/dotnet_trx.rs +593 -0
  98. package/temp/rtk/src/env_cmd.rs +204 -0
  99. package/temp/rtk/src/filter.rs +462 -0
  100. package/temp/rtk/src/filters/README.md +52 -0
  101. package/temp/rtk/src/filters/ansible-playbook.toml +34 -0
  102. package/temp/rtk/src/filters/basedpyright.toml +47 -0
  103. package/temp/rtk/src/filters/biome.toml +45 -0
  104. package/temp/rtk/src/filters/brew-install.toml +37 -0
  105. package/temp/rtk/src/filters/composer-install.toml +40 -0
  106. package/temp/rtk/src/filters/df.toml +16 -0
  107. package/temp/rtk/src/filters/dotnet-build.toml +64 -0
  108. package/temp/rtk/src/filters/du.toml +16 -0
  109. package/temp/rtk/src/filters/fail2ban-client.toml +15 -0
  110. package/temp/rtk/src/filters/gcc.toml +49 -0
  111. package/temp/rtk/src/filters/gcloud.toml +22 -0
  112. package/temp/rtk/src/filters/hadolint.toml +24 -0
  113. package/temp/rtk/src/filters/helm.toml +29 -0
  114. package/temp/rtk/src/filters/iptables.toml +27 -0
  115. package/temp/rtk/src/filters/jj.toml +28 -0
  116. package/temp/rtk/src/filters/jq.toml +24 -0
  117. package/temp/rtk/src/filters/make.toml +41 -0
  118. package/temp/rtk/src/filters/markdownlint.toml +24 -0
  119. package/temp/rtk/src/filters/mix-compile.toml +27 -0
  120. package/temp/rtk/src/filters/mix-format.toml +15 -0
  121. package/temp/rtk/src/filters/mvn-build.toml +44 -0
  122. package/temp/rtk/src/filters/oxlint.toml +43 -0
  123. package/temp/rtk/src/filters/ping.toml +63 -0
  124. package/temp/rtk/src/filters/pio-run.toml +40 -0
  125. package/temp/rtk/src/filters/poetry-install.toml +50 -0
  126. package/temp/rtk/src/filters/pre-commit.toml +35 -0
  127. package/temp/rtk/src/filters/ps.toml +16 -0
  128. package/temp/rtk/src/filters/quarto-render.toml +41 -0
  129. package/temp/rtk/src/filters/rsync.toml +48 -0
  130. package/temp/rtk/src/filters/shellcheck.toml +27 -0
  131. package/temp/rtk/src/filters/shopify-theme.toml +29 -0
  132. package/temp/rtk/src/filters/skopeo.toml +45 -0
  133. package/temp/rtk/src/filters/sops.toml +16 -0
  134. package/temp/rtk/src/filters/ssh.toml +44 -0
  135. package/temp/rtk/src/filters/stat.toml +34 -0
  136. package/temp/rtk/src/filters/swift-build.toml +41 -0
  137. package/temp/rtk/src/filters/systemctl-status.toml +33 -0
  138. package/temp/rtk/src/filters/terraform-plan.toml +35 -0
  139. package/temp/rtk/src/filters/tofu-fmt.toml +16 -0
  140. package/temp/rtk/src/filters/tofu-init.toml +38 -0
  141. package/temp/rtk/src/filters/tofu-plan.toml +35 -0
  142. package/temp/rtk/src/filters/tofu-validate.toml +17 -0
  143. package/temp/rtk/src/filters/trunk-build.toml +39 -0
  144. package/temp/rtk/src/filters/ty.toml +50 -0
  145. package/temp/rtk/src/filters/uv-sync.toml +37 -0
  146. package/temp/rtk/src/filters/xcodebuild.toml +99 -0
  147. package/temp/rtk/src/filters/yamllint.toml +25 -0
  148. package/temp/rtk/src/find_cmd.rs +598 -0
  149. package/temp/rtk/src/format_cmd.rs +386 -0
  150. package/temp/rtk/src/gain.rs +723 -0
  151. package/temp/rtk/src/gh_cmd.rs +1651 -0
  152. package/temp/rtk/src/git.rs +2012 -0
  153. package/temp/rtk/src/go_cmd.rs +592 -0
  154. package/temp/rtk/src/golangci_cmd.rs +254 -0
  155. package/temp/rtk/src/grep_cmd.rs +288 -0
  156. package/temp/rtk/src/gt_cmd.rs +810 -0
  157. package/temp/rtk/src/hook_audit_cmd.rs +283 -0
  158. package/temp/rtk/src/hook_check.rs +171 -0
  159. package/temp/rtk/src/init.rs +1859 -0
  160. package/temp/rtk/src/integrity.rs +537 -0
  161. package/temp/rtk/src/json_cmd.rs +231 -0
  162. package/temp/rtk/src/learn/detector.rs +628 -0
  163. package/temp/rtk/src/learn/mod.rs +119 -0
  164. package/temp/rtk/src/learn/report.rs +184 -0
  165. package/temp/rtk/src/lint_cmd.rs +694 -0
  166. package/temp/rtk/src/local_llm.rs +316 -0
  167. package/temp/rtk/src/log_cmd.rs +248 -0
  168. package/temp/rtk/src/ls.rs +324 -0
  169. package/temp/rtk/src/main.rs +2482 -0
  170. package/temp/rtk/src/mypy_cmd.rs +389 -0
  171. package/temp/rtk/src/next_cmd.rs +241 -0
  172. package/temp/rtk/src/npm_cmd.rs +236 -0
  173. package/temp/rtk/src/parser/README.md +267 -0
  174. package/temp/rtk/src/parser/error.rs +46 -0
  175. package/temp/rtk/src/parser/formatter.rs +336 -0
  176. package/temp/rtk/src/parser/mod.rs +311 -0
  177. package/temp/rtk/src/parser/types.rs +119 -0
  178. package/temp/rtk/src/pip_cmd.rs +302 -0
  179. package/temp/rtk/src/playwright_cmd.rs +479 -0
  180. package/temp/rtk/src/pnpm_cmd.rs +573 -0
  181. package/temp/rtk/src/prettier_cmd.rs +221 -0
  182. package/temp/rtk/src/prisma_cmd.rs +482 -0
  183. package/temp/rtk/src/psql_cmd.rs +382 -0
  184. package/temp/rtk/src/pytest_cmd.rs +384 -0
  185. package/temp/rtk/src/read.rs +217 -0
  186. package/temp/rtk/src/rewrite_cmd.rs +50 -0
  187. package/temp/rtk/src/ruff_cmd.rs +402 -0
  188. package/temp/rtk/src/runner.rs +271 -0
  189. package/temp/rtk/src/summary.rs +297 -0
  190. package/temp/rtk/src/tee.rs +405 -0
  191. package/temp/rtk/src/telemetry.rs +248 -0
  192. package/temp/rtk/src/toml_filter.rs +1655 -0
  193. package/temp/rtk/src/tracking.rs +1416 -0
  194. package/temp/rtk/src/tree.rs +209 -0
  195. package/temp/rtk/src/tsc_cmd.rs +259 -0
  196. package/temp/rtk/src/utils.rs +432 -0
  197. package/temp/rtk/src/verify_cmd.rs +47 -0
  198. package/temp/rtk/src/vitest_cmd.rs +385 -0
  199. package/temp/rtk/src/wc_cmd.rs +401 -0
  200. package/temp/rtk/src/wget_cmd.rs +260 -0
  201. package/temp/rtk/tests/fixtures/dotnet/build_failed.txt +11 -0
  202. package/temp/rtk/tests/fixtures/dotnet/format_changes.json +31 -0
  203. package/temp/rtk/tests/fixtures/dotnet/format_empty.json +1 -0
  204. package/temp/rtk/tests/fixtures/dotnet/format_success.json +12 -0
  205. 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
+ }