use crate::{ transport::{Event, Payload, Request, Transport}, types::*, Result, }; pub use log::{error, info}; use std::{ collections::HashMap, net::{IpAddr, Ipv4Addr, SocketAddr}, path::PathBuf, process::Stdio, sync::{ atomic::{AtomicU64, Ordering}, Arc, }, }; use tokio::{ io::{AsyncBufRead, AsyncWrite, BufReader, BufWriter}, net::TcpStream, process::{Child, Command}, sync::{ mpsc::{channel, Receiver, Sender, UnboundedReceiver, UnboundedSender}, Mutex, }, time, }; #[derive(Debug)] pub struct Client { id: usize, _process: Option, server_tx: UnboundedSender, request_counter: AtomicU64, capabilities: Option, awaited_events: Arc>>>, } impl Client { pub fn streams( rx: Box, tx: Box, id: usize, process: Option, ) -> Result { let (server_rx, server_tx) = Transport::start(rx, tx, id); let client = Self { id, _process: process, server_tx, request_counter: AtomicU64::new(0), capabilities: None, awaited_events: Arc::new(Mutex::new(HashMap::default())), }; tokio::spawn(Self::recv(Arc::clone(&client.awaited_events), server_rx)); Ok(client) } pub async fn tcp(addr: std::net::SocketAddr, id: usize) -> Result { let stream = TcpStream::connect(addr).await?; let (rx, tx) = stream.into_split(); Self::streams(Box::new(BufReader::new(rx)), Box::new(tx), id, None) } pub fn stdio(cmd: &str, args: Vec<&str>, id: usize) -> Result { let process = Command::new(cmd) .args(args) .stdin(Stdio::piped()) .stdout(Stdio::piped()) // make sure the process is reaped on drop .kill_on_drop(true) .spawn(); let mut process = process?; // TODO: do we need bufreader/writer here? or do we use async wrappers on unblock? let writer = BufWriter::new(process.stdin.take().expect("Failed to open stdin")); let reader = BufReader::new(process.stdout.take().expect("Failed to open stdout")); Self::streams( Box::new(BufReader::new(reader)), Box::new(writer), id, Some(process), ) } async fn get_port() -> Option { Some( tokio::net::TcpListener::bind(SocketAddr::new( IpAddr::V4(Ipv4Addr::new(127, 0, 0, 1)), 0, )) .await .ok()? .local_addr() .ok()? .port(), ) } pub async fn tcp_process( cmd: &str, args: Vec<&str>, port_format: &str, id: usize, ) -> Result { let port = Self::get_port().await.unwrap(); let process = Command::new(cmd) .args(args) .args(port_format.replace("{}", &port.to_string()).split(' ')) // silence messages .stdin(Stdio::null()) .stdout(Stdio::null()) .stderr(Stdio::null()) // make sure the process is reaped on drop .kill_on_drop(true) .spawn()?; // Wait for adapter to become ready for connection time::sleep(time::Duration::from_millis(500)).await; let stream = TcpStream::connect(SocketAddr::new( IpAddr::V4(Ipv4Addr::new(127, 0, 0, 1)), port, )) .await?; let (rx, tx) = stream.into_split(); Self::streams( Box::new(BufReader::new(rx)), Box::new(tx), id, Some(process), ) } async fn recv( awaited_events: Arc>>>, mut server_rx: UnboundedReceiver, ) { while let Some(msg) = server_rx.recv().await { match msg { Payload::Event(ev) => { let name = ev.event.clone(); let hashmap = awaited_events.lock().await; let tx = hashmap.get(&name); match tx { Some(tx) => match tx.send(ev).await { Ok(_) => (), Err(_) => error!( "Tried sending event into a closed channel (name={:?})", name ), }, None => { info!("unhandled event {}", name); // client_tx.send(Payload::Event(ev)).expect("Failed to send"); } } } Payload::Response(_) => unreachable!(), Payload::Request(_) => todo!(), } } } pub async fn listen_for_event(&self, name: String) -> Receiver { let (rx, tx) = channel(1); self.awaited_events.lock().await.insert(name.clone(), rx); tx } pub fn id(&self) -> usize { self.id } fn next_request_id(&self) -> u64 { self.request_counter.fetch_add(1, Ordering::Relaxed) } async fn request( &self, arguments: R::Arguments, ) -> Result { let (callback_tx, mut callback_rx) = channel(1); let arguments = Some(serde_json::to_value(arguments)?); let req = Request { back_ch: Some(callback_tx), seq: self.next_request_id(), command: R::COMMAND.to_string(), arguments, }; self.server_tx .send(req) .expect("Failed to send request to debugger"); let response = callback_rx.recv().await.unwrap()?; let response = serde_json::from_value(response.body.unwrap_or_default())?; Ok(response) } pub fn capabilities(&self) -> &DebuggerCapabilities { self.capabilities .as_ref() .expect("debugger not yet initialized!") } pub async fn initialize(&mut self, adapter_id: String) -> Result<()> { let args = requests::InitializeArguments { client_id: Some("hx".to_owned()), client_name: Some("helix".to_owned()), adapter_id, locale: Some("en-us".to_owned()), lines_start_at_one: Some(true), columns_start_at_one: Some(true), path_format: Some("path".to_owned()), supports_variable_type: Some(false), supports_variable_paging: Some(false), supports_run_in_terminal_request: Some(false), supports_memory_references: Some(false), supports_progress_reporting: Some(false), supports_invalidated_event: Some(false), }; let response = self.request::(args).await?; self.capabilities = Some(response); Ok(()) } pub async fn disconnect(&mut self) -> Result<()> { self.request::(()).await } pub async fn launch(&mut self, args: serde_json::Value) -> Result<()> { // TODO: buffer these until initialized arrives let response = self.request::(args).await?; log::error!("launch response {}", response); Ok(()) } pub async fn attach(&mut self, args: serde_json::Value) -> Result<()> { // TODO: buffer these until initialized arrives let response = self.request::(args).await?; log::error!("attach response {}", response); Ok(()) } pub async fn set_breakpoints( &mut self, file: String, breakpoints: Vec, ) -> Result>> { let args = requests::SetBreakpointsArguments { source: Source { path: Some(file), name: None, source_reference: None, presentation_hint: None, origin: None, sources: None, adapter_data: None, checksums: None, }, breakpoints: Some(breakpoints), source_modified: Some(false), }; let response = self.request::(args).await?; Ok(response.breakpoints) } pub async fn configuration_done(&mut self) -> Result<()> { self.request::(()).await } pub async fn continue_thread(&mut self, thread_id: usize) -> Result> { let args = requests::ContinueArguments { thread_id }; let response = self.request::(args).await?; Ok(response.all_threads_continued) } pub async fn stack_trace( &mut self, thread_id: usize, ) -> Result<(Vec, Option)> { let args = requests::StackTraceArguments { thread_id, start_frame: None, levels: None, format: None, }; let response = self.request::(args).await?; Ok((response.stack_frames, response.total_frames)) } pub async fn threads(&mut self) -> Result> { let response = self.request::(()).await?; Ok(response.threads) } pub async fn scopes(&mut self, frame_id: usize) -> Result> { let args = requests::ScopesArguments { frame_id }; let response = self.request::(args).await?; Ok(response.scopes) } pub async fn variables(&mut self, variables_reference: usize) -> Result> { let args = requests::VariablesArguments { variables_reference, filter: None, start: None, count: None, format: None, }; let response = self.request::(args).await?; Ok(response.variables) } }