From a6566832c2cf2bc25c644c601c7c5acd7fae1840 Mon Sep 17 00:00:00 2001 From: Yuya Nishihara Date: Mon, 12 Aug 2024 12:42:02 +0900 Subject: [PATCH] merged_tree: extract file-conflict resolution from merge_tree_values() I'll add a public function that resolves file conflicts. This function will take owned MergedTreeValue, and that's why the extracted function returns None instead of cloning the passed value. --- lib/src/merged_tree.rs | 34 ++++++++++++++++++++++------------ 1 file changed, 22 insertions(+), 12 deletions(-) diff --git a/lib/src/merged_tree.rs b/lib/src/merged_tree.rs index 7dc84a8a8..8ba5d687e 100644 --- a/lib/src/merged_tree.rs +++ b/lib/src/merged_tree.rs @@ -493,18 +493,28 @@ fn merge_tree_values( Ok(merged_tree .map(|tree| (tree.id() != empty_tree_id).then(|| TreeValue::Tree(tree.id().clone())))) } else { - // Try to resolve file conflicts by merging the file contents. Treats missing - // files as empty. The values may contain trees canceling each other (notably - // padded absent trees), so we need to simplify them first. - let simplified = values.clone().simplify(); - // No fast path for simplified.is_resolved(). If it could be resolved, it would - // have been caught by values.resolve_trivial() above. - if let Some(resolved) = try_resolve_file_conflict(store, path, &simplified)? { - Ok(Merge::normal(resolved)) - } else { - // Failed to merge the files, or the paths are not files - Ok(values.cloned()) - } + let maybe_resolved = try_resolve_file_values(store, path, values)?; + Ok(maybe_resolved.unwrap_or_else(|| values.cloned())) + } +} + +/// Tries to resolve file conflicts by merging the file contents. Treats missing +/// files as empty. +fn try_resolve_file_values( + store: &Arc, + path: &RepoPath, + values: &MergedTreeVal, +) -> BackendResult> { + // The values may contain trees canceling each other (notably padded absent + // trees), so we need to simplify them first. + let simplified = values.clone().simplify(); + // No fast path for simplified.is_resolved(). If it could be resolved, it would + // have been caught by values.resolve_trivial() above. + if let Some(resolved) = try_resolve_file_conflict(store, path, &simplified)? { + Ok(Some(Merge::normal(resolved))) + } else { + // Failed to merge the files, or the paths are not files + Ok(None) } }