op_store: auto-upgrade existing repos from Thrift to Protobuf

With this patch, we auto-upgrade existing repos that use Thrift format
for the operation log to use Protobuf format. That would only be repos
used with an unreleased version of jj after 0.5.1 (which may be the
majority of repos?).

The upgrade from Thrift is simpler because we now use the same hashing
scheme for the Protobuf-based storage, so the operation and view IDs
remain the same as they were in the Thrift-based storage. We could
simplify the code a bit more as a result, but since this code is
supposed to be short-lived, I didn't bother.

Since the change from the Protobuf format with the old hashing scheme
to a the (same) Protobuf format with the new hashing scheme shouldn't
impact users, I removed the entry we had in the changelog about the
format change.
This commit is contained in:
Martin von Zweigbergk 2022-12-02 12:29:25 -08:00 committed by Martin von Zweigbergk
parent 5a9d1e5fdd
commit c269b72fb3
4 changed files with 22 additions and 67 deletions

View file

@ -25,10 +25,6 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
description, even if there already was a description set. It now also only description, even if there already was a description set. It now also only
works on the working-copy commit (there's no `-r` argument). works on the working-copy commit (there's no `-r` argument).
* The storage format for the operation log has changed. It will be
automatically upgraded the first time you run a command in an existing repo.
The operation IDs will change in that process.
### New features ### New features
* Commands with long output are paginated. * Commands with long output are paginated.

View file

