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
|
use thiserror;
use std::path::Path;
#[derive(Debug, thiserror::Error)]
pub enum Error {
#[error("git error")]
Git(#[from] git2::Error),
}
/// Mirror a repository.
///
/// Works like:
///
/// ```shell
/// git clone --mirror URL
/// ```
pub fn mirror<P: AsRef<Path>>(
url: &str,
path: P,
description: Option<&str>,
) -> Result<(), Error> {
let description_str = match description {
Some(d) => d,
None => "",
};
let repo = git2::Repository::init_opts(
path,
&git2::RepositoryInitOptions::new()
.bare(true)
.description(description_str),
)?;
let remote_name = "origin";
let mut remote = repo.remote_with_fetch(
remote_name,
url,
"+refs/*:refs/*",
)?;
let mut config = repo.config()?;
config.set_bool(
&format!("remote.{}.mirror", remote_name),
true,
)?;
let refspecs: [&str; 0] = [];
remote.fetch(&refspecs, None, None)?;
Ok(())
}
/// Update remotes.
///
/// Works like:
///
/// ```shell
/// git remote update
/// ```
pub fn update<P: AsRef<Path>>(
path: P,
) -> Result<(), Error> {
let repo = git2::Repository::open_bare(path)?;
for remote_opt in &repo.remotes()? {
if let Some(remote_name) = remote_opt {
let mut remote = repo.find_remote(remote_name)?;
let mut fetch_options = git2::FetchOptions::new();
fetch_options
.prune(git2::FetchPrune::On)
.download_tags(git2::AutotagOption::All);
let refspecs: [&str; 0] = [];
remote.fetch(&refspecs, Some(&mut fetch_options), None)?;
}
}
Ok(())
}
|