mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-14 06:03:58 +00:00
Re-implement dependecny tracking for cargo script
This commit is contained in:
parent
2f828073aa
commit
0b24599cf9
10 changed files with 187 additions and 234 deletions
|
@ -335,7 +335,7 @@ fn load_crate_graph(
|
||||||
) -> RootDatabase {
|
) -> RootDatabase {
|
||||||
let (ProjectWorkspace::Cargo { toolchain, target_layout, .. }
|
let (ProjectWorkspace::Cargo { toolchain, target_layout, .. }
|
||||||
| ProjectWorkspace::Json { toolchain, target_layout, .. }
|
| ProjectWorkspace::Json { toolchain, target_layout, .. }
|
||||||
| ProjectWorkspace::DetachedFiles { toolchain, target_layout, .. }) = ws;
|
| ProjectWorkspace::DetachedFile { toolchain, target_layout, .. }) = ws;
|
||||||
|
|
||||||
let lru_cap = std::env::var("RA_LRU_CAP").ok().and_then(|it| it.parse::<usize>().ok());
|
let lru_cap = std::env::var("RA_LRU_CAP").ok().and_then(|it| it.parse::<usize>().ok());
|
||||||
let mut db = RootDatabase::new(lru_cap);
|
let mut db = RootDatabase::new(lru_cap);
|
||||||
|
|
|
@ -305,8 +305,10 @@ impl CargoWorkspace {
|
||||||
.collect(),
|
.collect(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
if cargo_toml.extension().is_some_and(|x| x == "rs") {
|
// The manifest is a rust file, so this means its a script manifest
|
||||||
// TODO: enable `+nightly` for cargo scripts
|
if cargo_toml.extension().is_some_and(|ext| ext == "rs") {
|
||||||
|
// Deliberately don't set up RUSTC_BOOTSTRAP or a nightly override here, the user should
|
||||||
|
// opt into it themselves.
|
||||||
other_options.push("-Zscript".to_owned());
|
other_options.push("-Zscript".to_owned());
|
||||||
}
|
}
|
||||||
meta.other_options(other_options);
|
meta.other_options(other_options);
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
//! metadata` or `rust-project.json`) into representation stored in the salsa
|
//! metadata` or `rust-project.json`) into representation stored in the salsa
|
||||||
//! database -- `CrateGraph`.
|
//! database -- `CrateGraph`.
|
||||||
|
|
||||||
use std::{collections::VecDeque, fmt, fs, io::BufRead, iter, sync};
|
use std::{collections::VecDeque, fmt, fs, iter, sync};
|
||||||
|
|
||||||
use anyhow::{format_err, Context};
|
use anyhow::{format_err, Context};
|
||||||
use base_db::{
|
use base_db::{
|
||||||
|
@ -99,9 +99,9 @@ pub enum ProjectWorkspace {
|
||||||
// //
|
// //
|
||||||
/// Project with a set of disjoint files, not belonging to any particular workspace.
|
/// Project with a set of disjoint files, not belonging to any particular workspace.
|
||||||
/// Backed by basic sysroot crates for basic completion and highlighting.
|
/// Backed by basic sysroot crates for basic completion and highlighting.
|
||||||
DetachedFiles {
|
DetachedFile {
|
||||||
/// The set of detached files.
|
/// The file in question.
|
||||||
files: Vec<AbsPathBuf>,
|
file: AbsPathBuf,
|
||||||
/// The sysroot loaded for this workspace.
|
/// The sysroot loaded for this workspace.
|
||||||
sysroot: Result<Sysroot, Option<String>>,
|
sysroot: Result<Sysroot, Option<String>>,
|
||||||
/// Holds cfg flags for the current target. We get those by running
|
/// Holds cfg flags for the current target. We get those by running
|
||||||
|
@ -120,50 +120,6 @@ pub enum ProjectWorkspace {
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Tracks the cargo toml parts in cargo scripts, to detect if they
|
|
||||||
/// changed and reload workspace in that case.
|
|
||||||
pub struct CargoScriptTomls(pub FxHashMap<AbsPathBuf, String>);
|
|
||||||
|
|
||||||
impl CargoScriptTomls {
|
|
||||||
fn extract_toml_part(p: &AbsPath) -> Option<String> {
|
|
||||||
let mut r = String::new();
|
|
||||||
let f = std::fs::File::open(p).ok()?;
|
|
||||||
let f = std::io::BufReader::new(f);
|
|
||||||
let mut started = false;
|
|
||||||
for line in f.lines() {
|
|
||||||
let line = line.ok()?;
|
|
||||||
if started {
|
|
||||||
if line.trim() == "//! ```" {
|
|
||||||
return Some(r);
|
|
||||||
}
|
|
||||||
r += &line;
|
|
||||||
} else {
|
|
||||||
if line.trim() == "//! ```cargo" {
|
|
||||||
started = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
None
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn track_file(&mut self, p: AbsPathBuf) {
|
|
||||||
let toml = CargoScriptTomls::extract_toml_part(&p).unwrap_or_default();
|
|
||||||
self.0.insert(p, toml);
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn need_reload(&mut self, p: &AbsPath) -> bool {
|
|
||||||
let Some(prev) = self.0.get_mut(p) else {
|
|
||||||
return false; // File is not tracked
|
|
||||||
};
|
|
||||||
let next = CargoScriptTomls::extract_toml_part(p).unwrap_or_default();
|
|
||||||
if *prev == next {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
*prev = next;
|
|
||||||
true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl fmt::Debug for ProjectWorkspace {
|
impl fmt::Debug for ProjectWorkspace {
|
||||||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
// Make sure this isn't too verbose.
|
// Make sure this isn't too verbose.
|
||||||
|
@ -213,8 +169,8 @@ impl fmt::Debug for ProjectWorkspace {
|
||||||
.field("n_cfg_overrides", &cfg_overrides.len());
|
.field("n_cfg_overrides", &cfg_overrides.len());
|
||||||
debug_struct.finish()
|
debug_struct.finish()
|
||||||
}
|
}
|
||||||
ProjectWorkspace::DetachedFiles {
|
ProjectWorkspace::DetachedFile {
|
||||||
files,
|
file,
|
||||||
sysroot,
|
sysroot,
|
||||||
rustc_cfg,
|
rustc_cfg,
|
||||||
toolchain,
|
toolchain,
|
||||||
|
@ -223,7 +179,8 @@ impl fmt::Debug for ProjectWorkspace {
|
||||||
cargo_script,
|
cargo_script,
|
||||||
} => f
|
} => f
|
||||||
.debug_struct("DetachedFiles")
|
.debug_struct("DetachedFiles")
|
||||||
.field("n_files", &files.len())
|
.field("file", &file)
|
||||||
|
.field("cargo_script", &cargo_script.is_some())
|
||||||
.field("sysroot", &sysroot.is_ok())
|
.field("sysroot", &sysroot.is_ok())
|
||||||
.field("cargo_script", &cargo_script.is_some())
|
.field("cargo_script", &cargo_script.is_some())
|
||||||
.field("n_rustc_cfg", &rustc_cfg.len())
|
.field("n_rustc_cfg", &rustc_cfg.len())
|
||||||
|
@ -479,71 +436,79 @@ impl ProjectWorkspace {
|
||||||
pub fn load_detached_files(
|
pub fn load_detached_files(
|
||||||
detached_files: Vec<AbsPathBuf>,
|
detached_files: Vec<AbsPathBuf>,
|
||||||
config: &CargoConfig,
|
config: &CargoConfig,
|
||||||
cargo_script_tomls: &mut CargoScriptTomls,
|
) -> Vec<anyhow::Result<ProjectWorkspace>> {
|
||||||
) -> anyhow::Result<ProjectWorkspace> {
|
dbg!(detached_files
|
||||||
let dir = detached_files
|
.into_iter()
|
||||||
.first()
|
.map(|detached_file| {
|
||||||
.and_then(|it| it.parent())
|
let dir = detached_file
|
||||||
.ok_or_else(|| format_err!("No detached files to load"))?;
|
.parent()
|
||||||
let sysroot = match &config.sysroot {
|
.ok_or_else(|| format_err!("detached file has no parent"))?;
|
||||||
Some(RustLibSource::Path(path)) => {
|
let sysroot = match &config.sysroot {
|
||||||
Sysroot::with_sysroot_dir(path.clone(), config.sysroot_query_metadata)
|
Some(RustLibSource::Path(path)) => {
|
||||||
.map_err(|e| Some(format!("Failed to find sysroot at {path}:{e}")))
|
Sysroot::with_sysroot_dir(path.clone(), config.sysroot_query_metadata)
|
||||||
}
|
.map_err(|e| Some(format!("Failed to find sysroot at {path}:{e}")))
|
||||||
Some(RustLibSource::Discover) => Sysroot::discover(
|
}
|
||||||
dir,
|
Some(RustLibSource::Discover) => {
|
||||||
&config.extra_env,
|
Sysroot::discover(dir, &config.extra_env, config.sysroot_query_metadata)
|
||||||
config.sysroot_query_metadata,
|
.map_err(|e| {
|
||||||
)
|
Some(format!(
|
||||||
.map_err(|e| {
|
"Failed to find sysroot for {dir}. Is rust-src installed? {e}"
|
||||||
Some(format!("Failed to find sysroot for {dir}. Is rust-src installed? {e}"))
|
))
|
||||||
}),
|
})
|
||||||
None => Err(None),
|
}
|
||||||
};
|
None => Err(None),
|
||||||
|
};
|
||||||
|
|
||||||
let sysroot_ref = sysroot.as_ref().ok();
|
let sysroot_ref = sysroot.as_ref().ok();
|
||||||
let toolchain =
|
let toolchain = match get_toolchain_version(
|
||||||
match get_toolchain_version(dir, sysroot_ref, Tool::Rustc, &config.extra_env, "rustc ")
|
dir,
|
||||||
{
|
sysroot_ref,
|
||||||
Ok(it) => it,
|
Tool::Rustc,
|
||||||
Err(e) => {
|
&config.extra_env,
|
||||||
tracing::error!("{e}");
|
"rustc ",
|
||||||
None
|
) {
|
||||||
}
|
Ok(it) => it,
|
||||||
};
|
Err(e) => {
|
||||||
|
tracing::error!("{e}");
|
||||||
|
None
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
let rustc_cfg = rustc_cfg::get(None, &config.extra_env, RustcCfgConfig::Rustc(sysroot_ref));
|
let rustc_cfg =
|
||||||
let data_layout = target_data_layout::get(
|
rustc_cfg::get(None, &config.extra_env, RustcCfgConfig::Rustc(sysroot_ref));
|
||||||
RustcDataLayoutConfig::Rustc(sysroot_ref),
|
let data_layout = target_data_layout::get(
|
||||||
None,
|
RustcDataLayoutConfig::Rustc(sysroot_ref),
|
||||||
&config.extra_env,
|
None,
|
||||||
);
|
&config.extra_env,
|
||||||
let cargo_toml = ManifestPath::try_from(detached_files[0].clone()).unwrap();
|
);
|
||||||
let meta = CargoWorkspace::fetch_metadata(
|
|
||||||
&cargo_toml,
|
|
||||||
cargo_toml.parent(),
|
|
||||||
config,
|
|
||||||
sysroot_ref,
|
|
||||||
&|_| (),
|
|
||||||
)
|
|
||||||
.with_context(|| {
|
|
||||||
format!("Failed to read Cargo metadata from Cargo.toml file {cargo_toml}")
|
|
||||||
})?;
|
|
||||||
let cargo = CargoWorkspace::new(meta);
|
|
||||||
|
|
||||||
for file in &detached_files {
|
let cargo_script = ManifestPath::try_from(detached_file.clone())
|
||||||
cargo_script_tomls.track_file(file.clone());
|
.ok()
|
||||||
}
|
.and_then(|file| {
|
||||||
|
CargoWorkspace::fetch_metadata(
|
||||||
|
&file,
|
||||||
|
file.parent(),
|
||||||
|
config,
|
||||||
|
sysroot_ref,
|
||||||
|
&|_| (),
|
||||||
|
)
|
||||||
|
.ok()
|
||||||
|
})
|
||||||
|
.map(CargoWorkspace::new);
|
||||||
|
|
||||||
Ok(ProjectWorkspace::DetachedFiles {
|
Ok(ProjectWorkspace::DetachedFile {
|
||||||
files: detached_files,
|
file: detached_file,
|
||||||
sysroot,
|
sysroot,
|
||||||
rustc_cfg,
|
rustc_cfg,
|
||||||
toolchain,
|
toolchain,
|
||||||
target_layout: data_layout.map(Arc::from).map_err(|it| Arc::from(it.to_string())),
|
target_layout: data_layout
|
||||||
cfg_overrides: config.cfg_overrides.clone(),
|
.map(Arc::from)
|
||||||
cargo_script: Some(cargo),
|
.map_err(|it| Arc::from(it.to_string())),
|
||||||
})
|
cfg_overrides: config.cfg_overrides.clone(),
|
||||||
|
cargo_script,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.collect())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Runs the build scripts for this [`ProjectWorkspace`].
|
/// Runs the build scripts for this [`ProjectWorkspace`].
|
||||||
|
@ -565,7 +530,7 @@ impl ProjectWorkspace {
|
||||||
format!("Failed to run build scripts for {}", cargo.workspace_root())
|
format!("Failed to run build scripts for {}", cargo.workspace_root())
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
ProjectWorkspace::Json { .. } | ProjectWorkspace::DetachedFiles { .. } => {
|
ProjectWorkspace::Json { .. } | ProjectWorkspace::DetachedFile { .. } => {
|
||||||
Ok(WorkspaceBuildScripts::default())
|
Ok(WorkspaceBuildScripts::default())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -623,11 +588,11 @@ impl ProjectWorkspace {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn workspace_definition_path(&self) -> Option<&AbsPath> {
|
pub fn workspace_definition_path(&self) -> &AbsPath {
|
||||||
match self {
|
match self {
|
||||||
ProjectWorkspace::Cargo { cargo, .. } => Some(cargo.workspace_root()),
|
ProjectWorkspace::Cargo { cargo, .. } => cargo.workspace_root(),
|
||||||
ProjectWorkspace::Json { project, .. } => Some(project.path()),
|
ProjectWorkspace::Json { project, .. } => project.path(),
|
||||||
ProjectWorkspace::DetachedFiles { .. } => None,
|
ProjectWorkspace::DetachedFile { file, .. } => file,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -635,10 +600,10 @@ impl ProjectWorkspace {
|
||||||
match self {
|
match self {
|
||||||
ProjectWorkspace::Cargo { sysroot: Ok(sysroot), .. }
|
ProjectWorkspace::Cargo { sysroot: Ok(sysroot), .. }
|
||||||
| ProjectWorkspace::Json { sysroot: Ok(sysroot), .. }
|
| ProjectWorkspace::Json { sysroot: Ok(sysroot), .. }
|
||||||
| ProjectWorkspace::DetachedFiles { sysroot: Ok(sysroot), .. } => {
|
| ProjectWorkspace::DetachedFile { sysroot: Ok(sysroot), .. } => {
|
||||||
sysroot.discover_proc_macro_srv()
|
sysroot.discover_proc_macro_srv()
|
||||||
}
|
}
|
||||||
ProjectWorkspace::DetachedFiles { .. } => {
|
ProjectWorkspace::DetachedFile { .. } => {
|
||||||
Err(anyhow::format_err!("cannot find proc-macro server, no sysroot was found"))
|
Err(anyhow::format_err!("cannot find proc-macro server, no sysroot was found"))
|
||||||
}
|
}
|
||||||
ProjectWorkspace::Cargo { cargo, .. } => Err(anyhow::format_err!(
|
ProjectWorkspace::Cargo { cargo, .. } => Err(anyhow::format_err!(
|
||||||
|
@ -769,15 +734,13 @@ impl ProjectWorkspace {
|
||||||
}))
|
}))
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
ProjectWorkspace::DetachedFiles { files, sysroot, .. } => files
|
ProjectWorkspace::DetachedFile { file, sysroot, .. } => iter::once(PackageRoot {
|
||||||
.iter()
|
is_local: true,
|
||||||
.map(|detached_file| PackageRoot {
|
include: vec![file.clone()],
|
||||||
is_local: true,
|
exclude: Vec::new(),
|
||||||
include: vec![detached_file.clone()],
|
})
|
||||||
exclude: Vec::new(),
|
.chain(mk_sysroot(sysroot.as_ref()))
|
||||||
})
|
.collect(),
|
||||||
.chain(mk_sysroot(sysroot.as_ref()))
|
|
||||||
.collect(),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -793,9 +756,9 @@ impl ProjectWorkspace {
|
||||||
let sysroot_package_len = sysroot.as_ref().map_or(0, |it| it.num_packages());
|
let sysroot_package_len = sysroot.as_ref().map_or(0, |it| it.num_packages());
|
||||||
cargo.packages().len() + sysroot_package_len + rustc_package_len
|
cargo.packages().len() + sysroot_package_len + rustc_package_len
|
||||||
}
|
}
|
||||||
ProjectWorkspace::DetachedFiles { sysroot, files, .. } => {
|
ProjectWorkspace::DetachedFile { sysroot, .. } => {
|
||||||
let sysroot_package_len = sysroot.as_ref().map_or(0, |it| it.num_packages());
|
let sysroot_package_len = sysroot.as_ref().map_or(0, |it| it.num_packages());
|
||||||
sysroot_package_len + files.len()
|
sysroot_package_len + 1
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -848,8 +811,8 @@ impl ProjectWorkspace {
|
||||||
),
|
),
|
||||||
sysroot,
|
sysroot,
|
||||||
),
|
),
|
||||||
ProjectWorkspace::DetachedFiles {
|
ProjectWorkspace::DetachedFile {
|
||||||
files,
|
file,
|
||||||
sysroot,
|
sysroot,
|
||||||
rustc_cfg,
|
rustc_cfg,
|
||||||
toolchain: _,
|
toolchain: _,
|
||||||
|
@ -868,10 +831,10 @@ impl ProjectWorkspace {
|
||||||
&WorkspaceBuildScripts::default(),
|
&WorkspaceBuildScripts::default(),
|
||||||
)
|
)
|
||||||
} else {
|
} else {
|
||||||
detached_files_to_crate_graph(
|
detached_file_to_crate_graph(
|
||||||
rustc_cfg.clone(),
|
rustc_cfg.clone(),
|
||||||
load,
|
load,
|
||||||
files,
|
file,
|
||||||
sysroot.as_ref().ok(),
|
sysroot.as_ref().ok(),
|
||||||
cfg_overrides,
|
cfg_overrides,
|
||||||
)
|
)
|
||||||
|
@ -949,8 +912,8 @@ impl ProjectWorkspace {
|
||||||
&& cfg_overrides == o_cfg_overrides
|
&& cfg_overrides == o_cfg_overrides
|
||||||
}
|
}
|
||||||
(
|
(
|
||||||
Self::DetachedFiles {
|
Self::DetachedFile {
|
||||||
files,
|
file,
|
||||||
sysroot,
|
sysroot,
|
||||||
rustc_cfg,
|
rustc_cfg,
|
||||||
cargo_script,
|
cargo_script,
|
||||||
|
@ -958,8 +921,8 @@ impl ProjectWorkspace {
|
||||||
target_layout,
|
target_layout,
|
||||||
cfg_overrides,
|
cfg_overrides,
|
||||||
},
|
},
|
||||||
Self::DetachedFiles {
|
Self::DetachedFile {
|
||||||
files: o_files,
|
file: o_file,
|
||||||
sysroot: o_sysroot,
|
sysroot: o_sysroot,
|
||||||
rustc_cfg: o_rustc_cfg,
|
rustc_cfg: o_rustc_cfg,
|
||||||
cargo_script: o_cargo_script,
|
cargo_script: o_cargo_script,
|
||||||
|
@ -968,7 +931,7 @@ impl ProjectWorkspace {
|
||||||
cfg_overrides: o_cfg_overrides,
|
cfg_overrides: o_cfg_overrides,
|
||||||
},
|
},
|
||||||
) => {
|
) => {
|
||||||
files == o_files
|
file == o_file
|
||||||
&& sysroot == o_sysroot
|
&& sysroot == o_sysroot
|
||||||
&& rustc_cfg == o_rustc_cfg
|
&& rustc_cfg == o_rustc_cfg
|
||||||
&& toolchain == o_toolchain
|
&& toolchain == o_toolchain
|
||||||
|
@ -1285,10 +1248,10 @@ fn cargo_to_crate_graph(
|
||||||
res
|
res
|
||||||
}
|
}
|
||||||
|
|
||||||
fn detached_files_to_crate_graph(
|
fn detached_file_to_crate_graph(
|
||||||
rustc_cfg: Vec<CfgFlag>,
|
rustc_cfg: Vec<CfgFlag>,
|
||||||
load: FileLoader<'_>,
|
load: FileLoader<'_>,
|
||||||
detached_files: &[AbsPathBuf],
|
detached_file: &AbsPathBuf,
|
||||||
sysroot: Option<&Sysroot>,
|
sysroot: Option<&Sysroot>,
|
||||||
override_cfg: &CfgOverrides,
|
override_cfg: &CfgOverrides,
|
||||||
) -> (CrateGraph, ProcMacroPaths) {
|
) -> (CrateGraph, ProcMacroPaths) {
|
||||||
|
@ -1305,34 +1268,32 @@ fn detached_files_to_crate_graph(
|
||||||
override_cfg.apply(&mut cfg_options, "");
|
override_cfg.apply(&mut cfg_options, "");
|
||||||
let cfg_options = Arc::new(cfg_options);
|
let cfg_options = Arc::new(cfg_options);
|
||||||
|
|
||||||
for detached_file in detached_files {
|
let file_id = match load(detached_file) {
|
||||||
let file_id = match load(detached_file) {
|
Some(file_id) => file_id,
|
||||||
Some(file_id) => file_id,
|
None => {
|
||||||
None => {
|
tracing::error!("Failed to load detached file {:?}", detached_file);
|
||||||
tracing::error!("Failed to load detached file {:?}", detached_file);
|
return (crate_graph, FxHashMap::default());
|
||||||
continue;
|
}
|
||||||
}
|
};
|
||||||
};
|
let display_name = detached_file
|
||||||
let display_name = detached_file
|
.file_stem()
|
||||||
.file_stem()
|
.map(|file_stem| CrateDisplayName::from_canonical_name(file_stem.to_owned()));
|
||||||
.map(|file_stem| CrateDisplayName::from_canonical_name(file_stem.to_owned()));
|
let detached_file_crate = crate_graph.add_crate_root(
|
||||||
let detached_file_crate = crate_graph.add_crate_root(
|
file_id,
|
||||||
file_id,
|
Edition::CURRENT,
|
||||||
Edition::CURRENT,
|
display_name.clone(),
|
||||||
display_name.clone(),
|
None,
|
||||||
None,
|
cfg_options.clone(),
|
||||||
cfg_options.clone(),
|
None,
|
||||||
None,
|
Env::default(),
|
||||||
Env::default(),
|
false,
|
||||||
false,
|
CrateOrigin::Local {
|
||||||
CrateOrigin::Local {
|
repo: None,
|
||||||
repo: None,
|
name: display_name.map(|n| n.canonical_name().to_owned()),
|
||||||
name: display_name.map(|n| n.canonical_name().to_owned()),
|
},
|
||||||
},
|
);
|
||||||
);
|
|
||||||
|
|
||||||
public_deps.add_to_crate_graph(&mut crate_graph, detached_file_crate);
|
public_deps.add_to_crate_graph(&mut crate_graph, detached_file_crate);
|
||||||
}
|
|
||||||
(crate_graph, FxHashMap::default())
|
(crate_graph, FxHashMap::default())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -75,8 +75,8 @@ impl Tester {
|
||||||
&cargo_config.extra_env,
|
&cargo_config.extra_env,
|
||||||
);
|
);
|
||||||
|
|
||||||
let workspace = ProjectWorkspace::DetachedFiles {
|
let workspace = ProjectWorkspace::DetachedFile {
|
||||||
files: vec![tmp_file],
|
file: tmp_file,
|
||||||
sysroot,
|
sysroot,
|
||||||
rustc_cfg: vec![],
|
rustc_cfg: vec![],
|
||||||
toolchain: None,
|
toolchain: None,
|
||||||
|
|
|
@ -18,9 +18,7 @@ use parking_lot::{
|
||||||
RwLockWriteGuard,
|
RwLockWriteGuard,
|
||||||
};
|
};
|
||||||
use proc_macro_api::ProcMacroServer;
|
use proc_macro_api::ProcMacroServer;
|
||||||
use project_model::{
|
use project_model::{CargoWorkspace, ProjectWorkspace, Target, WorkspaceBuildScripts};
|
||||||
CargoScriptTomls, CargoWorkspace, ProjectWorkspace, Target, WorkspaceBuildScripts,
|
|
||||||
};
|
|
||||||
use rustc_hash::{FxHashMap, FxHashSet};
|
use rustc_hash::{FxHashMap, FxHashSet};
|
||||||
use triomphe::Arc;
|
use triomphe::Arc;
|
||||||
use vfs::{AnchoredPathBuf, ChangedFile, Vfs};
|
use vfs::{AnchoredPathBuf, ChangedFile, Vfs};
|
||||||
|
@ -127,6 +125,7 @@ pub(crate) struct GlobalState {
|
||||||
/// to invalidate any salsa caches.
|
/// to invalidate any salsa caches.
|
||||||
pub(crate) workspaces: Arc<Vec<ProjectWorkspace>>,
|
pub(crate) workspaces: Arc<Vec<ProjectWorkspace>>,
|
||||||
pub(crate) crate_graph_file_dependencies: FxHashSet<vfs::VfsPath>,
|
pub(crate) crate_graph_file_dependencies: FxHashSet<vfs::VfsPath>,
|
||||||
|
pub(crate) detached_files: FxHashSet<vfs::AbsPathBuf>,
|
||||||
|
|
||||||
// op queues
|
// op queues
|
||||||
pub(crate) fetch_workspaces_queue:
|
pub(crate) fetch_workspaces_queue:
|
||||||
|
@ -146,7 +145,6 @@ pub(crate) struct GlobalState {
|
||||||
/// this queue should run only *after* [`GlobalState::process_changes`] has
|
/// this queue should run only *after* [`GlobalState::process_changes`] has
|
||||||
/// been called.
|
/// been called.
|
||||||
pub(crate) deferred_task_queue: TaskQueue,
|
pub(crate) deferred_task_queue: TaskQueue,
|
||||||
pub(crate) cargo_script_tomls: Arc<Mutex<CargoScriptTomls>>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// An immutable snapshot of the world's state at a point in time.
|
/// An immutable snapshot of the world's state at a point in time.
|
||||||
|
@ -236,6 +234,7 @@ impl GlobalState {
|
||||||
|
|
||||||
workspaces: Arc::from(Vec::new()),
|
workspaces: Arc::from(Vec::new()),
|
||||||
crate_graph_file_dependencies: FxHashSet::default(),
|
crate_graph_file_dependencies: FxHashSet::default(),
|
||||||
|
detached_files: FxHashSet::default(),
|
||||||
fetch_workspaces_queue: OpQueue::default(),
|
fetch_workspaces_queue: OpQueue::default(),
|
||||||
fetch_build_data_queue: OpQueue::default(),
|
fetch_build_data_queue: OpQueue::default(),
|
||||||
fetch_proc_macros_queue: OpQueue::default(),
|
fetch_proc_macros_queue: OpQueue::default(),
|
||||||
|
@ -243,7 +242,6 @@ impl GlobalState {
|
||||||
prime_caches_queue: OpQueue::default(),
|
prime_caches_queue: OpQueue::default(),
|
||||||
|
|
||||||
deferred_task_queue: task_queue,
|
deferred_task_queue: task_queue,
|
||||||
cargo_script_tomls: Arc::new(Mutex::new(CargoScriptTomls(FxHashMap::default()))),
|
|
||||||
};
|
};
|
||||||
// Apply any required database inputs from the config.
|
// Apply any required database inputs from the config.
|
||||||
this.update_configuration(config);
|
this.update_configuration(config);
|
||||||
|
@ -326,11 +324,7 @@ impl GlobalState {
|
||||||
if file.is_created_or_deleted() {
|
if file.is_created_or_deleted() {
|
||||||
workspace_structure_change.get_or_insert((path, false)).1 |=
|
workspace_structure_change.get_or_insert((path, false)).1 |=
|
||||||
self.crate_graph_file_dependencies.contains(vfs_path);
|
self.crate_graph_file_dependencies.contains(vfs_path);
|
||||||
} else if reload::should_refresh_for_change(
|
} else if reload::should_refresh_for_change(&path, file.kind()) {
|
||||||
&path,
|
|
||||||
file.kind(),
|
|
||||||
&mut self.cargo_script_tomls.lock(),
|
|
||||||
) {
|
|
||||||
workspace_structure_change.get_or_insert((path.clone(), false));
|
workspace_structure_change.get_or_insert((path.clone(), false));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -527,7 +521,7 @@ impl GlobalStateSnapshot {
|
||||||
cargo.target_by_root(path).map(|it| (cargo, it))
|
cargo.target_by_root(path).map(|it| (cargo, it))
|
||||||
}
|
}
|
||||||
ProjectWorkspace::Json { .. } => None,
|
ProjectWorkspace::Json { .. } => None,
|
||||||
ProjectWorkspace::DetachedFiles { .. } => None,
|
ProjectWorkspace::DetachedFile { .. } => None,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -150,14 +150,14 @@ pub(crate) fn handle_did_save_text_document(
|
||||||
if let Ok(vfs_path) = from_proto::vfs_path(¶ms.text_document.uri) {
|
if let Ok(vfs_path) = from_proto::vfs_path(¶ms.text_document.uri) {
|
||||||
// Re-fetch workspaces if a workspace related file has changed
|
// Re-fetch workspaces if a workspace related file has changed
|
||||||
if let Some(abs_path) = vfs_path.as_path() {
|
if let Some(abs_path) = vfs_path.as_path() {
|
||||||
if reload::should_refresh_for_change(
|
if reload::should_refresh_for_change(abs_path, ChangeKind::Modify) {
|
||||||
abs_path,
|
|
||||||
ChangeKind::Modify,
|
|
||||||
&mut state.cargo_script_tomls.lock(),
|
|
||||||
) {
|
|
||||||
state
|
state
|
||||||
.fetch_workspaces_queue
|
.fetch_workspaces_queue
|
||||||
.request_op(format!("workspace vfs file change saved {abs_path}"), false);
|
.request_op(format!("workspace vfs file change saved {abs_path}"), false);
|
||||||
|
} else if state.detached_files.contains(abs_path) {
|
||||||
|
state
|
||||||
|
.fetch_workspaces_queue
|
||||||
|
.request_op(format!("detached file saved {abs_path}"), false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -307,7 +307,7 @@ fn run_flycheck(state: &mut GlobalState, vfs_path: VfsPath) -> bool {
|
||||||
}
|
}
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
project_model::ProjectWorkspace::DetachedFiles { .. } => return None,
|
project_model::ProjectWorkspace::DetachedFile { .. } => return None,
|
||||||
};
|
};
|
||||||
Some((idx, package))
|
Some((idx, package))
|
||||||
});
|
});
|
||||||
|
|
|
@ -101,7 +101,7 @@ pub(crate) fn handle_analyzer_status(
|
||||||
"Workspace root folders: {:?}",
|
"Workspace root folders: {:?}",
|
||||||
snap.workspaces
|
snap.workspaces
|
||||||
.iter()
|
.iter()
|
||||||
.flat_map(|ws| ws.workspace_definition_path())
|
.map(|ws| ws.workspace_definition_path())
|
||||||
.collect::<Vec<&AbsPath>>()
|
.collect::<Vec<&AbsPath>>()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1761,7 +1761,7 @@ pub(crate) fn handle_open_docs(
|
||||||
let ws_and_sysroot = snap.workspaces.iter().find_map(|ws| match ws {
|
let ws_and_sysroot = snap.workspaces.iter().find_map(|ws| match ws {
|
||||||
ProjectWorkspace::Cargo { cargo, sysroot, .. } => Some((cargo, sysroot.as_ref().ok())),
|
ProjectWorkspace::Cargo { cargo, sysroot, .. } => Some((cargo, sysroot.as_ref().ok())),
|
||||||
ProjectWorkspace::Json { .. } => None,
|
ProjectWorkspace::Json { .. } => None,
|
||||||
ProjectWorkspace::DetachedFiles { .. } => None,
|
ProjectWorkspace::DetachedFile { .. } => None,
|
||||||
});
|
});
|
||||||
|
|
||||||
let (cargo, sysroot) = match ws_and_sysroot {
|
let (cargo, sysroot) = match ws_and_sysroot {
|
||||||
|
|
|
@ -25,7 +25,7 @@ use ide_db::{
|
||||||
use itertools::Itertools;
|
use itertools::Itertools;
|
||||||
use load_cargo::{load_proc_macro, ProjectFolders};
|
use load_cargo::{load_proc_macro, ProjectFolders};
|
||||||
use proc_macro_api::ProcMacroServer;
|
use proc_macro_api::ProcMacroServer;
|
||||||
use project_model::{CargoScriptTomls, ProjectWorkspace, WorkspaceBuildScripts};
|
use project_model::{ProjectWorkspace, WorkspaceBuildScripts};
|
||||||
use stdx::{format_to, thread::ThreadIntent};
|
use stdx::{format_to, thread::ThreadIntent};
|
||||||
use triomphe::Arc;
|
use triomphe::Arc;
|
||||||
use vfs::{AbsPath, AbsPathBuf, ChangeKind};
|
use vfs::{AbsPath, AbsPathBuf, ChangeKind};
|
||||||
|
@ -153,7 +153,7 @@ impl GlobalState {
|
||||||
for ws in self.workspaces.iter() {
|
for ws in self.workspaces.iter() {
|
||||||
let (ProjectWorkspace::Cargo { sysroot, .. }
|
let (ProjectWorkspace::Cargo { sysroot, .. }
|
||||||
| ProjectWorkspace::Json { sysroot, .. }
|
| ProjectWorkspace::Json { sysroot, .. }
|
||||||
| ProjectWorkspace::DetachedFiles { sysroot, .. }) = ws;
|
| ProjectWorkspace::DetachedFile { sysroot, .. }) = ws;
|
||||||
match sysroot {
|
match sysroot {
|
||||||
Err(None) => (),
|
Err(None) => (),
|
||||||
Err(Some(e)) => {
|
Err(Some(e)) => {
|
||||||
|
@ -206,7 +206,6 @@ impl GlobalState {
|
||||||
let linked_projects = self.config.linked_or_discovered_projects();
|
let linked_projects = self.config.linked_or_discovered_projects();
|
||||||
let detached_files = self.config.detached_files().to_vec();
|
let detached_files = self.config.detached_files().to_vec();
|
||||||
let cargo_config = self.config.cargo();
|
let cargo_config = self.config.cargo();
|
||||||
let cargo_script_tomls = self.cargo_script_tomls.clone();
|
|
||||||
|
|
||||||
move |sender| {
|
move |sender| {
|
||||||
let progress = {
|
let progress = {
|
||||||
|
@ -256,10 +255,9 @@ impl GlobalState {
|
||||||
}
|
}
|
||||||
|
|
||||||
if !detached_files.is_empty() {
|
if !detached_files.is_empty() {
|
||||||
workspaces.push(project_model::ProjectWorkspace::load_detached_files(
|
workspaces.extend(project_model::ProjectWorkspace::load_detached_files(
|
||||||
detached_files,
|
detached_files,
|
||||||
&cargo_config,
|
&cargo_config,
|
||||||
&mut cargo_script_tomls.lock(),
|
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -542,9 +540,6 @@ impl GlobalState {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn recreate_crate_graph(&mut self, cause: String) {
|
fn recreate_crate_graph(&mut self, cause: String) {
|
||||||
// crate graph construction relies on these paths, record them so when one of them gets
|
|
||||||
// deleted or created we trigger a reconstruction of the crate graph
|
|
||||||
let mut crate_graph_file_dependencies = mem::take(&mut self.crate_graph_file_dependencies);
|
|
||||||
self.report_progress(
|
self.report_progress(
|
||||||
"Building CrateGraph",
|
"Building CrateGraph",
|
||||||
crate::lsp::utils::Progress::Begin,
|
crate::lsp::utils::Progress::Begin,
|
||||||
|
@ -553,13 +548,25 @@ impl GlobalState {
|
||||||
None,
|
None,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// crate graph construction relies on these paths, record them so when one of them gets
|
||||||
|
// deleted or created we trigger a reconstruction of the crate graph
|
||||||
|
self.crate_graph_file_dependencies.clear();
|
||||||
|
self.detached_files = self
|
||||||
|
.workspaces
|
||||||
|
.iter()
|
||||||
|
.filter_map(|ws| match ws {
|
||||||
|
ProjectWorkspace::DetachedFile { file, .. } => Some(file.clone()),
|
||||||
|
_ => None,
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
let (crate_graph, proc_macro_paths, layouts, toolchains) = {
|
let (crate_graph, proc_macro_paths, layouts, toolchains) = {
|
||||||
// Create crate graph from all the workspaces
|
// Create crate graph from all the workspaces
|
||||||
let vfs = &mut self.vfs.write().0;
|
let vfs = &mut self.vfs.write().0;
|
||||||
|
|
||||||
let load = |path: &AbsPath| {
|
let load = |path: &AbsPath| {
|
||||||
let vfs_path = vfs::VfsPath::from(path.to_path_buf());
|
let vfs_path = vfs::VfsPath::from(path.to_path_buf());
|
||||||
crate_graph_file_dependencies.insert(vfs_path.clone());
|
self.crate_graph_file_dependencies.insert(vfs_path.clone());
|
||||||
vfs.file_id(&vfs_path)
|
vfs.file_id(&vfs_path)
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -579,7 +586,6 @@ impl GlobalState {
|
||||||
change.set_target_data_layouts(layouts);
|
change.set_target_data_layouts(layouts);
|
||||||
change.set_toolchains(toolchains);
|
change.set_toolchains(toolchains);
|
||||||
self.analysis_host.apply_change(change);
|
self.analysis_host.apply_change(change);
|
||||||
self.crate_graph_file_dependencies = crate_graph_file_dependencies;
|
|
||||||
self.report_progress(
|
self.report_progress(
|
||||||
"Building CrateGraph",
|
"Building CrateGraph",
|
||||||
crate::lsp::utils::Progress::End,
|
crate::lsp::utils::Progress::End,
|
||||||
|
@ -676,7 +682,7 @@ impl GlobalState {
|
||||||
_ => None,
|
_ => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ProjectWorkspace::DetachedFiles { .. } => None,
|
ProjectWorkspace::DetachedFile { .. } => None,
|
||||||
})
|
})
|
||||||
.map(|(id, root, sysroot_root)| {
|
.map(|(id, root, sysroot_root)| {
|
||||||
let sender = sender.clone();
|
let sender = sender.clone();
|
||||||
|
@ -712,18 +718,14 @@ pub fn ws_to_crate_graph(
|
||||||
let mut toolchains = Vec::default();
|
let mut toolchains = Vec::default();
|
||||||
let e = Err(Arc::from("missing layout"));
|
let e = Err(Arc::from("missing layout"));
|
||||||
for ws in workspaces {
|
for ws in workspaces {
|
||||||
|
dbg!(ws);
|
||||||
let (other, mut crate_proc_macros) = ws.to_crate_graph(&mut load, extra_env);
|
let (other, mut crate_proc_macros) = ws.to_crate_graph(&mut load, extra_env);
|
||||||
|
dbg!(&other);
|
||||||
let num_layouts = layouts.len();
|
let num_layouts = layouts.len();
|
||||||
let num_toolchains = toolchains.len();
|
let num_toolchains = toolchains.len();
|
||||||
let (toolchain, layout) = match ws {
|
let (ProjectWorkspace::Cargo { toolchain, target_layout, .. }
|
||||||
ProjectWorkspace::Cargo { toolchain, target_layout, .. }
|
| ProjectWorkspace::Json { toolchain, target_layout, .. }
|
||||||
| ProjectWorkspace::Json { toolchain, target_layout, .. } => {
|
| ProjectWorkspace::DetachedFile { toolchain, target_layout, .. }) = ws;
|
||||||
(toolchain.clone(), target_layout.clone())
|
|
||||||
}
|
|
||||||
ProjectWorkspace::DetachedFiles { .. } => {
|
|
||||||
(None, Err("detached files have no layout".into()))
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let mapping = crate_graph.extend(
|
let mapping = crate_graph.extend(
|
||||||
other,
|
other,
|
||||||
|
@ -732,7 +734,7 @@ pub fn ws_to_crate_graph(
|
||||||
// if the newly created crate graph's layout is equal to the crate of the merged graph, then
|
// if the newly created crate graph's layout is equal to the crate of the merged graph, then
|
||||||
// we can merge the crates.
|
// we can merge the crates.
|
||||||
let id = cg_id.into_raw().into_u32() as usize;
|
let id = cg_id.into_raw().into_u32() as usize;
|
||||||
layouts[id] == layout && toolchains[id] == toolchain && cg_data == o_data
|
layouts[id] == *target_layout && toolchains[id] == *toolchain && cg_data == o_data
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
// Populate the side tables for the newly merged crates
|
// Populate the side tables for the newly merged crates
|
||||||
|
@ -744,13 +746,13 @@ pub fn ws_to_crate_graph(
|
||||||
if layouts.len() <= idx {
|
if layouts.len() <= idx {
|
||||||
layouts.resize(idx + 1, e.clone());
|
layouts.resize(idx + 1, e.clone());
|
||||||
}
|
}
|
||||||
layouts[idx].clone_from(&layout);
|
layouts[idx].clone_from(target_layout);
|
||||||
}
|
}
|
||||||
if idx >= num_toolchains {
|
if idx >= num_toolchains {
|
||||||
if toolchains.len() <= idx {
|
if toolchains.len() <= idx {
|
||||||
toolchains.resize(idx + 1, None);
|
toolchains.resize(idx + 1, None);
|
||||||
}
|
}
|
||||||
toolchains[idx].clone_from(&toolchain);
|
toolchains[idx].clone_from(toolchain);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
proc_macro_paths.push(crate_proc_macros);
|
proc_macro_paths.push(crate_proc_macros);
|
||||||
|
@ -760,15 +762,7 @@ pub fn ws_to_crate_graph(
|
||||||
(crate_graph, proc_macro_paths, layouts, toolchains)
|
(crate_graph, proc_macro_paths, layouts, toolchains)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn should_refresh_for_change(
|
pub(crate) fn should_refresh_for_change(path: &AbsPath, change_kind: ChangeKind) -> bool {
|
||||||
path: &AbsPath,
|
|
||||||
change_kind: ChangeKind,
|
|
||||||
cargo_script_tomls: &mut CargoScriptTomls,
|
|
||||||
) -> bool {
|
|
||||||
if cargo_script_tomls.need_reload(path) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
const IMPLICIT_TARGET_FILES: &[&str] = &["build.rs", "src/main.rs", "src/lib.rs"];
|
const IMPLICIT_TARGET_FILES: &[&str] = &["build.rs", "src/main.rs", "src/lib.rs"];
|
||||||
const IMPLICIT_TARGET_DIRS: &[&str] = &["src/bin", "examples", "tests", "benches"];
|
const IMPLICIT_TARGET_DIRS: &[&str] = &["src/bin", "examples", "tests", "benches"];
|
||||||
|
|
||||||
|
|
|
@ -139,16 +139,16 @@ version = "0.1.0"
|
||||||
pub struct SpecialHashMap2;
|
pub struct SpecialHashMap2;
|
||||||
//- /src/lib.rs
|
//- /src/lib.rs
|
||||||
#!/usr/bin/env -S cargo +nightly -Zscript
|
#!/usr/bin/env -S cargo +nightly -Zscript
|
||||||
//! ```cargo
|
//! ---cargo
|
||||||
//! [dependencies]
|
//! [dependencies]
|
||||||
//! dependency = { path = "../dependency" }
|
//! dependency = { path = "../dependency" }
|
||||||
//! ```
|
//! ---
|
||||||
use dependency::Spam;
|
use dependency::Spam;
|
||||||
use dependency2::Spam;
|
use dependency2::Spam;
|
||||||
"#,
|
"#,
|
||||||
)
|
)
|
||||||
.with_config(serde_json::json!({
|
.with_config(serde_json::json!({
|
||||||
"cargo": { "sysroot": "discover" },
|
"cargo": { "sysroot": null },
|
||||||
}))
|
}))
|
||||||
.server()
|
.server()
|
||||||
.wait_until_workspace_is_loaded();
|
.wait_until_workspace_is_loaded();
|
||||||
|
@ -156,18 +156,18 @@ use dependency2::Spam;
|
||||||
let res = server.send_request::<Completion>(CompletionParams {
|
let res = server.send_request::<Completion>(CompletionParams {
|
||||||
text_document_position: TextDocumentPositionParams::new(
|
text_document_position: TextDocumentPositionParams::new(
|
||||||
server.doc_id("src/lib.rs"),
|
server.doc_id("src/lib.rs"),
|
||||||
Position::new(7, 18),
|
Position::new(5, 18),
|
||||||
),
|
),
|
||||||
context: None,
|
context: None,
|
||||||
partial_result_params: PartialResultParams::default(),
|
partial_result_params: PartialResultParams::default(),
|
||||||
work_done_progress_params: WorkDoneProgressParams::default(),
|
work_done_progress_params: WorkDoneProgressParams::default(),
|
||||||
});
|
});
|
||||||
assert!(res.to_string().contains("SpecialHashMap"));
|
assert!(res.to_string().contains("SpecialHashMap"), "{}", res.to_string());
|
||||||
|
|
||||||
let res = server.send_request::<Completion>(CompletionParams {
|
let res = server.send_request::<Completion>(CompletionParams {
|
||||||
text_document_position: TextDocumentPositionParams::new(
|
text_document_position: TextDocumentPositionParams::new(
|
||||||
server.doc_id("src/lib.rs"),
|
server.doc_id("src/lib.rs"),
|
||||||
Position::new(8, 18),
|
Position::new(6, 18),
|
||||||
),
|
),
|
||||||
context: None,
|
context: None,
|
||||||
partial_result_params: PartialResultParams::default(),
|
partial_result_params: PartialResultParams::default(),
|
||||||
|
@ -178,10 +178,10 @@ use dependency2::Spam;
|
||||||
server.write_file_and_save(
|
server.write_file_and_save(
|
||||||
"src/lib.rs",
|
"src/lib.rs",
|
||||||
r#"#!/usr/bin/env -S cargo +nightly -Zscript
|
r#"#!/usr/bin/env -S cargo +nightly -Zscript
|
||||||
//! ```cargo
|
//! ---cargo
|
||||||
//! [dependencies]
|
//! [dependencies]
|
||||||
//! dependency2 = { path = "../dependency2" }
|
//! dependency2 = { path = "../dependency2" }
|
||||||
//! ```
|
//! ---
|
||||||
use dependency::Spam;
|
use dependency::Spam;
|
||||||
use dependency2::Spam;
|
use dependency2::Spam;
|
||||||
"#
|
"#
|
||||||
|
@ -195,7 +195,7 @@ use dependency2::Spam;
|
||||||
let res = server.send_request::<Completion>(CompletionParams {
|
let res = server.send_request::<Completion>(CompletionParams {
|
||||||
text_document_position: TextDocumentPositionParams::new(
|
text_document_position: TextDocumentPositionParams::new(
|
||||||
server.doc_id("src/lib.rs"),
|
server.doc_id("src/lib.rs"),
|
||||||
Position::new(7, 18),
|
Position::new(5, 18),
|
||||||
),
|
),
|
||||||
context: None,
|
context: None,
|
||||||
partial_result_params: PartialResultParams::default(),
|
partial_result_params: PartialResultParams::default(),
|
||||||
|
@ -206,7 +206,7 @@ use dependency2::Spam;
|
||||||
let res = server.send_request::<Completion>(CompletionParams {
|
let res = server.send_request::<Completion>(CompletionParams {
|
||||||
text_document_position: TextDocumentPositionParams::new(
|
text_document_position: TextDocumentPositionParams::new(
|
||||||
server.doc_id("src/lib.rs"),
|
server.doc_id("src/lib.rs"),
|
||||||
Position::new(8, 18),
|
Position::new(6, 18),
|
||||||
),
|
),
|
||||||
context: None,
|
context: None,
|
||||||
partial_result_params: PartialResultParams::default(),
|
partial_result_params: PartialResultParams::default(),
|
||||||
|
|
|
@ -287,6 +287,7 @@ impl Server {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[track_caller]
|
||||||
pub(crate) fn send_request<R>(&self, params: R::Params) -> Value
|
pub(crate) fn send_request<R>(&self, params: R::Params) -> Value
|
||||||
where
|
where
|
||||||
R: lsp_types::request::Request,
|
R: lsp_types::request::Request,
|
||||||
|
@ -298,6 +299,7 @@ impl Server {
|
||||||
let r = Request::new(id.into(), R::METHOD.to_owned(), params);
|
let r = Request::new(id.into(), R::METHOD.to_owned(), params);
|
||||||
self.send_request_(r)
|
self.send_request_(r)
|
||||||
}
|
}
|
||||||
|
#[track_caller]
|
||||||
fn send_request_(&self, r: Request) -> Value {
|
fn send_request_(&self, r: Request) -> Value {
|
||||||
let id = r.id.clone();
|
let id = r.id.clone();
|
||||||
self.client.sender.send(r.clone().into()).unwrap();
|
self.client.sender.send(r.clone().into()).unwrap();
|
||||||
|
|
Loading…
Reference in a new issue