1use 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
28const DEFAULT_TIMEOUT: f64 = 5.0;
32
33const TIMEOUT_WHILE_OUTPUT: f64 = 20.0;
35
36const OUTPUT_WAIT_PATIENCE: i32 = 3;
38
39const COMMAND_CHUNK_SIZE: usize = 64;
41
42const TEXT_CHUNK_SIZE: usize = 128;
44
45const MAX_OUTPUT_LENGTH: usize = 100_000;
49
50const MAX_OUTPUT_TOKENS: usize = 25_000;
55
56fn 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 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 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
88const 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#[derive(Debug, Clone)]
102pub struct ExitedShellInfo {
103 pub last_command: String,
104 pub final_output: String,
105 pub exited_at: Instant,
106}
107
108#[derive(Debug, Default)]
110pub struct BackgroundShellManager {
111 shells: HashMap<String, SharedPtyShell>,
112 tombstones: HashMap<String, ExitedShellInfo>,
115}
116
117impl BackgroundShellManager {
118 const TOMBSTONE_TTL: Duration = Duration::from_secs(300);
120
121 pub fn new() -> Self {
123 Self { shells: HashMap::new(), tombstones: HashMap::new() }
124 }
125
126 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 pub fn get_shell(&self, bg_command_id: &str) -> Option<SharedPtyShell> {
142 self.shells.get(bg_command_id).cloned()
143 }
144
145 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 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 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 pub fn peek_tombstone(&self, bg_command_id: &str) -> Option<ExitedShellInfo> {
222 self.tombstones.get(bg_command_id).cloned()
223 }
224
225 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
255lazy_static::lazy_static! {
257 static ref BG_SHELL_MANAGER: StdMutex<BackgroundShellManager> = StdMutex::new(BackgroundShellManager::new());
258}
259
260fn 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 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
300fn 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 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 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
342fn rstrip_lines(lines: &[String]) -> String {
344 lines.iter().map(|line| line.trim_end()).collect::<Vec<_>>().join("\n")
345}
346
347fn 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 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#[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 }
414 _ => false,
415 }
416}
417
418#[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 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 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 if thread_id != bash_state.current_thread_id {
456 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 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 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 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
496async 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 let bg_shell: Option<SharedPtyShell> = match action {
507 BashCommandAction::Command { .. } => None, 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 drop(manager);
528 return finalize_tombstone(&bash_state.cwd, id, tombstone, action);
529 } else {
530 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 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
600async 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 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 let command = command.trim();
619 crate::utils::bash_parser::assert_single_statement(command)?;
620
621 if is_background {
623 return execute_in_background(bash_state, command, timeout_s).await;
624 }
625
626 {
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 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 {
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 {
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_utf8_in_byte_chunks(bash, command, COMMAND_CHUNK_SIZE)?;
668
669 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 let shell_arc = bash_state.pty_shell.clone();
680 wait_for_output(bash_state, &shell_arc, timeout_s, false, None, false).await
681}
682
683async 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 sleep(Duration::from_secs_f64(wait.min(DEFAULT_TIMEOUT))).await;
701
702 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 !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 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; }
758 last_incremental = new_incremental;
759
760 output = new_output;
761 remaining -= wait;
762 }
763
764 if !complete {
765 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 let rendered = wcgw_incremental_text(&output, &last_pending_output);
778
779 let rendered = truncate_to_token_budget(&rendered, MAX_OUTPUT_TOKENS).into_owned();
781
782 let running_for = if complete {
784 None
785 } else {
786 Some(format!("{} seconds", (start.elapsed().as_secs() + timeout_s as u64)))
787 };
788
789 let status = get_status(bash_state, is_bg, bg_id, !complete, running_for.as_deref());
791 Ok(format!("{rendered}{status}"))
792}
793
794fn 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 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 unreachable!("finalize_tombstone called for non-bg action")
827 }
828 }
829}
830
831async 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 let shell_arc = bg_shell.unwrap_or_else(|| bash_state.pty_shell.clone());
853
854 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 !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 let response = wait_for_output(bash_state, &shell_arc, timeout_s, is_bg, bg_id, true).await?;
876
877 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 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 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
918async 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 if text.is_empty() {
932 return Err(WinxError::CommandExecutionError(
933 "Failure: send_text cannot be empty".to_string(),
934 ));
935 }
936
937 let shell_arc = bg_shell.unwrap_or_else(|| bash_state.pty_shell.clone());
939
940 {
942 let mut guard = shell_arc.lock().await;
943
944 let bash = guard.as_mut().ok_or(WinxError::BashStateNotInitialized)?;
945
946 send_utf8_in_byte_chunks(bash, text, TEXT_CHUNK_SIZE)?;
948
949 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(bash_state, &shell_arc, timeout_s, is_bg, bg_id, false).await
961}
962
963async 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 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 for key in keys {
992 match key {
993 SpecialKey::KeyUp => {
994 bash.send_special_key("KeyUp").map_err(|e| {
996 WinxError::CommandExecutionError(format!("Failed to send KeyUp: {e}"))
997 })?;
998 }
999 SpecialKey::KeyDown => {
1000 bash.send_special_key("KeyDown").map_err(|e| {
1002 WinxError::CommandExecutionError(format!("Failed to send KeyDown: {e}"))
1003 })?;
1004 }
1005 SpecialKey::KeyLeft => {
1006 bash.send_special_key("KeyLeft").map_err(|e| {
1008 WinxError::CommandExecutionError(format!("Failed to send KeyLeft: {e}"))
1009 })?;
1010 }
1011 SpecialKey::KeyRight => {
1012 bash.send_special_key("KeyRight").map_err(|e| {
1014 WinxError::CommandExecutionError(format!("Failed to send KeyRight: {e}"))
1015 })?;
1016 }
1017 SpecialKey::Enter => {
1018 bash.send_special_key("Enter").map_err(|e| {
1020 WinxError::CommandExecutionError(format!("Failed to send Enter: {e}"))
1021 })?;
1022 }
1023 SpecialKey::CtrlC => {
1024 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 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 bash.send_suspend().map_err(|e| {
1040 WinxError::CommandExecutionError(format!("Failed to send Ctrl+Z: {e}"))
1041 })?;
1042 }
1043 }
1044 }
1045 if submit {
1047 bash.send_special_key("Enter")
1048 .map_err(|e| WinxError::CommandExecutionError(format!("Failed to submit: {e}")))?;
1049 }
1050 }
1051
1052 let mut output =
1060 wait_for_output(bash_state, &shell_arc, timeout_s, is_bg, bg_id, false).await?;
1061
1062 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
1070async 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 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 for &code in ascii_codes {
1099 bash.send_bytes(&[code]).map_err(|e| {
1101 WinxError::CommandExecutionError(format!("Failed to write ASCII code: {e}"))
1102 })?;
1103
1104 if code == 3 {
1106 is_interrupt = true;
1107 }
1108 }
1109 if submit {
1111 bash.send_special_key("Enter")
1112 .map_err(|e| WinxError::CommandExecutionError(format!("Failed to submit: {e}")))?;
1113 }
1114 }
1115
1116 let mut output =
1122 wait_for_output(bash_state, &shell_arc, timeout_s, is_bg, bg_id, false).await?;
1123
1124 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
1132async 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 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 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 {
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#[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 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#[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#[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}