|  | //! Defines a unit of change that can applied to the database to get the next | 
|  | //! state. Changes are transactional. | 
|  |  | 
|  | use std::fmt; | 
|  |  | 
|  | use salsa::Durability; | 
|  | use triomphe::Arc; | 
|  | use vfs::FileId; | 
|  |  | 
|  | use crate::{CrateGraphBuilder, CratesIdMap, RootQueryDb, SourceRoot, SourceRootId}; | 
|  |  | 
|  | /// Encapsulate a bunch of raw `.set` calls on the database. | 
|  | #[derive(Default)] | 
|  | pub struct FileChange { | 
|  | pub roots: Option<Vec<SourceRoot>>, | 
|  | pub files_changed: Vec<(FileId, Option<String>)>, | 
|  | pub crate_graph: Option<CrateGraphBuilder>, | 
|  | } | 
|  |  | 
|  | impl fmt::Debug for FileChange { | 
|  | fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result { | 
|  | let mut d = fmt.debug_struct("Change"); | 
|  | if let Some(roots) = &self.roots { | 
|  | d.field("roots", roots); | 
|  | } | 
|  | if !self.files_changed.is_empty() { | 
|  | d.field("files_changed", &self.files_changed.len()); | 
|  | } | 
|  | if self.crate_graph.is_some() { | 
|  | d.field("crate_graph", &self.crate_graph); | 
|  | } | 
|  | d.finish() | 
|  | } | 
|  | } | 
|  |  | 
|  | impl FileChange { | 
|  | pub fn new() -> Self { | 
|  | FileChange::default() | 
|  | } | 
|  |  | 
|  | pub fn set_roots(&mut self, roots: Vec<SourceRoot>) { | 
|  | self.roots = Some(roots); | 
|  | } | 
|  |  | 
|  | pub fn change_file(&mut self, file_id: FileId, new_text: Option<String>) { | 
|  | self.files_changed.push((file_id, new_text)) | 
|  | } | 
|  |  | 
|  | pub fn set_crate_graph(&mut self, graph: CrateGraphBuilder) { | 
|  | self.crate_graph = Some(graph); | 
|  | } | 
|  |  | 
|  | pub fn apply(self, db: &mut dyn RootQueryDb) -> Option<CratesIdMap> { | 
|  | let _p = tracing::info_span!("FileChange::apply").entered(); | 
|  | if let Some(roots) = self.roots { | 
|  | for (idx, root) in roots.into_iter().enumerate() { | 
|  | let root_id = SourceRootId(idx as u32); | 
|  | let durability = source_root_durability(&root); | 
|  | for file_id in root.iter() { | 
|  | db.set_file_source_root_with_durability(file_id, root_id, durability); | 
|  | } | 
|  |  | 
|  | db.set_source_root_with_durability(root_id, Arc::new(root), durability); | 
|  | } | 
|  | } | 
|  |  | 
|  | for (file_id, text) in self.files_changed { | 
|  | let source_root_id = db.file_source_root(file_id); | 
|  | let source_root = db.source_root(source_root_id.source_root_id(db)); | 
|  |  | 
|  | let durability = file_text_durability(&source_root.source_root(db)); | 
|  | // XXX: can't actually remove the file, just reset the text | 
|  | let text = text.unwrap_or_default(); | 
|  | db.set_file_text_with_durability(file_id, &text, durability) | 
|  | } | 
|  |  | 
|  | if let Some(crate_graph) = self.crate_graph { | 
|  | return Some(crate_graph.set_in_db(db)); | 
|  | } | 
|  | None | 
|  | } | 
|  | } | 
|  |  | 
|  | fn source_root_durability(source_root: &SourceRoot) -> Durability { | 
|  | if source_root.is_library { Durability::MEDIUM } else { Durability::LOW } | 
|  | } | 
|  |  | 
|  | fn file_text_durability(source_root: &SourceRoot) -> Durability { | 
|  | if source_root.is_library { Durability::HIGH } else { Durability::LOW } | 
|  | } |