// Copyright 2023 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. use std::any::Any; use std::path::Path; use std::path::PathBuf; use std::sync::Arc; use itertools::Itertools; use jj_cli::cli_util::CliRunner; use jj_cli::cli_util::CommandHelper; use jj_cli::command_error::CommandError; use jj_cli::ui::Ui; use jj_lib::backend::Backend; use jj_lib::backend::MergedTreeId; use jj_lib::commit::Commit; use jj_lib::git_backend::GitBackend; use jj_lib::local_working_copy::LocalWorkingCopy; use jj_lib::op_store::OperationId; use jj_lib::op_store::WorkspaceId; use jj_lib::repo::ReadonlyRepo; use jj_lib::repo_path::RepoPathBuf; use jj_lib::settings::UserSettings; use jj_lib::signing::Signer; use jj_lib::store::Store; use jj_lib::working_copy::CheckoutError; use jj_lib::working_copy::CheckoutOptions; use jj_lib::working_copy::CheckoutStats; use jj_lib::working_copy::LockedWorkingCopy; use jj_lib::working_copy::ResetError; use jj_lib::working_copy::SnapshotError; use jj_lib::working_copy::SnapshotOptions; use jj_lib::working_copy::SnapshotStats; use jj_lib::working_copy::WorkingCopy; use jj_lib::working_copy::WorkingCopyFactory; use jj_lib::working_copy::WorkingCopyStateError; use jj_lib::workspace::WorkingCopyFactories; use jj_lib::workspace::Workspace; use jj_lib::workspace::WorkspaceInitError; #[derive(clap::Parser, Clone, Debug)] enum CustomCommand { /// Initialize a workspace using the "conflicts" working copy InitConflicts, } fn run_custom_command( _ui: &mut Ui, command_helper: &CommandHelper, command: CustomCommand, ) -> Result<(), CommandError> { match command { CustomCommand::InitConflicts => { let wc_path = command_helper.cwd(); let backend_initializer = |settings: &UserSettings, store_path: &Path| { let backend: Box = Box::new(GitBackend::init_internal(settings, store_path)?); Ok(backend) }; Workspace::init_with_factories( command_helper.settings(), wc_path, &backend_initializer, Signer::from_settings(command_helper.settings()) .map_err(WorkspaceInitError::SignInit)?, &ReadonlyRepo::default_op_store_initializer(), &ReadonlyRepo::default_op_heads_store_initializer(), &ReadonlyRepo::default_index_store_initializer(), &ReadonlyRepo::default_submodule_store_initializer(), &ConflictsWorkingCopyFactory {}, WorkspaceId::default(), )?; Ok(()) } } } fn main() -> std::process::ExitCode { let mut working_copy_factories = WorkingCopyFactories::new(); working_copy_factories.insert( ConflictsWorkingCopy::name().to_owned(), Box::new(ConflictsWorkingCopyFactory {}), ); CliRunner::init() .add_working_copy_factories(working_copy_factories) .add_subcommand(run_custom_command) .run() } /// A working copy that adds a .conflicts file with a list of unresolved /// conflicts. /// /// Most functions below just delegate to the inner working-copy backend. The /// only interesting functions are `snapshot()` and `check_out()`. The former /// adds `.conflicts` to the .gitignores. The latter writes the `.conflicts` /// file to the working copy. struct ConflictsWorkingCopy { inner: Box, working_copy_path: PathBuf, } impl ConflictsWorkingCopy { fn name() -> &'static str { "conflicts" } fn init( store: Arc, working_copy_path: PathBuf, state_path: PathBuf, operation_id: OperationId, workspace_id: WorkspaceId, ) -> Result { let inner = LocalWorkingCopy::init( store, working_copy_path.clone(), state_path, operation_id, workspace_id, )?; Ok(ConflictsWorkingCopy { inner: Box::new(inner), working_copy_path, }) } fn load(store: Arc, working_copy_path: PathBuf, state_path: PathBuf) -> Self { let inner = LocalWorkingCopy::load(store, working_copy_path.clone(), state_path); ConflictsWorkingCopy { inner: Box::new(inner), working_copy_path, } } } impl WorkingCopy for ConflictsWorkingCopy { fn as_any(&self) -> &dyn Any { self } fn name(&self) -> &str { Self::name() } fn workspace_id(&self) -> &WorkspaceId { self.inner.workspace_id() } fn operation_id(&self) -> &OperationId { self.inner.operation_id() } fn tree_id(&self) -> Result<&MergedTreeId, WorkingCopyStateError> { self.inner.tree_id() } fn sparse_patterns(&self) -> Result<&[RepoPathBuf], WorkingCopyStateError> { self.inner.sparse_patterns() } fn start_mutation(&self) -> Result, WorkingCopyStateError> { let inner = self.inner.start_mutation()?; Ok(Box::new(LockedConflictsWorkingCopy { wc_path: self.working_copy_path.clone(), inner, })) } } struct ConflictsWorkingCopyFactory {} impl WorkingCopyFactory for ConflictsWorkingCopyFactory { fn init_working_copy( &self, store: Arc, working_copy_path: PathBuf, state_path: PathBuf, operation_id: OperationId, workspace_id: WorkspaceId, ) -> Result, WorkingCopyStateError> { Ok(Box::new(ConflictsWorkingCopy::init( store, working_copy_path, state_path, operation_id, workspace_id, )?)) } fn load_working_copy( &self, store: Arc, working_copy_path: PathBuf, state_path: PathBuf, ) -> Result, WorkingCopyStateError> { Ok(Box::new(ConflictsWorkingCopy::load( store, working_copy_path, state_path, ))) } } struct LockedConflictsWorkingCopy { wc_path: PathBuf, inner: Box, } impl LockedWorkingCopy for LockedConflictsWorkingCopy { fn as_any(&self) -> &dyn Any { self } fn as_any_mut(&mut self) -> &mut dyn Any { self } fn old_operation_id(&self) -> &OperationId { self.inner.old_operation_id() } fn old_tree_id(&self) -> &MergedTreeId { self.inner.old_tree_id() } fn snapshot( &mut self, options: &SnapshotOptions, ) -> Result<(MergedTreeId, SnapshotStats), SnapshotError> { let options = SnapshotOptions { base_ignores: options.base_ignores.chain("", "/.conflicts".as_bytes())?, ..options.clone() }; self.inner.snapshot(&options) } fn check_out( &mut self, commit: &Commit, options: &CheckoutOptions, ) -> Result { let conflicts = commit .tree()? .conflicts() .map(|(path, _value)| format!("{}\n", path.as_internal_file_string())) .join(""); std::fs::write(self.wc_path.join(".conflicts"), conflicts).unwrap(); self.inner.check_out(commit, options) } fn rename_workspace(&mut self, new_workspace_id: WorkspaceId) { self.inner.rename_workspace(new_workspace_id); } fn reset(&mut self, commit: &Commit) -> Result<(), ResetError> { self.inner.reset(commit) } fn recover(&mut self, commit: &Commit) -> Result<(), ResetError> { self.inner.recover(commit) } fn sparse_patterns(&self) -> Result<&[RepoPathBuf], WorkingCopyStateError> { self.inner.sparse_patterns() } fn set_sparse_patterns( &mut self, new_sparse_patterns: Vec, options: &CheckoutOptions, ) -> Result { self.inner.set_sparse_patterns(new_sparse_patterns, options) } fn finish( self: Box, operation_id: OperationId, ) -> Result, WorkingCopyStateError> { let inner = self.inner.finish(operation_id)?; Ok(Box::new(ConflictsWorkingCopy { inner, working_copy_path: self.wc_path, })) } }