python.rs

 1use std::path::Path;
 2
 3use anyhow::{Result, bail};
 4use async_trait::async_trait;
 5use dap::{DapLocator, DebugRequest, adapters::DebugAdapterName};
 6use gpui::SharedString;
 7
 8use task::{DebugScenario, SpawnInTerminal, TaskTemplate};
 9
10pub(crate) struct PythonLocator;
11
12#[async_trait]
13impl DapLocator for PythonLocator {
14    fn name(&self) -> SharedString {
15        SharedString::new_static("Python")
16    }
17
18    /// Determines whether this locator can generate debug target for given task.
19    async fn create_scenario(
20        &self,
21        build_config: &TaskTemplate,
22        resolved_label: &str,
23        adapter: &DebugAdapterName,
24    ) -> Option<DebugScenario> {
25        if adapter.0.as_ref() != "Debugpy" {
26            return None;
27        }
28        let valid_program = build_config.command.starts_with("\"$ZED_")
29            || Path::new(&build_config.command)
30                .file_name()
31                .is_some_and(|name| name.to_str().is_some_and(|path| path.starts_with("python")));
32        if !valid_program || build_config.args.iter().any(|arg| arg == "-c") {
33            // We cannot debug selections.
34            return None;
35        }
36        let command = build_config.command.clone();
37        let module_specifier_position = build_config
38            .args
39            .iter()
40            .position(|arg| arg == "-m")
41            .map(|position| position + 1);
42        // Skip the -m and module name, get all that's after.
43        let mut rest_of_the_args = module_specifier_position
44            .and_then(|position| build_config.args.get(position..))
45            .into_iter()
46            .flatten()
47            .fuse();
48        let mod_name = rest_of_the_args.next();
49        let args = rest_of_the_args.collect::<Vec<_>>();
50
51        let program_position = mod_name
52            .is_none()
53            .then(|| {
54                build_config
55                    .args
56                    .iter()
57                    .position(|arg| *arg == "\"$ZED_FILE\"")
58            })
59            .flatten();
60        let args = if let Some(position) = program_position {
61            args.into_iter().skip(position).collect::<Vec<_>>()
62        } else {
63            args
64        };
65        if program_position.is_none() && mod_name.is_none() {
66            return None;
67        }
68        let mut config = serde_json::json!({
69            "request": "launch",
70            "python": command,
71            "args": args,
72            "cwd": build_config.cwd.clone()
73        });
74        if let Some(config_obj) = config.as_object_mut() {
75            if let Some(module) = mod_name {
76                config_obj.insert("module".to_string(), module.clone().into());
77            }
78            if let Some(program) = program_position {
79                config_obj.insert(
80                    "program".to_string(),
81                    build_config.args[program].clone().into(),
82                );
83            }
84        }
85
86        Some(DebugScenario {
87            adapter: adapter.0.clone(),
88            label: resolved_label.to_string().into(),
89            build: None,
90            config,
91            tcp_connection: None,
92        })
93    }
94
95    async fn run(&self, _: SpawnInTerminal) -> Result<DebugRequest> {
96        bail!("Python locator should not require DapLocator::run to be ran");
97    }
98}