Skip to main content

winx_code_agent/tools/
bash_command.rs

1//! Implementation of the `BashCommand` tool with WCGW parity.
2//!
3//! This module provides the implementation for the `BashCommand` tool, which is used
4//! to execute shell commands, check command status, and interact with the shell.
5//! Matches the behavior of wcgw Python implementation 1:1.
6
7use anyhow::Context as AnyhowContext;
8use rand::RngExt;
9use regex::Regex;
10use std::collections::HashMap;
11use std::fmt::Write as FmtWrite;
12use std::path::{Path, PathBuf};
13use std::process::{Command, Stdio};
14use std::sync::{Arc, Mutex as StdMutex};
15use std::time::{Duration, Instant};
16use tokio::sync::Mutex;
17use tokio::time::sleep;
18use tracing::{debug, error, info, warn};
19
20use crate::errors::{Result, WinxError};
21use crate::state::bash_state::BashState;
22use crate::state::pty::PtyShell;
23use crate::state::terminal::{render_terminal_output, strip_ansi_codes};
24use crate::types::{normalize_thread_id, BashCommand, BashCommandAction, SpecialKey};
25
26type SharedPtyShell = Arc<Mutex<Option<PtyShell>>>;
27
28// ==================== WCGW-Style Constants ====================
29
30/// Default timeout for command execution (seconds) - matches WCGW Python Config.timeout
31const DEFAULT_TIMEOUT: f64 = 5.0;
32
33/// Extended timeout while output is still being produced - matches WCGW Python `Config.timeout_while_output`
34const TIMEOUT_WHILE_OUTPUT: f64 = 20.0;
35
36/// Number of iterations to wait without new output before giving up - matches WCGW Python `Config.output_wait_patience`
37const OUTPUT_WAIT_PATIENCE: i32 = 3;
38
39/// Chunk size for sending commands (characters) - matches WCGW Python (64 chars)
40const COMMAND_CHUNK_SIZE: usize = 64;
41
42/// Chunk size for sending text input (characters) - matches WCGW Python (128 chars)
43const TEXT_CHUNK_SIZE: usize = 128;
44
45/// Cheap byte-level safety net. We never even consider token counting if the
46/// raw payload is smaller than this — `tiktoken_rs` is fast but not free, and
47/// the vast majority of responses are tiny status updates.
48const MAX_OUTPUT_LENGTH: usize = 100_000;
49
50/// Token budget reserved for a single PTY response when token-aware truncation
51/// kicks in. Picked to leave plenty of room for the surrounding context — most
52/// frontier models have 128k+ windows, so 25k for one shell payload is generous
53/// without monopolizing the conversation.
54const MAX_OUTPUT_TOKENS: usize = 25_000;
55
56/// Truncate `text` so its `cl100k_base` token count stays under `max_tokens`.
57///
58/// We tokenize the tail of the string only when the raw byte length already
59/// exceeds the byte cap; otherwise we trust the byte budget and return as-is.
60/// When the tail still overshoots, we keep the last `max_tokens - reserve`
61/// tokens and prepend a "(...truncated)" marker — exactly what wcgw does in
62/// `_incremental_text`.
63fn truncate_to_token_budget(text: &str, max_tokens: usize) -> std::borrow::Cow<'_, str> {
64    if text.len() <= MAX_OUTPUT_LENGTH {
65        return std::borrow::Cow::Borrowed(text);
66    }
67
68    let Ok(bpe) = tiktoken_rs::cl100k_base() else {
69        // Fallback to the byte-based truncation we used before tiktoken.
70        return std::borrow::Cow::Owned(format!(
71            "(...truncated)\n{}",
72            &text[text.len() - MAX_OUTPUT_LENGTH..]
73        ));
74    };
75
76    let tokens = bpe.encode_with_special_tokens(text);
77    if tokens.len() <= max_tokens {
78        return std::borrow::Cow::Borrowed(text);
79    }
80
81    // Reserve one token slot for the marker overhead.
82    let keep = max_tokens.saturating_sub(1);
83    let tail = &tokens[tokens.len() - keep..];
84    let decoded = bpe.decode(tail).unwrap_or_default();
85    std::borrow::Cow::Owned(format!("(...truncated)\n{decoded}"))
86}
87
88/// Message when a command is already running - matches WCGW Python `WAITING_INPUT_MESSAGE`
89const WAITING_INPUT_MESSAGE: &str = "A command is already running. NOTE: You can't run multiple shell commands in main shell, likely a previous program hasn't exited.
901. Get its output using status check.
912. Use `send_ascii` or `send_specials` to give inputs to the running program OR
923. kill the previous program by sending ctrl+c first using `send_ascii` or `send_specials`
934. Interrupt and run the process in background
94";
95
96// ==================== Background Shell Manager ====================
97
98/// Snapshot of a background shell that has exited but whose final output has not
99/// yet been consumed by the caller. We keep it around so the next call (typically
100/// a `status_check`) can return the trailing output before the entry is gone.
101#[derive(Debug, Clone)]
102pub struct ExitedShellInfo {
103    pub last_command: String,
104    pub final_output: String,
105    pub exited_at: Instant,
106}
107
108/// Manages background shell sessions - matches WCGW Python's `background_shells` dict
109#[derive(Debug, Default)]
110pub struct BackgroundShellManager {
111    shells: HashMap<String, SharedPtyShell>,
112    /// Recently exited shells that still owe their final output to the caller.
113    /// Entries are consumed the first time the caller queries the id, then dropped.
114    tombstones: HashMap<String, ExitedShellInfo>,
115}
116
117impl BackgroundShellManager {
118    /// Tombstones older than this are garbage-collected on the next prune pass.
119    const TOMBSTONE_TTL: Duration = Duration::from_secs(300);
120
121    /// Create a new background shell manager
122    pub fn new() -> Self {
123        Self { shells: HashMap::new(), tombstones: HashMap::new() }
124    }
125
126    /// Start a new background shell and return its command ID
127    pub fn start_new_shell(&mut self, working_dir: &Path, restricted_mode: bool) -> Result<String> {
128        let cid = format!("{:010x}", rand::rng().random::<u32>());
129
130        let shell = PtyShell::new(working_dir, restricted_mode).map_err(|e| {
131            WinxError::CommandExecutionError(format!("Failed to start background shell: {e}"))
132        })?;
133
134        self.shells.insert(cid.clone(), Arc::new(Mutex::new(Some(shell))));
135
136        info!("Started background shell with id: {}", cid);
137        Ok(cid)
138    }
139
140    /// Get a background shell by its command ID
141    pub fn get_shell(&self, bg_command_id: &str) -> Option<SharedPtyShell> {
142        self.shells.get(bg_command_id).cloned()
143    }
144
145    /// Remove and cleanup a background shell
146    pub fn remove_shell(&mut self, bg_command_id: &str) -> bool {
147        if let Some(shell_arc) = self.shells.remove(bg_command_id) {
148            if let Ok(mut guard) = shell_arc.try_lock() {
149                *guard = None;
150            }
151            info!("Removed background shell: {}", bg_command_id);
152            true
153        } else {
154            false
155        }
156    }
157
158    fn prune_finished_shells(&mut self) {
159        // GC old tombstones first.
160        let now = Instant::now();
161        self.tombstones.retain(|_, info| now.duration_since(info.exited_at) < Self::TOMBSTONE_TTL);
162
163        let mut finished: Vec<(String, Option<ExitedShellInfo>)> = Vec::new();
164
165        for (id, shell_arc) in &self.shells {
166            let Ok(mut guard) = shell_arc.try_lock() else {
167                continue;
168            };
169
170            let Some(shell) = guard.as_mut() else {
171                finished.push((id.clone(), None));
172                continue;
173            };
174
175            if !shell.is_alive() {
176                let tombstone = ExitedShellInfo {
177                    last_command: shell.last_command.clone(),
178                    final_output: shell.output_buffer.clone(),
179                    exited_at: now,
180                };
181                finished.push((id.clone(), Some(tombstone)));
182                continue;
183            }
184
185            // Never prune shells that haven't received a command yet.
186            // The global BG_SHELL_MANAGER is shared across parallel tests; a freshly
187            // spawned shell would otherwise be evicted between start_new_shell and
188            // the first send_command, leading to "Failed to get background shell".
189            if shell.last_command.is_empty() {
190                continue;
191            }
192
193            if shell.command_running {
194                let _ = shell.read_output(0.1);
195            }
196
197            if !shell.command_running {
198                let tombstone = ExitedShellInfo {
199                    last_command: shell.last_command.clone(),
200                    final_output: shell.output_buffer.clone(),
201                    exited_at: now,
202                };
203                finished.push((id.clone(), Some(tombstone)));
204            }
205        }
206
207        for (id, tombstone) in finished {
208            self.remove_shell(&id);
209            if let Some(info) = tombstone {
210                self.tombstones.insert(id, info);
211            }
212        }
213    }
214
215    /// Look up the tombstone for a recently-exited shell, if any.
216    ///
217    /// The entry stays in the map until the TTL expires (see
218    /// `prune_finished_shells`), so repeated `status_check` calls on the same
219    /// `bg_command_id` keep returning the cached final output instead of
220    /// flipping to "shell not found" after the first read.
221    pub fn peek_tombstone(&self, bg_command_id: &str) -> Option<ExitedShellInfo> {
222        self.tombstones.get(bg_command_id).cloned()
223    }
224
225    /// Get info about all running background shells - matches WCGW Python `get_bg_running_commandsinfo`
226    pub fn get_running_info(&mut self) -> String {
227        self.prune_finished_shells();
228
229        if self.shells.is_empty() {
230            return "No command running in background.\n".to_string();
231        }
232
233        let mut running = Vec::new();
234        for (id, shell_arc) in &self.shells {
235            if let Ok(guard) = shell_arc.try_lock() {
236                if let Some(bash) = guard.as_ref() {
237                    if bash.command_running {
238                        running
239                            .push(format!("Command: {}, bg_command_id: {}", bash.last_command, id));
240                    }
241                }
242            } else {
243                running.push(format!("Command: <busy>, bg_command_id: {id}"));
244            }
245        }
246
247        if running.is_empty() {
248            "No command running in background.\n".to_string()
249        } else {
250            format!("Following background commands are attached:\n{}\n", running.join("\n"))
251        }
252    }
253}
254
255// Global background shell manager (thread-safe) - matches WCGW Python's BashState.background_shells
256lazy_static::lazy_static! {
257    static ref BG_SHELL_MANAGER: StdMutex<BackgroundShellManager> = StdMutex::new(BackgroundShellManager::new());
258}
259
260// ==================== WCGW-Style Helper Functions ====================
261
262/// Get WCGW-style status string - matches WCGW Python's `get_status()`
263fn get_status(
264    bash_state: &BashState,
265    is_bg: bool,
266    bg_id: Option<&str>,
267    is_running: bool,
268    running_for: Option<&str>,
269) -> String {
270    let mut status = "\n\n---\n\n".to_string();
271
272    if is_bg {
273        if let Some(id) = bg_id {
274            let _ = writeln!(status, "bg_command_id = {id}");
275        }
276    }
277
278    if is_running {
279        status.push_str("status = still running\n");
280        if let Some(duration) = running_for {
281            let _ = writeln!(status, "running for = {duration}");
282        }
283    } else {
284        status.push_str("status = process exited\n");
285    }
286
287    let _ = writeln!(status, "cwd = {}", bash_state.cwd.display());
288
289    if !is_bg {
290        // Add background shell info for main shell - matches WCGW Python
291        if let Ok(mut manager) = BG_SHELL_MANAGER.lock() {
292            status.push_str("This is the main shell. ");
293            status.push_str(&manager.get_running_info());
294        }
295    }
296
297    status.trim_end().to_string()
298}
299
300/// Process output with WCGW-style incremental text handling - matches WCGW Python _`incremental_text`
301fn wcgw_incremental_text(text: &str, last_pending_output: &str) -> String {
302    let truncated = truncate_to_token_budget(text, MAX_OUTPUT_TOKENS);
303    let text = truncated.as_ref();
304
305    if last_pending_output.is_empty() {
306        let rendered = render_terminal_output(text);
307        return rstrip_lines(&rendered).trim_start().to_string();
308    }
309
310    let last_rendered = render_terminal_output(last_pending_output);
311    if last_rendered.is_empty() {
312        return rstrip_lines(&render_terminal_output(text));
313    }
314
315    // Get text after last pending output
316    let text_after_last = if text.len() > last_pending_output.len() {
317        &text[last_pending_output.len()..]
318    } else {
319        text
320    };
321
322    let combined = format!("{}\n{}", last_rendered.join("\n"), text_after_last);
323    let new_rendered = render_terminal_output(&combined);
324
325    // Get incremental part - matches WCGW Python get_incremental_output
326    let incremental = get_incremental_output(&last_rendered, &new_rendered);
327    rstrip_lines(&incremental)
328}
329
330fn extract_prompt_cwd(output: &str) -> Option<PathBuf> {
331    let stripped = strip_ansi_codes(output);
332    let prompt_regex = Regex::new(r"◉ (?P<cwd>[^\r\n]*?)──➤").ok()?;
333
334    prompt_regex
335        .captures_iter(&stripped)
336        .filter_map(|captures| captures.name("cwd").map(|cwd| cwd.as_str().trim()))
337        .filter(|cwd| !cwd.is_empty())
338        .last()
339        .map(PathBuf::from)
340}
341
342/// Right-strip each line and join - matches WCGW Python rstrip
343fn rstrip_lines(lines: &[String]) -> String {
344    lines.iter().map(|line| line.trim_end()).collect::<Vec<_>>().join("\n")
345}
346
347/// Get incremental output between old and new - matches WCGW Python `get_incremental_output`
348fn get_incremental_output(old_output: &[String], new_output: &[String]) -> Vec<String> {
349    if old_output.is_empty() {
350        return new_output.to_vec();
351    }
352
353    let nold = old_output.len();
354    let nnew = new_output.len();
355
356    // Find where old output ends in new output
357    for i in (0..nnew).rev() {
358        if new_output[i] != old_output[nold - 1] {
359            continue;
360        }
361
362        let mut matched = true;
363        for j in (0..i).rev() {
364            let old_idx = (nold as i64 - 1 + j as i64 - i as i64) as isize;
365            if old_idx < 0 {
366                break;
367            }
368            if new_output[j] != old_output[old_idx as usize] {
369                matched = false;
370                break;
371            }
372        }
373
374        if matched {
375            return new_output[i + 1..].to_vec();
376        }
377    }
378
379    new_output.to_vec()
380}
381
382fn send_utf8_in_byte_chunks(shell: &mut PtyShell, text: &str, chunk_size: usize) -> Result<()> {
383    let mut start = 0;
384
385    while start < text.len() {
386        let mut end = (start + chunk_size).min(text.len());
387        while !text.is_char_boundary(end) {
388            end -= 1;
389        }
390        if end == start {
391            end = text[start..].char_indices().nth(1).map_or(text.len(), |(idx, _)| start + idx);
392        }
393
394        shell.send_text(&text[start..end]).map_err(|e| {
395            WinxError::CommandExecutionError(format!("Failed to write PTY input: {e}"))
396        })?;
397        start = end;
398    }
399
400    Ok(())
401}
402
403/// Check if action is effectively a status check - matches WCGW Python `is_status_check`
404#[allow(dead_code)]
405fn is_status_check_action(action: &BashCommandAction) -> bool {
406    match action {
407        BashCommandAction::StatusCheck { .. } => true,
408        BashCommandAction::SendSpecials { send_specials, .. } => {
409            send_specials.len() == 1 && send_specials[0] == SpecialKey::Enter
410        }
411        BashCommandAction::SendAscii { send_ascii, .. } => {
412            send_ascii.len() == 1 && send_ascii[0] == 10 // newline
413        }
414        _ => false,
415    }
416}
417
418// ==================== Main Tool Handler ====================
419
420/// Handles the `BashCommand` tool call with WCGW parity
421///
422/// This function processes the `BashCommand` tool call following WCGW Python's
423/// `execute_bash()` function behavior exactly.
424#[tracing::instrument(level = "info", skip(bash_state_arc, bash_command))]
425pub async fn handle_tool_call(
426    bash_state_arc: &Arc<Mutex<Option<BashState>>>,
427    bash_command: BashCommand,
428) -> Result<String> {
429    info!("BashCommand tool called with: {:?}", bash_command);
430
431    let thread_id = normalize_thread_id(&bash_command.thread_id);
432
433    // Check if thread_id is empty
434    if thread_id.is_empty() {
435        error!("Empty thread_id provided in BashCommand");
436        return Err(WinxError::ThreadIdMismatch(
437            "Error: No saved bash state found for thread ID \"\". Please initialize first with this ID.".to_string()
438        ));
439    }
440
441    // Extract bash_state data
442    let mut bash_state: BashState;
443    {
444        let bash_state_guard = bash_state_arc.lock().await;
445
446        let Some(state) = &*bash_state_guard else {
447            error!("BashState not initialized");
448            return Err(WinxError::BashStateNotInitialized);
449        };
450
451        bash_state = state.clone();
452    }
453
454    // Verify thread ID matches - matches WCGW Python thread_id check
455    if thread_id != bash_state.current_thread_id {
456        // Try to load state from thread_id - matches WCGW Python load_state_from_thread_id
457        if !bash_state.load_state_from_disk(&thread_id).unwrap_or(false) {
458            return Err(WinxError::ThreadIdMismatch(format!(
459                "Error: No saved bash state found for thread_id `{thread_id}`. Please initialize first with this ID."
460            )));
461        }
462    }
463
464    // Calculate effective timeout - matches WCGW Python
465    // SECURITY: Ensure timeout is not negative to prevent unexpected behavior
466    let timeout_s = bash_command
467        .wait_for_seconds
468        .map_or(DEFAULT_TIMEOUT, |t| f64::from(t).max(0.0))
469        .min(TIMEOUT_WHILE_OUTPUT);
470
471    // Execute the action based on type - matches WCGW Python's _execute_bash()
472    let result = execute_bash_action(&mut bash_state, &bash_command.action_json, timeout_s).await;
473
474    {
475        let mut bash_state_guard = bash_state_arc.lock().await;
476        if let Some(state) = bash_state_guard.as_mut() {
477            state.cwd.clone_from(&bash_state.cwd);
478        }
479    }
480
481    // Remove echo if it's a command - matches WCGW Python
482    match result {
483        Ok(mut output) => {
484            if let BashCommandAction::Command { ref command, .. } = bash_command.action_json {
485                let cmd_trimmed = command.trim();
486                if output.starts_with(cmd_trimmed) {
487                    output = output[cmd_trimmed.len()..].to_string();
488                }
489            }
490            Ok(output)
491        }
492        Err(e) => Err(e),
493    }
494}
495
496/// Execute a bash action - matches WCGW Python's _`execute_bash()` function
497async fn execute_bash_action(
498    bash_state: &mut BashState,
499    action: &BashCommandAction,
500    timeout_s: f64,
501) -> Result<String> {
502    let mut is_bg = false;
503    let mut bg_id: Option<String> = None;
504
505    // Handle bg_command_id routing - matches WCGW Python
506    let bg_shell: Option<SharedPtyShell> = match action {
507        BashCommandAction::Command { .. } => None, // Commands don't use bg_command_id for routing
508        BashCommandAction::StatusCheck { bg_command_id, .. }
509        | BashCommandAction::SendText { bg_command_id, .. }
510        | BashCommandAction::SendSpecials { bg_command_id, .. }
511        | BashCommandAction::SendAscii { bg_command_id, .. } => {
512            if let Some(id) = bg_command_id {
513                let mut manager = BG_SHELL_MANAGER.lock().map_err(|e| {
514                    WinxError::BashStateLockError(format!("Failed to lock bg manager: {e}"))
515                })?;
516                manager.prune_finished_shells();
517
518                if let Some(shell) = manager.get_shell(id) {
519                    is_bg = true;
520                    bg_id = Some(id.clone());
521                    Some(shell)
522                } else if let Some(tombstone) = manager.peek_tombstone(id) {
523                    // Shell already exited. For a status check we can hand back the
524                    // final cached output exactly once. For anything else (send_text,
525                    // send_specials, send_ascii) tell the caller the shell is gone
526                    // and include the captured output so they can recover state.
527                    drop(manager);
528                    return finalize_tombstone(&bash_state.cwd, id, tombstone, action);
529                } else {
530                    // Error message matches WCGW Python
531                    let error = format!(
532                        "No shell found running with command id {}.\n{}",
533                        id,
534                        manager.get_running_info()
535                    );
536                    return Err(WinxError::CommandExecutionError(error));
537                }
538            } else {
539                None
540            }
541        }
542    };
543
544    // Process based on action type - matches WCGW Python _execute_bash dispatch
545    match action {
546        BashCommandAction::Command { command, is_background } => {
547            execute_command(bash_state, command, *is_background, timeout_s).await
548        }
549        BashCommandAction::StatusCheck { scrollback_lines, verbose, .. } => {
550            execute_status_check(
551                bash_state,
552                bg_shell,
553                is_bg,
554                bg_id.as_deref(),
555                timeout_s,
556                *scrollback_lines,
557                *verbose,
558            )
559            .await
560        }
561        BashCommandAction::SendText { send_text, submit, .. } => {
562            execute_send_text(
563                bash_state,
564                send_text,
565                *submit,
566                bg_shell,
567                is_bg,
568                bg_id.as_deref(),
569                timeout_s,
570            )
571            .await
572        }
573        BashCommandAction::SendSpecials { send_specials, submit, .. } => {
574            execute_send_specials(
575                bash_state,
576                send_specials,
577                *submit,
578                bg_shell,
579                is_bg,
580                bg_id.as_deref(),
581                timeout_s,
582            )
583            .await
584        }
585        BashCommandAction::SendAscii { send_ascii, submit, .. } => {
586            execute_send_ascii(
587                bash_state,
588                send_ascii,
589                *submit,
590                bg_shell,
591                is_bg,
592                bg_id.as_deref(),
593                timeout_s,
594            )
595            .await
596        }
597    }
598}
599
600/// Execute a command - matches WCGW Python's Command handling in _`execute_bash`
601async fn execute_command(
602    bash_state: &mut BashState,
603    command: &str,
604    is_background: bool,
605    timeout_s: f64,
606) -> Result<String> {
607    debug!("Processing Command action: {}", command);
608
609    // Check mode permissions - matches WCGW Python bash_command_mode check
610    if !bash_state.is_command_allowed(command) {
611        error!("Command '{}' not allowed in current mode", command);
612        return Err(WinxError::CommandNotAllowed(
613            "Error: BashCommand not allowed in current mode".to_string(),
614        ));
615    }
616
617    // Validate single statement - matches WCGW Python assert_single_statement
618    let command = command.trim();
619    crate::utils::bash_parser::assert_single_statement(command)?;
620
621    // If background execution requested, start new shell - matches WCGW Python is_background handling
622    if is_background {
623        return execute_in_background(bash_state, command, timeout_s).await;
624    }
625
626    // Check if a command is already running - matches WCGW Python state check
627    {
628        let bash_guard = bash_state.pty_shell.lock().await;
629
630        if let Some(ref bash) = *bash_guard {
631            if bash.command_running {
632                return Err(WinxError::CommandExecutionError(WAITING_INPUT_MESSAGE.to_string()));
633            }
634        }
635    }
636
637    // Initialize bash if needed
638    if bash_state.pty_shell.lock().await.is_none() {
639        bash_state
640            .init_pty_shell()
641            .await
642            .map_err(|e| WinxError::CommandExecutionError(format!("Failed to init bash: {e}")))?;
643    }
644
645    // Clear prompt before sending - matches WCGW Python clear_to_run.
646    // Drain any leftover output and, if the shell still looks busy, send
647    // Ctrl-C so the new command lands on a fresh prompt instead of being
648    // appended to whatever was hanging on stdin.
649    {
650        let mut bash_guard = bash_state.pty_shell.lock().await;
651        if let Some(bash) = bash_guard.as_mut() {
652            if let Err(e) = bash.clear_to_run(DEFAULT_TIMEOUT as f32) {
653                warn!("clear_to_run failed before send: {e}");
654            }
655        }
656    }
657
658    // Send command in chunks of 64 characters - matches WCGW Python exactly
659    {
660        let mut bash_guard = bash_state.pty_shell.lock().await;
661
662        let bash = bash_guard.as_mut().ok_or(WinxError::BashStateNotInitialized)?;
663
664        bash.output_buffer.clear();
665        bash.output_truncated = false;
666        // Send in chunks - matches WCGW Python: for i in range(0, len(command), 64)
667        send_utf8_in_byte_chunks(bash, command, COMMAND_CHUNK_SIZE)?;
668
669        // Send linesep to execute - matches WCGW Python bash_state.send(bash_state.linesep, ...)
670        bash.send_special_key("Enter").map_err(|e| {
671            WinxError::CommandExecutionError(format!("Failed to send newline: {e}"))
672        })?;
673
674        bash.last_command = command.to_string();
675        bash.command_running = true;
676    }
677
678    // Wait for output with WCGW-style patience handling
679    let shell_arc = bash_state.pty_shell.clone();
680    wait_for_output(bash_state, &shell_arc, timeout_s, false, None, false).await
681}
682
683/// Wait for command output with WCGW-style patience handling - matches WCGW Python expect/wait logic.
684///
685/// `shell_arc` selects which shell to read from (main shell or a bg shell handle).
686async fn wait_for_output(
687    bash_state: &mut BashState,
688    shell_arc: &SharedPtyShell,
689    timeout_s: f64,
690    is_bg: bool,
691    bg_id: Option<&str>,
692    is_status_check: bool,
693) -> Result<String> {
694    let start = Instant::now();
695    let wait = timeout_s.min(TIMEOUT_WHILE_OUTPUT);
696    let mut last_pending_output = String::new();
697    let mut complete = false;
698
699    // Initial wait - matches WCGW Python wait = min(timeout_s or CONFIG.timeout, CONFIG.timeout_while_output)
700    sleep(Duration::from_secs_f64(wait.min(DEFAULT_TIMEOUT))).await;
701
702    // Read initial output
703    let mut output = {
704        let mut bash_guard = shell_arc.lock().await;
705
706        if let Some(bash) = bash_guard.as_mut() {
707            let (out, done) = bash.read_output(0.5).map_err(|e| {
708                WinxError::CommandExecutionError(format!("Failed to read output: {e}"))
709            })?;
710            complete = done;
711            out
712        } else {
713            String::new()
714        }
715    };
716
717    // If not complete and this is a status check, use WCGW-style patience waiting
718    // Matches WCGW Python: if is_status_check(bash_arg) block
719    if !complete && is_status_check {
720        let mut remaining = TIMEOUT_WHILE_OUTPUT - wait;
721        let mut patience = OUTPUT_WAIT_PATIENCE;
722
723        let incremental = wcgw_incremental_text(&output, &last_pending_output);
724        if incremental.is_empty() {
725            patience -= 1;
726        }
727
728        let mut last_incremental = incremental;
729
730        while remaining > 0.0 && patience > 0 {
731            sleep(Duration::from_secs_f64(wait.min(remaining))).await;
732
733            let (new_output, done) = {
734                let mut bash_guard = shell_arc.lock().await;
735
736                if let Some(bash) = bash_guard.as_mut() {
737                    bash.read_output(0.5).map_err(|e| {
738                        WinxError::CommandExecutionError(format!("Failed to read output: {e}"))
739                    })?
740                } else {
741                    (String::new(), true)
742                }
743            };
744
745            if done {
746                complete = true;
747                output = new_output;
748                break;
749            }
750
751            // Check if output changed - matches WCGW Python patience logic
752            let new_incremental = wcgw_incremental_text(&new_output, &last_pending_output);
753            if new_incremental == last_incremental {
754                patience -= 1;
755            } else {
756                patience = OUTPUT_WAIT_PATIENCE; // Reset patience on new output
757            }
758            last_incremental = new_incremental;
759
760            output = new_output;
761            remaining -= wait;
762        }
763
764        if !complete {
765            // Update pending output - matches WCGW Python bash_state.set_pending(text)
766            last_pending_output = output.clone();
767        }
768    }
769
770    if complete {
771        if let Some(cwd) = extract_prompt_cwd(&output) {
772            bash_state.cwd = cwd;
773        }
774    }
775
776    // Process output through terminal emulation - matches WCGW Python _incremental_text
777    let rendered = wcgw_incremental_text(&output, &last_pending_output);
778
779    // Truncate if needed - matches WCGW Python token truncation
780    let rendered = truncate_to_token_budget(&rendered, MAX_OUTPUT_TOKENS).into_owned();
781
782    // Calculate running duration for status
783    let running_for = if complete {
784        None
785    } else {
786        Some(format!("{} seconds", (start.elapsed().as_secs() + timeout_s as u64)))
787    };
788
789    // Add status - matches WCGW Python get_status
790    let status = get_status(bash_state, is_bg, bg_id, !complete, running_for.as_deref());
791    Ok(format!("{rendered}{status}"))
792}
793
794/// Render the final cached output of an exited background shell.
795///
796/// `status_check` is allowed to "consume" the tombstone and return the trailing
797/// output exactly once. Send-style actions (`send_text`, `send_specials`,
798/// `send_ascii`) cannot interact with a dead shell, so we return an explicit
799/// error that still includes the captured output so the agent can recover state.
800fn finalize_tombstone(
801    cwd: &Path,
802    id: &str,
803    tombstone: ExitedShellInfo,
804    action: &BashCommandAction,
805) -> Result<String> {
806    let ExitedShellInfo { last_command, final_output, .. } = tombstone;
807    match action {
808        BashCommandAction::StatusCheck { .. } => {
809            let rendered = wcgw_incremental_text(&final_output, "");
810            let rendered = truncate_to_token_budget(&rendered, MAX_OUTPUT_TOKENS).into_owned();
811            // Build a compact status block matching `get_status` for a finished bg shell.
812            let mut status = "\n\n---\n\n".to_string();
813            let _ = writeln!(status, "bg_command_id = {id}");
814            status.push_str("status = process exited\n");
815            let _ = writeln!(status, "cwd = {}", cwd.display());
816            Ok(format!("{rendered}{}", status.trim_end()))
817        }
818        BashCommandAction::SendText { .. }
819        | BashCommandAction::SendSpecials { .. }
820        | BashCommandAction::SendAscii { .. } => Err(WinxError::CommandExecutionError(format!(
821            "Background shell {id} already exited (last command: {last_command}).\nFinal captured output:\n{final_output}"
822        ))),
823        BashCommandAction::Command { .. } => {
824            // We only enter `finalize_tombstone` from the bg routing path, which
825            // never matches Command. Treat this as a programmer error.
826            unreachable!("finalize_tombstone called for non-bg action")
827        }
828    }
829}
830
831/// Execute a status check - matches WCGW Python's `StatusCheck` handling.
832///
833/// New behavior (v0.2.308):
834/// - Deduplicates against the last response by fingerprint; when nothing
835///   changed and `verbose=false`, returns a compact "no new output" payload
836///   instead of resending the same screen.
837/// - Optional `scrollback_lines` pulls bounded history from the `PtyShell`
838///   ringbuffer so agents can reorient after a long pause.
839async fn execute_status_check(
840    bash_state: &mut BashState,
841    bg_shell: Option<SharedPtyShell>,
842    is_bg: bool,
843    bg_id: Option<&str>,
844    timeout_s: f64,
845    scrollback_lines: Option<usize>,
846    verbose: bool,
847) -> Result<String> {
848    debug!("Processing StatusCheck action (verbose={verbose}, scrollback={scrollback_lines:?})");
849
850    // Pick the shell we're going to inspect: bg shell when bg_command_id was provided,
851    // otherwise fall back to the main interactive shell.
852    let shell_arc = bg_shell.unwrap_or_else(|| bash_state.pty_shell.clone());
853
854    // Check if there's a running command - matches WCGW Python state check
855    let is_running = {
856        let guard = shell_arc.lock().await;
857        if let Some(ref bash) = *guard {
858            bash.command_running
859        } else {
860            false
861        }
862    };
863
864    // If no command running and not background, return error - matches WCGW Python
865    if !is_running && !is_bg {
866        let mut manager = BG_SHELL_MANAGER.lock().map_err(|e| {
867            WinxError::BashStateLockError(format!("Failed to lock bg manager: {e}"))
868        })?;
869        let error =
870            format!("No running command to check status of.\n{}", manager.get_running_info());
871        return Err(WinxError::CommandExecutionError(error));
872    }
873
874    // Read output with patience handling - this IS a status check
875    let response = wait_for_output(bash_state, &shell_arc, timeout_s, is_bg, bg_id, true).await?;
876
877    // Inter-call dedup: hash only the response *body* (the chunk before the
878    // `\n\n---\n` status footer). The footer contains a live "running for"
879    // counter that would otherwise defeat the comparison.
880    let body = response.split("\n\n---\n").next().unwrap_or(&response);
881    if !verbose && scrollback_lines.is_none() {
882        let mut guard = shell_arc.lock().await;
883        if let Some(bash) = guard.as_mut() {
884            let fingerprint = PtyShell::fingerprint(body);
885            if Some(fingerprint) == bash.last_returned_hash {
886                let status = get_status(bash_state, is_bg, bg_id, is_running, None);
887                return Ok(format!("no new output since last check{status}"));
888            }
889            bash.last_returned_hash = Some(fingerprint);
890        }
891    } else if !verbose {
892        // Still record the hash so subsequent non-scrollback calls can dedup.
893        let mut guard = shell_arc.lock().await;
894        if let Some(bash) = guard.as_mut() {
895            bash.last_returned_hash = Some(PtyShell::fingerprint(body));
896        }
897    }
898
899    // Optional scrollback prefix — only ever pulled when the caller asks for it.
900    if let Some(lines) = scrollback_lines {
901        if lines > 0 {
902            let scrollback = {
903                let guard = shell_arc.lock().await;
904                guard.as_ref().map(|s| s.collect_scrollback(lines)).unwrap_or_default()
905            };
906            if !scrollback.is_empty() {
907                let count = scrollback.lines().count();
908                return Ok(format!(
909                    "--- scrollback ({count} lines) ---\n{scrollback}\n--- latest ---\n{response}"
910                ));
911            }
912        }
913    }
914
915    Ok(response)
916}
917
918/// Execute `send_text` - matches WCGW Python's `SendText` handling
919async fn execute_send_text(
920    bash_state: &mut BashState,
921    text: &str,
922    submit: bool,
923    bg_shell: Option<SharedPtyShell>,
924    is_bg: bool,
925    bg_id: Option<&str>,
926    timeout_s: f64,
927) -> Result<String> {
928    debug!("Processing SendText action: {text:?} (submit={submit})");
929
930    // Validate - matches WCGW Python
931    if text.is_empty() {
932        return Err(WinxError::CommandExecutionError(
933            "Failure: send_text cannot be empty".to_string(),
934        ));
935    }
936
937    // Get the target shell
938    let shell_arc = bg_shell.unwrap_or_else(|| bash_state.pty_shell.clone());
939
940    // Send text in chunks of 128 characters - matches WCGW Python exactly
941    {
942        let mut guard = shell_arc.lock().await;
943
944        let bash = guard.as_mut().ok_or(WinxError::BashStateNotInitialized)?;
945
946        // Send in chunks - matches WCGW Python: for i in range(0, len(command_data.send_text), 128)
947        send_utf8_in_byte_chunks(bash, text, TEXT_CHUNK_SIZE)?;
948
949        // Only append Enter when the caller explicitly asks to submit. Many TUIs
950        // (e.g., Claude Code) treat a bare CR as a soft newline inside the input
951        // box, so blindly auto-Entering interferes with multi-step interaction.
952        if submit {
953            bash.send_special_key("Enter").map_err(|e| {
954                WinxError::CommandExecutionError(format!("Failed to send newline: {e}"))
955            })?;
956        }
957    }
958
959    // Wait for output
960    wait_for_output(bash_state, &shell_arc, timeout_s, is_bg, bg_id, false).await
961}
962
963/// Execute `send_specials` - matches WCGW Python's `SendSpecials` handling exactly
964async fn execute_send_specials(
965    bash_state: &mut BashState,
966    keys: &[SpecialKey],
967    submit: bool,
968    bg_shell: Option<SharedPtyShell>,
969    is_bg: bool,
970    bg_id: Option<&str>,
971    timeout_s: f64,
972) -> Result<String> {
973    debug!("Processing SendSpecials action: {keys:?} (submit={submit})");
974
975    // Validate - matches WCGW Python
976    if keys.is_empty() {
977        return Err(WinxError::CommandExecutionError(
978            "Failure: send_specials cannot be empty".to_string(),
979        ));
980    }
981
982    let shell_arc = bg_shell.unwrap_or_else(|| bash_state.pty_shell.clone());
983    let mut is_interrupt = false;
984
985    {
986        let mut guard = shell_arc.lock().await;
987
988        let bash = guard.as_mut().ok_or(WinxError::BashStateNotInitialized)?;
989
990        // Send each special key - matches WCGW Python exactly
991        for key in keys {
992            match key {
993                SpecialKey::KeyUp => {
994                    // matches WCGW Python: bash_state.send("\033[A", ...)
995                    bash.send_special_key("KeyUp").map_err(|e| {
996                        WinxError::CommandExecutionError(format!("Failed to send KeyUp: {e}"))
997                    })?;
998                }
999                SpecialKey::KeyDown => {
1000                    // matches WCGW Python: bash_state.send("\033[B", ...)
1001                    bash.send_special_key("KeyDown").map_err(|e| {
1002                        WinxError::CommandExecutionError(format!("Failed to send KeyDown: {e}"))
1003                    })?;
1004                }
1005                SpecialKey::KeyLeft => {
1006                    // matches WCGW Python: bash_state.send("\033[D", ...)
1007                    bash.send_special_key("KeyLeft").map_err(|e| {
1008                        WinxError::CommandExecutionError(format!("Failed to send KeyLeft: {e}"))
1009                    })?;
1010                }
1011                SpecialKey::KeyRight => {
1012                    // matches WCGW Python: bash_state.send("\033[C", ...)
1013                    bash.send_special_key("KeyRight").map_err(|e| {
1014                        WinxError::CommandExecutionError(format!("Failed to send KeyRight: {e}"))
1015                    })?;
1016                }
1017                SpecialKey::Enter => {
1018                    // matches WCGW Python: bash_state.send("\x0d", ...) - carriage return
1019                    bash.send_special_key("Enter").map_err(|e| {
1020                        WinxError::CommandExecutionError(format!("Failed to send Enter: {e}"))
1021                    })?;
1022                }
1023                SpecialKey::CtrlC => {
1024                    // matches WCGW Python: bash_state.sendintr()
1025                    bash.send_interrupt().map_err(|e| {
1026                        WinxError::CommandExecutionError(format!("Failed to send interrupt: {e}"))
1027                    })?;
1028                    is_interrupt = true;
1029                }
1030                SpecialKey::CtrlD => {
1031                    // matches WCGW Python: bash_state.sendintr() - same as Ctrl+C in WCGW
1032                    bash.send_eof().map_err(|e| {
1033                        WinxError::CommandExecutionError(format!("Failed to send Ctrl+D: {e}"))
1034                    })?;
1035                    is_interrupt = true;
1036                }
1037                SpecialKey::CtrlZ => {
1038                    // Ctrl+Z = SIGTSTP (suspend) - ASCII 0x1a
1039                    bash.send_suspend().map_err(|e| {
1040                        WinxError::CommandExecutionError(format!("Failed to send Ctrl+Z: {e}"))
1041                    })?;
1042                }
1043            }
1044        }
1045        // Submit (append Enter) only when explicitly requested by the caller.
1046        if submit {
1047            bash.send_special_key("Enter")
1048                .map_err(|e| WinxError::CommandExecutionError(format!("Failed to submit: {e}")))?;
1049        }
1050    }
1051
1052    // NOTE: wcgw treats a bare Enter as a status check and applies its
1053    // patience loop. We deliberately diverge: for a driving agent (e.g.,
1054    // pushing Enter to submit text in a TUI) the patience loop swallows the
1055    // immediate response. Callers that want patience semantics should use the
1056    // explicit `status_check` action instead.
1057
1058    // Wait for output
1059    let mut output =
1060        wait_for_output(bash_state, &shell_arc, timeout_s, is_bg, bg_id, false).await?;
1061
1062    // Add interrupt failure message if still running - matches WCGW Python exactly
1063    if is_interrupt && output.contains("status = still running") {
1064        output.push_str("\n---\n----\nFailure interrupting.\nYou may want to try Ctrl-c again or program specific exit interactive commands.\n");
1065    }
1066
1067    Ok(output)
1068}
1069
1070/// Execute `send_ascii` - matches WCGW Python's `SendAscii` handling
1071async fn execute_send_ascii(
1072    bash_state: &mut BashState,
1073    ascii_codes: &[u8],
1074    submit: bool,
1075    bg_shell: Option<SharedPtyShell>,
1076    is_bg: bool,
1077    bg_id: Option<&str>,
1078    timeout_s: f64,
1079) -> Result<String> {
1080    debug!("Processing SendAscii action: {ascii_codes:?} (submit={submit})");
1081
1082    // Validate - matches WCGW Python
1083    if ascii_codes.is_empty() {
1084        return Err(WinxError::CommandExecutionError(
1085            "Failure: send_ascii cannot be empty".to_string(),
1086        ));
1087    }
1088
1089    let shell_arc = bg_shell.unwrap_or_else(|| bash_state.pty_shell.clone());
1090    let mut is_interrupt = false;
1091
1092    {
1093        let mut guard = shell_arc.lock().await;
1094
1095        let bash = guard.as_mut().ok_or(WinxError::BashStateNotInitialized)?;
1096
1097        // Send each ASCII code - matches WCGW Python
1098        for &code in ascii_codes {
1099            // matches WCGW Python: bash_state.send(chr(ascii_char), ...)
1100            bash.send_bytes(&[code]).map_err(|e| {
1101                WinxError::CommandExecutionError(format!("Failed to write ASCII code: {e}"))
1102            })?;
1103
1104            // Check for interrupt - matches WCGW Python: if ascii_char == 3: is_interrupt = True
1105            if code == 3 {
1106                is_interrupt = true;
1107            }
1108        }
1109        // Submit (append Enter) only when explicitly requested by the caller.
1110        if submit {
1111            bash.send_special_key("Enter")
1112                .map_err(|e| WinxError::CommandExecutionError(format!("Failed to submit: {e}")))?;
1113        }
1114    }
1115
1116    // Same divergence from wcgw as in `execute_send_specials`: send_ascii [10]
1117    // or [13] is treated as a direct write, not a status check. Callers that
1118    // need patience-aware reads should use `status_check`.
1119
1120    // Wait for output
1121    let mut output =
1122        wait_for_output(bash_state, &shell_arc, timeout_s, is_bg, bg_id, false).await?;
1123
1124    // Add interrupt failure message if still running - matches WCGW Python
1125    if is_interrupt && output.contains("status = still running") {
1126        output.push_str("\n---\n----\nFailure interrupting.\nYou may want to try Ctrl-c again or program specific exit interactive commands.\n");
1127    }
1128
1129    Ok(output)
1130}
1131
1132/// Execute command in background - matches WCGW Python's `is_background` handling
1133async fn execute_in_background(
1134    bash_state: &mut BashState,
1135    command: &str,
1136    timeout_s: f64,
1137) -> Result<String> {
1138    debug!("Executing command in background: {}", command);
1139
1140    // Start a new background shell - matches WCGW Python bash_state.start_new_bg_shell
1141    let restricted_mode =
1142        matches!(bash_state.bash_command_mode.bash_mode, crate::types::BashMode::RestrictedMode);
1143
1144    let bg_id = {
1145        let mut manager = BG_SHELL_MANAGER.lock().map_err(|e| {
1146            WinxError::BashStateLockError(format!("Failed to lock bg manager: {e}"))
1147        })?;
1148        manager.start_new_shell(&bash_state.cwd, restricted_mode)?
1149    };
1150
1151    // Get the shell
1152    let shell_arc = {
1153        let manager = BG_SHELL_MANAGER.lock().map_err(|e| {
1154            WinxError::BashStateLockError(format!("Failed to lock bg manager: {e}"))
1155        })?;
1156        manager.get_shell(&bg_id).ok_or_else(|| {
1157            WinxError::CommandExecutionError("Failed to get background shell".to_string())
1158        })?
1159    };
1160
1161    // Send command via the same PTY path used by foreground execute_command.
1162    {
1163        let mut guard = shell_arc.lock().await;
1164        let bash = guard.as_mut().ok_or(WinxError::BashStateNotInitialized)?;
1165        bash.send_command(command).map_err(|e| {
1166            WinxError::CommandExecutionError(format!("Failed to send bg command: {e}"))
1167        })?;
1168    }
1169    debug!("bg[{}]: send_command returned, replying with bg_command_id", bg_id);
1170
1171    let _ = timeout_s;
1172    let _ = shell_arc;
1173    Ok(get_status(bash_state, true, Some(&bg_id), true, None))
1174}
1175
1176// ==================== Legacy Screen-based Functions (kept for backward compatibility) ====================
1177
1178/// Process simple command execution for a bash command (legacy)
1179#[allow(dead_code)]
1180#[tracing::instrument(level = "debug", skip(command, cwd))]
1181async fn execute_simple_command(command: &str, cwd: &Path) -> Result<String> {
1182    debug!("Executing command: {}", command);
1183
1184    let start_time = Instant::now();
1185    let mut cmd = Command::new("sh");
1186    cmd.arg("-c")
1187        .arg(command)
1188        .current_dir(cwd)
1189        .stdin(Stdio::null())
1190        .stdout(Stdio::piped())
1191        .stderr(Stdio::piped());
1192
1193    let output = cmd.output().context("Failed to execute command")?;
1194    let elapsed = start_time.elapsed();
1195
1196    let stdout = String::from_utf8_lossy(&output.stdout).to_string();
1197    let stderr = String::from_utf8_lossy(&output.stderr).to_string();
1198
1199    let raw_result = format!("{stdout}{stderr}");
1200    let mut result = raw_result.clone();
1201    if !raw_result.is_empty() {
1202        let rendered_lines = render_terminal_output(&raw_result);
1203        if rendered_lines.is_empty() {
1204            // Fallback: just strip ANSI codes if rendering failed or wasn't needed
1205            result = strip_ansi_codes(&raw_result);
1206        } else {
1207            result = rendered_lines.join("\n");
1208        }
1209    }
1210
1211    result = truncate_to_token_budget(&result, MAX_OUTPUT_TOKENS).into_owned();
1212
1213    let exit_status = if output.status.success() {
1214        "Command completed successfully".to_string()
1215    } else {
1216        format!("Command failed with status: {}", output.status)
1217    };
1218
1219    let current_dir = std::env::current_dir()
1220        .map_or_else(|_| "Unknown".to_string(), |p| p.to_string_lossy().into_owned());
1221
1222    debug!("Command executed in {:.2?}", elapsed);
1223    Ok(format!("{result}\n\n---\n\nstatus = {exit_status}\ncwd = {current_dir}\n"))
1224}
1225
1226/// Execute command in screen (legacy)
1227#[allow(dead_code)]
1228#[tracing::instrument(level = "debug", skip(command, cwd, screen_name))]
1229async fn execute_in_screen(command: &str, cwd: &Path, screen_name: &str) -> Result<String> {
1230    debug!("Executing command in screen session '{}': {}", screen_name, command);
1231
1232    let screen_check = Command::new("which")
1233        .arg("screen")
1234        .output()
1235        .context("Failed to check for screen command")?;
1236
1237    if !screen_check.status.success() {
1238        warn!("Screen command not found, falling back to direct execution");
1239        return execute_simple_command(command, cwd).await;
1240    }
1241
1242    let _cleanup = Command::new("screen").args(["-X", "-S", screen_name, "quit"]).output();
1243
1244    let screen_cmd = format!(
1245        "screen -dmS {} bash -c '{} ; ec=$? ; echo \"Command completed with exit code: $ec\" ; sleep 1 ; exit $ec'",
1246        screen_name,
1247        command.replace('\'', "'\\''")
1248    );
1249
1250    let screen_start = Command::new("sh")
1251        .arg("-c")
1252        .arg(&screen_cmd)
1253        .current_dir(cwd)
1254        .output()
1255        .context("Failed to start screen session")?;
1256
1257    if !screen_start.status.success() {
1258        let stderr = String::from_utf8_lossy(&screen_start.stderr).to_string();
1259        error!("Failed to start screen session: {}", stderr);
1260        return Err(WinxError::CommandExecutionError(format!(
1261            "Failed to start screen session: {stderr}"
1262        )));
1263    }
1264
1265    sleep(Duration::from_millis(300)).await;
1266
1267    let screen_check =
1268        Command::new("screen").args(["-ls"]).output().context("Failed to list screen sessions")?;
1269
1270    let screen_list = String::from_utf8_lossy(&screen_check.stdout).to_string();
1271
1272    let current_dir = std::env::current_dir()
1273        .map_or_else(|_| "Unknown".to_string(), |p| p.to_string_lossy().into_owned());
1274
1275    Ok(format!(
1276        "Started command in background screen session '{screen_name}'.\n\
1277        Use status_check to get output.\n\n\
1278        Screen sessions:\n{screen_list}\n\
1279        ---\n\n\
1280        status = running in background\n\
1281        cwd = {current_dir}\n"
1282    ))
1283}
1284
1285/// Converts a `SpecialKey` to its screen stuff input representation (legacy)
1286#[allow(dead_code)]
1287fn special_key_to_screen_input(key: SpecialKey) -> String {
1288    match key {
1289        SpecialKey::Enter => String::from("\r"),
1290        SpecialKey::KeyUp => String::from("\x1b[A"),
1291        SpecialKey::KeyDown => String::from("\x1b[B"),
1292        SpecialKey::KeyLeft => String::from("\x1b[D"),
1293        SpecialKey::KeyRight => String::from("\x1b[C"),
1294        SpecialKey::CtrlC => String::from("\x03"),
1295        SpecialKey::CtrlD => String::from("\x04"),
1296        SpecialKey::CtrlZ => String::from("\x1a"),
1297    }
1298}