ok/jj
1
0
Fork 0
forked from mirrors/jj
jj/lib/src/working_copy.rs
Martin von Zweigbergk e1f00d9426 working copy: pass commit instead of tree into check_out()
Our internal working copy implementations at Google will need the
commit so they can walk history backwards until they get to a "public"
commit. They'll then use that to tell build tools and virtual file
systems to present that as a base.

I'm not sure if we'll need to update `reset()` too. It's currently
only used by `jj untrack`, which doesn't change the commit's parent,
so it wouldn't affect any history walks.
2023-10-16 22:33:44 -07:00

269 lines
10 KiB
Rust

// 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.
//! Defines the interface for the working copy. See `LocalWorkingCopy` for the
//! default local-disk implementation.
use std::any::Any;
use std::ffi::OsString;
use std::path::{Path, PathBuf};
use std::sync::Arc;
use thiserror::Error;
use crate::backend::{BackendError, MergedTreeId};
use crate::commit::Commit;
use crate::fsmonitor::FsmonitorKind;
use crate::gitignore::GitIgnoreFile;
use crate::merged_tree::MergedTree;
use crate::op_store::{OperationId, WorkspaceId};
use crate::repo_path::RepoPath;
use crate::settings::HumanByteSize;
/// The trait all working-copy implementations must implement.
pub trait WorkingCopy {
/// Should return `self`. For down-casting purposes.
fn as_any(&self) -> &dyn Any;
/// The name/id of the implementation. Used for choosing the right
/// implementation when loading a working copy.
fn name(&self) -> &str;
/// The working copy's root directory.
fn path(&self) -> &Path;
/// The working copy's workspace ID.
fn workspace_id(&self) -> &WorkspaceId;
/// The operation this working copy was most recently updated to.
fn operation_id(&self) -> &OperationId;
/// The ID of the tree this working copy was most recently updated to.
fn tree_id(&self) -> Result<&MergedTreeId, WorkingCopyStateError>;
/// Patterns that decide which paths from the current tree should be checked
/// out in the working copy. An empty list means that no paths should be
/// checked out in the working copy. A single `RepoPath::root()` entry means
/// that all files should be checked out.
fn sparse_patterns(&self) -> Result<&[RepoPath], WorkingCopyStateError>;
/// Locks the working copy and returns an instance with methods for updating
/// the working copy files and state.
fn start_mutation(&self) -> Result<Box<dyn LockedWorkingCopy>, WorkingCopyStateError>;
}
/// A working copy that's being modified.
pub trait LockedWorkingCopy {
/// Should return `self`. For down-casting purposes.
fn as_any(&self) -> &dyn Any;
/// Should return `self`. For down-casting purposes.
fn as_any_mut(&mut self) -> &mut dyn Any;
/// The operation at the time the lock was taken
fn old_operation_id(&self) -> &OperationId;
/// The tree at the time the lock was taken
fn old_tree_id(&self) -> &MergedTreeId;
/// Snapshot the working copy and return the tree id.
fn snapshot(&mut self, options: SnapshotOptions) -> Result<MergedTreeId, SnapshotError>;
/// Check out the specified commit in the working copy.
fn check_out(&mut self, commit: &Commit) -> Result<CheckoutStats, CheckoutError>;
/// Update to another tree without touching the files in the working copy.
fn reset(&mut self, new_tree: &MergedTree) -> Result<(), ResetError>;
/// See `WorkingCopy::sparse_patterns()`
fn sparse_patterns(&self) -> Result<&[RepoPath], WorkingCopyStateError>;
/// Updates the patterns that decide which paths from the current tree
/// should be checked out in the working copy.
// TODO: Use a different error type here so we can include a
// `SparseNotSupported` variants for working copies that don't support sparse
// checkouts (e.g. because they use a virtual file system so there's no reason
// to use sparse).
fn set_sparse_patterns(
&mut self,
new_sparse_patterns: Vec<RepoPath>,
) -> Result<CheckoutStats, CheckoutError>;
/// Finish the modifications to the working copy by writing the updated
/// states to disk. Returns the new (unlocked) working copy.
fn finish(
self: Box<Self>,
operation_id: OperationId,
) -> Result<Box<dyn WorkingCopy>, WorkingCopyStateError>;
}
/// An error while snapshotting the working copy.
#[derive(Debug, Error)]
pub enum SnapshotError {
/// A path in the working copy was not valid UTF-8.
#[error("Working copy path {} is not valid UTF-8", path.to_string_lossy())]
InvalidUtf8Path {
/// The path with invalid UTF-8.
path: OsString,
},
/// A symlink target in the working copy was not valid UTF-8.
#[error("Symlink {path} target is not valid UTF-8")]
InvalidUtf8SymlinkTarget {
/// The path of the symlink that has a target that's not valid UTF-8.
/// This path itself is valid UTF-8.
path: PathBuf,
/// The symlink target with invalid UTF-8.
target: PathBuf,
},
/// Reading or writing from the commit backend failed.
#[error("Internal backend error: {0}")]
InternalBackendError(#[from] BackendError),
/// A file was larger than the specified maximum file size for new
/// (previously untracked) files.
#[error("New file {path} of size ~{size} exceeds snapshot.max-new-file-size ({max_size})")]
NewFileTooLarge {
/// The path of the large file.
path: PathBuf,
/// The size of the large file.
size: HumanByteSize,
/// The maximum allowed size.
max_size: HumanByteSize,
},
/// Some other error happened while snapshotting the working copy.
#[error("{message}: {err:?}")]
Other {
/// Error message.
message: String,
/// The underlying error.
#[source]
err: Box<dyn std::error::Error + Send + Sync>,
},
}
/// Options used when snapshotting the working copy. Some of them may be ignored
/// by some `WorkingCopy` implementations.
pub struct SnapshotOptions<'a> {
/// The `.gitignore`s to use while snapshotting. The typically come from the
/// user's configured patterns combined with per-repo patterns.
// The base_ignores are passed in here rather than being set on the TreeState
// because the TreeState may be long-lived if the library is used in a
// long-lived process.
pub base_ignores: Arc<GitIgnoreFile>,
/// The fsmonitor (e.g. Watchman) to use, if any.
// TODO: Should we make this a field on `LocalWorkingCopy` instead since it's quite specific to
// that implementation?
pub fsmonitor_kind: Option<FsmonitorKind>,
/// A callback for the UI to display progress.
pub progress: Option<&'a SnapshotProgress<'a>>,
/// The size of the largest file that should be allowed to become tracked
/// (already tracked files are always snapshotted). If there are larger
/// files in the working copy, then `LockedWorkingCopy::snapshot()` may
/// (depending on implementation)
/// return `SnapshotError::NewFileTooLarge`.
pub max_new_file_size: u64,
}
impl SnapshotOptions<'_> {
/// Create an instance for use in tests.
pub fn empty_for_test() -> Self {
SnapshotOptions {
base_ignores: GitIgnoreFile::empty(),
fsmonitor_kind: None,
progress: None,
max_new_file_size: u64::MAX,
}
}
}
/// A callback for getting progress updates.
pub type SnapshotProgress<'a> = dyn Fn(&RepoPath) + 'a + Sync;
/// Stats about a checkout operation on a working copy. All "files" mentioned
/// below may also be symlinks or materialized conflicts.
#[derive(Debug, PartialEq, Eq, Clone)]
pub struct CheckoutStats {
/// The number of files that were updated in the working copy.
/// These files existed before and after the checkout.
pub updated_files: u32,
/// The number of files added in the working copy.
pub added_files: u32,
/// The number of files removed in the working copy.
pub removed_files: u32,
/// The number of files that were supposed to be updated or added in the
/// working copy but were skipped because there was an untracked (probably
/// ignored) file in its place.
pub skipped_files: u32,
}
/// The working-copy checkout failed.
#[derive(Debug, Error)]
pub enum CheckoutError {
/// The current working-copy commit was deleted, maybe by an overly
/// aggressive GC that happened while the current process was running.
#[error("Current working-copy commit not found: {source}")]
SourceNotFound {
/// The underlying error.
source: Box<dyn std::error::Error + Send + Sync>,
},
/// Another process checked out a commit while the current process was
/// running (after the working copy was read by the current process).
#[error("Concurrent checkout")]
ConcurrentCheckout,
/// Reading or writing from the commit backend failed.
#[error("Internal backend error: {0}")]
InternalBackendError(#[from] BackendError),
/// Some other error happened while checking out the working copy.
#[error("{message}: {err:?}")]
Other {
/// Error message.
message: String,
/// The underlying error.
#[source]
err: Box<dyn std::error::Error + Send + Sync>,
},
}
/// An error while resetting the working copy.
#[derive(Debug, Error)]
pub enum ResetError {
/// The current working-copy commit was deleted, maybe by an overly
/// aggressive GC that happened while the current process was running.
#[error("Current working-copy commit not found: {source}")]
SourceNotFound {
/// The underlying error.
source: Box<dyn std::error::Error + Send + Sync>,
},
/// Reading or writing from the commit backend failed.
#[error("Internal error: {0}")]
InternalBackendError(#[from] BackendError),
/// Some other error happened while checking out the working copy.
#[error("{message}: {err:?}")]
Other {
/// Error message.
message: String,
#[source]
/// The underlying error.
err: Box<dyn std::error::Error + Send + Sync>,
},
}
/// An error while reading the working copy state.
#[derive(Debug, Error)]
#[error("{message}: {err:?}")]
pub struct WorkingCopyStateError {
/// Error message.
pub message: String,
/// The underlying error.
pub err: Box<dyn std::error::Error + Send + Sync>,
}