
- Normalize `node-terminal` to `pwa-node` before sending to DAP - Split `command` into `program` and `args` - Run in external console Release Notes: - debugger: Fixed debugging JavaScript tasks that used `"type": "node-terminal"`.
540 lines
24 KiB
Rust
540 lines
24 KiB
Rust
use adapters::latest_github_release;
|
|
use anyhow::Context as _;
|
|
use dap::{StartDebuggingRequestArguments, adapters::DebugTaskDefinition};
|
|
use gpui::AsyncApp;
|
|
use serde_json::Value;
|
|
use std::{collections::HashMap, path::PathBuf, sync::OnceLock};
|
|
use task::DebugRequest;
|
|
use util::{ResultExt, maybe};
|
|
|
|
use crate::*;
|
|
|
|
#[derive(Debug, Default)]
|
|
pub(crate) struct JsDebugAdapter {
|
|
checked: OnceLock<()>,
|
|
}
|
|
|
|
impl JsDebugAdapter {
|
|
const ADAPTER_NAME: &'static str = "JavaScript";
|
|
const ADAPTER_NPM_NAME: &'static str = "vscode-js-debug";
|
|
const ADAPTER_PATH: &'static str = "js-debug/src/dapDebugServer.js";
|
|
|
|
async fn fetch_latest_adapter_version(
|
|
&self,
|
|
delegate: &Arc<dyn DapDelegate>,
|
|
) -> Result<AdapterVersion> {
|
|
let release = latest_github_release(
|
|
&format!("microsoft/{}", Self::ADAPTER_NPM_NAME),
|
|
true,
|
|
false,
|
|
delegate.http_client(),
|
|
)
|
|
.await?;
|
|
|
|
let asset_name = format!("js-debug-dap-{}.tar.gz", release.tag_name);
|
|
|
|
Ok(AdapterVersion {
|
|
tag_name: release.tag_name,
|
|
url: release
|
|
.assets
|
|
.iter()
|
|
.find(|asset| asset.name == asset_name)
|
|
.with_context(|| format!("no asset found matching {asset_name:?}"))?
|
|
.browser_download_url
|
|
.clone(),
|
|
})
|
|
}
|
|
|
|
async fn get_installed_binary(
|
|
&self,
|
|
delegate: &Arc<dyn DapDelegate>,
|
|
task_definition: &DebugTaskDefinition,
|
|
user_installed_path: Option<PathBuf>,
|
|
user_args: Option<Vec<String>>,
|
|
_: &mut AsyncApp,
|
|
) -> Result<DebugAdapterBinary> {
|
|
let adapter_path = if let Some(user_installed_path) = user_installed_path {
|
|
user_installed_path
|
|
} else {
|
|
let adapter_path = paths::debug_adapters_dir().join(self.name().as_ref());
|
|
|
|
let file_name_prefix = format!("{}_", self.name());
|
|
|
|
util::fs::find_file_name_in_dir(adapter_path.as_path(), |file_name| {
|
|
file_name.starts_with(&file_name_prefix)
|
|
})
|
|
.await
|
|
.context("Couldn't find JavaScript dap directory")?
|
|
};
|
|
|
|
let tcp_connection = task_definition.tcp_connection.clone().unwrap_or_default();
|
|
let (host, port, timeout) = crate::configure_tcp_connection(tcp_connection).await?;
|
|
|
|
let mut configuration = task_definition.config.clone();
|
|
if let Some(configuration) = configuration.as_object_mut() {
|
|
maybe!({
|
|
configuration
|
|
.get("type")
|
|
.filter(|value| value == &"node-terminal")?;
|
|
let command = configuration.get("command")?.as_str()?.to_owned();
|
|
let mut args = shlex::split(&command)?.into_iter();
|
|
let program = args.next()?;
|
|
configuration.insert("program".to_owned(), program.into());
|
|
configuration.insert(
|
|
"args".to_owned(),
|
|
args.map(Value::from).collect::<Vec<_>>().into(),
|
|
);
|
|
configuration.insert("console".to_owned(), "externalTerminal".into());
|
|
Some(())
|
|
});
|
|
|
|
configuration.entry("type").and_modify(normalize_task_type);
|
|
|
|
if let Some(program) = configuration
|
|
.get("program")
|
|
.cloned()
|
|
.and_then(|value| value.as_str().map(str::to_owned))
|
|
{
|
|
match program.as_str() {
|
|
"npm" | "pnpm" | "yarn" | "bun"
|
|
if !configuration.contains_key("runtimeExecutable")
|
|
&& !configuration.contains_key("runtimeArgs") =>
|
|
{
|
|
configuration.remove("program");
|
|
configuration.insert("runtimeExecutable".to_owned(), program.into());
|
|
if let Some(args) = configuration.remove("args") {
|
|
configuration.insert("runtimeArgs".to_owned(), args);
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
configuration
|
|
.entry("cwd")
|
|
.or_insert(delegate.worktree_root_path().to_string_lossy().into());
|
|
|
|
configuration
|
|
.entry("console")
|
|
.or_insert("externalTerminal".into());
|
|
|
|
configuration.entry("sourceMaps").or_insert(true.into());
|
|
configuration
|
|
.entry("pauseForSourceMap")
|
|
.or_insert(true.into());
|
|
configuration
|
|
.entry("sourceMapRenames")
|
|
.or_insert(true.into());
|
|
}
|
|
|
|
let arguments = if let Some(mut args) = user_args {
|
|
args.insert(
|
|
0,
|
|
adapter_path
|
|
.join(Self::ADAPTER_PATH)
|
|
.to_string_lossy()
|
|
.to_string(),
|
|
);
|
|
args
|
|
} else {
|
|
vec![
|
|
adapter_path
|
|
.join(Self::ADAPTER_PATH)
|
|
.to_string_lossy()
|
|
.to_string(),
|
|
port.to_string(),
|
|
host.to_string(),
|
|
]
|
|
};
|
|
|
|
Ok(DebugAdapterBinary {
|
|
command: Some(
|
|
delegate
|
|
.node_runtime()
|
|
.binary_path()
|
|
.await?
|
|
.to_string_lossy()
|
|
.into_owned(),
|
|
),
|
|
arguments,
|
|
cwd: Some(delegate.worktree_root_path().to_path_buf()),
|
|
envs: HashMap::default(),
|
|
connection: Some(adapters::TcpArguments {
|
|
host,
|
|
port,
|
|
timeout,
|
|
}),
|
|
request_args: StartDebuggingRequestArguments {
|
|
configuration,
|
|
request: self.request_kind(&task_definition.config).await?,
|
|
},
|
|
})
|
|
}
|
|
}
|
|
|
|
#[async_trait(?Send)]
|
|
impl DebugAdapter for JsDebugAdapter {
|
|
fn name(&self) -> DebugAdapterName {
|
|
DebugAdapterName(Self::ADAPTER_NAME.into())
|
|
}
|
|
|
|
async fn config_from_zed_format(&self, zed_scenario: ZedDebugConfig) -> Result<DebugScenario> {
|
|
let mut args = json!({
|
|
"type": "pwa-node",
|
|
"request": match zed_scenario.request {
|
|
DebugRequest::Launch(_) => "launch",
|
|
DebugRequest::Attach(_) => "attach",
|
|
},
|
|
});
|
|
|
|
let map = args.as_object_mut().unwrap();
|
|
match &zed_scenario.request {
|
|
DebugRequest::Attach(attach) => {
|
|
map.insert("processId".into(), attach.process_id.into());
|
|
}
|
|
DebugRequest::Launch(launch) => {
|
|
if launch.program.starts_with("http://") {
|
|
map.insert("url".into(), launch.program.clone().into());
|
|
} else {
|
|
map.insert("program".into(), launch.program.clone().into());
|
|
}
|
|
|
|
if !launch.args.is_empty() {
|
|
map.insert("args".into(), launch.args.clone().into());
|
|
}
|
|
if !launch.env.is_empty() {
|
|
map.insert("env".into(), launch.env_json());
|
|
}
|
|
|
|
if let Some(stop_on_entry) = zed_scenario.stop_on_entry {
|
|
map.insert("stopOnEntry".into(), stop_on_entry.into());
|
|
}
|
|
if let Some(cwd) = launch.cwd.as_ref() {
|
|
map.insert("cwd".into(), cwd.to_string_lossy().into_owned().into());
|
|
}
|
|
}
|
|
};
|
|
|
|
Ok(DebugScenario {
|
|
adapter: zed_scenario.adapter,
|
|
label: zed_scenario.label,
|
|
build: None,
|
|
config: args,
|
|
tcp_connection: None,
|
|
})
|
|
}
|
|
|
|
fn dap_schema(&self) -> serde_json::Value {
|
|
json!({
|
|
"oneOf": [
|
|
{
|
|
"allOf": [
|
|
{
|
|
"type": "object",
|
|
"required": ["request"],
|
|
"properties": {
|
|
"request": {
|
|
"type": "string",
|
|
"enum": ["launch"],
|
|
"description": "Request to launch a new process"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"type": "object",
|
|
"properties": {
|
|
"type": {
|
|
"type": "string",
|
|
"enum": ["pwa-node", "node", "chrome", "pwa-chrome", "msedge", "pwa-msedge"],
|
|
"description": "The type of debug session",
|
|
"default": "pwa-node"
|
|
},
|
|
"program": {
|
|
"type": "string",
|
|
"description": "Path to the program or file to debug"
|
|
},
|
|
"cwd": {
|
|
"type": "string",
|
|
"description": "Absolute path to the working directory of the program being debugged"
|
|
},
|
|
"args": {
|
|
"type": ["array", "string"],
|
|
"description": "Command line arguments passed to the program",
|
|
"items": {
|
|
"type": "string"
|
|
},
|
|
"default": []
|
|
},
|
|
"env": {
|
|
"type": "object",
|
|
"description": "Environment variables passed to the program",
|
|
"default": {}
|
|
},
|
|
"envFile": {
|
|
"type": ["string", "array"],
|
|
"description": "Path to a file containing environment variable definitions",
|
|
"items": {
|
|
"type": "string"
|
|
}
|
|
},
|
|
"stopOnEntry": {
|
|
"type": "boolean",
|
|
"description": "Automatically stop program after launch",
|
|
"default": false
|
|
},
|
|
"runtimeExecutable": {
|
|
"type": ["string", "null"],
|
|
"description": "Runtime to use, an absolute path or the name of a runtime available on PATH",
|
|
"default": "node"
|
|
},
|
|
"runtimeArgs": {
|
|
"type": ["array", "null"],
|
|
"description": "Arguments passed to the runtime executable",
|
|
"items": {
|
|
"type": "string"
|
|
},
|
|
"default": []
|
|
},
|
|
"outFiles": {
|
|
"type": "array",
|
|
"description": "Glob patterns for locating generated JavaScript files",
|
|
"items": {
|
|
"type": "string"
|
|
},
|
|
"default": ["${ZED_WORKTREE_ROOT}/**/*.js", "!**/node_modules/**"]
|
|
},
|
|
"sourceMaps": {
|
|
"type": "boolean",
|
|
"description": "Use JavaScript source maps if they exist",
|
|
"default": true
|
|
},
|
|
"pauseForSourceMap": {
|
|
"type": "boolean",
|
|
"description": "Wait for source maps to load before setting breakpoints.",
|
|
"default": true
|
|
},
|
|
"sourceMapRenames": {
|
|
"type": "boolean",
|
|
"description": "Whether to use the \"names\" mapping in sourcemaps.",
|
|
"default": true
|
|
},
|
|
"sourceMapPathOverrides": {
|
|
"type": "object",
|
|
"description": "Rewrites the locations of source files from what the sourcemap says to their locations on disk",
|
|
"default": {}
|
|
},
|
|
"restart": {
|
|
"type": ["boolean", "object"],
|
|
"description": "Restart session after Node.js has terminated",
|
|
"default": false
|
|
},
|
|
"trace": {
|
|
"type": ["boolean", "object"],
|
|
"description": "Enables logging of the Debug Adapter",
|
|
"default": false
|
|
},
|
|
"console": {
|
|
"type": "string",
|
|
"enum": ["internalConsole", "integratedTerminal"],
|
|
"description": "Where to launch the debug target",
|
|
"default": "internalConsole"
|
|
},
|
|
// Browser-specific
|
|
"url": {
|
|
"type": ["string", "null"],
|
|
"description": "Will navigate to this URL and attach to it (browser debugging)"
|
|
},
|
|
"webRoot": {
|
|
"type": "string",
|
|
"description": "Workspace absolute path to the webserver root",
|
|
"default": "${ZED_WORKTREE_ROOT}"
|
|
},
|
|
"userDataDir": {
|
|
"type": ["string", "boolean"],
|
|
"description": "Path to a custom Chrome user profile (browser debugging)",
|
|
"default": true
|
|
},
|
|
"skipFiles": {
|
|
"type": "array",
|
|
"description": "An array of glob patterns for files to skip when debugging",
|
|
"items": {
|
|
"type": "string"
|
|
},
|
|
"default": ["<node_internals>/**"]
|
|
},
|
|
"timeout": {
|
|
"type": "number",
|
|
"description": "Retry for this number of milliseconds to connect to the debug adapter",
|
|
"default": 10000
|
|
},
|
|
"resolveSourceMapLocations": {
|
|
"type": ["array", "null"],
|
|
"description": "A list of minimatch patterns for source map resolution",
|
|
"items": {
|
|
"type": "string"
|
|
}
|
|
}
|
|
},
|
|
"oneOf": [
|
|
{ "required": ["program"] },
|
|
{ "required": ["url"] }
|
|
]
|
|
}
|
|
]
|
|
},
|
|
{
|
|
"allOf": [
|
|
{
|
|
"type": "object",
|
|
"required": ["request"],
|
|
"properties": {
|
|
"request": {
|
|
"type": "string",
|
|
"enum": ["attach"],
|
|
"description": "Request to attach to an existing process"
|
|
}
|
|
}
|
|
},
|
|
{
|
|
"type": "object",
|
|
"properties": {
|
|
"type": {
|
|
"type": "string",
|
|
"enum": ["pwa-node", "node", "chrome", "pwa-chrome", "edge", "pwa-edge"],
|
|
"description": "The type of debug session",
|
|
"default": "pwa-node"
|
|
},
|
|
"processId": {
|
|
"type": ["string", "number"],
|
|
"description": "ID of process to attach to (Node.js debugging)"
|
|
},
|
|
"port": {
|
|
"type": "number",
|
|
"description": "Debug port to attach to",
|
|
"default": 9229
|
|
},
|
|
"address": {
|
|
"type": "string",
|
|
"description": "TCP/IP address of the process to be debugged",
|
|
"default": "localhost"
|
|
},
|
|
"restart": {
|
|
"type": ["boolean", "object"],
|
|
"description": "Restart session after Node.js has terminated",
|
|
"default": false
|
|
},
|
|
"sourceMaps": {
|
|
"type": "boolean",
|
|
"description": "Use JavaScript source maps if they exist",
|
|
"default": true
|
|
},
|
|
"sourceMapPathOverrides": {
|
|
"type": "object",
|
|
"description": "Rewrites the locations of source files from what the sourcemap says to their locations on disk",
|
|
"default": {}
|
|
},
|
|
"outFiles": {
|
|
"type": "array",
|
|
"description": "Glob patterns for locating generated JavaScript files",
|
|
"items": {
|
|
"type": "string"
|
|
},
|
|
"default": ["${ZED_WORKTREE_ROOT}/**/*.js", "!**/node_modules/**"]
|
|
},
|
|
"url": {
|
|
"type": "string",
|
|
"description": "Will search for a page with this URL and attach to it (browser debugging)"
|
|
},
|
|
"webRoot": {
|
|
"type": "string",
|
|
"description": "Workspace absolute path to the webserver root",
|
|
"default": "${ZED_WORKTREE_ROOT}"
|
|
},
|
|
"skipFiles": {
|
|
"type": "array",
|
|
"description": "An array of glob patterns for files to skip when debugging",
|
|
"items": {
|
|
"type": "string"
|
|
},
|
|
"default": ["<node_internals>/**"]
|
|
},
|
|
"timeout": {
|
|
"type": "number",
|
|
"description": "Retry for this number of milliseconds to connect to the debug adapter",
|
|
"default": 10000
|
|
},
|
|
"resolveSourceMapLocations": {
|
|
"type": ["array", "null"],
|
|
"description": "A list of minimatch patterns for source map resolution",
|
|
"items": {
|
|
"type": "string"
|
|
}
|
|
},
|
|
"remoteRoot": {
|
|
"type": ["string", "null"],
|
|
"description": "Path to the remote directory containing the program"
|
|
},
|
|
"localRoot": {
|
|
"type": ["string", "null"],
|
|
"description": "Path to the local directory containing the program"
|
|
}
|
|
},
|
|
"oneOf": [
|
|
{ "required": ["processId"] },
|
|
{ "required": ["port"] }
|
|
]
|
|
}
|
|
]
|
|
}
|
|
]
|
|
})
|
|
}
|
|
|
|
async fn get_binary(
|
|
&self,
|
|
delegate: &Arc<dyn DapDelegate>,
|
|
config: &DebugTaskDefinition,
|
|
user_installed_path: Option<PathBuf>,
|
|
user_args: Option<Vec<String>>,
|
|
cx: &mut AsyncApp,
|
|
) -> Result<DebugAdapterBinary> {
|
|
if self.checked.set(()).is_ok() {
|
|
delegate.output_to_console(format!("Checking latest version of {}...", self.name()));
|
|
if let Some(version) = self.fetch_latest_adapter_version(delegate).await.log_err() {
|
|
adapters::download_adapter_from_github(
|
|
self.name(),
|
|
version,
|
|
adapters::DownloadedFileType::GzipTar,
|
|
delegate.as_ref(),
|
|
)
|
|
.await?;
|
|
} else {
|
|
delegate.output_to_console(format!("{} debug adapter is up to date", self.name()));
|
|
}
|
|
}
|
|
|
|
self.get_installed_binary(delegate, &config, user_installed_path, user_args, cx)
|
|
.await
|
|
}
|
|
|
|
fn label_for_child_session(&self, args: &StartDebuggingRequestArguments) -> Option<String> {
|
|
let label = args.configuration.get("name")?.as_str()?;
|
|
Some(label.to_owned())
|
|
}
|
|
}
|
|
|
|
fn normalize_task_type(task_type: &mut Value) {
|
|
let Some(task_type_str) = task_type.as_str() else {
|
|
return;
|
|
};
|
|
|
|
let new_name = match task_type_str {
|
|
"node" | "pwa-node" | "node-terminal" => "pwa-node",
|
|
"chrome" | "pwa-chrome" => "pwa-chrome",
|
|
"edge" | "msedge" | "pwa-edge" | "pwa-msedge" => "pwa-msedge",
|
|
_ => task_type_str,
|
|
}
|
|
.to_owned();
|
|
|
|
*task_type = Value::String(new_name);
|
|
}
|