parsers.rs
435 lines
| 13.7 KiB
| application/rls-services+xml
|
RustLexer
Raphaël Gomès
|
r42617 | // Copyright 2019 Raphaël Gomès <rgomes@octobus.net> | ||
// | ||||
// This software may be used and distributed according to the terms of the | ||||
// GNU General Public License version 2 or any later version. | ||||
Raphaël Gomès
|
r43227 | use crate::utils::hg_path::HgPath; | ||
Raphaël Gomès
|
r42828 | use crate::{ | ||
Raphaël Gomès
|
r42994 | dirstate::{CopyMap, EntryState, StateMap}, | ||
Raphaël Gomès
|
r42993 | DirstateEntry, DirstatePackError, DirstateParents, DirstateParseError, | ||
Raphaël Gomès
|
r42828 | }; | ||
Raphaël Gomès
|
r42617 | use byteorder::{BigEndian, ReadBytesExt, WriteBytesExt}; | ||
Raphaël Gomès
|
r45028 | use micro_timer::timed; | ||
Raphaël Gomès
|
r42994 | use std::convert::{TryFrom, TryInto}; | ||
Raphaël Gomès
|
r42617 | use std::io::Cursor; | ||
Raphaël Gomès
|
r42993 | use std::time::Duration; | ||
Raphaël Gomès
|
r42617 | |||
/// Parents are stored in the dirstate as byte hashes. | ||||
Raphaël Gomès
|
r42993 | pub const PARENT_SIZE: usize = 20; | ||
Raphaël Gomès
|
r42617 | /// Dirstate entries have a static part of 8 + 32 + 32 + 32 + 32 bits. | ||
const MIN_ENTRY_SIZE: usize = 17; | ||||
Raphaël Gomès
|
r42993 | // TODO parse/pack: is mutate-on-loop better for performance? | ||
Raphaël Gomès
|
r45028 | #[timed] | ||
Raphaël Gomès
|
r42617 | pub fn parse_dirstate( | ||
Raphaël Gomès
|
r42993 | state_map: &mut StateMap, | ||
copy_map: &mut CopyMap, | ||||
Raphaël Gomès
|
r42617 | contents: &[u8], | ||
Raphaël Gomès
|
r42993 | ) -> Result<DirstateParents, DirstateParseError> { | ||
Raphaël Gomès
|
r42617 | if contents.len() < PARENT_SIZE * 2 { | ||
return Err(DirstateParseError::TooLittleData); | ||||
} | ||||
let mut curr_pos = PARENT_SIZE * 2; | ||||
let parents = DirstateParents { | ||||
Yuya Nishihara
|
r43067 | p1: contents[..PARENT_SIZE].try_into().unwrap(), | ||
p2: contents[PARENT_SIZE..curr_pos].try_into().unwrap(), | ||||
Raphaël Gomès
|
r42617 | }; | ||
while curr_pos < contents.len() { | ||||
if curr_pos + MIN_ENTRY_SIZE > contents.len() { | ||||
return Err(DirstateParseError::Overflow); | ||||
} | ||||
let entry_bytes = &contents[curr_pos..]; | ||||
let mut cursor = Cursor::new(entry_bytes); | ||||
Raphaël Gomès
|
r42994 | let state = EntryState::try_from(cursor.read_u8()?)?; | ||
Raphaël Gomès
|
r42617 | let mode = cursor.read_i32::<BigEndian>()?; | ||
let size = cursor.read_i32::<BigEndian>()?; | ||||
let mtime = cursor.read_i32::<BigEndian>()?; | ||||
let path_len = cursor.read_i32::<BigEndian>()? as usize; | ||||
if path_len > contents.len() - curr_pos { | ||||
return Err(DirstateParseError::Overflow); | ||||
} | ||||
// Slice instead of allocating a Vec needed for `read_exact` | ||||
let path = &entry_bytes[MIN_ENTRY_SIZE..MIN_ENTRY_SIZE + (path_len)]; | ||||
let (path, copy) = match memchr::memchr(0, path) { | ||||
None => (path, None), | ||||
Some(i) => (&path[..i], Some(&path[(i + 1)..])), | ||||
}; | ||||
if let Some(copy_path) = copy { | ||||
Raphaël Gomès
|
r43227 | copy_map.insert( | ||
HgPath::new(path).to_owned(), | ||||
HgPath::new(copy_path).to_owned(), | ||||
); | ||||
Raphaël Gomès
|
r42617 | }; | ||
Raphaël Gomès
|
r42993 | state_map.insert( | ||
Raphaël Gomès
|
r43227 | HgPath::new(path).to_owned(), | ||
Raphaël Gomès
|
r42617 | DirstateEntry { | ||
state, | ||||
mode, | ||||
size, | ||||
mtime, | ||||
}, | ||||
Raphaël Gomès
|
r42993 | ); | ||
Raphaël Gomès
|
r42617 | curr_pos = curr_pos + MIN_ENTRY_SIZE + (path_len); | ||
} | ||||
Raphaël Gomès
|
r42993 | Ok(parents) | ||
Raphaël Gomès
|
r42617 | } | ||
Raphaël Gomès
|
r42993 | /// `now` is the duration in seconds since the Unix epoch | ||
Raphaël Gomès
|
r42617 | pub fn pack_dirstate( | ||
Raphaël Gomès
|
r42993 | state_map: &mut StateMap, | ||
copy_map: &CopyMap, | ||||
Raphaël Gomès
|
r42617 | parents: DirstateParents, | ||
Raphaël Gomès
|
r42993 | now: Duration, | ||
) -> Result<Vec<u8>, DirstatePackError> { | ||||
// TODO move away from i32 before 2038. | ||||
let now: i32 = now.as_secs().try_into().expect("time overflow"); | ||||
Raphaël Gomès
|
r42617 | |||
Raphaël Gomès
|
r42993 | let expected_size: usize = state_map | ||
Raphaël Gomès
|
r42617 | .iter() | ||
Raphaël Gomès
|
r42993 | .map(|(filename, _)| { | ||
Raphaël Gomès
|
r42617 | let mut length = MIN_ENTRY_SIZE + filename.len(); | ||
Yuya Nishihara
|
r43062 | if let Some(copy) = copy_map.get(filename) { | ||
Raphaël Gomès
|
r42617 | length += copy.len() + 1; | ||
} | ||||
length | ||||
}) | ||||
.sum(); | ||||
let expected_size = expected_size + PARENT_SIZE * 2; | ||||
let mut packed = Vec::with_capacity(expected_size); | ||||
Raphaël Gomès
|
r42993 | let mut new_state_map = vec![]; | ||
Raphaël Gomès
|
r42617 | |||
Raphaël Gomès
|
r42993 | packed.extend(&parents.p1); | ||
packed.extend(&parents.p2); | ||||
Raphaël Gomès
|
r42617 | |||
Yuya Nishihara
|
r43062 | for (filename, entry) in state_map.iter() { | ||
Raphaël Gomès
|
r43227 | let new_filename = filename.to_owned(); | ||
Raphaël Gomès
|
r42617 | let mut new_mtime: i32 = entry.mtime; | ||
Raphaël Gomès
|
r42994 | if entry.state == EntryState::Normal && entry.mtime == now { | ||
Raphaël Gomès
|
r42617 | // The file was last modified "simultaneously" with the current | ||
// write to dirstate (i.e. within the same second for file- | ||||
// systems with a granularity of 1 sec). This commonly happens | ||||
// for at least a couple of files on 'update'. | ||||
// The user could change the file without changing its size | ||||
// within the same second. Invalidate the file's mtime in | ||||
// dirstate, forcing future 'status' calls to compare the | ||||
// contents of the file if the size is the same. This prevents | ||||
// mistakenly treating such files as clean. | ||||
new_mtime = -1; | ||||
Raphaël Gomès
|
r42993 | new_state_map.push(( | ||
Yuya Nishihara
|
r43062 | filename.to_owned(), | ||
Raphaël Gomès
|
r42617 | DirstateEntry { | ||
mtime: new_mtime, | ||||
..*entry | ||||
}, | ||||
)); | ||||
} | ||||
Raphaël Gomès
|
r43227 | let mut new_filename = new_filename.into_vec(); | ||
Yuya Nishihara
|
r43062 | if let Some(copy) = copy_map.get(filename) { | ||
Raphaël Gomès
|
r42617 | new_filename.push('\0' as u8); | ||
Raphaël Gomès
|
r43227 | new_filename.extend(copy.bytes()); | ||
Raphaël Gomès
|
r42617 | } | ||
Raphaël Gomès
|
r42994 | packed.write_u8(entry.state.into())?; | ||
Raphaël Gomès
|
r42617 | packed.write_i32::<BigEndian>(entry.mode)?; | ||
packed.write_i32::<BigEndian>(entry.size)?; | ||||
packed.write_i32::<BigEndian>(new_mtime)?; | ||||
packed.write_i32::<BigEndian>(new_filename.len() as i32)?; | ||||
packed.extend(new_filename) | ||||
} | ||||
if packed.len() != expected_size { | ||||
return Err(DirstatePackError::BadSize(expected_size, packed.len())); | ||||
} | ||||
Raphaël Gomès
|
r42993 | state_map.extend(new_state_map); | ||
Ok(packed) | ||||
Raphaël Gomès
|
r42617 | } | ||
#[cfg(test)] | ||||
mod tests { | ||||
use super::*; | ||||
Raphaël Gomès
|
r44278 | use crate::{utils::hg_path::HgPathBuf, FastHashMap}; | ||
Raphaël Gomès
|
r42617 | |||
#[test] | ||||
fn test_pack_dirstate_empty() { | ||||
Raphaël Gomès
|
r44278 | let mut state_map: StateMap = FastHashMap::default(); | ||
let copymap = FastHashMap::default(); | ||||
Raphaël Gomès
|
r42617 | let parents = DirstateParents { | ||
Raphaël Gomès
|
r42993 | p1: *b"12345678910111213141", | ||
p2: *b"00000000000000000000", | ||||
Raphaël Gomès
|
r42617 | }; | ||
Raphaël Gomès
|
r42993 | let now = Duration::new(15000000, 0); | ||
let expected = b"1234567891011121314100000000000000000000".to_vec(); | ||||
Raphaël Gomès
|
r42617 | |||
assert_eq!( | ||||
expected, | ||||
Raphaël Gomès
|
r42993 | pack_dirstate(&mut state_map, ©map, parents, now).unwrap() | ||
Raphaël Gomès
|
r42617 | ); | ||
Raphaël Gomès
|
r42993 | |||
assert!(state_map.is_empty()) | ||||
Raphaël Gomès
|
r42617 | } | ||
#[test] | ||||
fn test_pack_dirstate_one_entry() { | ||||
Raphaël Gomès
|
r42993 | let expected_state_map: StateMap = [( | ||
Raphaël Gomès
|
r43227 | HgPathBuf::from_bytes(b"f1"), | ||
Raphaël Gomès
|
r42617 | DirstateEntry { | ||
Raphaël Gomès
|
r42994 | state: EntryState::Normal, | ||
Raphaël Gomès
|
r42617 | mode: 0o644, | ||
size: 0, | ||||
mtime: 791231220, | ||||
}, | ||||
Raphaël Gomès
|
r42993 | )] | ||
.iter() | ||||
.cloned() | ||||
.collect(); | ||||
let mut state_map = expected_state_map.clone(); | ||||
Raphaël Gomès
|
r44278 | let copymap = FastHashMap::default(); | ||
Raphaël Gomès
|
r42617 | let parents = DirstateParents { | ||
Raphaël Gomès
|
r42993 | p1: *b"12345678910111213141", | ||
p2: *b"00000000000000000000", | ||||
Raphaël Gomès
|
r42617 | }; | ||
Raphaël Gomès
|
r42993 | let now = Duration::new(15000000, 0); | ||
let expected = [ | ||||
49, 50, 51, 52, 53, 54, 55, 56, 57, 49, 48, 49, 49, 49, 50, 49, | ||||
51, 49, 52, 49, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, | ||||
48, 48, 48, 48, 48, 48, 48, 48, 110, 0, 0, 1, 164, 0, 0, 0, 0, 47, | ||||
41, 58, 244, 0, 0, 0, 2, 102, 49, | ||||
] | ||||
.to_vec(); | ||||
Raphaël Gomès
|
r42617 | |||
assert_eq!( | ||||
expected, | ||||
Raphaël Gomès
|
r42993 | pack_dirstate(&mut state_map, ©map, parents, now).unwrap() | ||
Raphaël Gomès
|
r42617 | ); | ||
Raphaël Gomès
|
r42993 | assert_eq!(expected_state_map, state_map); | ||
} | ||||
Raphaël Gomès
|
r42617 | #[test] | ||
Raphaël Gomès
|
r42993 | fn test_pack_dirstate_one_entry_with_copy() { | ||
let expected_state_map: StateMap = [( | ||||
Raphaël Gomès
|
r43227 | HgPathBuf::from_bytes(b"f1"), | ||
Raphaël Gomès
|
r42617 | DirstateEntry { | ||
Raphaël Gomès
|
r42994 | state: EntryState::Normal, | ||
Raphaël Gomès
|
r42617 | mode: 0o644, | ||
size: 0, | ||||
mtime: 791231220, | ||||
}, | ||||
Raphaël Gomès
|
r42993 | )] | ||
.iter() | ||||
.cloned() | ||||
.collect(); | ||||
let mut state_map = expected_state_map.clone(); | ||||
Raphaël Gomès
|
r44278 | let mut copymap = FastHashMap::default(); | ||
Raphaël Gomès
|
r43227 | copymap.insert( | ||
HgPathBuf::from_bytes(b"f1"), | ||||
HgPathBuf::from_bytes(b"copyname"), | ||||
); | ||||
Raphaël Gomès
|
r42617 | let parents = DirstateParents { | ||
Raphaël Gomès
|
r42993 | p1: *b"12345678910111213141", | ||
p2: *b"00000000000000000000", | ||||
Raphaël Gomès
|
r42617 | }; | ||
Raphaël Gomès
|
r42993 | let now = Duration::new(15000000, 0); | ||
let expected = [ | ||||
49, 50, 51, 52, 53, 54, 55, 56, 57, 49, 48, 49, 49, 49, 50, 49, | ||||
51, 49, 52, 49, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, 48, | ||||
48, 48, 48, 48, 48, 48, 48, 48, 110, 0, 0, 1, 164, 0, 0, 0, 0, 47, | ||||
41, 58, 244, 0, 0, 0, 11, 102, 49, 0, 99, 111, 112, 121, 110, 97, | ||||
109, 101, | ||||
] | ||||
.to_vec(); | ||||
Raphaël Gomès
|
r42617 | |||
assert_eq!( | ||||
Raphaël Gomès
|
r42993 | expected, | ||
pack_dirstate(&mut state_map, ©map, parents, now).unwrap() | ||||
); | ||||
assert_eq!(expected_state_map, state_map); | ||||
} | ||||
#[test] | ||||
fn test_parse_pack_one_entry_with_copy() { | ||||
let mut state_map: StateMap = [( | ||||
Raphaël Gomès
|
r43227 | HgPathBuf::from_bytes(b"f1"), | ||
Raphaël Gomès
|
r42993 | DirstateEntry { | ||
Raphaël Gomès
|
r42994 | state: EntryState::Normal, | ||
Raphaël Gomès
|
r42993 | mode: 0o644, | ||
size: 0, | ||||
mtime: 791231220, | ||||
}, | ||||
)] | ||||
.iter() | ||||
.cloned() | ||||
.collect(); | ||||
Raphaël Gomès
|
r44278 | let mut copymap = FastHashMap::default(); | ||
Raphaël Gomès
|
r43227 | copymap.insert( | ||
HgPathBuf::from_bytes(b"f1"), | ||||
HgPathBuf::from_bytes(b"copyname"), | ||||
); | ||||
Raphaël Gomès
|
r42993 | let parents = DirstateParents { | ||
p1: *b"12345678910111213141", | ||||
p2: *b"00000000000000000000", | ||||
}; | ||||
let now = Duration::new(15000000, 0); | ||||
let result = | ||||
pack_dirstate(&mut state_map, ©map, parents.clone(), now) | ||||
.unwrap(); | ||||
Raphaël Gomès
|
r44278 | let mut new_state_map: StateMap = FastHashMap::default(); | ||
let mut new_copy_map: CopyMap = FastHashMap::default(); | ||||
Raphaël Gomès
|
r42993 | let new_parents = parse_dirstate( | ||
&mut new_state_map, | ||||
&mut new_copy_map, | ||||
result.as_slice(), | ||||
) | ||||
.unwrap(); | ||||
assert_eq!( | ||||
(parents, state_map, copymap), | ||||
(new_parents, new_state_map, new_copy_map) | ||||
Raphaël Gomès
|
r42617 | ) | ||
} | ||||
#[test] | ||||
fn test_parse_pack_multiple_entries_with_copy() { | ||||
Raphaël Gomès
|
r42993 | let mut state_map: StateMap = [ | ||
Raphaël Gomès
|
r42617 | ( | ||
Raphaël Gomès
|
r43227 | HgPathBuf::from_bytes(b"f1"), | ||
Raphaël Gomès
|
r42617 | DirstateEntry { | ||
Raphaël Gomès
|
r42994 | state: EntryState::Normal, | ||
Raphaël Gomès
|
r42617 | mode: 0o644, | ||
size: 0, | ||||
mtime: 791231220, | ||||
}, | ||||
), | ||||
( | ||||
Raphaël Gomès
|
r43227 | HgPathBuf::from_bytes(b"f2"), | ||
Raphaël Gomès
|
r42617 | DirstateEntry { | ||
Raphaël Gomès
|
r42994 | state: EntryState::Merged, | ||
Raphaël Gomès
|
r42617 | mode: 0o777, | ||
size: 1000, | ||||
mtime: 791231220, | ||||
}, | ||||
), | ||||
( | ||||
Raphaël Gomès
|
r43227 | HgPathBuf::from_bytes(b"f3"), | ||
Raphaël Gomès
|
r42617 | DirstateEntry { | ||
Raphaël Gomès
|
r42994 | state: EntryState::Removed, | ||
Raphaël Gomès
|
r42617 | mode: 0o644, | ||
size: 234553, | ||||
mtime: 791231220, | ||||
}, | ||||
), | ||||
( | ||||
Raphaël Gomès
|
r43227 | HgPathBuf::from_bytes(b"f4\xF6"), | ||
Raphaël Gomès
|
r42617 | DirstateEntry { | ||
Raphaël Gomès
|
r42994 | state: EntryState::Added, | ||
Raphaël Gomès
|
r42617 | mode: 0o644, | ||
size: -1, | ||||
mtime: -1, | ||||
}, | ||||
), | ||||
Raphaël Gomès
|
r42993 | ] | ||
.iter() | ||||
.cloned() | ||||
.collect(); | ||||
Raphaël Gomès
|
r44278 | let mut copymap = FastHashMap::default(); | ||
Raphaël Gomès
|
r43227 | copymap.insert( | ||
HgPathBuf::from_bytes(b"f1"), | ||||
HgPathBuf::from_bytes(b"copyname"), | ||||
); | ||||
copymap.insert( | ||||
HgPathBuf::from_bytes(b"f4\xF6"), | ||||
HgPathBuf::from_bytes(b"copyname2"), | ||||
); | ||||
Raphaël Gomès
|
r42617 | let parents = DirstateParents { | ||
Raphaël Gomès
|
r42993 | p1: *b"12345678910111213141", | ||
p2: *b"00000000000000000000", | ||||
Raphaël Gomès
|
r42617 | }; | ||
Raphaël Gomès
|
r42993 | let now = Duration::new(15000000, 0); | ||
Raphaël Gomès
|
r42617 | let result = | ||
Raphaël Gomès
|
r42993 | pack_dirstate(&mut state_map, ©map, parents.clone(), now) | ||
.unwrap(); | ||||
Raphaël Gomès
|
r42617 | |||
Raphaël Gomès
|
r44278 | let mut new_state_map: StateMap = FastHashMap::default(); | ||
let mut new_copy_map: CopyMap = FastHashMap::default(); | ||||
Raphaël Gomès
|
r42993 | let new_parents = parse_dirstate( | ||
&mut new_state_map, | ||||
&mut new_copy_map, | ||||
result.as_slice(), | ||||
) | ||||
.unwrap(); | ||||
Raphaël Gomès
|
r42617 | assert_eq!( | ||
Raphaël Gomès
|
r42993 | (parents, state_map, copymap), | ||
(new_parents, new_state_map, new_copy_map) | ||||
Raphaël Gomès
|
r42617 | ) | ||
} | ||||
#[test] | ||||
/// https://www.mercurial-scm.org/repo/hg/rev/af3f26b6bba4 | ||||
fn test_parse_pack_one_entry_with_copy_and_time_conflict() { | ||||
Raphaël Gomès
|
r42993 | let mut state_map: StateMap = [( | ||
Raphaël Gomès
|
r43227 | HgPathBuf::from_bytes(b"f1"), | ||
Raphaël Gomès
|
r42617 | DirstateEntry { | ||
Raphaël Gomès
|
r42994 | state: EntryState::Normal, | ||
Raphaël Gomès
|
r42617 | mode: 0o644, | ||
size: 0, | ||||
mtime: 15000000, | ||||
}, | ||||
Raphaël Gomès
|
r42993 | )] | ||
.iter() | ||||
.cloned() | ||||
.collect(); | ||||
Raphaël Gomès
|
r44278 | let mut copymap = FastHashMap::default(); | ||
Raphaël Gomès
|
r43227 | copymap.insert( | ||
HgPathBuf::from_bytes(b"f1"), | ||||
HgPathBuf::from_bytes(b"copyname"), | ||||
); | ||||
Raphaël Gomès
|
r42617 | let parents = DirstateParents { | ||
Raphaël Gomès
|
r42993 | p1: *b"12345678910111213141", | ||
p2: *b"00000000000000000000", | ||||
Raphaël Gomès
|
r42617 | }; | ||
Raphaël Gomès
|
r42993 | let now = Duration::new(15000000, 0); | ||
Raphaël Gomès
|
r42617 | let result = | ||
Raphaël Gomès
|
r42993 | pack_dirstate(&mut state_map, ©map, parents.clone(), now) | ||
.unwrap(); | ||||
Raphaël Gomès
|
r44278 | let mut new_state_map: StateMap = FastHashMap::default(); | ||
let mut new_copy_map: CopyMap = FastHashMap::default(); | ||||
Raphaël Gomès
|
r42993 | let new_parents = parse_dirstate( | ||
&mut new_state_map, | ||||
&mut new_copy_map, | ||||
result.as_slice(), | ||||
) | ||||
.unwrap(); | ||||
Raphaël Gomès
|
r42617 | |||
assert_eq!( | ||||
( | ||||
parents, | ||||
Raphaël Gomès
|
r42993 | [( | ||
Raphaël Gomès
|
r43227 | HgPathBuf::from_bytes(b"f1"), | ||
Raphaël Gomès
|
r42617 | DirstateEntry { | ||
Raphaël Gomès
|
r42994 | state: EntryState::Normal, | ||
Raphaël Gomès
|
r42617 | mode: 0o644, | ||
size: 0, | ||||
mtime: -1 | ||||
} | ||||
Raphaël Gomès
|
r42993 | )] | ||
.iter() | ||||
.cloned() | ||||
.collect::<StateMap>(), | ||||
copymap, | ||||
Raphaël Gomès
|
r42617 | ), | ||
Raphaël Gomès
|
r42993 | (new_parents, new_state_map, new_copy_map) | ||
Raphaël Gomès
|
r42617 | ) | ||
} | ||||
} | ||||