// Copyright 2020 The Jujutsu Authors // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // https://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #![allow(missing_docs)] use std::collections::BTreeMap; use std::sync::Arc; use pollster::FutureExt; use crate::backend; use crate::backend::BackendResult; use crate::backend::TreeId; use crate::backend::TreeValue; use crate::repo_path::RepoPath; use crate::repo_path::RepoPathBuf; use crate::store::Store; use crate::tree::Tree; #[derive(Debug)] enum Override { Tombstone, Replace(TreeValue), } #[derive(Debug)] pub struct TreeBuilder { store: Arc, base_tree_id: TreeId, overrides: BTreeMap, } impl TreeBuilder { pub fn new(store: Arc, base_tree_id: TreeId) -> TreeBuilder { let overrides = BTreeMap::new(); TreeBuilder { store, base_tree_id, overrides, } } pub fn store(&self) -> &Store { self.store.as_ref() } pub fn set(&mut self, path: RepoPathBuf, value: TreeValue) { assert!(!path.is_root()); self.overrides.insert(path, Override::Replace(value)); } pub fn remove(&mut self, path: RepoPathBuf) { assert!(!path.is_root()); self.overrides.insert(path, Override::Tombstone); } pub fn set_or_remove(&mut self, path: RepoPathBuf, value: Option) { assert!(!path.is_root()); if let Some(value) = value { self.overrides.insert(path, Override::Replace(value)); } else { self.overrides.insert(path, Override::Tombstone); } } pub fn write_tree(self) -> BackendResult { if self.overrides.is_empty() { return Ok(self.base_tree_id); } let mut trees_to_write = self.get_base_trees()?; // Update entries in parent trees for file overrides for (path, file_override) in self.overrides { let (dir, basename) = path.split().unwrap(); let tree = trees_to_write.get_mut(dir).unwrap(); match file_override { Override::Replace(value) => { tree.set(basename.to_owned(), value); } Override::Tombstone => { tree.remove(basename); } } } // Write trees in reverse lexicographical order, starting with trees without // children. // TODO: Writing trees concurrently should help on high-latency backends let store = &self.store; while let Some((dir, tree)) = trees_to_write.pop_last() { if let Some((parent, basename)) = dir.split() { let parent_tree = trees_to_write.get_mut(parent).unwrap(); if tree.is_empty() { if let Some(TreeValue::Tree(_)) = parent_tree.value(basename) { parent_tree.remove(basename); } else { // Entry would have been replaced with file (see above) } } else { let tree = store.write_tree(&dir, tree).block_on()?; parent_tree.set(basename.to_owned(), TreeValue::Tree(tree.id().clone())); } } else { // We're writing the root tree. Write it even if empty. Return its id. assert!(trees_to_write.is_empty()); let written_tree = store.write_tree(&dir, tree).block_on()?; return Ok(written_tree.id().clone()); } } unreachable!("trees_to_write must contain the root tree"); } fn get_base_trees(&self) -> BackendResult> { let store = &self.store; let mut tree_cache = { let dir = RepoPathBuf::root(); let tree = store.get_tree(&dir, &self.base_tree_id)?; BTreeMap::from([(dir, tree)]) }; fn populate_trees<'a>( tree_cache: &'a mut BTreeMap, store: &Arc, dir: &RepoPath, ) -> BackendResult<&'a Tree> { // `if let Some(tree) = ...` doesn't pass lifetime check as of Rust 1.76.0 if tree_cache.contains_key(dir) { return Ok(tree_cache.get(dir).unwrap()); } let (parent, basename) = dir.split().expect("root must be populated"); let tree = populate_trees(tree_cache, store, parent)? .sub_tree(basename)? .unwrap_or_else(|| Tree::empty(store.clone(), dir.to_owned())); Ok(tree_cache.entry(dir.to_owned()).or_insert(tree)) } for path in self.overrides.keys() { let parent = path.parent().unwrap(); populate_trees(&mut tree_cache, store, parent)?; } Ok(tree_cache .into_iter() .map(|(dir, tree)| (dir, tree.data().clone())) .collect()) } }