@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,432 @@
1
+ //! Utility functions for text processing and command execution.
2
+ //!
3
+ //! Provides common helpers used across rtk commands:
4
+ //! - ANSI color code stripping
5
+ //! - Text truncation
6
+ //! - Command execution with error context
7
+
8
+ use anyhow::{Context, Result};
9
+ use regex::Regex;
10
+ use std::process::Command;
11
+
12
+ /// Truncates a string to `max_len` characters, appending `...` if needed.
13
+ ///
14
+ /// # Arguments
15
+ /// * `s` - The string to truncate
16
+ /// * `max_len` - Maximum length before truncation (minimum 3 to include "...")
17
+ ///
18
+ /// # Examples
19
+ /// ```
20
+ /// use rtk::utils::truncate;
21
+ /// assert_eq!(truncate("hello world", 8), "hello...");
22
+ /// assert_eq!(truncate("hi", 10), "hi");
23
+ /// ```
24
+ pub fn truncate(s: &str, max_len: usize) -> String {
25
+ let char_count = s.chars().count();
26
+ if char_count <= max_len {
27
+ s.to_string()
28
+ } else if max_len < 3 {
29
+ // If max_len is too small, just return "..."
30
+ "...".to_string()
31
+ } else {
32
+ format!("{}...", s.chars().take(max_len - 3).collect::<String>())
33
+ }
34
+ }
35
+
36
+ /// Strips ANSI escape codes (colors, styles) from a string.
37
+ ///
38
+ /// # Arguments
39
+ /// * `text` - Text potentially containing ANSI escape codes
40
+ ///
41
+ /// # Examples
42
+ /// ```
43
+ /// use rtk::utils::strip_ansi;
44
+ /// let colored = "\x1b[31mError\x1b[0m";
45
+ /// assert_eq!(strip_ansi(colored), "Error");
46
+ /// ```
47
+ pub fn strip_ansi(text: &str) -> String {
48
+ lazy_static::lazy_static! {
49
+ static ref ANSI_RE: Regex = Regex::new(r"\x1b\[[0-9;]*[a-zA-Z]").unwrap();
50
+ }
51
+ ANSI_RE.replace_all(text, "").to_string()
52
+ }
53
+
54
+ /// Executes a command and returns cleaned stdout/stderr.
55
+ ///
56
+ /// # Arguments
57
+ /// * `cmd` - Command to execute (e.g. "eslint")
58
+ /// * `args` - Command arguments
59
+ ///
60
+ /// # Returns
61
+ /// `(stdout: String, stderr: String, exit_code: i32)`
62
+ ///
63
+ /// # Examples
64
+ /// ```no_run
65
+ /// use rtk::utils::execute_command;
66
+ /// let (stdout, stderr, code) = execute_command("echo", &["test"]).unwrap();
67
+ /// assert_eq!(code, 0);
68
+ /// ```
69
+ #[allow(dead_code)]
70
+ pub fn execute_command(cmd: &str, args: &[&str]) -> Result<(String, String, i32)> {
71
+ let output = Command::new(cmd)
72
+ .args(args)
73
+ .output()
74
+ .context(format!("Failed to execute {}", cmd))?;
75
+
76
+ let stdout = String::from_utf8_lossy(&output.stdout).to_string();
77
+ let stderr = String::from_utf8_lossy(&output.stderr).to_string();
78
+ let exit_code = output.status.code().unwrap_or(-1);
79
+
80
+ Ok((stdout, stderr, exit_code))
81
+ }
82
+
83
+ /// Formats a token count with K/M suffixes for readability.
84
+ ///
85
+ /// # Arguments
86
+ /// * `n` - Token count
87
+ ///
88
+ /// # Returns
89
+ /// Formatted string (e.g. "1.2M", "59.2K", "694")
90
+ ///
91
+ /// # Examples
92
+ /// ```
93
+ /// use rtk::utils::format_tokens;
94
+ /// assert_eq!(format_tokens(1_234_567), "1.2M");
95
+ /// assert_eq!(format_tokens(59_234), "59.2K");
96
+ /// assert_eq!(format_tokens(694), "694");
97
+ /// ```
98
+ pub fn format_tokens(n: usize) -> String {
99
+ if n >= 1_000_000 {
100
+ format!("{:.1}M", n as f64 / 1_000_000.0)
101
+ } else if n >= 1_000 {
102
+ format!("{:.1}K", n as f64 / 1_000.0)
103
+ } else {
104
+ format!("{}", n)
105
+ }
106
+ }
107
+
108
+ /// Formats a USD amount with adaptive precision.
109
+ ///
110
+ /// # Arguments
111
+ /// * `amount` - Amount in dollars
112
+ ///
113
+ /// # Returns
114
+ /// Formatted string with $ prefix
115
+ ///
116
+ /// # Examples
117
+ /// ```
118
+ /// use rtk::utils::format_usd;
119
+ /// assert_eq!(format_usd(1234.567), "$1234.57");
120
+ /// assert_eq!(format_usd(12.345), "$12.35");
121
+ /// assert_eq!(format_usd(0.123), "$0.12");
122
+ /// assert_eq!(format_usd(0.0096), "$0.0096");
123
+ /// ```
124
+ pub fn format_usd(amount: f64) -> String {
125
+ if !amount.is_finite() {
126
+ return "$0.00".to_string();
127
+ }
128
+ if amount >= 0.01 {
129
+ format!("${:.2}", amount)
130
+ } else {
131
+ format!("${:.4}", amount)
132
+ }
133
+ }
134
+
135
+ /// Format cost-per-token as $/MTok (e.g., "$3.86/MTok")
136
+ ///
137
+ /// # Arguments
138
+ /// * `cpt` - Cost per token (not per million tokens)
139
+ ///
140
+ /// # Returns
141
+ /// Formatted string like "$3.86/MTok"
142
+ ///
143
+ /// # Examples
144
+ /// ```
145
+ /// use rtk::utils::format_cpt;
146
+ /// assert_eq!(format_cpt(0.000003), "$3.00/MTok");
147
+ /// assert_eq!(format_cpt(0.0000038), "$3.80/MTok");
148
+ /// assert_eq!(format_cpt(0.00000386), "$3.86/MTok");
149
+ /// ```
150
+ pub fn format_cpt(cpt: f64) -> String {
151
+ if !cpt.is_finite() || cpt <= 0.0 {
152
+ return "$0.00/MTok".to_string();
153
+ }
154
+ let cpt_per_million = cpt * 1_000_000.0;
155
+ format!("${:.2}/MTok", cpt_per_million)
156
+ }
157
+
158
+ /// Join items into a newline-separated string, appending an overflow hint when total > max.
159
+ ///
160
+ /// # Examples
161
+ /// ```
162
+ /// use rtk::utils::join_with_overflow;
163
+ /// let items = vec!["a".to_string(), "b".to_string()];
164
+ /// assert_eq!(join_with_overflow(&items, 5, 3, "items"), "a\nb\n... +2 more items");
165
+ /// assert_eq!(join_with_overflow(&items, 2, 3, "items"), "a\nb");
166
+ /// ```
167
+ pub fn join_with_overflow(items: &[String], total: usize, max: usize, label: &str) -> String {
168
+ let mut out = items.join("\n");
169
+ if total > max {
170
+ out.push_str(&format!("\n... +{} more {}", total - max, label));
171
+ }
172
+ out
173
+ }
174
+
175
+ /// Truncate an ISO 8601 datetime string to just the date portion (first 10 chars).
176
+ ///
177
+ /// # Examples
178
+ /// ```
179
+ /// use rtk::utils::truncate_iso_date;
180
+ /// assert_eq!(truncate_iso_date("2024-01-15T10:30:00Z"), "2024-01-15");
181
+ /// assert_eq!(truncate_iso_date("2024-01-15"), "2024-01-15");
182
+ /// assert_eq!(truncate_iso_date("short"), "short");
183
+ /// ```
184
+ pub fn truncate_iso_date(date: &str) -> &str {
185
+ if date.len() >= 10 {
186
+ &date[..10]
187
+ } else {
188
+ date
189
+ }
190
+ }
191
+
192
+ /// Format a confirmation message: "ok \<action\> \<detail\>"
193
+ /// Used for write operations (merge, create, comment, edit, etc.)
194
+ ///
195
+ /// # Examples
196
+ /// ```
197
+ /// use rtk::utils::ok_confirmation;
198
+ /// assert_eq!(ok_confirmation("merged", "#42"), "ok merged #42");
199
+ /// assert_eq!(ok_confirmation("created", "PR #5 https://..."), "ok created PR #5 https://...");
200
+ /// ```
201
+ pub fn ok_confirmation(action: &str, detail: &str) -> String {
202
+ if detail.is_empty() {
203
+ format!("ok {}", action)
204
+ } else {
205
+ format!("ok {} {}", action, detail)
206
+ }
207
+ }
208
+
209
+ /// Detect the package manager used in the current directory.
210
+ /// Returns "pnpm", "yarn", or "npm" based on lockfile presence.
211
+ ///
212
+ /// # Examples
213
+ /// ```no_run
214
+ /// use rtk::utils::detect_package_manager;
215
+ /// let pm = detect_package_manager();
216
+ /// // Returns "pnpm" if pnpm-lock.yaml exists, "yarn" if yarn.lock, else "npm"
217
+ /// ```
218
+ #[allow(dead_code)]
219
+ pub fn detect_package_manager() -> &'static str {
220
+ if std::path::Path::new("pnpm-lock.yaml").exists() {
221
+ "pnpm"
222
+ } else if std::path::Path::new("yarn.lock").exists() {
223
+ "yarn"
224
+ } else {
225
+ "npm"
226
+ }
227
+ }
228
+
229
+ /// Build a Command using the detected package manager's exec mechanism.
230
+ /// Returns a Command ready to have tool-specific args appended.
231
+ pub fn package_manager_exec(tool: &str) -> Command {
232
+ let tool_exists = Command::new("which")
233
+ .arg(tool)
234
+ .output()
235
+ .map(|o| o.status.success())
236
+ .unwrap_or(false);
237
+
238
+ if tool_exists {
239
+ Command::new(tool)
240
+ } else {
241
+ let pm = detect_package_manager();
242
+ match pm {
243
+ "pnpm" => {
244
+ let mut c = Command::new("pnpm");
245
+ c.arg("exec").arg("--").arg(tool);
246
+ c
247
+ }
248
+ "yarn" => {
249
+ let mut c = Command::new("yarn");
250
+ c.arg("exec").arg("--").arg(tool);
251
+ c
252
+ }
253
+ _ => {
254
+ let mut c = Command::new("npx");
255
+ c.arg("--no-install").arg("--").arg(tool);
256
+ c
257
+ }
258
+ }
259
+ }
260
+ }
261
+
262
+ #[cfg(test)]
263
+ mod tests {
264
+ use super::*;
265
+
266
+ #[test]
267
+ fn test_truncate_short_string() {
268
+ assert_eq!(truncate("hello", 10), "hello");
269
+ }
270
+
271
+ #[test]
272
+ fn test_truncate_long_string() {
273
+ let result = truncate("hello world", 8);
274
+ assert_eq!(result, "hello...");
275
+ }
276
+
277
+ #[test]
278
+ fn test_truncate_exact_length() {
279
+ assert_eq!(truncate("hello", 5), "hello");
280
+ }
281
+
282
+ #[test]
283
+ fn test_truncate_edge_case() {
284
+ // max_len < 3 returns just "..."
285
+ assert_eq!(truncate("hello", 2), "...");
286
+ // When string length equals max_len, return as is
287
+ assert_eq!(truncate("abc", 3), "abc");
288
+ // When string is longer and max_len is exactly 3, return "..."
289
+ assert_eq!(truncate("hello world", 3), "...");
290
+ }
291
+
292
+ #[test]
293
+ fn test_strip_ansi_simple() {
294
+ let input = "\x1b[31mError\x1b[0m";
295
+ assert_eq!(strip_ansi(input), "Error");
296
+ }
297
+
298
+ #[test]
299
+ fn test_strip_ansi_multiple() {
300
+ let input = "\x1b[1m\x1b[32mSuccess\x1b[0m\x1b[0m";
301
+ assert_eq!(strip_ansi(input), "Success");
302
+ }
303
+
304
+ #[test]
305
+ fn test_strip_ansi_no_codes() {
306
+ assert_eq!(strip_ansi("plain text"), "plain text");
307
+ }
308
+
309
+ #[test]
310
+ fn test_strip_ansi_complex() {
311
+ let input = "\x1b[32mGreen\x1b[0m normal \x1b[31mRed\x1b[0m";
312
+ assert_eq!(strip_ansi(input), "Green normal Red");
313
+ }
314
+
315
+ #[test]
316
+ fn test_execute_command_success() {
317
+ let result = execute_command("echo", &["test"]);
318
+ assert!(result.is_ok());
319
+ let (stdout, _, code) = result.unwrap();
320
+ assert_eq!(code, 0);
321
+ assert!(stdout.contains("test"));
322
+ }
323
+
324
+ #[test]
325
+ fn test_execute_command_failure() {
326
+ let result = execute_command("nonexistent_command_xyz_12345", &[]);
327
+ assert!(result.is_err());
328
+ }
329
+
330
+ #[test]
331
+ fn test_format_tokens_millions() {
332
+ assert_eq!(format_tokens(1_234_567), "1.2M");
333
+ assert_eq!(format_tokens(12_345_678), "12.3M");
334
+ }
335
+
336
+ #[test]
337
+ fn test_format_tokens_thousands() {
338
+ assert_eq!(format_tokens(59_234), "59.2K");
339
+ assert_eq!(format_tokens(1_000), "1.0K");
340
+ }
341
+
342
+ #[test]
343
+ fn test_format_tokens_small() {
344
+ assert_eq!(format_tokens(694), "694");
345
+ assert_eq!(format_tokens(0), "0");
346
+ }
347
+
348
+ #[test]
349
+ fn test_format_usd_large() {
350
+ assert_eq!(format_usd(1234.567), "$1234.57");
351
+ assert_eq!(format_usd(1000.0), "$1000.00");
352
+ }
353
+
354
+ #[test]
355
+ fn test_format_usd_medium() {
356
+ assert_eq!(format_usd(12.345), "$12.35");
357
+ assert_eq!(format_usd(0.99), "$0.99");
358
+ }
359
+
360
+ #[test]
361
+ fn test_format_usd_small() {
362
+ assert_eq!(format_usd(0.0096), "$0.0096");
363
+ assert_eq!(format_usd(0.0001), "$0.0001");
364
+ }
365
+
366
+ #[test]
367
+ fn test_format_usd_edge() {
368
+ assert_eq!(format_usd(0.01), "$0.01");
369
+ assert_eq!(format_usd(0.009), "$0.0090");
370
+ }
371
+
372
+ #[test]
373
+ fn test_ok_confirmation_with_detail() {
374
+ assert_eq!(ok_confirmation("merged", "#42"), "ok merged #42");
375
+ assert_eq!(
376
+ ok_confirmation("created", "PR #5 https://github.com/foo/bar/pull/5"),
377
+ "ok created PR #5 https://github.com/foo/bar/pull/5"
378
+ );
379
+ }
380
+
381
+ #[test]
382
+ fn test_ok_confirmation_no_detail() {
383
+ assert_eq!(ok_confirmation("commented", ""), "ok commented");
384
+ }
385
+
386
+ #[test]
387
+ fn test_format_cpt_normal() {
388
+ assert_eq!(format_cpt(0.000003), "$3.00/MTok");
389
+ assert_eq!(format_cpt(0.0000038), "$3.80/MTok");
390
+ assert_eq!(format_cpt(0.00000386), "$3.86/MTok");
391
+ }
392
+
393
+ #[test]
394
+ fn test_format_cpt_edge_cases() {
395
+ assert_eq!(format_cpt(0.0), "$0.00/MTok"); // zero
396
+ assert_eq!(format_cpt(-0.000001), "$0.00/MTok"); // negative
397
+ assert_eq!(format_cpt(f64::INFINITY), "$0.00/MTok"); // infinite
398
+ assert_eq!(format_cpt(f64::NAN), "$0.00/MTok"); // NaN
399
+ }
400
+
401
+ #[test]
402
+ fn test_detect_package_manager_default() {
403
+ // In the test environment (rtk repo), there's no JS lockfile
404
+ // so it should default to "npm"
405
+ let pm = detect_package_manager();
406
+ assert!(["pnpm", "yarn", "npm"].contains(&pm));
407
+ }
408
+
409
+ #[test]
410
+ fn test_truncate_multibyte_thai() {
411
+ // Thai characters are 3 bytes each
412
+ let thai = "สวัสดีครับ";
413
+ let result = truncate(thai, 5);
414
+ // Should not panic, should produce valid UTF-8
415
+ assert!(result.len() <= thai.len());
416
+ assert!(result.ends_with("..."));
417
+ }
418
+
419
+ #[test]
420
+ fn test_truncate_multibyte_emoji() {
421
+ let emoji = "🎉🎊🎈🎁🎂🎄🎃🎆🎇✨";
422
+ let result = truncate(emoji, 5);
423
+ assert!(result.ends_with("..."));
424
+ }
425
+
426
+ #[test]
427
+ fn test_truncate_multibyte_cjk() {
428
+ let cjk = "你好世界测试字符串";
429
+ let result = truncate(cjk, 6);
430
+ assert!(result.ends_with("..."));
431
+ }
432
+ }
@@ -0,0 +1,47 @@
1
+ use anyhow::Result;
2
+
3
+ use crate::toml_filter;
4
+
5
+ /// Run TOML filter inline tests.
6
+ ///
7
+ /// - `filter`: if `Some`, only run tests for that filter name
8
+ /// - `require_all`: fail if any filter has no inline tests
9
+ pub fn run(filter: Option<String>, require_all: bool) -> Result<()> {
10
+ let results = toml_filter::run_filter_tests(filter.as_deref());
11
+
12
+ let total = results.outcomes.len();
13
+ let passed = results.outcomes.iter().filter(|o| o.passed).count();
14
+ let failed = total - passed;
15
+
16
+ // Print failures with details
17
+ for outcome in &results.outcomes {
18
+ if !outcome.passed {
19
+ eprintln!(
20
+ "FAIL [{}] {}\n expected: {:?}\n actual: {:?}",
21
+ outcome.filter_name, outcome.test_name, outcome.expected, outcome.actual
22
+ );
23
+ }
24
+ }
25
+
26
+ if total == 0 {
27
+ println!("No inline tests found.");
28
+ } else {
29
+ println!("{}/{} tests passed", passed, total);
30
+ }
31
+
32
+ if require_all && !results.filters_without_tests.is_empty() {
33
+ for name in &results.filters_without_tests {
34
+ eprintln!("MISSING tests for filter: {}", name);
35
+ }
36
+ anyhow::bail!(
37
+ "{} filter(s) have no inline tests (use --require-all in CI)",
38
+ results.filters_without_tests.len()
39
+ );
40
+ }
41
+
42
+ if failed > 0 {
43
+ anyhow::bail!("{} test(s) failed", failed);
44
+ }
45
+
46
+ Ok(())
47
+ }