use super::DapLocator; use anyhow::{Result, anyhow}; use async_trait::async_trait; use serde_json::{Value, json}; use smol::{ io::AsyncReadExt, process::{Command, Stdio}, }; use task::DebugTaskDefinition; use util::maybe; pub(super) struct CargoLocator; async fn find_best_executable(executables: &[String], test_name: &str) -> Option { if executables.len() == 1 { return executables.first().cloned(); } for executable in executables { let Some(mut child) = Command::new(&executable) .arg("--list") .stdout(Stdio::piped()) .spawn() .ok() else { continue; }; let mut test_lines = String::default(); if let Some(mut stdout) = child.stdout.take() { stdout.read_to_string(&mut test_lines).await.ok(); for line in test_lines.lines() { if line.contains(&test_name) { return Some(executable.clone()); } } } } None } #[async_trait] impl DapLocator for CargoLocator { async fn run_locator(&self, debug_config: &mut DebugTaskDefinition) -> Result<()> { let Some(launch_config) = (match &mut debug_config.request { task::DebugRequestType::Launch(launch_config) => Some(launch_config), _ => None, }) else { return Err(anyhow!("Couldn't get launch config in locator")); }; let Some(cwd) = launch_config.cwd.clone() else { return Err(anyhow!( "Couldn't get cwd from debug config which is needed for locators" )); }; let mut child = Command::new("cargo") .args(&launch_config.args) .arg("--message-format=json") .current_dir(cwd) .stdout(Stdio::piped()) .spawn()?; let mut output = String::new(); if let Some(mut stdout) = child.stdout.take() { stdout.read_to_string(&mut output).await?; } let status = child.status().await?; if !status.success() { return Err(anyhow::anyhow!("Cargo command failed")); } let executables = output .lines() .filter(|line| !line.trim().is_empty()) .filter_map(|line| serde_json::from_str(line).ok()) .filter_map(|json: Value| { json.get("executable") .and_then(Value::as_str) .map(String::from) }) .collect::>(); if executables.is_empty() { return Err(anyhow!("Couldn't get executable in cargo locator")); }; let is_test = launch_config .args .first() .map_or(false, |arg| arg == "test"); let mut test_name = None; if is_test { if let Some(package_index) = launch_config .args .iter() .position(|arg| arg == "-p" || arg == "--package") { test_name = launch_config .args .get(package_index + 2) .filter(|name| !name.starts_with("--")) .cloned(); } } let executable = { if let Some(ref name) = test_name { find_best_executable(&executables, &name).await } else { None } }; let Some(executable) = executable.or_else(|| executables.first().cloned()) else { return Err(anyhow!("Couldn't get executable in cargo locator")); }; launch_config.program = executable; if debug_config.adapter == "LLDB" && debug_config.initialize_args.is_none() { // Find Rust pretty-printers in current toolchain's sysroot let cwd = launch_config.cwd.clone(); debug_config.initialize_args = maybe!(async move { let cwd = cwd?; let output = Command::new("rustc") .arg("--print") .arg("sysroot") .current_dir(cwd) .output() .await .ok()?; if !output.status.success() { return None; } let sysroot_path = String::from_utf8(output.stdout).ok()?; let sysroot_path = sysroot_path.trim_end(); let first_command = format!( r#"command script import "{sysroot_path}/lib/rustlib/etc/lldb_lookup.py"# ); let second_command = format!(r#"command source -s 0 '{sysroot_path}/lib/rustlib/etc/lldb_commands"#); Some(json!({"initCommands": [first_command, second_command]})) }) .await; } launch_config.args.clear(); if let Some(test_name) = test_name { launch_config.args.push(test_name); } Ok(()) } }