1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
|
use tokio::{fs, process::Command};
/// Confirm the current operation
/// Waits for user input of 'y' or 'n'
pub async fn confirm_hint(text: impl Into<String>) -> bool {
use tokio::io::{self, AsyncBufReadExt, AsyncWriteExt, BufReader};
let prompt = text.into().trim().to_string();
let mut stdout = io::stdout();
let mut stdin = BufReader::new(io::stdin());
stdout
.write_all(prompt.as_bytes())
.await
.expect("Failed to write prompt");
stdout.flush().await.expect("Failed to flush stdout");
let mut input = String::new();
stdin
.read_line(&mut input)
.await
.expect("Failed to read input");
input.trim().eq_ignore_ascii_case("y")
}
/// Confirm the current operation, or execute a closure if rejected
/// Waits for user input of 'y' or 'n'
/// If 'n' is entered, executes the provided closure and returns false
pub async fn confirm_hint_or<F>(text: impl Into<String>, on_reject: F) -> bool
where
F: FnOnce(),
{
let confirmed = confirm_hint(text).await;
if !confirmed {
on_reject();
}
confirmed
}
/// Confirm the current operation, and execute a closure if confirmed
/// Waits for user input of 'y' or 'n'
/// If 'y' is entered, executes the provided closure and returns true
pub async fn confirm_hint_then<F>(text: impl Into<String>, on_confirm: F) -> bool
where
F: FnOnce(),
{
let confirmed = confirm_hint(text).await;
if confirmed {
on_confirm();
}
confirmed
}
/// Input text using the system editor
/// Opens the system editor (from EDITOR environment variable) with default text in a cache file,
/// then reads back the modified content after the editor closes, removing comment lines
pub async fn input_with_editor(
default_text: impl AsRef<str>,
cache_file: impl AsRef<std::path::Path>,
comment_char: impl AsRef<str>,
) -> Result<String, std::io::Error> {
let cache_path = cache_file.as_ref();
let default_content = default_text.as_ref();
let comment_prefix = comment_char.as_ref();
// Write default text to cache file
fs::write(cache_path, default_content).await?;
// Get editor from environment variable
let editor = std::env::var("EDITOR").unwrap_or_else(|_| "vi".to_string());
// Open editor with cache file
let status = Command::new(editor).arg(cache_path).status().await?;
if !status.success() {
return Err(std::io::Error::new(
std::io::ErrorKind::Other,
"Editor exited with non-zero status",
));
}
// Read the modified content
let content = fs::read_to_string(cache_path).await?;
// Remove comment lines and trim
let processed_content: String = content
.lines()
.filter_map(|line| {
let trimmed = line.trim();
if trimmed.starts_with(comment_prefix) {
None
} else {
Some(line)
}
})
.collect::<Vec<&str>>()
.join("\n");
// Delete the cache file
let _ = fs::remove_file(cache_path).await;
Ok(processed_content)
}
|