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 byteorder::WriteBytesExt;
use git_object as object;
use quick_error::quick_error;

quick_error! {
    #[derive(Debug)]
    pub enum Error {
        ParseIntegerError(msg: &'static str, number: Vec<u8>, err: btoi::ParseIntegerError) {
            display("{}: {:?}", msg, std::str::from_utf8(number))
            source(err)
        }
        InvalidHeader(msg: &'static str) {
            display("{}", msg)
        }
        ObjectHeader(err: object::Error) {
            display("Could not parse object kind")
            from()
            source(err)
        }
    }
}

pub fn decode(input: &[u8]) -> Result<(object::Kind, u64, usize), Error> {
    let header_end = input
        .iter()
        .position(|&b| b == 0)
        .ok_or_else(|| Error::InvalidHeader("Did not find 0 byte in header"))?;
    let header = &input[..header_end];
    let mut split = header.split(|&b| b == b' ');
    match (split.next(), split.next()) {
        (Some(kind), Some(size)) => Ok((
            object::Kind::from_bytes(kind)?,
            btoi::btoi(size).map_err(|e| {
                Error::ParseIntegerError("Object size in header could not be parsed", size.to_owned(), e)
            })?,
            header_end + 1, // account for 0 byte
        )),
        _ => Err(Error::InvalidHeader("Expected '<type> <size>'")),
    }
}

fn kind_to_bytes_with_space(object: object::Kind) -> &'static [u8] {
    use object::Kind::*;
    match object {
        Tree => b"tree ",
        Blob => b"blob ",
        Commit => b"commit ",
        Tag => b"tag ",
    }
}

pub fn encode(object: object::Kind, size: u64, mut out: impl std::io::Write) -> Result<usize, std::io::Error> {
    let mut written = out.write(kind_to_bytes_with_space(object))?;
    written += itoa::write(&mut out, size)?;
    out.write_u8(0)?;
    Ok(written + 1)
}

#[cfg(test)]
mod tests {
    mod encode_decode_round_trip {
        use crate::loose::object::header;
        use git_object::bstr::ByteSlice;

        #[test]
        fn all() -> Result<(), Box<dyn std::error::Error>> {
            let mut buf = [0; 20];
            for (kind, size, expected) in &[
                (git_object::Kind::Tree, 1234, &b"tree 1234\0"[..]),
                (git_object::Kind::Blob, 0, b"blob 0\0"),
                (git_object::Kind::Commit, 24241, b"commit 24241\0"),
                (git_object::Kind::Tag, 9999999999, b"tag 9999999999\0"),
            ] {
                let written = header::encode(*kind, *size, &mut buf[..])?;
                assert_eq!(buf[..written].as_bstr(), expected.as_bstr());
                let (actual_kind, actual_size, actual_read) = header::decode(&buf[..written])?;
                assert_eq!(actual_kind, *kind);
                assert_eq!(actual_size, *size);
                assert_eq!(actual_read, written);
            }
            Ok(())
        }
    }
}