@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.
- package/dist/cli.js +64 -16
- package/package.json +1 -1
- package/src/ai.ts +8 -0
- package/src/cli.tsx +57 -18
- package/src/output-processor.ts +6 -1
- package/src/output-store.ts +58 -12
- package/src/tool-profiles.ts +139 -0
- package/temp/rtk/.claude/agents/code-reviewer.md +0 -221
- package/temp/rtk/.claude/agents/debugger.md +0 -519
- package/temp/rtk/.claude/agents/rtk-testing-specialist.md +0 -461
- package/temp/rtk/.claude/agents/rust-rtk.md +0 -511
- package/temp/rtk/.claude/agents/technical-writer.md +0 -355
- package/temp/rtk/.claude/commands/diagnose.md +0 -352
- package/temp/rtk/.claude/commands/test-routing.md +0 -362
- package/temp/rtk/.claude/hooks/bash/pre-commit-format.sh +0 -16
- package/temp/rtk/.claude/hooks/rtk-rewrite.sh +0 -70
- package/temp/rtk/.claude/hooks/rtk-suggest.sh +0 -152
- package/temp/rtk/.claude/rules/cli-testing.md +0 -526
- package/temp/rtk/.claude/skills/issue-triage/SKILL.md +0 -348
- package/temp/rtk/.claude/skills/issue-triage/templates/issue-comment.md +0 -134
- package/temp/rtk/.claude/skills/performance.md +0 -435
- package/temp/rtk/.claude/skills/pr-triage/SKILL.md +0 -315
- package/temp/rtk/.claude/skills/pr-triage/templates/review-comment.md +0 -71
- package/temp/rtk/.claude/skills/repo-recap.md +0 -206
- package/temp/rtk/.claude/skills/rtk-tdd/SKILL.md +0 -78
- package/temp/rtk/.claude/skills/rtk-tdd/references/testing-patterns.md +0 -124
- package/temp/rtk/.claude/skills/security-guardian.md +0 -503
- package/temp/rtk/.claude/skills/ship.md +0 -404
- package/temp/rtk/.github/workflows/benchmark.yml +0 -34
- package/temp/rtk/.github/workflows/dco-check.yaml +0 -12
- package/temp/rtk/.github/workflows/release-please.yml +0 -51
- package/temp/rtk/.github/workflows/release.yml +0 -343
- package/temp/rtk/.github/workflows/security-check.yml +0 -135
- package/temp/rtk/.github/workflows/validate-docs.yml +0 -78
- package/temp/rtk/.release-please-manifest.json +0 -3
- package/temp/rtk/ARCHITECTURE.md +0 -1491
- package/temp/rtk/CHANGELOG.md +0 -640
- package/temp/rtk/CLAUDE.md +0 -605
- package/temp/rtk/CONTRIBUTING.md +0 -199
- package/temp/rtk/Cargo.lock +0 -1668
- package/temp/rtk/Cargo.toml +0 -64
- package/temp/rtk/Formula/rtk.rb +0 -43
- package/temp/rtk/INSTALL.md +0 -390
- package/temp/rtk/LICENSE +0 -21
- package/temp/rtk/README.md +0 -386
- package/temp/rtk/README_es.md +0 -159
- package/temp/rtk/README_fr.md +0 -197
- package/temp/rtk/README_ja.md +0 -159
- package/temp/rtk/README_ko.md +0 -159
- package/temp/rtk/README_zh.md +0 -167
- package/temp/rtk/ROADMAP.md +0 -15
- package/temp/rtk/SECURITY.md +0 -217
- package/temp/rtk/TEST_EXEC_TIME.md +0 -102
- package/temp/rtk/build.rs +0 -57
- package/temp/rtk/docs/AUDIT_GUIDE.md +0 -432
- package/temp/rtk/docs/FEATURES.md +0 -1410
- package/temp/rtk/docs/TROUBLESHOOTING.md +0 -309
- package/temp/rtk/docs/filter-workflow.md +0 -102
- package/temp/rtk/docs/images/gain-dashboard.jpg +0 -0
- package/temp/rtk/docs/tracking.md +0 -583
- package/temp/rtk/hooks/opencode-rtk.ts +0 -39
- package/temp/rtk/hooks/rtk-awareness.md +0 -29
- package/temp/rtk/hooks/rtk-rewrite.sh +0 -61
- package/temp/rtk/hooks/test-rtk-rewrite.sh +0 -442
- package/temp/rtk/install.sh +0 -124
- package/temp/rtk/release-please-config.json +0 -10
- package/temp/rtk/scripts/benchmark.sh +0 -592
- package/temp/rtk/scripts/check-installation.sh +0 -162
- package/temp/rtk/scripts/install-local.sh +0 -37
- package/temp/rtk/scripts/rtk-economics.sh +0 -137
- package/temp/rtk/scripts/test-all.sh +0 -561
- package/temp/rtk/scripts/test-aristote.sh +0 -227
- package/temp/rtk/scripts/test-tracking.sh +0 -79
- package/temp/rtk/scripts/update-readme-metrics.sh +0 -32
- package/temp/rtk/scripts/validate-docs.sh +0 -73
- package/temp/rtk/src/aws_cmd.rs +0 -880
- package/temp/rtk/src/binlog.rs +0 -1645
- package/temp/rtk/src/cargo_cmd.rs +0 -1727
- package/temp/rtk/src/cc_economics.rs +0 -1157
- package/temp/rtk/src/ccusage.rs +0 -340
- package/temp/rtk/src/config.rs +0 -187
- package/temp/rtk/src/container.rs +0 -855
- package/temp/rtk/src/curl_cmd.rs +0 -134
- package/temp/rtk/src/deps.rs +0 -268
- package/temp/rtk/src/diff_cmd.rs +0 -367
- package/temp/rtk/src/discover/mod.rs +0 -274
- package/temp/rtk/src/discover/provider.rs +0 -388
- package/temp/rtk/src/discover/registry.rs +0 -2022
- package/temp/rtk/src/discover/report.rs +0 -202
- package/temp/rtk/src/discover/rules.rs +0 -667
- package/temp/rtk/src/display_helpers.rs +0 -402
- package/temp/rtk/src/dotnet_cmd.rs +0 -1771
- package/temp/rtk/src/dotnet_format_report.rs +0 -133
- package/temp/rtk/src/dotnet_trx.rs +0 -593
- package/temp/rtk/src/env_cmd.rs +0 -204
- package/temp/rtk/src/filter.rs +0 -462
- package/temp/rtk/src/filters/README.md +0 -52
- package/temp/rtk/src/filters/ansible-playbook.toml +0 -34
- package/temp/rtk/src/filters/basedpyright.toml +0 -47
- package/temp/rtk/src/filters/biome.toml +0 -45
- package/temp/rtk/src/filters/brew-install.toml +0 -37
- package/temp/rtk/src/filters/composer-install.toml +0 -40
- package/temp/rtk/src/filters/df.toml +0 -16
- package/temp/rtk/src/filters/dotnet-build.toml +0 -64
- package/temp/rtk/src/filters/du.toml +0 -16
- package/temp/rtk/src/filters/fail2ban-client.toml +0 -15
- package/temp/rtk/src/filters/gcc.toml +0 -49
- package/temp/rtk/src/filters/gcloud.toml +0 -22
- package/temp/rtk/src/filters/hadolint.toml +0 -24
- package/temp/rtk/src/filters/helm.toml +0 -29
- package/temp/rtk/src/filters/iptables.toml +0 -27
- package/temp/rtk/src/filters/jj.toml +0 -28
- package/temp/rtk/src/filters/jq.toml +0 -24
- package/temp/rtk/src/filters/make.toml +0 -41
- package/temp/rtk/src/filters/markdownlint.toml +0 -24
- package/temp/rtk/src/filters/mix-compile.toml +0 -27
- package/temp/rtk/src/filters/mix-format.toml +0 -15
- package/temp/rtk/src/filters/mvn-build.toml +0 -44
- package/temp/rtk/src/filters/oxlint.toml +0 -43
- package/temp/rtk/src/filters/ping.toml +0 -63
- package/temp/rtk/src/filters/pio-run.toml +0 -40
- package/temp/rtk/src/filters/poetry-install.toml +0 -50
- package/temp/rtk/src/filters/pre-commit.toml +0 -35
- package/temp/rtk/src/filters/ps.toml +0 -16
- package/temp/rtk/src/filters/quarto-render.toml +0 -41
- package/temp/rtk/src/filters/rsync.toml +0 -48
- package/temp/rtk/src/filters/shellcheck.toml +0 -27
- package/temp/rtk/src/filters/shopify-theme.toml +0 -29
- package/temp/rtk/src/filters/skopeo.toml +0 -45
- package/temp/rtk/src/filters/sops.toml +0 -16
- package/temp/rtk/src/filters/ssh.toml +0 -44
- package/temp/rtk/src/filters/stat.toml +0 -34
- package/temp/rtk/src/filters/swift-build.toml +0 -41
- package/temp/rtk/src/filters/systemctl-status.toml +0 -33
- package/temp/rtk/src/filters/terraform-plan.toml +0 -35
- package/temp/rtk/src/filters/tofu-fmt.toml +0 -16
- package/temp/rtk/src/filters/tofu-init.toml +0 -38
- package/temp/rtk/src/filters/tofu-plan.toml +0 -35
- package/temp/rtk/src/filters/tofu-validate.toml +0 -17
- package/temp/rtk/src/filters/trunk-build.toml +0 -39
- package/temp/rtk/src/filters/ty.toml +0 -50
- package/temp/rtk/src/filters/uv-sync.toml +0 -37
- package/temp/rtk/src/filters/xcodebuild.toml +0 -99
- package/temp/rtk/src/filters/yamllint.toml +0 -25
- package/temp/rtk/src/find_cmd.rs +0 -598
- package/temp/rtk/src/format_cmd.rs +0 -386
- package/temp/rtk/src/gain.rs +0 -723
- package/temp/rtk/src/gh_cmd.rs +0 -1651
- package/temp/rtk/src/git.rs +0 -2012
- package/temp/rtk/src/go_cmd.rs +0 -592
- package/temp/rtk/src/golangci_cmd.rs +0 -254
- package/temp/rtk/src/grep_cmd.rs +0 -288
- package/temp/rtk/src/gt_cmd.rs +0 -810
- package/temp/rtk/src/hook_audit_cmd.rs +0 -283
- package/temp/rtk/src/hook_check.rs +0 -171
- package/temp/rtk/src/init.rs +0 -1859
- package/temp/rtk/src/integrity.rs +0 -537
- package/temp/rtk/src/json_cmd.rs +0 -231
- package/temp/rtk/src/learn/detector.rs +0 -628
- package/temp/rtk/src/learn/mod.rs +0 -119
- package/temp/rtk/src/learn/report.rs +0 -184
- package/temp/rtk/src/lint_cmd.rs +0 -694
- package/temp/rtk/src/local_llm.rs +0 -316
- package/temp/rtk/src/log_cmd.rs +0 -248
- package/temp/rtk/src/ls.rs +0 -324
- package/temp/rtk/src/main.rs +0 -2482
- package/temp/rtk/src/mypy_cmd.rs +0 -389
- package/temp/rtk/src/next_cmd.rs +0 -241
- package/temp/rtk/src/npm_cmd.rs +0 -236
- package/temp/rtk/src/parser/README.md +0 -267
- package/temp/rtk/src/parser/error.rs +0 -46
- package/temp/rtk/src/parser/formatter.rs +0 -336
- package/temp/rtk/src/parser/mod.rs +0 -311
- package/temp/rtk/src/parser/types.rs +0 -119
- package/temp/rtk/src/pip_cmd.rs +0 -302
- package/temp/rtk/src/playwright_cmd.rs +0 -479
- package/temp/rtk/src/pnpm_cmd.rs +0 -573
- package/temp/rtk/src/prettier_cmd.rs +0 -221
- package/temp/rtk/src/prisma_cmd.rs +0 -482
- package/temp/rtk/src/psql_cmd.rs +0 -382
- package/temp/rtk/src/pytest_cmd.rs +0 -384
- package/temp/rtk/src/read.rs +0 -217
- package/temp/rtk/src/rewrite_cmd.rs +0 -50
- package/temp/rtk/src/ruff_cmd.rs +0 -402
- package/temp/rtk/src/runner.rs +0 -271
- package/temp/rtk/src/summary.rs +0 -297
- package/temp/rtk/src/tee.rs +0 -405
- package/temp/rtk/src/telemetry.rs +0 -248
- package/temp/rtk/src/toml_filter.rs +0 -1655
- package/temp/rtk/src/tracking.rs +0 -1416
- package/temp/rtk/src/tree.rs +0 -209
- package/temp/rtk/src/tsc_cmd.rs +0 -259
- package/temp/rtk/src/utils.rs +0 -432
- package/temp/rtk/src/verify_cmd.rs +0 -47
- package/temp/rtk/src/vitest_cmd.rs +0 -385
- package/temp/rtk/src/wc_cmd.rs +0 -401
- package/temp/rtk/src/wget_cmd.rs +0 -260
- package/temp/rtk/tests/fixtures/dotnet/build_failed.txt +0 -11
- package/temp/rtk/tests/fixtures/dotnet/format_changes.json +0 -31
- package/temp/rtk/tests/fixtures/dotnet/format_empty.json +0 -1
- package/temp/rtk/tests/fixtures/dotnet/format_success.json +0 -12
- package/temp/rtk/tests/fixtures/dotnet/test_failed.txt +0 -18
package/temp/rtk/src/utils.rs
DELETED
|
@@ -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
|
-
}
|