mirror of
https://github.com/martinvonz/jj.git
synced 2025-01-28 15:26:25 +00:00
repo: load index eagerly to simplify error handling
If readonly_index() and index() returned Result, it would propagate to many call sites. That seems bad for API ergonomics. Suppose most "repo" commands depend on an index, I think it's okay to load index eagerly: - "jj config" doesn't load repo (nor index) - "jj workspace root" doesn't load repo (nor index) - some other mutation commands load index when printing commit summary - many other commands load index when resolving revset
This commit is contained in:
parent
626aa90610
commit
da221eb888
1 changed files with 23 additions and 20 deletions
|
@ -17,7 +17,6 @@
|
||||||
use std::collections::hash_map::Entry;
|
use std::collections::hash_map::Entry;
|
||||||
use std::collections::{HashMap, HashSet};
|
use std::collections::{HashMap, HashSet};
|
||||||
use std::fmt::{Debug, Formatter};
|
use std::fmt::{Debug, Formatter};
|
||||||
use std::ops::Deref;
|
|
||||||
use std::path::{Path, PathBuf};
|
use std::path::{Path, PathBuf};
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::{fs, slice};
|
use std::{fs, slice};
|
||||||
|
@ -37,7 +36,7 @@ use crate::commit_builder::{CommitBuilder, DetachedCommitBuilder};
|
||||||
use crate::default_index::{DefaultIndexStore, DefaultMutableIndex};
|
use crate::default_index::{DefaultIndexStore, DefaultMutableIndex};
|
||||||
use crate::default_submodule_store::DefaultSubmoduleStore;
|
use crate::default_submodule_store::DefaultSubmoduleStore;
|
||||||
use crate::file_util::{IoResultExt as _, PathError};
|
use crate::file_util::{IoResultExt as _, PathError};
|
||||||
use crate::index::{ChangeIdIndex, Index, IndexStore, MutableIndex, ReadonlyIndex};
|
use crate::index::{ChangeIdIndex, Index, IndexReadError, IndexStore, MutableIndex, ReadonlyIndex};
|
||||||
use crate::local_backend::LocalBackend;
|
use crate::local_backend::LocalBackend;
|
||||||
use crate::object_id::{HexPrefix, ObjectId, PrefixResolution};
|
use crate::object_id::{HexPrefix, ObjectId, PrefixResolution};
|
||||||
use crate::op_heads_store::{self, OpHeadResolutionError, OpHeadsStore};
|
use crate::op_heads_store::{self, OpHeadResolutionError, OpHeadsStore};
|
||||||
|
@ -95,7 +94,7 @@ pub struct ReadonlyRepo {
|
||||||
settings: RepoSettings,
|
settings: RepoSettings,
|
||||||
index_store: Arc<dyn IndexStore>,
|
index_store: Arc<dyn IndexStore>,
|
||||||
submodule_store: Arc<dyn SubmoduleStore>,
|
submodule_store: Arc<dyn SubmoduleStore>,
|
||||||
index: OnceCell<Box<dyn ReadonlyIndex>>,
|
index: Box<dyn ReadonlyIndex>,
|
||||||
change_id_index: OnceCell<Box<dyn ChangeIdIndex>>,
|
change_id_index: OnceCell<Box<dyn ChangeIdIndex>>,
|
||||||
// TODO: This should eventually become part of the index and not be stored fully in memory.
|
// TODO: This should eventually become part of the index and not be stored fully in memory.
|
||||||
view: View,
|
view: View,
|
||||||
|
@ -179,7 +178,7 @@ impl ReadonlyRepo {
|
||||||
let index_store = index_store_initializer(user_settings, &index_path)?;
|
let index_store = index_store_initializer(user_settings, &index_path)?;
|
||||||
let index_type_path = index_path.join("type");
|
let index_type_path = index_path.join("type");
|
||||||
fs::write(&index_type_path, index_store.name()).context(&index_type_path)?;
|
fs::write(&index_type_path, index_store.name()).context(&index_type_path)?;
|
||||||
let index_store = Arc::from(index_store);
|
let index_store: Arc<dyn IndexStore> = Arc::from(index_store);
|
||||||
|
|
||||||
let submodule_store_path = repo_path.join("submodule_store");
|
let submodule_store_path = repo_path.join("submodule_store");
|
||||||
fs::create_dir(&submodule_store_path).context(&submodule_store_path)?;
|
fs::create_dir(&submodule_store_path).context(&submodule_store_path)?;
|
||||||
|
@ -198,6 +197,11 @@ impl ReadonlyRepo {
|
||||||
root_operation_data,
|
root_operation_data,
|
||||||
);
|
);
|
||||||
let root_view = root_operation.view().expect("failed to read root view");
|
let root_view = root_operation.view().expect("failed to read root view");
|
||||||
|
let index = index_store
|
||||||
|
.get_index_at_op(&root_operation, &store)
|
||||||
|
// If the root op index couldn't be read, the index backend wouldn't
|
||||||
|
// be initialized properly.
|
||||||
|
.map_err(|err| BackendInitError(err.into()))?;
|
||||||
let repo = Arc::new(ReadonlyRepo {
|
let repo = Arc::new(ReadonlyRepo {
|
||||||
repo_path,
|
repo_path,
|
||||||
store,
|
store,
|
||||||
|
@ -206,7 +210,7 @@ impl ReadonlyRepo {
|
||||||
operation: root_operation,
|
operation: root_operation,
|
||||||
settings: repo_settings,
|
settings: repo_settings,
|
||||||
index_store,
|
index_store,
|
||||||
index: OnceCell::new(),
|
index,
|
||||||
change_id_index: OnceCell::new(),
|
change_id_index: OnceCell::new(),
|
||||||
view: root_view,
|
view: root_view,
|
||||||
submodule_store,
|
submodule_store,
|
||||||
|
@ -247,15 +251,7 @@ impl ReadonlyRepo {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn readonly_index(&self) -> &dyn ReadonlyIndex {
|
pub fn readonly_index(&self) -> &dyn ReadonlyIndex {
|
||||||
self.index
|
self.index.as_ref()
|
||||||
.get_or_init(|| {
|
|
||||||
// TODO: somehow propagate error, but it's weird if all callers
|
|
||||||
// had Result<T, IndexReadError> signature.
|
|
||||||
self.index_store
|
|
||||||
.get_index_at_op(&self.operation, &self.store)
|
|
||||||
.unwrap()
|
|
||||||
})
|
|
||||||
.deref()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn change_id_index(&self) -> &dyn ChangeIdIndex {
|
fn change_id_index(&self) -> &dyn ChangeIdIndex {
|
||||||
|
@ -589,6 +585,8 @@ pub enum RepoLoaderError {
|
||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
Backend(#[from] BackendError),
|
Backend(#[from] BackendError),
|
||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
|
IndexRead(#[from] IndexReadError),
|
||||||
|
#[error(transparent)]
|
||||||
OpHeadResolution(#[from] OpHeadResolutionError),
|
OpHeadResolution(#[from] OpHeadResolutionError),
|
||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
OpStore(#[from] OpStoreError),
|
OpStore(#[from] OpStoreError),
|
||||||
|
@ -669,13 +667,13 @@ impl RepoLoader {
|
||||||
|op_heads| self._resolve_op_heads(op_heads, user_settings),
|
|op_heads| self._resolve_op_heads(op_heads, user_settings),
|
||||||
)?;
|
)?;
|
||||||
let view = op.view()?;
|
let view = op.view()?;
|
||||||
Ok(self._finish_load(op, view))
|
self._finish_load(op, view)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[instrument(skip(self))]
|
#[instrument(skip(self))]
|
||||||
pub fn load_at(&self, op: &Operation) -> Result<Arc<ReadonlyRepo>, RepoLoaderError> {
|
pub fn load_at(&self, op: &Operation) -> Result<Arc<ReadonlyRepo>, RepoLoaderError> {
|
||||||
let view = op.view()?;
|
let view = op.view()?;
|
||||||
Ok(self._finish_load(op.clone(), view))
|
self._finish_load(op.clone(), view)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn create_from(
|
pub fn create_from(
|
||||||
|
@ -693,7 +691,7 @@ impl RepoLoader {
|
||||||
settings: self.repo_settings.clone(),
|
settings: self.repo_settings.clone(),
|
||||||
index_store: self.index_store.clone(),
|
index_store: self.index_store.clone(),
|
||||||
submodule_store: self.submodule_store.clone(),
|
submodule_store: self.submodule_store.clone(),
|
||||||
index: OnceCell::with_value(index),
|
index,
|
||||||
change_id_index: OnceCell::new(),
|
change_id_index: OnceCell::new(),
|
||||||
view,
|
view,
|
||||||
};
|
};
|
||||||
|
@ -743,7 +741,12 @@ impl RepoLoader {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn _finish_load(&self, operation: Operation, view: View) -> Arc<ReadonlyRepo> {
|
fn _finish_load(
|
||||||
|
&self,
|
||||||
|
operation: Operation,
|
||||||
|
view: View,
|
||||||
|
) -> Result<Arc<ReadonlyRepo>, RepoLoaderError> {
|
||||||
|
let index = self.index_store.get_index_at_op(&operation, &self.store)?;
|
||||||
let repo = ReadonlyRepo {
|
let repo = ReadonlyRepo {
|
||||||
repo_path: self.repo_path.clone(),
|
repo_path: self.repo_path.clone(),
|
||||||
store: self.store.clone(),
|
store: self.store.clone(),
|
||||||
|
@ -753,11 +756,11 @@ impl RepoLoader {
|
||||||
settings: self.repo_settings.clone(),
|
settings: self.repo_settings.clone(),
|
||||||
index_store: self.index_store.clone(),
|
index_store: self.index_store.clone(),
|
||||||
submodule_store: self.submodule_store.clone(),
|
submodule_store: self.submodule_store.clone(),
|
||||||
index: OnceCell::new(),
|
index,
|
||||||
change_id_index: OnceCell::new(),
|
change_id_index: OnceCell::new(),
|
||||||
view,
|
view,
|
||||||
};
|
};
|
||||||
Arc::new(repo)
|
Ok(Arc::new(repo))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue