2020-12-12 08:00:42 +00:00
|
|
|
// Copyright 2020 Google LLC
|
|
|
|
//
|
|
|
|
// 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::fmt::{Debug, Formatter};
|
|
|
|
use std::fs;
|
|
|
|
use std::fs::File;
|
|
|
|
use std::io::{Read, Write};
|
2021-01-04 17:40:46 +00:00
|
|
|
use std::path::{Path, PathBuf};
|
2020-12-12 08:00:42 +00:00
|
|
|
use std::sync::{Arc, Mutex, MutexGuard};
|
|
|
|
|
|
|
|
use thiserror::Error;
|
|
|
|
|
|
|
|
use crate::commit_builder::{new_change_id, signature};
|
2021-02-01 02:13:37 +00:00
|
|
|
use crate::evolution::{EvolutionRef, MutableEvolution, ReadonlyEvolution};
|
2020-12-12 08:00:42 +00:00
|
|
|
use crate::git_store::GitStore;
|
2021-01-31 02:50:27 +00:00
|
|
|
use crate::index::{IndexRef, MutableIndex, ReadonlyIndex};
|
2021-03-03 05:43:13 +00:00
|
|
|
use crate::index_store::IndexStore;
|
2020-12-12 08:00:42 +00:00
|
|
|
use crate::local_store::LocalStore;
|
2021-02-27 07:37:42 +00:00
|
|
|
use crate::op_store::OpStore;
|
2020-12-12 08:00:42 +00:00
|
|
|
use crate::operation::Operation;
|
|
|
|
use crate::settings::{RepoSettings, UserSettings};
|
2021-02-27 07:37:42 +00:00
|
|
|
use crate::simple_op_store::SimpleOpStore;
|
2020-12-12 08:00:42 +00:00
|
|
|
use crate::store;
|
|
|
|
use crate::store::{Store, StoreError};
|
|
|
|
use crate::store_wrapper::StoreWrapper;
|
2021-02-01 02:00:14 +00:00
|
|
|
use crate::transaction::Transaction;
|
2021-02-01 07:41:59 +00:00
|
|
|
use crate::view::{MutableView, ReadonlyView, ViewRef};
|
2020-12-12 08:00:42 +00:00
|
|
|
use crate::working_copy::WorkingCopy;
|
|
|
|
|
|
|
|
#[derive(Debug, Error, PartialEq, Eq)]
|
|
|
|
pub enum RepoError {
|
|
|
|
#[error("Object not found")]
|
|
|
|
NotFound,
|
|
|
|
#[error("Error: {0}")]
|
|
|
|
Other(String),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<StoreError> for RepoError {
|
|
|
|
fn from(err: StoreError) -> Self {
|
|
|
|
match err {
|
|
|
|
StoreError::NotFound => RepoError::NotFound,
|
|
|
|
StoreError::Other(description) => RepoError::Other(description),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub type RepoResult<T> = Result<T, RepoError>;
|
|
|
|
|
2021-01-31 07:44:31 +00:00
|
|
|
// TODO: Should we implement From<&ReadonlyRepo> and From<&MutableRepo> for
|
|
|
|
// RepoRef?
|
|
|
|
#[derive(Clone, Copy)]
|
|
|
|
pub enum RepoRef<'a, 'r: 'a> {
|
|
|
|
Readonly(&'a ReadonlyRepo),
|
|
|
|
Mutable(&'a MutableRepo<'r>),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, 'r> RepoRef<'a, 'r> {
|
|
|
|
pub fn store(&self) -> &'a Arc<StoreWrapper> {
|
|
|
|
match self {
|
|
|
|
RepoRef::Readonly(repo) => repo.store(),
|
|
|
|
RepoRef::Mutable(repo) => repo.store(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-31 02:50:27 +00:00
|
|
|
pub fn index(&self) -> IndexRef {
|
|
|
|
match self {
|
|
|
|
RepoRef::Readonly(repo) => IndexRef::Readonly(repo.index()),
|
|
|
|
RepoRef::Mutable(repo) => IndexRef::Mutable(repo.index()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-01 07:41:59 +00:00
|
|
|
pub fn view(&self) -> ViewRef<'a> {
|
2021-01-31 07:44:31 +00:00
|
|
|
match self {
|
2021-02-01 07:41:59 +00:00
|
|
|
RepoRef::Readonly(repo) => ViewRef::Readonly(repo.view()),
|
|
|
|
RepoRef::Mutable(repo) => ViewRef::Mutable(repo.view()),
|
2021-01-31 07:44:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-01 07:41:59 +00:00
|
|
|
pub fn evolution(&self) -> EvolutionRef<'a, 'a, 'r> {
|
2021-01-31 07:44:31 +00:00
|
|
|
match self {
|
2021-02-01 02:13:37 +00:00
|
|
|
RepoRef::Readonly(repo) => EvolutionRef::Readonly(repo.evolution()),
|
|
|
|
RepoRef::Mutable(repo) => EvolutionRef::Mutable(repo.evolution()),
|
2021-01-31 07:44:31 +00:00
|
|
|
}
|
|
|
|
}
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub struct ReadonlyRepo {
|
|
|
|
repo_path: PathBuf,
|
|
|
|
wc_path: PathBuf,
|
|
|
|
store: Arc<StoreWrapper>,
|
|
|
|
settings: RepoSettings,
|
2021-03-06 18:37:57 +00:00
|
|
|
index_store: Arc<IndexStore>,
|
2020-12-18 16:32:15 +00:00
|
|
|
index: Mutex<Option<Arc<ReadonlyIndex>>>,
|
2020-12-12 08:00:42 +00:00
|
|
|
working_copy: Arc<Mutex<WorkingCopy>>,
|
|
|
|
view: ReadonlyView,
|
|
|
|
evolution: Option<ReadonlyEvolution<'static>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Debug for ReadonlyRepo {
|
|
|
|
fn fmt(&self, f: &mut Formatter<'_>) -> Result<(), std::fmt::Error> {
|
|
|
|
f.debug_struct("Repo")
|
|
|
|
.field("repo_path", &self.repo_path)
|
|
|
|
.field("wc_path", &self.wc_path)
|
|
|
|
.field("store", &self.store)
|
|
|
|
.finish()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-04 07:11:22 +00:00
|
|
|
#[derive(Error, Debug, PartialEq)]
|
|
|
|
pub enum RepoLoadError {
|
|
|
|
#[error("There is no Jujube repo in {0}")]
|
|
|
|
NoRepoHere(PathBuf),
|
|
|
|
}
|
|
|
|
|
2020-12-12 08:00:42 +00:00
|
|
|
impl ReadonlyRepo {
|
|
|
|
pub fn init_local(settings: &UserSettings, wc_path: PathBuf) -> Arc<ReadonlyRepo> {
|
|
|
|
let repo_path = wc_path.join(".jj");
|
|
|
|
fs::create_dir(repo_path.clone()).unwrap();
|
|
|
|
let store_path = repo_path.join("store");
|
|
|
|
fs::create_dir(&store_path).unwrap();
|
|
|
|
let store = Box::new(LocalStore::init(store_path));
|
|
|
|
ReadonlyRepo::init(settings, repo_path, wc_path, store)
|
|
|
|
}
|
|
|
|
|
2020-12-28 08:26:37 +00:00
|
|
|
/// Initializes a repo with a new Git store in .jj/git/ (bare Git repo)
|
|
|
|
pub fn init_internal_git(settings: &UserSettings, wc_path: PathBuf) -> Arc<ReadonlyRepo> {
|
|
|
|
let repo_path = wc_path.join(".jj");
|
|
|
|
fs::create_dir(repo_path.clone()).unwrap();
|
|
|
|
let git_store_path = repo_path.join("git");
|
|
|
|
git2::Repository::init_bare(&git_store_path).unwrap();
|
|
|
|
let store_path = repo_path.join("store");
|
|
|
|
let git_store_path = fs::canonicalize(git_store_path).unwrap();
|
|
|
|
let mut store_file = File::create(store_path).unwrap();
|
|
|
|
store_file.write_all(b"git: git").unwrap();
|
2021-02-27 07:00:46 +00:00
|
|
|
let store = Box::new(GitStore::load(&git_store_path));
|
2020-12-28 08:26:37 +00:00
|
|
|
ReadonlyRepo::init(settings, repo_path, wc_path, store)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Initializes a repo with an existing Git store at the specified path
|
|
|
|
pub fn init_external_git(
|
2020-12-12 08:00:42 +00:00
|
|
|
settings: &UserSettings,
|
|
|
|
wc_path: PathBuf,
|
|
|
|
git_store_path: PathBuf,
|
|
|
|
) -> Arc<ReadonlyRepo> {
|
|
|
|
let repo_path = wc_path.join(".jj");
|
|
|
|
fs::create_dir(repo_path.clone()).unwrap();
|
|
|
|
let store_path = repo_path.join("store");
|
|
|
|
let git_store_path = fs::canonicalize(git_store_path).unwrap();
|
|
|
|
let mut store_file = File::create(store_path).unwrap();
|
|
|
|
store_file
|
2020-12-28 08:26:37 +00:00
|
|
|
.write_all(format!("git: {}", git_store_path.to_str().unwrap()).as_bytes())
|
2020-12-12 08:00:42 +00:00
|
|
|
.unwrap();
|
2021-02-27 07:00:46 +00:00
|
|
|
let store = Box::new(GitStore::load(&git_store_path));
|
2020-12-12 08:00:42 +00:00
|
|
|
ReadonlyRepo::init(settings, repo_path, wc_path, store)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn init(
|
|
|
|
user_settings: &UserSettings,
|
|
|
|
repo_path: PathBuf,
|
|
|
|
wc_path: PathBuf,
|
|
|
|
store: Box<dyn Store>,
|
|
|
|
) -> Arc<ReadonlyRepo> {
|
|
|
|
let repo_settings = user_settings.with_repo(&repo_path).unwrap();
|
|
|
|
let store = StoreWrapper::new(store);
|
|
|
|
|
|
|
|
fs::create_dir(repo_path.join("working_copy")).unwrap();
|
2020-12-19 07:50:10 +00:00
|
|
|
let working_copy = WorkingCopy::init(
|
|
|
|
store.clone(),
|
|
|
|
wc_path.clone(),
|
|
|
|
repo_path.join("working_copy"),
|
|
|
|
);
|
2020-12-12 08:00:42 +00:00
|
|
|
|
|
|
|
fs::create_dir(repo_path.join("view")).unwrap();
|
|
|
|
let signature = signature(user_settings);
|
|
|
|
let checkout_commit = store::Commit {
|
|
|
|
parents: vec![],
|
|
|
|
predecessors: vec![],
|
|
|
|
root_tree: store.empty_tree_id().clone(),
|
|
|
|
change_id: new_change_id(),
|
|
|
|
description: "".to_string(),
|
|
|
|
author: signature.clone(),
|
|
|
|
committer: signature,
|
|
|
|
is_open: true,
|
|
|
|
is_pruned: false,
|
|
|
|
};
|
|
|
|
let checkout_commit = store.write_commit(checkout_commit);
|
2021-02-27 07:37:42 +00:00
|
|
|
|
|
|
|
std::fs::create_dir(repo_path.join("op_store")).unwrap();
|
|
|
|
let op_store: Arc<dyn OpStore> = Arc::new(SimpleOpStore::init(repo_path.join("op_store")));
|
2021-03-06 18:37:57 +00:00
|
|
|
|
|
|
|
fs::create_dir(repo_path.join("index")).unwrap();
|
|
|
|
let index_store = Arc::new(IndexStore::init(repo_path.join("index")));
|
|
|
|
|
2020-12-12 08:00:42 +00:00
|
|
|
let view = ReadonlyView::init(
|
|
|
|
store.clone(),
|
2021-02-27 07:37:42 +00:00
|
|
|
op_store,
|
2021-03-06 18:37:57 +00:00
|
|
|
index_store.clone(),
|
2020-12-12 08:00:42 +00:00
|
|
|
repo_path.join("view"),
|
|
|
|
checkout_commit.id().clone(),
|
|
|
|
);
|
|
|
|
|
|
|
|
let repo = ReadonlyRepo {
|
2021-03-03 05:43:13 +00:00
|
|
|
repo_path,
|
2020-12-12 08:00:42 +00:00
|
|
|
wc_path,
|
|
|
|
store,
|
|
|
|
settings: repo_settings,
|
2021-03-03 05:43:13 +00:00
|
|
|
index_store,
|
2020-12-12 08:00:42 +00:00
|
|
|
index: Mutex::new(None),
|
|
|
|
working_copy: Arc::new(Mutex::new(working_copy)),
|
|
|
|
view,
|
|
|
|
evolution: None,
|
|
|
|
};
|
|
|
|
let mut repo = Arc::new(repo);
|
|
|
|
let repo_ref: &ReadonlyRepo = repo.as_ref();
|
|
|
|
let static_lifetime_repo: &'static ReadonlyRepo = unsafe { std::mem::transmute(repo_ref) };
|
|
|
|
|
|
|
|
let evolution = ReadonlyEvolution::new(static_lifetime_repo);
|
2021-01-04 19:13:51 +00:00
|
|
|
Arc::get_mut(&mut repo).unwrap().evolution = Some(evolution);
|
2020-12-12 08:00:42 +00:00
|
|
|
|
|
|
|
repo.working_copy_locked()
|
2020-12-19 07:50:10 +00:00
|
|
|
.check_out(checkout_commit)
|
2020-12-12 08:00:42 +00:00
|
|
|
.expect("failed to check out root commit");
|
|
|
|
repo
|
|
|
|
}
|
|
|
|
|
2021-01-04 07:11:22 +00:00
|
|
|
pub fn load(
|
|
|
|
user_settings: &UserSettings,
|
|
|
|
wc_path: PathBuf,
|
|
|
|
) -> Result<Arc<ReadonlyRepo>, RepoLoadError> {
|
2021-01-04 17:40:46 +00:00
|
|
|
ReadonlyRepo::loader(user_settings, wc_path)?.load_at_head()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn loader(
|
|
|
|
user_settings: &UserSettings,
|
|
|
|
wc_path: PathBuf,
|
|
|
|
) -> Result<RepoLoader, RepoLoadError> {
|
|
|
|
RepoLoader::init(user_settings, wc_path)
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2021-01-31 07:44:31 +00:00
|
|
|
pub fn as_repo_ref(&self) -> RepoRef {
|
|
|
|
RepoRef::Readonly(&self)
|
|
|
|
}
|
|
|
|
|
2020-12-12 08:00:42 +00:00
|
|
|
pub fn repo_path(&self) -> &PathBuf {
|
|
|
|
&self.repo_path
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn working_copy_path(&self) -> &PathBuf {
|
|
|
|
&self.wc_path
|
|
|
|
}
|
|
|
|
|
2021-01-31 07:44:31 +00:00
|
|
|
pub fn view(&self) -> &ReadonlyView {
|
|
|
|
&self.view
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn evolution<'a>(&'a self) -> &ReadonlyEvolution<'a> {
|
|
|
|
let evolution: &ReadonlyEvolution<'static> = self.evolution.as_ref().unwrap();
|
|
|
|
let evolution: &ReadonlyEvolution<'a> = unsafe { std::mem::transmute(evolution) };
|
|
|
|
evolution
|
|
|
|
}
|
|
|
|
|
2020-12-18 16:32:15 +00:00
|
|
|
pub fn index(&self) -> Arc<ReadonlyIndex> {
|
2020-12-12 08:00:42 +00:00
|
|
|
let mut locked_index = self.index.lock().unwrap();
|
|
|
|
if locked_index.is_none() {
|
|
|
|
let op_id = self.view.base_op_head_id().clone();
|
2021-03-06 18:37:57 +00:00
|
|
|
let op = self.view.op_store().read_operation(&op_id).unwrap();
|
|
|
|
let op = Operation::new(self.view.op_store().clone(), op_id, op);
|
|
|
|
locked_index.replace(self.index_store.get_index_at_op(&op, self.store.as_ref()));
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
2020-12-18 06:57:41 +00:00
|
|
|
locked_index.as_ref().unwrap().clone()
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
2020-12-18 16:32:15 +00:00
|
|
|
pub fn reindex(&mut self) -> Arc<ReadonlyIndex> {
|
2021-03-03 05:43:13 +00:00
|
|
|
self.index_store.reinit();
|
2020-12-12 08:00:42 +00:00
|
|
|
{
|
|
|
|
let mut locked_index = self.index.lock().unwrap();
|
|
|
|
locked_index.take();
|
|
|
|
}
|
|
|
|
self.index()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn working_copy(&self) -> &Arc<Mutex<WorkingCopy>> {
|
|
|
|
&self.working_copy
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn working_copy_locked(&self) -> MutexGuard<WorkingCopy> {
|
|
|
|
self.working_copy.as_ref().lock().unwrap()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn store(&self) -> &Arc<StoreWrapper> {
|
|
|
|
&self.store
|
|
|
|
}
|
|
|
|
|
2021-03-06 18:37:57 +00:00
|
|
|
pub fn index_store(&self) -> &Arc<IndexStore> {
|
2021-03-03 06:53:20 +00:00
|
|
|
&self.index_store
|
|
|
|
}
|
|
|
|
|
2020-12-12 08:00:42 +00:00
|
|
|
pub fn settings(&self) -> &RepoSettings {
|
|
|
|
&self.settings
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn start_transaction(&self, description: &str) -> Transaction {
|
2021-01-31 02:50:27 +00:00
|
|
|
let mut_repo = MutableRepo::new(
|
|
|
|
self,
|
|
|
|
self.index(),
|
|
|
|
&self.view,
|
|
|
|
&self.evolution.as_ref().unwrap(),
|
|
|
|
);
|
2021-02-01 02:07:37 +00:00
|
|
|
Transaction::new(mut_repo, description)
|
2020-12-12 08:00:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn reload(&mut self) {
|
|
|
|
self.view.reload();
|
|
|
|
let repo_ref: &ReadonlyRepo = self;
|
|
|
|
let static_lifetime_repo: &'static ReadonlyRepo = unsafe { std::mem::transmute(repo_ref) };
|
|
|
|
{
|
|
|
|
let mut locked_index = self.index.lock().unwrap();
|
|
|
|
locked_index.take();
|
|
|
|
}
|
|
|
|
self.evolution = Some(ReadonlyEvolution::new(static_lifetime_repo));
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn reload_at(&mut self, operation: &Operation) {
|
|
|
|
self.view.reload_at(operation);
|
|
|
|
let repo_ref: &ReadonlyRepo = self;
|
|
|
|
let static_lifetime_repo: &'static ReadonlyRepo = unsafe { std::mem::transmute(repo_ref) };
|
|
|
|
{
|
|
|
|
let mut locked_index = self.index.lock().unwrap();
|
|
|
|
locked_index.take();
|
|
|
|
}
|
|
|
|
self.evolution = Some(ReadonlyEvolution::new(static_lifetime_repo));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-04 17:40:46 +00:00
|
|
|
pub struct RepoLoader {
|
|
|
|
wc_path: PathBuf,
|
|
|
|
repo_path: PathBuf,
|
|
|
|
repo_settings: RepoSettings,
|
|
|
|
store: Arc<StoreWrapper>,
|
|
|
|
op_store: Arc<dyn OpStore>,
|
2021-03-06 18:37:57 +00:00
|
|
|
index_store: Arc<IndexStore>,
|
2021-01-04 17:40:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl RepoLoader {
|
|
|
|
fn init(user_settings: &UserSettings, wc_path: PathBuf) -> Result<RepoLoader, RepoLoadError> {
|
|
|
|
let repo_path = wc_path.join(".jj");
|
|
|
|
// TODO: Check if ancestor directory has a .jj/
|
|
|
|
if !repo_path.is_dir() {
|
|
|
|
return Err(RepoLoadError::NoRepoHere(wc_path));
|
|
|
|
}
|
|
|
|
let store = RepoLoader::load_store(&repo_path);
|
|
|
|
let repo_settings = user_settings.with_repo(&repo_path).unwrap();
|
|
|
|
let op_store: Arc<dyn OpStore> = Arc::new(SimpleOpStore::load(repo_path.join("op_store")));
|
2021-03-06 18:37:57 +00:00
|
|
|
let index_store = Arc::new(IndexStore::load(repo_path.join("index")));
|
2021-01-04 17:40:46 +00:00
|
|
|
Ok(RepoLoader {
|
|
|
|
wc_path,
|
|
|
|
repo_path,
|
|
|
|
repo_settings,
|
|
|
|
store,
|
|
|
|
op_store,
|
2021-03-03 05:43:13 +00:00
|
|
|
index_store,
|
2021-01-04 17:40:46 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: This probably belongs in StoreWrapper (once that type has a better
|
|
|
|
// name)
|
|
|
|
fn load_store(repo_path: &Path) -> Arc<StoreWrapper> {
|
|
|
|
let store_path = repo_path.join("store");
|
|
|
|
let store: Box<dyn Store>;
|
|
|
|
if store_path.is_dir() {
|
|
|
|
store = Box::new(LocalStore::load(store_path));
|
|
|
|
} else {
|
|
|
|
let mut store_file = File::open(store_path).unwrap();
|
|
|
|
let mut buf = Vec::new();
|
|
|
|
store_file.read_to_end(&mut buf).unwrap();
|
|
|
|
let contents = String::from_utf8(buf).unwrap();
|
|
|
|
assert!(contents.starts_with("git: "));
|
|
|
|
let git_store_path_str = contents[5..].to_string();
|
|
|
|
let git_store_path =
|
|
|
|
fs::canonicalize(repo_path.join(PathBuf::from(git_store_path_str))).unwrap();
|
|
|
|
store = Box::new(GitStore::load(&git_store_path));
|
|
|
|
}
|
|
|
|
StoreWrapper::new(store)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn op_store(&self) -> &Arc<dyn OpStore> {
|
|
|
|
&self.op_store
|
|
|
|
}
|
|
|
|
|
2021-01-04 17:40:46 +00:00
|
|
|
pub fn load_at_head(self) -> Result<Arc<ReadonlyRepo>, RepoLoadError> {
|
2021-01-04 17:40:46 +00:00
|
|
|
let view = ReadonlyView::load(
|
|
|
|
self.store.clone(),
|
|
|
|
self.op_store.clone(),
|
2021-03-06 18:37:57 +00:00
|
|
|
self.index_store.clone(),
|
2021-01-04 17:40:46 +00:00
|
|
|
self.repo_path.join("view"),
|
|
|
|
);
|
2021-01-04 17:40:46 +00:00
|
|
|
self._finish_load(view)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn load_at(self, op: &Operation) -> Result<Arc<ReadonlyRepo>, RepoLoadError> {
|
|
|
|
let view = ReadonlyView::load_at(
|
|
|
|
self.store.clone(),
|
|
|
|
self.op_store.clone(),
|
2021-03-06 18:37:57 +00:00
|
|
|
self.index_store.clone(),
|
2021-01-04 17:40:46 +00:00
|
|
|
self.repo_path.join("view"),
|
|
|
|
op,
|
|
|
|
);
|
|
|
|
self._finish_load(view)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn _finish_load(self, view: ReadonlyView) -> Result<Arc<ReadonlyRepo>, RepoLoadError> {
|
2021-01-04 17:40:46 +00:00
|
|
|
let working_copy = WorkingCopy::load(
|
|
|
|
self.store.clone(),
|
|
|
|
self.wc_path.clone(),
|
|
|
|
self.repo_path.join("working_copy"),
|
|
|
|
);
|
|
|
|
let repo = ReadonlyRepo {
|
|
|
|
repo_path: self.repo_path,
|
|
|
|
wc_path: self.wc_path,
|
|
|
|
store: self.store,
|
|
|
|
settings: self.repo_settings,
|
2021-03-03 05:43:13 +00:00
|
|
|
index_store: self.index_store,
|
2021-01-04 17:40:46 +00:00
|
|
|
index: Mutex::new(None),
|
|
|
|
working_copy: Arc::new(Mutex::new(working_copy)),
|
|
|
|
view,
|
|
|
|
evolution: None,
|
|
|
|
};
|
|
|
|
let mut repo = Arc::new(repo);
|
|
|
|
let repo_ref: &ReadonlyRepo = repo.as_ref();
|
|
|
|
let static_lifetime_repo: &'static ReadonlyRepo = unsafe { std::mem::transmute(repo_ref) };
|
|
|
|
let evolution = ReadonlyEvolution::new(static_lifetime_repo);
|
|
|
|
Arc::get_mut(&mut repo).unwrap().evolution = Some(evolution);
|
|
|
|
Ok(repo)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-01 02:00:14 +00:00
|
|
|
pub struct MutableRepo<'r> {
|
|
|
|
repo: &'r ReadonlyRepo,
|
2021-01-31 02:50:27 +00:00
|
|
|
index: Option<MutableIndex>,
|
2021-02-01 02:00:14 +00:00
|
|
|
view: Option<MutableView>,
|
|
|
|
evolution: Option<MutableEvolution<'static, 'static>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'r> MutableRepo<'r> {
|
|
|
|
pub fn new(
|
|
|
|
repo: &'r ReadonlyRepo,
|
2021-01-31 02:50:27 +00:00
|
|
|
index: Arc<ReadonlyIndex>,
|
2021-02-01 02:00:14 +00:00
|
|
|
view: &ReadonlyView,
|
|
|
|
evolution: &ReadonlyEvolution<'r>,
|
|
|
|
) -> Arc<MutableRepo<'r>> {
|
|
|
|
let mut_view = view.start_modification();
|
2021-01-31 02:50:27 +00:00
|
|
|
let mut_index = MutableIndex::incremental(index);
|
2021-02-01 02:00:14 +00:00
|
|
|
let mut mut_repo = Arc::new(MutableRepo {
|
|
|
|
repo,
|
2021-01-31 02:50:27 +00:00
|
|
|
index: Some(mut_index),
|
2021-02-01 02:00:14 +00:00
|
|
|
view: Some(mut_view),
|
|
|
|
evolution: None,
|
|
|
|
});
|
|
|
|
let repo_ref: &MutableRepo = mut_repo.as_ref();
|
|
|
|
let static_lifetime_repo: &'static MutableRepo = unsafe { std::mem::transmute(repo_ref) };
|
|
|
|
let mut_evolution: MutableEvolution<'_, '_> =
|
|
|
|
evolution.start_modification(static_lifetime_repo);
|
|
|
|
let static_lifetime_mut_evolution: MutableEvolution<'static, 'static> =
|
|
|
|
unsafe { std::mem::transmute(mut_evolution) };
|
|
|
|
Arc::get_mut(&mut mut_repo).unwrap().evolution = Some(static_lifetime_mut_evolution);
|
|
|
|
mut_repo
|
|
|
|
}
|
|
|
|
|
2021-01-31 07:44:31 +00:00
|
|
|
pub fn as_repo_ref(&self) -> RepoRef {
|
|
|
|
RepoRef::Mutable(&self)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn store(&self) -> &Arc<StoreWrapper> {
|
|
|
|
self.repo.store()
|
|
|
|
}
|
|
|
|
|
2021-01-31 02:50:27 +00:00
|
|
|
pub fn index(&self) -> &MutableIndex {
|
|
|
|
self.index.as_ref().unwrap()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn index_mut(&mut self) -> &mut MutableIndex {
|
|
|
|
self.index.as_mut().unwrap()
|
|
|
|
}
|
|
|
|
|
2021-02-01 02:00:14 +00:00
|
|
|
pub fn base_repo(&self) -> &'r ReadonlyRepo {
|
|
|
|
self.repo
|
|
|
|
}
|
|
|
|
|
2021-02-01 07:41:59 +00:00
|
|
|
pub fn view(&self) -> &MutableView {
|
2021-01-31 07:44:31 +00:00
|
|
|
self.view.as_ref().unwrap()
|
|
|
|
}
|
|
|
|
|
2021-02-01 02:00:14 +00:00
|
|
|
pub fn view_mut(&mut self) -> &mut MutableView {
|
|
|
|
self.view.as_mut().unwrap()
|
|
|
|
}
|
|
|
|
|
2021-02-08 01:59:30 +00:00
|
|
|
pub fn consume(mut self) -> (MutableIndex, MutableView) {
|
|
|
|
(self.index.take().unwrap(), self.view.take().unwrap())
|
2021-02-01 02:00:14 +00:00
|
|
|
}
|
|
|
|
|
2021-02-01 02:13:37 +00:00
|
|
|
pub fn evolution<'m>(&'m self) -> &MutableEvolution<'r, 'm> {
|
|
|
|
let evolution: &MutableEvolution<'static, 'static> = self.evolution.as_ref().unwrap();
|
|
|
|
let evolution: &MutableEvolution<'r, 'm> = unsafe { std::mem::transmute(evolution) };
|
|
|
|
evolution
|
2021-01-31 07:44:31 +00:00
|
|
|
}
|
|
|
|
|
2021-02-01 02:00:14 +00:00
|
|
|
pub fn evolution_mut<'m>(&'m mut self) -> &'m mut MutableEvolution<'r, 'm> {
|
|
|
|
let evolution: &mut MutableEvolution<'static, 'static> = self.evolution.as_mut().unwrap();
|
|
|
|
let evolution: &mut MutableEvolution<'r, 'm> = unsafe { std::mem::transmute(evolution) };
|
|
|
|
evolution
|
|
|
|
}
|
|
|
|
}
|