@hasna/terminal 2.3.0 → 2.3.1

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