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
//! Layout of a repository (files in `.pijul`) on the disk. This
//! module exports both high-level functions that require no knowledge
//! of the repository, and lower-level constants documented on
//! [pijul.org/documentation/repository](https://pijul.org/documentation/repository),
//! used for instance for downloading files from remote repositories.

use std::path::{Path, PathBuf};
use std::fs::{metadata, create_dir_all, File};
use std::io::BufReader;
use rustc_serialize::base64::{ToBase64, URL_SAFE};
use std;
use backend::HashRef;
use patch::{Patch};
use error::Error;

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() + &b.as_bytes().to_base64(URL_SAFE)
}

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 find_repo_root<'a>(dir: &'a Path) -> Option<PathBuf> {
    let c: Vec<&std::ffi::OsStr> = dir.iter().collect();
    let mut i = c.len();
    while i > 0 {
        let mut p = PathBuf::new();
        for j in 0..i {
            p.push(c[j])
        }
        p.push(PIJUL_DIR_NAME);
        debug!("trying {:?}", p);
        match metadata(&p) {
            Ok(ref attr) if attr.is_dir() => {
                p.pop();
                return Some(p);
            }
            _ => {}
        }
        p.pop();
        i -= 1;
    }
    None
}

pub fn create(dir: &Path) -> std::io::Result<()> {
    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));
    Ok(())
}


pub fn patch_file_name(hash: HashRef) -> String {
    hash.to_base64(URL_SAFE) + ".cbor.gz"
}

pub fn read_patch(repo: &Path, hash: HashRef) -> Result<Patch, Error> {
    let patch_dir = patches_dir(repo);
    let path = patch_dir.join(&patch_file_name(hash));
    let f = File::open(path)?;
    let mut f = BufReader::new(f);
    let (_, _, patch) = Patch::from_reader_compressed(&mut f)?;
    Ok(patch)
}