ok/jj
1
0
Fork 0
forked from mirrors/jj

cli: add op waypoint command to mark points in the op log to easily restore to at a later point

This commit is contained in:
Noah Mayr 2024-04-02 19:10:24 +02:00
parent 361b4ca425
commit cc5ad3aceb
3 changed files with 138 additions and 3 deletions

View file

@ -18,10 +18,10 @@ use std::slice;
use clap::Subcommand;
use itertools::Itertools as _;
use jj_lib::object_id::ObjectId;
use jj_lib::op_store::OperationId;
use jj_lib::op_store::{OpStoreResult, OperationId};
use jj_lib::op_walk;
use jj_lib::operation::Operation;
use jj_lib::repo::Repo;
use jj_lib::repo::{Repo, RepoLoader};
use crate::cli_util::{format_template, short_operation_hash, CommandHelper, LogContentFormat};
use crate::command_error::{user_error, user_error_with_hint, CommandError};
@ -39,6 +39,7 @@ pub enum OperationCommand {
Log(OperationLogArgs),
Undo(OperationUndoArgs),
Restore(OperationRestoreArgs),
Waypoint(OperationWaypointArgs),
}
/// Show the operation log
@ -70,6 +71,10 @@ pub struct OperationRestoreArgs {
/// state of the repo at that operation.
operation: String,
/// Restore using a waypoint instead of an operation id
#[arg(long, short)]
waypoint: bool,
/// What portions of the local state to restore (can be repeated)
///
/// This option is EXPERIMENTAL.
@ -113,6 +118,22 @@ pub struct OperationAbandonArgs {
operation: String,
}
/// Create a new operation that restores the repo to an earlier state
///
/// This restores the repo to the state at the specified operation, effectively
/// undoing all later operations. It does so by creating a new operation.
#[derive(clap::Args, Clone, Debug)]
pub struct OperationWaypointArgs {
/// The name for the waypoint
waypoint: String,
/// The operation to add a named waypoint to
///
/// Use `jj op log` to find an operation to undo.
#[arg(default_value = "@")]
operation: String,
}
#[derive(Copy, Clone, Debug, PartialEq, Eq, PartialOrd, Ord, clap::ValueEnum)]
enum UndoWhatToRestore {
/// The jj repo state and local branches
@ -291,7 +312,25 @@ fn cmd_op_restore(
args: &OperationRestoreArgs,
) -> Result<(), CommandError> {
let mut workspace_command = command.workspace_helper(ui)?;
let target_op = workspace_command.resolve_single_op(&args.operation)?;
let target_op = if args.waypoint {
let workspace = command.load_workspace()?;
let repo_loader = workspace.repo_loader();
let ops = find_tagged_ops(repo_loader, "waypoint", &args.operation)?;
let [op] = ops.as_slice() else {
return Err(if ops.is_empty() {
user_error("There no op log entries with this waypoint!")
} else {
user_error("There is more than one op log entry with this waypoint!").hinted(
"Try setting the waypoint again, this should clear all other waypoints with \
this name.",
)
});
};
op.to_owned()
} else {
workspace_command.resolve_single_op(&args.operation)?
};
let mut tx = workspace_command.start_transaction();
let new_view = view_with_desired_portions_restored(
target_op.view()?.store_view(),
@ -395,6 +434,56 @@ fn cmd_op_abandon(
Ok(())
}
pub fn cmd_op_waypoint(
ui: &mut Ui,
command: &CommandHelper,
args: &OperationWaypointArgs,
) -> Result<(), CommandError> {
let workspace = command.load_workspace()?;
let repo_loader = workspace.repo_loader();
let workspace_command = command.workspace_helper(ui)?;
let target_op = workspace_command.resolve_single_op(&args.operation)?;
let op_store = workspace_command.repo().op_store();
let tagged_ops = find_tagged_ops(repo_loader, "waypoint", &args.waypoint)?;
for op in tagged_ops {
op_store.update_tag(op.id(), "waypoint".to_string(), None)?;
}
op_store.update_tag(
target_op.id(),
"waypoint".to_string(),
Some(args.waypoint.clone()),
)?;
Ok(())
}
fn find_tagged_ops(
repo_loader: &RepoLoader,
tag: &str,
value: &str,
) -> OpStoreResult<Vec<Operation>> {
let current_op = op_walk::resolve_op_for_load(repo_loader, "@").ok();
let head_ops = if let Some(op) = current_op {
vec![op]
} else {
op_walk::get_current_head_ops(
repo_loader.op_store(),
repo_loader.op_heads_store().as_ref(),
)?
};
let ops = op_walk::walk_ancestors(&head_ops);
ops.filter_ok(|op| {
op.metadata()
.tags
.get(tag)
.map(|val| val == value)
.unwrap_or(false)
})
.collect()
}
pub fn cmd_operation(
ui: &mut Ui,
command: &CommandHelper,
@ -405,5 +494,6 @@ pub fn cmd_operation(
OperationCommand::Log(args) => cmd_op_log(ui, command, args),
OperationCommand::Restore(args) => cmd_op_restore(ui, command, args),
OperationCommand::Undo(args) => cmd_op_undo(ui, command, args),
OperationCommand::Waypoint(args) => cmd_op_waypoint(ui, command, args),
}
}

View file

@ -432,6 +432,13 @@ pub trait OpStore: Send + Sync + Debug {
fn write_operation(&self, contents: &Operation) -> OpStoreResult<OperationId>;
fn update_tag(
&self,
id: &OperationId,
name: String,
value: Option<String>,
) -> OpStoreResult<()>;
/// Resolves an unambiguous operation ID prefix.
fn resolve_operation_id_prefix(
&self,

View file

@ -181,6 +181,44 @@ impl OpStore for SimpleOpStore {
Ok(id)
}
fn update_tag(
&self,
id: &OperationId,
tag: String,
value: Option<String>,
) -> OpStoreResult<()> {
let operation = self.read_operation(id)?;
let temp_file =
NamedTempFile::new_in(&self.path).map_err(|err| io_to_write_error(err, "operation"))?;
let mut tags = operation.metadata.tags;
if let Some(value) = value {
tags.insert(tag, value);
} else {
tags.remove(&tag);
}
let operation = op_store::Operation {
metadata: op_store::OperationMetadata {
tags,
..operation.metadata
},
..operation
};
let proto = operation_to_proto(&operation);
temp_file
.as_file()
.write_all(&proto.encode_to_vec())
.map_err(|err| io_to_write_error(err, "operation"))?;
persist_content_addressed_temp_file(temp_file, self.operation_path(id))
.map_err(|err| io_to_write_error(err, "operation"))?;
Ok(())
}
fn resolve_operation_id_prefix(
&self,
prefix: &HexPrefix,