@ -14,7 +14,6 @@
use std::collections::BTreeMap; use std::collections::BTreeMap;
use std::fmt::Debug; use std::fmt::Debug;
use std::fs;
use std::fs::File; use std::fs::File;
use std::io::{ErrorKind, Read, Write}; use std::io::{ErrorKind, Read, Write};
use std::path::PathBuf; use std::path::PathBuf;
@ -52,12 +51,6 @@ pub struct ThriftOpStore {
} }
impl ThriftOpStore { impl ThriftOpStore {
pub fn init(store_path: PathBuf) -> Self {
fs::create_dir(store_path.join("views")).unwrap();
fs::create_dir(store_path.join("operations")).unwrap();
Self::load(store_path)
}
pub fn load(store_path: PathBuf) -> Self { pub fn load(store_path: PathBuf) -> Self {
ThriftOpStore { path: store_path } ThriftOpStore { path: store_path }
} }

View file

@ -14,6 +14,7 @@
use std::collections::BTreeMap; use std::collections::BTreeMap;
use std::fmt::Debug; use std::fmt::Debug;
use std::fs;
use std::fs::File; use std::fs::File;
use std::io::ErrorKind; use std::io::ErrorKind;
use std::path::PathBuf; use std::path::PathBuf;
@ -42,6 +43,12 @@ pub struct ProtoOpStore {
} }
impl ProtoOpStore { impl ProtoOpStore {
pub fn init(store_path: PathBuf) -> Self {
fs::create_dir(store_path.join("views")).unwrap();
fs::create_dir(store_path.join("operations")).unwrap();
ProtoOpStore { path: store_path }
}
pub fn load(store_path: PathBuf) -> Self { pub fn load(store_path: PathBuf) -> Self {
ProtoOpStore { path: store_path } ProtoOpStore { path: store_path }
} }

View file

@ -36,25 +36,23 @@ impl From<PersistError> for OpStoreError {
} }
} }
// TODO: In version 0.7.0 or so, inline ThriftOpStore into this type and drop // TODO: In version 0.7.0 or so, inline ProtoOpStore into this type and drop
// support for upgrading from the proto format // support for upgrading from the thrift format
#[derive(Debug)] #[derive(Debug)]
pub struct SimpleOpStore { pub struct SimpleOpStore {
delegate: ThriftOpStore, delegate: ProtoOpStore,
} }
fn upgrade_to_thrift(store_path: PathBuf) -> std::io::Result<()> { fn upgrade_from_thrift(store_path: PathBuf) -> std::io::Result<()> {
println!("Upgrading operation log to Thrift format..."); println!("Upgrading operation log to Protobuf format...");
let old_store = ProtoOpStore::load(store_path.clone()); let old_store = ThriftOpStore::load(store_path.clone());
let tmp_store_dir = tempfile::Builder::new() let tmp_store_dir = tempfile::Builder::new()
.prefix("jj-op-store-upgrade-") .prefix("jj-op-store-upgrade-")
.tempdir_in(store_path.parent().unwrap()) .tempdir_in(store_path.parent().unwrap())
.unwrap(); .unwrap();
let tmp_store_path = tmp_store_dir.path().to_path_buf(); let tmp_store_path = tmp_store_dir.path().to_path_buf();
// Find the current operation head(s) of the operation log. Because the hash is // Find the current operation head(s) of the operation log
// based on the serialized format, it will be different after conversion, so
// we need to rewrite these later.
let op_heads_store_path = store_path.parent().unwrap().join("op_heads"); let op_heads_store_path = store_path.parent().unwrap().join("op_heads");
let mut old_op_heads = HashSet::new(); let mut old_op_heads = HashSet::new();
for entry in fs::read_dir(&op_heads_store_path)? { for entry in fs::read_dir(&op_heads_store_path)? {
@ -66,7 +64,7 @@ fn upgrade_to_thrift(store_path: PathBuf) -> std::io::Result<()> {
} }
// Do a DFS to rewrite the operations // Do a DFS to rewrite the operations
let new_store = ThriftOpStore::init(tmp_store_path.clone()); let new_store = ProtoOpStore::init(tmp_store_path.clone());
let mut converted: HashMap<OperationId, OperationId> = HashMap::new(); let mut converted: HashMap<OperationId, OperationId> = HashMap::new();
// The DFS stack // The DFS stack
let mut to_convert = old_op_heads let mut to_convert = old_op_heads
@ -102,67 +100,28 @@ fn upgrade_to_thrift(store_path: PathBuf) -> std::io::Result<()> {
} }
} }
fs::write(tmp_store_path.join("thrift_store"), "")?;
let backup_store_path = store_path.parent().unwrap().join("op_store_old"); let backup_store_path = store_path.parent().unwrap().join("op_store_old");
// Delete existing backup (probably from an earlier upgrade to Thrift)
fs::remove_dir_all(&backup_store_path).ok();
fs::rename(&store_path, backup_store_path)?; fs::rename(&store_path, backup_store_path)?;
fs::rename(&tmp_store_path, &store_path)?; fs::rename(&tmp_store_path, &store_path)?;
// Update the pointers to the head(s) of the operation log
for old_op_head in old_op_heads {
let new_op_head = converted.get(&old_op_head).unwrap().clone();
fs::write(op_heads_store_path.join(new_op_head.hex()), "")?;
fs::remove_file(op_heads_store_path.join(old_op_head.hex()))?;
}
// Update the pointers from operations to index files
let index_operations_path = store_path
.parent()
.unwrap()
.join("index")
.join("operations");
for entry in fs::read_dir(&index_operations_path)? {
let basename = entry?.file_name();
let op_id_str = basename.to_str().unwrap();
if let Ok(op_id_bytes) = hex::decode(op_id_str) {
let old_op_id = OperationId::new(op_id_bytes);
// This should always succeed, but just skip it if it doesn't. We'll index
// the commits on demand if we don't have an pointer to an index file.
if let Some(new_op_id) = converted.get(&old_op_id) {
fs::rename(
index_operations_path.join(basename),
index_operations_path.join(new_op_id.hex()),
)?;
}
}
}
// Update the pointer to the last operation exported to Git
let git_export_path = store_path.parent().unwrap().join("git_export_operation_id");
if let Ok(op_id_string) = fs::read_to_string(&git_export_path) {
if let Ok(op_id_bytes) = hex::decode(op_id_string) {
let old_op_id = OperationId::new(op_id_bytes);
let new_op_id = converted.get(&old_op_id).unwrap();
fs::write(&git_export_path, new_op_id.hex())?;
}
}
println!("Upgrade complete"); println!("Upgrade complete");
Ok(()) Ok(())
} }
impl SimpleOpStore { impl SimpleOpStore {
pub fn init(store_path: PathBuf) -> Self { pub fn init(store_path: PathBuf) -> Self {
fs::write(store_path.join("thrift_store"), "").unwrap(); let delegate = ProtoOpStore::init(store_path);
let delegate = ThriftOpStore::init(store_path);
SimpleOpStore { delegate } SimpleOpStore { delegate }
} }
pub fn load(store_path: PathBuf) -> Self { pub fn load(store_path: PathBuf) -> Self {
if !store_path.join("thrift_store").exists() { if store_path.join("thrift_store").exists() {
upgrade_to_thrift(store_path.clone()) upgrade_from_thrift(store_path.clone())
.expect("Failed to upgrade operation log to Thrift format"); .expect("Failed to upgrade operation log to Protobuf format");
} }
let delegate = ThriftOpStore::load(store_path); let delegate = ProtoOpStore::load(store_path);
SimpleOpStore { delegate } SimpleOpStore { delegate }
} }
} }