mirror of
https://github.com/denoland/deno.git
synced 2025-01-12 00:54:02 -05:00
fix(inspector): don't panic if port is not free (#22745)
Closes https://github.com/denoland/deno/issues/22113 Closes https://github.com/denoland/deno/issues/23177 Closes https://github.com/denoland/deno/issues/22883 Closes https://github.com/denoland/deno/issues/22377
This commit is contained in:
parent
ade0cd5e97
commit
f358ae6278
7 changed files with 64 additions and 31 deletions
|
@ -1186,14 +1186,20 @@ impl CliOptions {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn resolve_inspector_server(&self) -> Option<InspectorServer> {
|
pub fn resolve_inspector_server(
|
||||||
|
&self,
|
||||||
|
) -> Result<Option<InspectorServer>, AnyError> {
|
||||||
let maybe_inspect_host = self
|
let maybe_inspect_host = self
|
||||||
.flags
|
.flags
|
||||||
.inspect
|
.inspect
|
||||||
.or(self.flags.inspect_brk)
|
.or(self.flags.inspect_brk)
|
||||||
.or(self.flags.inspect_wait);
|
.or(self.flags.inspect_wait);
|
||||||
maybe_inspect_host
|
|
||||||
.map(|host| InspectorServer::new(host, version::get_user_agent()))
|
let Some(host) = maybe_inspect_host else {
|
||||||
|
return Ok(None);
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(Some(InspectorServer::new(host, version::get_user_agent())?))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn maybe_lockfile(&self) -> Option<Arc<Mutex<Lockfile>>> {
|
pub fn maybe_lockfile(&self) -> Option<Arc<Mutex<Lockfile>>> {
|
||||||
|
|
|
@ -672,11 +672,15 @@ impl CliFactory {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn maybe_inspector_server(&self) -> &Option<Arc<InspectorServer>> {
|
pub fn maybe_inspector_server(
|
||||||
self
|
&self,
|
||||||
.services
|
) -> Result<&Option<Arc<InspectorServer>>, AnyError> {
|
||||||
.maybe_inspector_server
|
self.services.maybe_inspector_server.get_or_try_init(|| {
|
||||||
.get_or_init(|| self.options.resolve_inspector_server().map(Arc::new))
|
match self.options.resolve_inspector_server() {
|
||||||
|
Ok(server) => Ok(server.map(Arc::new)),
|
||||||
|
Err(err) => Err(err),
|
||||||
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn module_load_preparer(
|
pub async fn module_load_preparer(
|
||||||
|
@ -789,7 +793,7 @@ impl CliFactory {
|
||||||
self.root_cert_store_provider().clone(),
|
self.root_cert_store_provider().clone(),
|
||||||
self.fs().clone(),
|
self.fs().clone(),
|
||||||
maybe_file_watcher_communicator,
|
maybe_file_watcher_communicator,
|
||||||
self.maybe_inspector_server().clone(),
|
self.maybe_inspector_server()?.clone(),
|
||||||
self.maybe_lockfile().clone(),
|
self.maybe_lockfile().clone(),
|
||||||
self.feature_checker().clone(),
|
self.feature_checker().clone(),
|
||||||
self.create_cli_main_worker_options()?,
|
self.create_cli_main_worker_options()?,
|
||||||
|
|
|
@ -1692,7 +1692,7 @@ let c: number = "a";
|
||||||
let cache =
|
let cache =
|
||||||
Arc::new(GlobalHttpCache::new(cache_location, RealDenoCacheEnv));
|
Arc::new(GlobalHttpCache::new(cache_location, RealDenoCacheEnv));
|
||||||
let ts_server = TsServer::new(Default::default(), cache);
|
let ts_server = TsServer::new(Default::default(), cache);
|
||||||
ts_server.start(None);
|
ts_server.start(None).unwrap();
|
||||||
|
|
||||||
// test enabled
|
// test enabled
|
||||||
{
|
{
|
||||||
|
|
|
@ -971,9 +971,14 @@ impl Inner {
|
||||||
self.config.update_capabilities(¶ms.capabilities);
|
self.config.update_capabilities(¶ms.capabilities);
|
||||||
}
|
}
|
||||||
|
|
||||||
self
|
if let Err(e) = self
|
||||||
.ts_server
|
.ts_server
|
||||||
.start(self.config.internal_inspect().to_address());
|
.start(self.config.internal_inspect().to_address())
|
||||||
|
{
|
||||||
|
lsp_warn!("{}", e);
|
||||||
|
self.client.show_message(MessageType::ERROR, e);
|
||||||
|
return Err(tower_lsp::jsonrpc::Error::internal_error());
|
||||||
|
};
|
||||||
|
|
||||||
self.update_debug_flag();
|
self.update_debug_flag();
|
||||||
self.refresh_workspace_files();
|
self.refresh_workspace_files();
|
||||||
|
|
|
@ -36,6 +36,7 @@ use crate::util::path::to_percent_decoded_str;
|
||||||
use dashmap::DashMap;
|
use dashmap::DashMap;
|
||||||
use deno_ast::MediaType;
|
use deno_ast::MediaType;
|
||||||
use deno_core::anyhow::anyhow;
|
use deno_core::anyhow::anyhow;
|
||||||
|
use deno_core::anyhow::Context as _;
|
||||||
use deno_core::error::custom_error;
|
use deno_core::error::custom_error;
|
||||||
use deno_core::error::AnyError;
|
use deno_core::error::AnyError;
|
||||||
use deno_core::futures::FutureExt;
|
use deno_core::futures::FutureExt;
|
||||||
|
@ -268,17 +269,20 @@ impl TsServer {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn start(&self, inspector_server_addr: Option<String>) {
|
pub fn start(
|
||||||
let maybe_inspector_server = inspector_server_addr.and_then(|addr| {
|
&self,
|
||||||
let addr: SocketAddr = match addr.parse() {
|
inspector_server_addr: Option<String>,
|
||||||
Ok(addr) => addr,
|
) -> Result<(), AnyError> {
|
||||||
Err(err) => {
|
let maybe_inspector_server = match inspector_server_addr {
|
||||||
lsp_warn!("Invalid inspector server address \"{}\": {}", &addr, err);
|
Some(addr) => {
|
||||||
return None;
|
let addr: SocketAddr = addr.parse().with_context(|| {
|
||||||
}
|
format!("Invalid inspector server address \"{}\"", &addr)
|
||||||
};
|
})?;
|
||||||
Some(Arc::new(InspectorServer::new(addr, "deno-lsp-tsc")))
|
let server = InspectorServer::new(addr, "deno-lsp-tsc")?;
|
||||||
});
|
Some(Arc::new(server))
|
||||||
|
}
|
||||||
|
None => None,
|
||||||
|
};
|
||||||
*self.inspector_server.lock() = maybe_inspector_server.clone();
|
*self.inspector_server.lock() = maybe_inspector_server.clone();
|
||||||
// TODO(bartlomieju): why is the join_handle ignored here? Should we store it
|
// TODO(bartlomieju): why is the join_handle ignored here? Should we store it
|
||||||
// on the `TsServer` struct.
|
// on the `TsServer` struct.
|
||||||
|
@ -295,6 +299,7 @@ impl TsServer {
|
||||||
maybe_inspector_server,
|
maybe_inspector_server,
|
||||||
)
|
)
|
||||||
});
|
});
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn project_changed(
|
pub async fn project_changed(
|
||||||
|
@ -4750,7 +4755,7 @@ mod tests {
|
||||||
Arc::new(mock_state_snapshot(sources, &location, config).await);
|
Arc::new(mock_state_snapshot(sources, &location, config).await);
|
||||||
let performance = Arc::new(Performance::default());
|
let performance = Arc::new(Performance::default());
|
||||||
let ts_server = TsServer::new(performance, cache.clone());
|
let ts_server = TsServer::new(performance, cache.clone());
|
||||||
ts_server.start(None);
|
ts_server.start(None).unwrap();
|
||||||
(ts_server, snapshot, cache)
|
(ts_server, snapshot, cache)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -2,6 +2,8 @@
|
||||||
|
|
||||||
// Alias for the future `!` type.
|
// Alias for the future `!` type.
|
||||||
use core::convert::Infallible as Never;
|
use core::convert::Infallible as Never;
|
||||||
|
use deno_core::anyhow::Context;
|
||||||
|
use deno_core::error::AnyError;
|
||||||
use deno_core::futures::channel::mpsc;
|
use deno_core::futures::channel::mpsc;
|
||||||
use deno_core::futures::channel::mpsc::UnboundedReceiver;
|
use deno_core::futures::channel::mpsc::UnboundedReceiver;
|
||||||
use deno_core::futures::channel::mpsc::UnboundedSender;
|
use deno_core::futures::channel::mpsc::UnboundedSender;
|
||||||
|
@ -45,27 +47,38 @@ pub struct InspectorServer {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl InspectorServer {
|
impl InspectorServer {
|
||||||
pub fn new(host: SocketAddr, name: &'static str) -> Self {
|
pub fn new(host: SocketAddr, name: &'static str) -> Result<Self, AnyError> {
|
||||||
let (register_inspector_tx, register_inspector_rx) =
|
let (register_inspector_tx, register_inspector_rx) =
|
||||||
mpsc::unbounded::<InspectorInfo>();
|
mpsc::unbounded::<InspectorInfo>();
|
||||||
|
|
||||||
let (shutdown_server_tx, shutdown_server_rx) = broadcast::channel(1);
|
let (shutdown_server_tx, shutdown_server_rx) = broadcast::channel(1);
|
||||||
|
|
||||||
|
let tcp_listener =
|
||||||
|
std::net::TcpListener::bind(host).with_context(|| {
|
||||||
|
format!("Failed to start inspector server at \"{}\"", host)
|
||||||
|
})?;
|
||||||
|
tcp_listener.set_nonblocking(true)?;
|
||||||
|
|
||||||
let thread_handle = thread::spawn(move || {
|
let thread_handle = thread::spawn(move || {
|
||||||
let rt = crate::tokio_util::create_basic_runtime();
|
let rt = crate::tokio_util::create_basic_runtime();
|
||||||
let local = tokio::task::LocalSet::new();
|
let local = tokio::task::LocalSet::new();
|
||||||
local.block_on(
|
local.block_on(
|
||||||
&rt,
|
&rt,
|
||||||
server(host, register_inspector_rx, shutdown_server_rx, name),
|
server(
|
||||||
|
tcp_listener,
|
||||||
|
register_inspector_rx,
|
||||||
|
shutdown_server_rx,
|
||||||
|
name,
|
||||||
|
),
|
||||||
)
|
)
|
||||||
});
|
});
|
||||||
|
|
||||||
Self {
|
Ok(Self {
|
||||||
host,
|
host,
|
||||||
register_inspector_tx,
|
register_inspector_tx,
|
||||||
shutdown_server_tx: Some(shutdown_server_tx),
|
shutdown_server_tx: Some(shutdown_server_tx),
|
||||||
thread_handle: Some(thread_handle),
|
thread_handle: Some(thread_handle),
|
||||||
}
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn register_inspector(
|
pub fn register_inspector(
|
||||||
|
@ -220,7 +233,7 @@ fn handle_json_version_request(
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn server(
|
async fn server(
|
||||||
host: SocketAddr,
|
listener: std::net::TcpListener,
|
||||||
register_inspector_rx: UnboundedReceiver<InspectorInfo>,
|
register_inspector_rx: UnboundedReceiver<InspectorInfo>,
|
||||||
shutdown_server_rx: broadcast::Receiver<()>,
|
shutdown_server_rx: broadcast::Receiver<()>,
|
||||||
name: &str,
|
name: &str,
|
||||||
|
@ -261,7 +274,7 @@ async fn server(
|
||||||
});
|
});
|
||||||
|
|
||||||
// Create the server manually so it can use the Local Executor
|
// Create the server manually so it can use the Local Executor
|
||||||
let listener = match TcpListener::bind(&host).await {
|
let listener = match TcpListener::from_std(listener) {
|
||||||
Ok(l) => l,
|
Ok(l) => l,
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
eprintln!("Cannot start inspector server: {:?}", err);
|
eprintln!("Cannot start inspector server: {:?}", err);
|
||||||
|
|
|
@ -465,7 +465,7 @@ async fn inspector_port_collision() {
|
||||||
.lines()
|
.lines()
|
||||||
.map(|r| r.unwrap())
|
.map(|r| r.unwrap())
|
||||||
.inspect(|line| assert!(!line.contains("Debugger listening")))
|
.inspect(|line| assert!(!line.contains("Debugger listening")))
|
||||||
.find(|line| line.contains("Cannot start inspector server"));
|
.find(|line| line.contains("Failed to start inspector server"));
|
||||||
assert!(stderr_2_error_message.is_some());
|
assert!(stderr_2_error_message.is_some());
|
||||||
|
|
||||||
child1.kill().unwrap();
|
child1.kill().unwrap();
|
||||||
|
|
Loading…
Reference in a new issue