Skip to content

Commit

Permalink
remove feature flag and enable CLI --shell arg and config option to a…
Browse files Browse the repository at this point in the history
…ctivate shelling out to git
  • Loading branch information
bsdinis committed Jan 2, 2025
1 parent 36e60bb commit 807bc10
Show file tree
Hide file tree
Showing 15 changed files with 675 additions and 50 deletions.
26 changes: 26 additions & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@ crossterm = { version = "0.27", default-features = false }
digest = "0.10.7"
dirs = "5.0.1"
dunce = "1.0.5"
duplicate = "2.0.0"
either = "1.13.0"
futures = "0.3.31"
git2 = { version = "0.19.0", features = [
Expand Down
2 changes: 1 addition & 1 deletion cli/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -101,6 +101,7 @@ anyhow = { workspace = true }
assert_cmd = { workspace = true }
assert_matches = { workspace = true }
async-trait = { workspace = true }
duplicate = { workspace = true }
insta = { workspace = true }
test-case = { workspace = true }
testutils = { workspace = true }
Expand All @@ -115,7 +116,6 @@ packaging = ["gix-max-performance"]
test-fakes = ["jj-lib/testing"]
vendored-openssl = ["git2/vendored-openssl", "jj-lib/vendored-openssl"]
watchman = ["jj-lib/watchman"]
shell = []

[package.metadata.binstall]
# The archive name is jj, not jj-cli. Also, `cargo binstall` gets
Expand Down
10 changes: 9 additions & 1 deletion cli/src/commands/git/clone.rs
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ use crate::command_error::user_error;
use crate::command_error::user_error_with_message;
use crate::command_error::CommandError;
use crate::commands::git::maybe_add_gitignore;
use crate::git_util::get_config_git_shell;
use crate::git_util::get_git_repo;
use crate::git_util::map_git_error;
use crate::git_util::print_git_import_stats;
Expand Down Expand Up @@ -63,6 +64,9 @@ pub struct GitCloneArgs {
/// Create a shallow clone of the given depth
#[arg(long)]
depth: Option<NonZeroU32>,
/// Run git clone as a forked git process
#[arg(long)]
shell: bool,
}

fn absolute_git_source(cwd: &Path, source: &str) -> Result<String, CommandError> {
Expand Down Expand Up @@ -129,6 +133,7 @@ pub fn cmd_git_clone(
// `/some/path/.`
let canonical_wc_path = dunce::canonicalize(&wc_path)
.map_err(|err| user_error_with_message(format!("Failed to create {wc_path_str}"), err))?;
let shell = args.shell || get_config_git_shell(command)?;
let clone_result = do_git_clone(
ui,
command,
Expand All @@ -137,6 +142,7 @@ pub fn cmd_git_clone(
remote_name,
&source,
&canonical_wc_path,
shell,
);
if clone_result.is_err() {
let clean_up_dirs = || -> io::Result<()> {
Expand Down Expand Up @@ -196,6 +202,7 @@ fn do_git_clone(
remote_name: &str,
source: &str,
wc_path: &Path,
shell: bool,
) -> Result<(WorkspaceCommandHelper, GitFetchStats), CommandError> {
let (workspace, repo) = if colocate {
Workspace::init_colocated_git(command.settings(), wc_path)?
Expand All @@ -214,7 +221,7 @@ fn do_git_clone(
let mut fetch_tx = workspace_command.start_transaction();
let git_settings = command.settings().git_settings()?;

let stats = with_remote_git_callbacks(ui, None, |cb| {
let stats = with_remote_git_callbacks(ui, None, shell, |cb| {
git::fetch(
fetch_tx.repo_mut(),
&git_repo,
Expand All @@ -223,6 +230,7 @@ fn do_git_clone(
cb,
&git_settings,
depth,
shell,
)
})
.map_err(|err| match err {
Expand Down
7 changes: 6 additions & 1 deletion cli/src/commands/git/fetch.rs
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ use crate::cli_util::CommandHelper;
use crate::command_error::CommandError;
use crate::commands::git::get_single_remote;
use crate::complete;
use crate::git_util::get_config_git_shell;
use crate::git_util::get_git_repo;
use crate::git_util::git_fetch;
use crate::ui::Ui;
Expand Down Expand Up @@ -60,6 +61,9 @@ pub struct GitFetchArgs {
/// Fetch from all remotes
#[arg(long, conflicts_with = "remotes")]
all_remotes: bool,
/// Run git fetch as a forked git process
#[arg(long)]
shell: bool,
}

#[tracing::instrument(skip(ui, command))]
Expand All @@ -69,6 +73,7 @@ pub fn cmd_git_fetch(
args: &GitFetchArgs,
) -> Result<(), CommandError> {
let mut workspace_command = command.workspace_helper(ui)?;
let shell_flag = args.shell || get_config_git_shell(command)?;
let git_repo = get_git_repo(workspace_command.repo().store())?;
let remotes = if args.all_remotes {
get_all_remotes(&git_repo)?
Expand All @@ -78,7 +83,7 @@ pub fn cmd_git_fetch(
args.remotes.clone()
};
let mut tx = workspace_command.start_transaction();
git_fetch(ui, &mut tx, &git_repo, &remotes, &args.branch)?;
git_fetch(ui, &mut tx, &git_repo, &remotes, &args.branch, shell_flag)?;
tx.finish(
ui,
format!("fetch from git remote(s) {}", remotes.iter().join(",")),
Expand Down
15 changes: 12 additions & 3 deletions cli/src/commands/git/push.rs
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@ use crate::command_error::CommandError;
use crate::commands::git::get_single_remote;
use crate::complete;
use crate::formatter::Formatter;
use crate::git_util::get_config_git_shell;
use crate::git_util::get_git_repo;
use crate::git_util::map_git_error;
use crate::git_util::with_remote_git_callbacks;
Expand Down Expand Up @@ -142,6 +143,9 @@ pub struct GitPushArgs {
/// Only display what will change on the remote
#[arg(long)]
dry_run: bool,
/// Run git push as a forked git process
#[arg(long)]
shell: bool,
}

fn make_bookmark_term(bookmark_names: &[impl fmt::Display]) -> String {
Expand All @@ -166,6 +170,7 @@ pub fn cmd_git_push(
args: &GitPushArgs,
) -> Result<(), CommandError> {
let mut workspace_command = command.workspace_helper(ui)?;
let shell_flag = args.shell || get_config_git_shell(command)?;
let git_repo = get_git_repo(workspace_command.repo().store())?;

let remote = if let Some(name) = &args.remote {
Expand Down Expand Up @@ -313,9 +318,13 @@ pub fn cmd_git_push(
let mut sideband_progress_callback = |progress_message: &[u8]| {
_ = writer.write(ui, progress_message);
};
with_remote_git_callbacks(ui, Some(&mut sideband_progress_callback), |cb| {
git::push_branches(tx.repo_mut(), &git_repo, &remote, &targets, cb)
})

with_remote_git_callbacks(
ui,
Some(&mut sideband_progress_callback),
shell_flag,
|cb| git::push_branches(tx.repo_mut(), &git_repo, &remote, &targets, cb, shell_flag),
)
.map_err(|err| match err {
GitPushError::InternalGitError(err) => map_git_error(err),
GitPushError::RefInUnexpectedLocation(refs) => user_error_with_hint(
Expand Down
31 changes: 25 additions & 6 deletions cli/src/git_util.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ use std::process::Stdio;
use std::time::Instant;

use itertools::Itertools;
use jj_lib::config::ConfigGetError;
use jj_lib::git;
use jj_lib::git::FailedRefExport;
use jj_lib::git::FailedRefExportReason;
Expand All @@ -40,6 +41,7 @@ use jj_lib::str_util::StringPattern;
use jj_lib::workspace::Workspace;
use unicode_width::UnicodeWidthStr;

use crate::cli_util::CommandHelper;
use crate::cli_util::WorkspaceCommandTransaction;
use crate::command_error::user_error;
use crate::command_error::user_error_with_hint;
Expand Down Expand Up @@ -257,11 +259,14 @@ type SidebandProgressCallback<'a> = &'a mut dyn FnMut(&[u8]);
pub fn with_remote_git_callbacks<T>(
ui: &Ui,
sideband_progress_callback: Option<SidebandProgressCallback<'_>>,
shell: bool,
f: impl FnOnce(git::RemoteCallbacks<'_>) -> T,
) -> T {
let mut callbacks = git::RemoteCallbacks::default();
let mut progress_callback = None;
if !cfg!(feature = "shell") {
if shell {
f(git::RemoteCallbacks::default())
} else {
let mut callbacks = git::RemoteCallbacks::default();
let mut progress_callback = None;
if let Some(mut output) = ui.progress_output() {
let mut progress = Progress::new(Instant::now());
progress_callback = Some(move |x: &git::Progress| {
Expand All @@ -282,8 +287,6 @@ pub fn with_remote_git_callbacks<T>(
|url: &str| Some((terminal_get_username(ui, url)?, terminal_get_pw(ui, url)?));
callbacks.get_username_password = Some(&mut get_user_pw);
f(callbacks)
} else {
f(callbacks)
}
}

Expand Down Expand Up @@ -467,11 +470,12 @@ pub fn git_fetch(
git_repo: &git2::Repository,
remotes: &[String],
branch: &[StringPattern],
shell: bool,
) -> Result<(), CommandError> {
let git_settings = tx.settings().git_settings()?;

for remote in remotes {
let stats = with_remote_git_callbacks(ui, None, |cb| {
let stats = with_remote_git_callbacks(ui, None, shell, |cb| {
git::fetch(
tx.repo_mut(),
git_repo,
Expand All @@ -480,6 +484,7 @@ pub fn git_fetch(
cb,
&git_settings,
None,
shell,
)
})
.map_err(|err| match err {
Expand Down Expand Up @@ -540,3 +545,17 @@ fn warn_if_branches_not_found(

Ok(())
}

pub(crate) fn get_config_git_shell(command: &CommandHelper) -> Result<bool, ConfigGetError> {
command
.config_env()
.resolve_config(command.raw_config())?
.get::<bool>("git.shell")
.or_else(|e| {
if matches!(e, ConfigGetError::NotFound { .. }) {
Ok(false)
} else {
Err(e)
}
})
}
4 changes: 4 additions & 0 deletions cli/tests/[email protected]
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
---
source: cli/tests/test_generate_md_cli_help.rs
description: "AUTO-GENERATED FILE, DO NOT EDIT. This cli reference is generated by a test as an `insta` snapshot. MkDocs includes this snapshot from docs/cli-reference.md."
snapshot_kind: text
---
<!-- BEGIN MARKDOWN-->

Expand Down Expand Up @@ -1073,6 +1074,7 @@ The Git repo will be a bare git repo stored inside the `.jj/` directory.
Default value: `origin`
* `--colocate` — Whether or not to colocate the Jujutsu repo with the git repo
* `--depth <DEPTH>` — Create a shallow clone of the given depth
* `--shell` — Run git clone as a forked git process
Expand Down Expand Up @@ -1103,6 +1105,7 @@ If a working-copy commit gets abandoned, it will be given a new, empty commit. T
This defaults to the `git.fetch` setting. If that is not configured, and if there are multiple remotes, the remote named "origin" will be used.
* `--all-remotes` — Fetch from all remotes
* `--shell` — Run git fetch as a forked git process
Expand Down Expand Up @@ -1186,6 +1189,7 @@ Before the command actually moves, creates, or deletes a remote bookmark, it mak
The created bookmark will be tracked automatically. Use the `git.push-bookmark-prefix` setting to change the prefix for generated names.
* `--dry-run` — Only display what will change on the remote
* `--shell` — Run git push as a forked git process
Expand Down
Loading

0 comments on commit 807bc10

Please sign in to comment.