diff_edit: load command arguments from merge-tools.<name> config

Apparently, I need to pass `--merge` option to use kdiff3 as a diff editor.

We could add `diff-editor-args` or extend `diff-editor` to a list of command
arguments, but we'll eventually add stock merge tools and the configuration
would look like:

    [merge-tools.<name>]
    program = ...
    diff-args = [...]
    edit-args = [...]
    merge-args = [...]
This commit is contained in:
Yuya Nishihara 2022-05-01 10:45:22 +09:00
parent 1f2753fd3f
commit f6acee41df
4 changed files with 56 additions and 4 deletions

View file

@ -66,6 +66,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
will create a branch named after the revision's change ID, so you don't have will create a branch named after the revision's change ID, so you don't have
to create a branch yourself. to create a branch yourself.
* Diff editor command arguments can now be specified by config file.
Example:
[merge-tools.kdiff3]
program = "kdiff3"
edit-args = ["--merge", "--cs", "CreateBakFiles=0"]
### Fixed bugs ### Fixed bugs
* When rebasing a conflict where one side modified a file and the other side * When rebasing a conflict where one side modified a file and the other side

1
Cargo.lock generated
View file

@ -748,6 +748,7 @@ dependencies = [
"predicates", "predicates",
"rand", "rand",
"regex", "regex",
"serde",
"tempfile", "tempfile",
"test-case", "test-case",
"textwrap 0.15.0", "textwrap 0.15.0",

View file

@ -52,6 +52,7 @@ pest = "2.1.3"
pest_derive = "2.1.0" pest_derive = "2.1.0"
rand = "0.8.5" rand = "0.8.5"
regex = "1.5.5" regex = "1.5.5"
serde = { version = "1.0", features = ["derive"] }
tempfile = "3.3.0" tempfile = "3.3.0"
textwrap = "0.15.0" textwrap = "0.15.0"
thiserror = "1.0.31" thiserror = "1.0.31"

View file

@ -18,6 +18,7 @@ use std::path::{Path, PathBuf};
use std::process::Command; use std::process::Command;
use std::sync::Arc; use std::sync::Arc;
use config::ConfigError;
use itertools::Itertools; use itertools::Itertools;
use jujutsu_lib::backend::TreeId; use jujutsu_lib::backend::TreeId;
use jujutsu_lib::gitignore::GitIgnoreFile; use jujutsu_lib::gitignore::GitIgnoreFile;
@ -34,6 +35,8 @@ use crate::ui::Ui;
#[derive(Debug, Error)] #[derive(Debug, Error)]
pub enum DiffEditError { pub enum DiffEditError {
#[error("Invalid config: {0}")]
ConfigError(#[from] ConfigError),
#[error("The diff tool exited with a non-zero code")] #[error("The diff tool exited with a non-zero code")]
DifftoolAborted, DifftoolAborted,
#[error("Failed to write directories to diff: {0:?}")] #[error("Failed to write directories to diff: {0:?}")]
@ -138,7 +141,7 @@ pub fn edit_diff(
// TODO: Make this configuration have a table of possible editors and detect the // TODO: Make this configuration have a table of possible editors and detect the
// best one here. // best one here.
let editor_binary = match settings.config().get_string("ui.diff-editor") { let editor_name = match settings.config().get_string("ui.diff-editor") {
Ok(editor_binary) => editor_binary, Ok(editor_binary) => editor_binary,
Err(_) => { Err(_) => {
let default_editor = "meld".to_string(); let default_editor = "meld".to_string();
@ -150,14 +153,15 @@ pub fn edit_diff(
default_editor default_editor
} }
}; };
let editor = get_tool(settings, &editor_name)?;
// Start a diff editor on the two directories. // Start a diff editor on the two directories.
let exit_status = Command::new(&editor_binary) let exit_status = Command::new(&editor.program)
.args(&editor.edit_args)
.arg(&left_wc_dir) .arg(&left_wc_dir)
.arg(&right_wc_dir) .arg(&right_wc_dir)
.status() .status()
.map_err(|e| DiffEditError::ExecuteEditorError { .map_err(|e| DiffEditError::ExecuteEditorError {
editor_binary, editor_binary: editor.program,
source: e, source: e,
})?; })?;
if !exit_status.success() { if !exit_status.success() {
@ -169,3 +173,42 @@ pub fn edit_diff(
Ok(right_tree_state.snapshot(base_ignores)?) Ok(right_tree_state.snapshot(base_ignores)?)
} }
/// Merge/diff tool loaded from the settings.
#[derive(Clone, Debug, serde::Deserialize)]
#[serde(rename_all = "kebab-case")]
struct MergeTool {
/// Program to execute.
pub program: String,
/// Arguments to pass to the program when editing diffs.
#[serde(default)]
pub edit_args: Vec<String>,
}
impl MergeTool {
pub fn with_program(program: &str) -> Self {
MergeTool {
program: program.to_owned(),
edit_args: vec![],
}
}
}
/// Loads merge tool options from `[merge-tools.<name>]`. The given name is used
/// as an executable name if no configuration found for that name.
fn get_tool(settings: &UserSettings, name: &str) -> Result<MergeTool, ConfigError> {
const TABLE_KEY: &'static str = "merge-tools";
let tools_table = match settings.config().get_table(TABLE_KEY) {
Ok(table) => table,
Err(ConfigError::NotFound(_)) => return Ok(MergeTool::with_program(name)),
Err(err) => return Err(err),
};
if let Some(v) = tools_table.get(name) {
v.clone()
.try_deserialize()
// add config key, deserialize error is otherwise unclear
.map_err(|e| ConfigError::Message(format!("{TABLE_KEY}.{name}: {e}")))
} else {
Ok(MergeTool::with_program(name))
}
}