1use std::{path::Path, sync::Arc};
2
3use dap::client::DebugAdapterClient;
4use gpui::{App, AppContext, Subscription};
5
6use super::session::{Session, SessionStateEvent};
7
8pub fn intercept_debug_sessions<T: Fn(&Arc<DebugAdapterClient>) + 'static>(
9 cx: &mut gpui::TestAppContext,
10 configure: T,
11) -> Subscription {
12 cx.update(|cx| {
13 let configure = Arc::new(configure);
14 cx.observe_new::<Session>(move |_, _, cx| {
15 let configure = configure.clone();
16 cx.subscribe_self(move |session, event, cx| {
17 let configure = configure.clone();
18 if matches!(event, SessionStateEvent::Running) {
19 let client = session.adapter_client().unwrap();
20 register_default_handlers(session, &client, cx);
21 configure(&client);
22 cx.background_spawn(async move {
23 client
24 .fake_event(dap::messages::Events::Initialized(
25 Some(Default::default()),
26 ))
27 .await
28 })
29 .detach();
30 }
31 })
32 .detach();
33 })
34 })
35}
36
37fn register_default_handlers(session: &Session, client: &Arc<DebugAdapterClient>, cx: &mut App) {
38 client.on_request::<dap::requests::Initialize, _>(move |_, _| Ok(Default::default()));
39 let paths = session.breakpoint_store.read(cx).breakpoint_paths();
40
41 client.on_request::<dap::requests::SetBreakpoints, _>(move |_, args| {
42 let p = Arc::from(Path::new(&args.source.path.unwrap()));
43 if !paths.contains(&p) {
44 panic!("Sent breakpoints for path without any")
45 }
46
47 Ok(dap::SetBreakpointsResponse {
48 breakpoints: Vec::default(),
49 })
50 });
51
52 client.on_request::<dap::requests::Launch, _>(move |_, _| Ok(()));
53
54 client.on_request::<dap::requests::SetExceptionBreakpoints, _>(move |_, _| {
55 Ok(dap::SetExceptionBreakpointsResponse { breakpoints: None })
56 });
57
58 client.on_request::<dap::requests::Disconnect, _>(move |_, _| Ok(()));
59
60 client.on_request::<dap::requests::Threads, _>(move |_, _| {
61 Ok(dap::ThreadsResponse { threads: vec![] })
62 });
63}