1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91
use crate::index::{CloneOptions, LAST_SEEN_REFNAME};
use crate::Index;
use git_repository as git;
use std::path::Path;
use std::sync::atomic::AtomicBool;
/// The error returned by various initialization methods.
#[derive(Debug, thiserror::Error)]
#[allow(missing_docs)]
pub enum Error {
#[error(transparent)]
PrepareClone(#[from] git::clone::Error),
#[error(transparent)]
Fetch(#[from] git::clone::fetch::Error),
#[error(transparent)]
Open(#[from] git::open::Error),
}
/// Initialization
impl Index {
/// Return a new `Index` instance from the given `path`, which should contain a bare clone of the `crates.io` index.
/// If the directory does not contain the repository or does not exist, it will be cloned from
/// the official location automatically (with complete history).
///
/// An error will occour if the repository exists and the remote URL does not match the given repository URL.
///
/// # Examples
///
/// ```no_run
/// use std::sync::atomic::AtomicBool;
/// use crates_index_diff::{Index, index, git};
///
/// # let path = tempdir::TempDir::new("index").unwrap();
/// // Note that credentials are automatically picked up from the standard git configuration.
/// let mut options = index::CloneOptions {
/// url: "https://github.com/rust-lang/staging.crates.io-index".into(),
/// };
///
///
/// let index = Index::from_path_or_cloned_with_options(path, git::progress::Discard, &AtomicBool::default(), options)?;
/// # Ok::<(), crates_index_diff::index::init::Error>(())
/// ```
pub fn from_path_or_cloned_with_options<P>(
path: impl AsRef<Path>,
progress: P,
should_interrupt: &AtomicBool,
CloneOptions { url }: CloneOptions,
) -> Result<Index, Error>
where
P: git::Progress,
P::SubProgress: 'static,
{
let path = path.as_ref();
let mut repo = match git::open(path) {
Ok(repo) => repo,
Err(git::open::Error::NotARepository(_)) => {
let (repo, _out) =
git::prepare_clone_bare(url, path)?.fetch_only(progress, should_interrupt)?;
repo
}
Err(err) => return Err(err.into()),
}
.apply_environment();
repo.object_cache_size_if_unset(4 * 1024 * 1024);
let remote_name = repo
.remote_names()
.into_iter()
.next()
.map(ToOwned::to_owned);
Ok(Index {
repo,
remote_name,
branch_name: "master",
seen_ref_name: LAST_SEEN_REFNAME,
})
}
/// Return a new `Index` instance from the given `path`, which should contain a bare or non-bare
/// clone of the `crates.io` index.
/// If the directory does not contain the repository or does not exist, it will be cloned from
/// the official location automatically (with complete history).
pub fn from_path_or_cloned(path: impl AsRef<Path>) -> Result<Index, Error> {
Index::from_path_or_cloned_with_options(
path,
git::progress::Discard,
&AtomicBool::default(),
CloneOptions::default(),
)
}
}