|  | use dot::{Id, LabelText}; | 
|  | use ide_db::base_db::salsa::plumbing::AsId; | 
|  | use ide_db::{ | 
|  | FxHashMap, RootDatabase, | 
|  | base_db::{ | 
|  | BuiltCrateData, BuiltDependency, Crate, ExtraCrateData, RootQueryDb, SourceDatabase, | 
|  | }, | 
|  | }; | 
|  |  | 
|  | // Feature: View Crate Graph | 
|  | // | 
|  | // Renders the currently loaded crate graph as an SVG graphic. Requires the `dot` tool, which | 
|  | // is part of graphviz, to be installed. | 
|  | // | 
|  | // Only workspace crates are included, no crates.io dependencies or sysroot crates. | 
|  | // | 
|  | // | Editor  | Action Name | | 
|  | // |---------|-------------| | 
|  | // | VS Code | **rust-analyzer: View Crate Graph** | | 
|  | pub(crate) fn view_crate_graph(db: &RootDatabase, full: bool) -> Result<String, String> { | 
|  | let all_crates = db.all_crates(); | 
|  | let crates_to_render = all_crates | 
|  | .iter() | 
|  | .copied() | 
|  | .map(|krate| (krate, (krate.data(db), krate.extra_data(db)))) | 
|  | .filter(|(_, (crate_data, _))| { | 
|  | if full { | 
|  | true | 
|  | } else { | 
|  | // Only render workspace crates | 
|  | let root_id = db.file_source_root(crate_data.root_file_id).source_root_id(db); | 
|  | !db.source_root(root_id).source_root(db).is_library | 
|  | } | 
|  | }) | 
|  | .collect(); | 
|  | let graph = DotCrateGraph { crates_to_render }; | 
|  |  | 
|  | let mut dot = Vec::new(); | 
|  | dot::render(&graph, &mut dot).unwrap(); | 
|  | Ok(String::from_utf8(dot).unwrap()) | 
|  | } | 
|  |  | 
|  | struct DotCrateGraph<'db> { | 
|  | crates_to_render: FxHashMap<Crate, (&'db BuiltCrateData, &'db ExtraCrateData)>, | 
|  | } | 
|  |  | 
|  | type Edge<'a> = (Crate, &'a BuiltDependency); | 
|  |  | 
|  | impl<'a> dot::GraphWalk<'a, Crate, Edge<'a>> for DotCrateGraph<'_> { | 
|  | fn nodes(&'a self) -> dot::Nodes<'a, Crate> { | 
|  | self.crates_to_render.keys().copied().collect() | 
|  | } | 
|  |  | 
|  | fn edges(&'a self) -> dot::Edges<'a, Edge<'a>> { | 
|  | self.crates_to_render | 
|  | .iter() | 
|  | .flat_map(|(krate, (crate_data, _))| { | 
|  | crate_data | 
|  | .dependencies | 
|  | .iter() | 
|  | .filter(|dep| self.crates_to_render.contains_key(&dep.crate_id)) | 
|  | .map(move |dep| (*krate, dep)) | 
|  | }) | 
|  | .collect() | 
|  | } | 
|  |  | 
|  | fn source(&'a self, edge: &Edge<'a>) -> Crate { | 
|  | edge.0 | 
|  | } | 
|  |  | 
|  | fn target(&'a self, edge: &Edge<'a>) -> Crate { | 
|  | edge.1.crate_id | 
|  | } | 
|  | } | 
|  |  | 
|  | impl<'a> dot::Labeller<'a, Crate, Edge<'a>> for DotCrateGraph<'_> { | 
|  | fn graph_id(&'a self) -> Id<'a> { | 
|  | Id::new("rust_analyzer_crate_graph").unwrap() | 
|  | } | 
|  |  | 
|  | fn node_id(&'a self, n: &Crate) -> Id<'a> { | 
|  | let id = n.as_id().index(); | 
|  | Id::new(format!("_{id:?}")).unwrap() | 
|  | } | 
|  |  | 
|  | fn node_shape(&'a self, _node: &Crate) -> Option<LabelText<'a>> { | 
|  | Some(LabelText::LabelStr("box".into())) | 
|  | } | 
|  |  | 
|  | fn node_label(&'a self, n: &Crate) -> LabelText<'a> { | 
|  | let name = self.crates_to_render[n] | 
|  | .1 | 
|  | .display_name | 
|  | .as_ref() | 
|  | .map_or("(unnamed crate)", |name| name.as_str()); | 
|  | LabelText::LabelStr(name.into()) | 
|  | } | 
|  | } |