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
use std::path::{Path, PathBuf};
use std::fs::{metadata, create_dir_all, File};
use std::io::{Write, BufReader};
use base64::{encode_config, URL_SAFE_NO_PAD};
use std;
use backend::HashRef;
use patch::{Patch, PatchHeader};
use Result;
use flate2;
use rand::Rng;
pub const PIJUL_DIR_NAME: &'static str = ".pijul";
pub fn repo_dir<P: AsRef<Path>>(p: P) -> PathBuf {
p.as_ref().join(PIJUL_DIR_NAME)
}
pub fn pristine_dir<P: AsRef<Path>>(p: P) -> PathBuf {
return p.as_ref().join(PIJUL_DIR_NAME).join("pristine");
}
pub const PATCHES_DIR_NAME: &'static str = "patches";
pub fn patches_dir<P: AsRef<Path>>(p: P) -> PathBuf {
return p.as_ref().join(PIJUL_DIR_NAME).join(PATCHES_DIR_NAME);
}
pub fn branch_changes_base_path(b: &str) -> String {
"changes.".to_string() + &encode_config(b.as_bytes(), URL_SAFE_NO_PAD)
}
pub fn branch_changes_file(p: &Path, b: &str) -> PathBuf {
p.join(PIJUL_DIR_NAME).join(branch_changes_base_path(b))
}
pub fn meta_file(p: &Path) -> PathBuf {
p.join(PIJUL_DIR_NAME).join("meta.toml")
}
pub fn id_file(p: &Path) -> PathBuf {
p.join(PIJUL_DIR_NAME).join("id")
}
pub fn find_repo_root<'a>(dir: &'a Path) -> Option<PathBuf> {
let mut p = dir.to_path_buf();
loop {
p.push(PIJUL_DIR_NAME);
match metadata(&p) {
Ok(ref attr) if attr.is_dir() => {
p.pop();
return Some(p);
}
_ => {}
}
p.pop();
if !p.pop() {
return None
}
}
}
pub const ID_LENGTH: usize = 100;
pub fn create<R:Rng>(dir: &Path, mut rng: R) -> std::io::Result<()> {
debug!("create: {:?}", dir);
let mut repo_dir = repo_dir(dir);
try!(create_dir_all(&repo_dir));
repo_dir.push("pristine");
try!(create_dir_all(&repo_dir));
repo_dir.pop();
repo_dir.push("patches");
try!(create_dir_all(&repo_dir));
repo_dir.pop();
repo_dir.push("id");
let mut f = std::fs::File::create(&repo_dir)?;
let mut x = String::new();
x.extend(rng.gen_ascii_chars().take(ID_LENGTH));
f.write_all(x.as_bytes())?;
repo_dir.pop();
repo_dir.push("version");
let mut f = std::fs::File::create(&repo_dir)?;
writeln!(f, "{}", env!("CARGO_PKG_VERSION"))?;
repo_dir.pop();
Ok(())
}
pub fn patch_file_name(hash: HashRef) -> String {
hash.to_base64(URL_SAFE_NO_PAD) + ".gz"
}
pub fn read_patch(repo: &Path, hash: HashRef) -> Result<Patch> {
let patch_dir = patches_dir(repo);
let path = patch_dir.join(&patch_file_name(hash));
debug!("read_patch, reading from {:?}", path);
let f = File::open(path)?;
let mut f = BufReader::new(f);
let (_, _, patch) = Patch::from_reader_compressed(&mut f)?;
Ok(patch)
}
pub fn read_patch_nochanges(repo: &Path, hash: HashRef) -> Result<PatchHeader> {
let patch_dir = patches_dir(repo);
let path = patch_dir.join(&patch_file_name(hash));
debug!("read_patch_nochanges, reading from {:?}", path);
let f = File::open(path)?;
let mut f = flate2::bufread::GzDecoder::new(BufReader::new(f)).unwrap();
Ok(PatchHeader::from_reader_nochanges(&mut f)?)
}