mirror of
https://github.com/martinvonz/jj.git
synced 2025-05-29 02:51:12 +00:00
138 lines
5.1 KiB
Rust
138 lines
5.1 KiB
Rust
// Copyright 2020 The Jujutsu Authors
|
|
//
|
|
// 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::fs;
|
|
use std::io::Write;
|
|
|
|
use clap::ArgGroup;
|
|
use jj_lib::file_util;
|
|
use jj_lib::git_backend::GitBackend;
|
|
use jj_lib::repo::Repo;
|
|
use jj_lib::workspace::Workspace;
|
|
use tracing::instrument;
|
|
|
|
use super::git;
|
|
use crate::cli_util::{
|
|
print_git_import_stats, user_error, user_error_with_hint, CommandError, CommandHelper,
|
|
};
|
|
use crate::ui::Ui;
|
|
|
|
/// Create a new repo in the given directory
|
|
///
|
|
/// If the given directory does not exist, it will be created. If no directory
|
|
/// is given, the current directory is used.
|
|
#[derive(clap::Args, Clone, Debug)]
|
|
#[command(group(ArgGroup::new("backend").args(&["git", "git_repo"])))]
|
|
pub(crate) struct InitArgs {
|
|
/// The destination directory
|
|
#[arg(default_value = ".", value_hint = clap::ValueHint::DirPath)]
|
|
destination: String,
|
|
/// Use the Git backend, creating a jj repo backed by a Git repo
|
|
#[arg(long)]
|
|
git: bool,
|
|
/// Path to a git repo the jj repo will be backed by
|
|
#[arg(long, value_hint = clap::ValueHint::DirPath)]
|
|
git_repo: Option<String>,
|
|
}
|
|
|
|
#[instrument(skip_all)]
|
|
pub(crate) fn cmd_init(
|
|
ui: &mut Ui,
|
|
command: &CommandHelper,
|
|
args: &InitArgs,
|
|
) -> Result<(), CommandError> {
|
|
if command.global_args().repository.is_some() {
|
|
return Err(user_error("'--repository' cannot be used with 'init'"));
|
|
}
|
|
let wc_path = command.cwd().join(&args.destination);
|
|
match fs::create_dir(&wc_path) {
|
|
Ok(()) => {}
|
|
Err(_) if wc_path.is_dir() => {}
|
|
Err(e) => return Err(user_error(format!("Failed to create workspace: {e}"))),
|
|
}
|
|
let wc_path = wc_path
|
|
.canonicalize()
|
|
.map_err(|e| user_error(format!("Failed to create workspace: {e}")))?; // raced?
|
|
|
|
if let Some(git_store_str) = &args.git_repo {
|
|
let mut git_store_path = command.cwd().join(git_store_str);
|
|
git_store_path = git_store_path
|
|
.canonicalize()
|
|
.map_err(|_| user_error(format!("{} doesn't exist", git_store_path.display())))?;
|
|
if !git_store_path.ends_with(".git") {
|
|
git_store_path.push(".git");
|
|
// Undo if .git doesn't exist - likely a bare repo.
|
|
if !git_store_path.exists() {
|
|
git_store_path.pop();
|
|
}
|
|
}
|
|
let (workspace, repo) =
|
|
Workspace::init_external_git(command.settings(), &wc_path, &git_store_path)?;
|
|
let git_repo = repo
|
|
.store()
|
|
.backend_impl()
|
|
.downcast_ref::<GitBackend>()
|
|
.unwrap()
|
|
.open_git_repo()?;
|
|
let mut workspace_command = command.for_loaded_repo(ui, workspace, repo)?;
|
|
git::maybe_add_gitignore(&workspace_command)?;
|
|
workspace_command.snapshot(ui)?;
|
|
if !workspace_command.working_copy_shared_with_git() {
|
|
let mut tx = workspace_command.start_transaction("import git refs");
|
|
let stats = jj_lib::git::import_some_refs(
|
|
tx.mut_repo(),
|
|
&git_repo,
|
|
&command.settings().git_settings(),
|
|
|ref_name| !jj_lib::git::is_reserved_git_remote_ref(ref_name),
|
|
)?;
|
|
print_git_import_stats(ui, &stats)?;
|
|
if let Some(git_head_id) = tx.mut_repo().view().git_head().as_normal().cloned() {
|
|
let git_head_commit = tx.mut_repo().store().get_commit(&git_head_id)?;
|
|
tx.check_out(&git_head_commit)?;
|
|
}
|
|
if tx.mut_repo().has_changes() {
|
|
tx.finish(ui)?;
|
|
}
|
|
}
|
|
} else if args.git {
|
|
Workspace::init_internal_git(command.settings(), &wc_path)?;
|
|
} else {
|
|
if !command.settings().allow_native_backend() {
|
|
return Err(user_error_with_hint(
|
|
"The native backend is disallowed by default.",
|
|
"Did you mean to pass `--git`?
|
|
Set `ui.allow-init-native` to allow initializing a repo with the native backend.",
|
|
));
|
|
}
|
|
Workspace::init_local(command.settings(), &wc_path)?;
|
|
};
|
|
let cwd = command.cwd().canonicalize().unwrap();
|
|
let relative_wc_path = file_util::relative_path(&cwd, &wc_path);
|
|
writeln!(
|
|
ui.stderr(),
|
|
"Initialized repo in \"{}\"",
|
|
relative_wc_path.display()
|
|
)?;
|
|
if args.git && wc_path.join(".git").exists() {
|
|
writeln!(ui.warning(), "Empty repo created.")?;
|
|
writeln!(
|
|
ui.hint(),
|
|
"Hint: To create a repo backed by the existing Git repo, run `jj init --git-repo={}` \
|
|
instead.",
|
|
relative_wc_path.display()
|
|
)?;
|
|
}
|
|
Ok(())
|
|
}
|