use super::*;
impl DebugAdapter {
pub(crate) fn run(&mut self) -> io::Result<()> {
self.run_with_io(io::stdin(), io::stdout())
}
pub(crate) fn run_socket(&mut self, port: u16) -> io::Result<()> {
let listener = TcpListener::bind(("127.0.0.1", port))?;
tracing::info!(port, "DAP socket transport listening on 127.0.0.1");
let (stream, peer_addr) = listener.accept()?;
tracing::info!(peer_addr = %peer_addr, "DAP socket client connected");
let reader_stream = stream.try_clone()?;
self.run_with_io(reader_stream, stream)
}
pub(super) fn run_with_io<R, W>(&mut self, input: R, output: W) -> io::Result<()>
where
R: Read,
W: Write + Send + 'static,
{
let shared_writer: Arc<Mutex<W>> = Arc::new(Mutex::new(output));
let event_writer = Arc::clone(&shared_writer);
let (tx, rx) = channel::<DapMessage>();
self.event_sender = Some(tx.clone());
thread::spawn(move || {
while let Ok(msg) = rx.recv() {
let framed = match serde_json::to_vec(&msg) {
Ok(payload) => frame(&payload),
Err(e) => {
tracing::error!(error = %e, message = ?msg, "Failed to serialize DAP message");
continue;
}
};
let mut writer = lock_or_recover(&event_writer, "event_writer");
if let Err(e) = writer.write_all(&framed) {
tracing::error!(error = %e, "Failed to write DAP frame in event handler");
continue;
}
if let Err(e) = writer.flush() {
tracing::error!(error = %e, "Failed to flush DAP frame in event handler");
}
}
tracing::debug!("Event handler thread terminating - channel closed");
});
let mut reader = BufReader::new(input);
let mut framer = ContentLengthFramer::new();
let mut read_buf = [0u8; 8 * 1024];
loop {
let bytes_read = reader.read(&mut read_buf)?;
if bytes_read == 0 {
return Ok(());
}
framer.push(&read_buf[..bytes_read]);
loop {
let body = match framer.try_next() {
Ok(Some(body)) => body,
Ok(None) => break,
Err(error) => {
tracing::warn!(%error, "Failed to parse DAP transport frame");
continue;
}
};
let msg = match serde_json::from_slice::<DapMessage>(&body) {
Ok(msg) => msg,
Err(_) => {
tracing::warn!(body = %String::from_utf8_lossy(&body), "Failed to parse DAP message");
continue;
}
};
let DapMessage::Request { seq, command, arguments } = msg else {
continue;
};
let response = self.dispatch_request(seq, &command, arguments);
let payload = match serde_json::to_vec(&response) {
Ok(payload) => payload,
Err(e) => {
tracing::error!(error = %e, "Failed to serialize DAP response");
continue;
}
};
let framed = frame(&payload);
let mut writer = lock_or_recover(&shared_writer, "response_writer");
writer.write_all(&framed)?;
writer.flush()?;
if command == "initialize"
&& Self::response_succeeded_for_command(&response, "initialize")
{
self.send_event("initialized", None);
}
}
}
}
}