Auto merge of #17915 - Veykril:offline-no-deps, r=Veykril

feat: Make rust-analyzer work partially when offline

Helps out with https://github.com/rust-lang/rust-analyzer/issues/12499 a bit
This commit is contained in:
bors 2024-08-17 17:20:39 +00:00
commit fa00326247
7 changed files with 85 additions and 24 deletions

View file

@ -33,6 +33,8 @@ pub struct CargoWorkspace {
workspace_root: AbsPathBuf, workspace_root: AbsPathBuf,
target_directory: AbsPathBuf, target_directory: AbsPathBuf,
manifest_path: ManifestPath, manifest_path: ManifestPath,
// Whether this workspace was queried with `--no-deps`.
no_deps: bool,
} }
impl ops::Index<Package> for CargoWorkspace { impl ops::Index<Package> for CargoWorkspace {
@ -259,6 +261,18 @@ impl CargoWorkspace {
sysroot: &Sysroot, sysroot: &Sysroot,
locked: bool, locked: bool,
progress: &dyn Fn(String), progress: &dyn Fn(String),
) -> anyhow::Result<cargo_metadata::Metadata> {
Self::fetch_metadata_(cargo_toml, current_dir, config, sysroot, locked, false, progress)
}
fn fetch_metadata_(
cargo_toml: &ManifestPath,
current_dir: &AbsPath,
config: &CargoConfig,
sysroot: &Sysroot,
locked: bool,
no_deps: bool,
progress: &dyn Fn(String),
) -> anyhow::Result<cargo_metadata::Metadata> { ) -> anyhow::Result<cargo_metadata::Metadata> {
let targets = find_list_of_build_targets(config, cargo_toml, sysroot); let targets = find_list_of_build_targets(config, cargo_toml, sysroot);
@ -314,6 +328,9 @@ impl CargoWorkspace {
if locked { if locked {
other_options.push("--locked".to_owned()); other_options.push("--locked".to_owned());
} }
if no_deps {
other_options.push("--no-deps".to_owned());
}
meta.other_options(other_options); meta.other_options(other_options);
// FIXME: Fetching metadata is a slow process, as it might require // FIXME: Fetching metadata is a slow process, as it might require
@ -324,6 +341,22 @@ impl CargoWorkspace {
(|| -> Result<cargo_metadata::Metadata, cargo_metadata::Error> { (|| -> Result<cargo_metadata::Metadata, cargo_metadata::Error> {
let output = meta.cargo_command().output()?; let output = meta.cargo_command().output()?;
if !output.status.success() { if !output.status.success() {
if !no_deps {
// If we failed to fetch metadata with deps, try again without them.
// This makes r-a still work partially when offline.
if let Ok(metadata) = Self::fetch_metadata_(
cargo_toml,
current_dir,
config,
sysroot,
locked,
true,
progress,
) {
return Ok(metadata);
}
}
return Err(cargo_metadata::Error::CargoMetadata { return Err(cargo_metadata::Error::CargoMetadata {
stderr: String::from_utf8(output.stderr)?, stderr: String::from_utf8(output.stderr)?,
}); });
@ -431,8 +464,8 @@ impl CargoWorkspace {
pkg_data.targets.push(tgt); pkg_data.targets.push(tgt);
} }
} }
let resolve = meta.resolve.expect("metadata executed with deps"); let no_deps = meta.resolve.is_none();
for mut node in resolve.nodes { for mut node in meta.resolve.map_or_else(Vec::new, |it| it.nodes) {
let &source = pkg_by_id.get(&node.id).unwrap(); let &source = pkg_by_id.get(&node.id).unwrap();
node.deps.sort_by(|a, b| a.pkg.cmp(&b.pkg)); node.deps.sort_by(|a, b| a.pkg.cmp(&b.pkg));
let dependencies = node let dependencies = node
@ -451,7 +484,14 @@ impl CargoWorkspace {
let target_directory = AbsPathBuf::assert(meta.target_directory); let target_directory = AbsPathBuf::assert(meta.target_directory);
CargoWorkspace { packages, targets, workspace_root, target_directory, manifest_path } CargoWorkspace {
packages,
targets,
workspace_root,
target_directory,
manifest_path,
no_deps,
}
} }
pub fn packages(&self) -> impl ExactSizeIterator<Item = Package> + '_ { pub fn packages(&self) -> impl ExactSizeIterator<Item = Package> + '_ {
@ -533,6 +573,10 @@ impl CargoWorkspace {
fn is_unique(&self, name: &str) -> bool { fn is_unique(&self, name: &str) -> bool {
self.packages.iter().filter(|(_, v)| v.name == name).count() == 1 self.packages.iter().filter(|(_, v)| v.name == name).count() == 1
} }
pub fn no_deps(&self) -> bool {
self.no_deps
}
} }
fn find_list_of_build_targets( fn find_list_of_build_targets(

View file

@ -372,18 +372,19 @@ impl Sysroot {
.flatten() .flatten()
}; };
let resolve = res.resolve.as_mut().expect("metadata executed with deps"); if let Some(resolve) = res.resolve.as_mut() {
resolve.nodes.retain_mut(|node| { resolve.nodes.retain_mut(|node| {
// Replace `rustc-std-workspace` crate with the actual one in the dependency list // Replace `rustc-std-workspace` crate with the actual one in the dependency list
node.deps.iter_mut().for_each(|dep| { node.deps.iter_mut().for_each(|dep| {
let real_pkg = patches.clone().find(|((_, fake_id), _)| *fake_id == dep.pkg); let real_pkg = patches.clone().find(|((_, fake_id), _)| *fake_id == dep.pkg);
if let Some((_, real)) = real_pkg { if let Some((_, real)) = real_pkg {
dep.pkg = real; dep.pkg = real;
} }
});
// Remove this node if it's a fake one
!patches.clone().any(|((_, fake), _)| fake == node.id)
}); });
// Remove this node if it's a fake one }
!patches.clone().any(|((_, fake), _)| fake == node.id)
});
// Remove the fake ones from the package list // Remove the fake ones from the package list
patches.map(|((idx, _), _)| idx).sorted().rev().for_each(|idx| { patches.map(|((idx, _), _)| idx).sorted().rev().for_each(|idx| {
res.packages.remove(idx); res.packages.remove(idx);

View file

@ -441,14 +441,15 @@ impl ProjectWorkspace {
) -> anyhow::Result<WorkspaceBuildScripts> { ) -> anyhow::Result<WorkspaceBuildScripts> {
match &self.kind { match &self.kind {
ProjectWorkspaceKind::DetachedFile { cargo: Some((cargo, _)), .. } ProjectWorkspaceKind::DetachedFile { cargo: Some((cargo, _)), .. }
| ProjectWorkspaceKind::Cargo { cargo, .. } => { | ProjectWorkspaceKind::Cargo { cargo, .. }
if !cargo.no_deps() =>
{
WorkspaceBuildScripts::run_for_workspace(config, cargo, progress, &self.sysroot) WorkspaceBuildScripts::run_for_workspace(config, cargo, progress, &self.sysroot)
.with_context(|| { .with_context(|| {
format!("Failed to run build scripts for {}", cargo.workspace_root()) format!("Failed to run build scripts for {}", cargo.workspace_root())
}) })
} }
ProjectWorkspaceKind::DetachedFile { cargo: None, .. } _ => Ok(WorkspaceBuildScripts::default()),
| ProjectWorkspaceKind::Json { .. } => Ok(WorkspaceBuildScripts::default()),
} }
} }

View file

@ -75,7 +75,7 @@ impl DiagnosticCollection {
flycheck_id: usize, flycheck_id: usize,
file_id: FileId, file_id: FileId,
diagnostic: lsp_types::Diagnostic, diagnostic: lsp_types::Diagnostic,
fix: Option<Fix>, fix: Option<Box<Fix>>,
) { ) {
let diagnostics = self.check.entry(flycheck_id).or_default().entry(file_id).or_default(); let diagnostics = self.check.entry(flycheck_id).or_default().entry(file_id).or_default();
for existing_diagnostic in diagnostics.iter() { for existing_diagnostic in diagnostics.iter() {
@ -84,8 +84,10 @@ impl DiagnosticCollection {
} }
} }
let check_fixes = Arc::make_mut(&mut self.check_fixes); if let Some(fix) = fix {
check_fixes.entry(flycheck_id).or_default().entry(file_id).or_default().extend(fix); let check_fixes = Arc::make_mut(&mut self.check_fixes);
check_fixes.entry(flycheck_id).or_default().entry(file_id).or_default().push(*fix);
}
diagnostics.push(diagnostic); diagnostics.push(diagnostic);
self.changes.insert(file_id); self.changes.insert(file_id);
} }

View file

@ -170,7 +170,7 @@ fn resolve_path(
struct SubDiagnostic { struct SubDiagnostic {
related: lsp_types::DiagnosticRelatedInformation, related: lsp_types::DiagnosticRelatedInformation,
suggested_fix: Option<Fix>, suggested_fix: Option<Box<Fix>>,
} }
enum MappedRustChildDiagnostic { enum MappedRustChildDiagnostic {
@ -241,7 +241,7 @@ fn map_rust_child_diagnostic(
location: location(config, workspace_root, spans[0], snap), location: location(config, workspace_root, spans[0], snap),
message: message.clone(), message: message.clone(),
}, },
suggested_fix: Some(Fix { suggested_fix: Some(Box::new(Fix {
ranges: spans ranges: spans
.iter() .iter()
.map(|&span| location(config, workspace_root, span, snap).range) .map(|&span| location(config, workspace_root, span, snap).range)
@ -260,7 +260,7 @@ fn map_rust_child_diagnostic(
data: None, data: None,
command: None, command: None,
}, },
}), })),
}) })
} }
} }
@ -269,7 +269,7 @@ fn map_rust_child_diagnostic(
pub(crate) struct MappedRustDiagnostic { pub(crate) struct MappedRustDiagnostic {
pub(crate) url: lsp_types::Url, pub(crate) url: lsp_types::Url,
pub(crate) diagnostic: lsp_types::Diagnostic, pub(crate) diagnostic: lsp_types::Diagnostic,
pub(crate) fix: Option<Fix>, pub(crate) fix: Option<Box<Fix>>,
} }
/// Converts a Rust root diagnostic to LSP form /// Converts a Rust root diagnostic to LSP form

View file

@ -218,6 +218,7 @@ struct FlycheckActor {
status: FlycheckStatus, status: FlycheckStatus,
} }
#[allow(clippy::large_enum_variant)]
enum Event { enum Event {
RequestStateChange(StateChange), RequestStateChange(StateChange),
CheckEvent(Option<CargoCheckMessage>), CheckEvent(Option<CargoCheckMessage>),

View file

@ -165,6 +165,18 @@ impl GlobalState {
self.proc_macro_clients.iter().map(Some).chain(iter::repeat_with(|| None)); self.proc_macro_clients.iter().map(Some).chain(iter::repeat_with(|| None));
for (ws, proc_macro_client) in self.workspaces.iter().zip(proc_macro_clients) { for (ws, proc_macro_client) in self.workspaces.iter().zip(proc_macro_clients) {
if matches!(
&ws.kind,
ProjectWorkspaceKind::Cargo { cargo, .. } | ProjectWorkspaceKind::DetachedFile { cargo: Some((cargo, _)), .. }
if cargo.no_deps()
) {
status.health |= lsp_ext::Health::Warning;
format_to!(
message,
"Workspace `{}` has been queried without dependencies, connecting to crates.io might have failed.\n\n",
ws.manifest_or_root()
);
}
if let Some(err) = ws.sysroot.error() { if let Some(err) = ws.sysroot.error() {
status.health |= lsp_ext::Health::Warning; status.health |= lsp_ext::Health::Warning;
format_to!( format_to!(