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
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
mod file_objects;
mod git_objects;
mod http_objects;
mod cached_objects;
pub use self::cached_objects::CachedObjects;
pub use self::file_objects::FileObjects;
pub use self::git_objects::GitObjects;
pub use self::http_objects::HttpObjects;
use checksum::Checksum;
use core::Object;
use errors::*;
use git;
use std::io::Read;
use std::path::{Path, PathBuf};
use std::rc::Rc;
use std::time::Duration;
use tokio_core::reactor::Core;
use update::Update;
use url::Url;
pub struct ObjectsConfig {
pub repo_dir: PathBuf,
pub cache_dir: Option<PathBuf>,
pub missing_cache_time: Option<Duration>,
}
pub trait Objects {
fn put_object(&mut self, checksum: &Checksum, source: &mut Read, force: bool) -> Result<()>;
fn get_object(&mut self, checksum: &Checksum) -> Result<Option<Box<Object>>>;
fn update(&self) -> Result<Vec<Update>> {
Ok(vec![])
}
}
pub struct NoObjects;
impl Objects for NoObjects {
fn put_object(&mut self, _: &Checksum, _: &mut Read, _: bool) -> Result<()> {
Err(ErrorKind::EmptyObjects.into())
}
fn get_object(&mut self, _: &Checksum) -> Result<Option<Box<Object>>> {
Err(ErrorKind::EmptyObjects.into())
}
}
pub fn objects_from_path<P: AsRef<Path>>(path: P) -> Result<FileObjects> {
let path = path.as_ref();
if !path.is_dir() {
return Err(format!("no such directory: {}", path.display()).into());
}
Ok(FileObjects::new(path))
}
pub fn objects_from_git<'a, I>(
config: ObjectsConfig,
scheme: I,
url: &'a Url,
) -> Result<Box<Objects>>
where
I: IntoIterator<Item = &'a str>,
{
let mut scheme = scheme.into_iter();
let sub_scheme = scheme.next().ok_or_else(|| {
format!("bad scheme ({}), expected git+scheme", url.scheme())
})?;
let git_repo = git::setup_git_repo(&config.repo_dir, sub_scheme, url)?;
let file_objects = FileObjects::new(git_repo.path());
let git_repo = Rc::new(git_repo);
let objects = GitObjects::new(url.clone(), git_repo, file_objects);
Ok(Box::new(objects))
}
pub fn objects_from_http(config: ObjectsConfig, url: &Url) -> Result<Box<Objects>> {
let core = Core::new()?;
let http_objects = HttpObjects::new(url.clone(), core);
if let Some(cache_dir) = config.cache_dir {
let missing_cache_time = config.missing_cache_time.unwrap_or_else(
|| Duration::new(60, 0),
);
return Ok(Box::new(CachedObjects::new(
cache_dir,
missing_cache_time,
http_objects,
)));
}
Ok(Box::new(http_objects))
}
pub fn objects_from_url(config: ObjectsConfig, url: &Url) -> Result<Box<Objects>> {
let mut scheme = url.scheme().split("+");
let first = scheme.next().ok_or_else(|| format!("bad scheme: {}", url))?;
match first {
"file" => objects_from_path(Path::new(url.path())).map(|o| Box::new(o) as Box<Objects>),
"git" => objects_from_git(config, scheme, url),
"http" => objects_from_http(config, url),
scheme => Err(format!("bad scheme: {}", scheme).into()),
}.chain_err(|| format!("load objects from url: {}", url))
}