Show More
@@ -0,0 +1,23 b'' | |||||
|
1 | Create a repo such that the changelog entry refers to a null manifest node: | |||
|
2 | ||||
|
3 | $ hg init a | |||
|
4 | $ cd a | |||
|
5 | $ hg log | |||
|
6 | $ touch x | |||
|
7 | $ hg add x | |||
|
8 | $ hg commit -m "init" | |||
|
9 | $ hg rm x | |||
|
10 | $ hg commit -q --amend | |||
|
11 | ||||
|
12 | $ wc -c < .hg/store/00manifest.i | |||
|
13 | 0 | |||
|
14 | ||||
|
15 | Make sure that the manifest can be read (and is empty): | |||
|
16 | ||||
|
17 | $ hg --config rhg.on-unsupported=abort files -r . | |||
|
18 | [1] | |||
|
19 | ||||
|
20 | Test a null changelog rev, too: | |||
|
21 | ||||
|
22 | $ hg --config rhg.on-unsupported=abort files -r 0000000000000000000000000000000000000000 | |||
|
23 | [1] |
@@ -1,5 +1,6 b'' | |||||
1 | use crate::errors::HgError; |
|
1 | use crate::errors::HgError; | |
2 | use crate::repo::Repo; |
|
2 | use crate::repo::Repo; | |
|
3 | use crate::revlog::node::NULL_NODE; | |||
3 | use crate::revlog::revlog::{Revlog, RevlogError}; |
|
4 | use crate::revlog::revlog::{Revlog, RevlogError}; | |
4 | use crate::revlog::Revision; |
|
5 | use crate::revlog::Revision; | |
5 | use crate::revlog::{Node, NodePrefix}; |
|
6 | use crate::revlog::{Node, NodePrefix}; | |
@@ -58,10 +59,9 b' impl ChangelogEntry {' | |||||
58 | /// Return the node id of the `manifest` referenced by this `changelog` |
|
59 | /// Return the node id of the `manifest` referenced by this `changelog` | |
59 | /// entry. |
|
60 | /// entry. | |
60 | pub fn manifest_node(&self) -> Result<Node, HgError> { |
|
61 | pub fn manifest_node(&self) -> Result<Node, HgError> { | |
61 | Node::from_hex_for_repo( |
|
62 | match self.lines().next() { | |
62 | self.lines() |
|
63 | None => Ok(NULL_NODE), | |
63 | .next() |
|
64 | Some(x) => Node::from_hex_for_repo(x), | |
64 | .ok_or_else(|| HgError::corrupted("empty changelog entry"))?, |
|
|||
65 | ) |
|
|||
66 | } |
|
65 | } | |
67 | } |
|
66 | } | |
|
67 | } |
@@ -208,6 +208,9 b" impl<'a> IndexEntry<'a> {" | |||||
208 |
|
208 | |||
209 | /// Value of the inline flag. |
|
209 | /// Value of the inline flag. | |
210 | pub fn is_inline(index_bytes: &[u8]) -> bool { |
|
210 | pub fn is_inline(index_bytes: &[u8]) -> bool { | |
|
211 | if index_bytes.len() < 4 { | |||
|
212 | return true; | |||
|
213 | } | |||
211 | match &index_bytes[0..=1] { |
|
214 | match &index_bytes[0..=1] { | |
212 | [0, 0] | [0, 2] => false, |
|
215 | [0, 0] | [0, 2] => false, | |
213 | _ => true, |
|
216 | _ => true, |
@@ -72,7 +72,7 b' impl Revlog {' | |||||
72 | let index_path = index_path.as_ref(); |
|
72 | let index_path = index_path.as_ref(); | |
73 | let index_mmap = repo.store_vfs().mmap_open(&index_path)?; |
|
73 | let index_mmap = repo.store_vfs().mmap_open(&index_path)?; | |
74 |
|
74 | |||
75 | let version = get_version(&index_mmap); |
|
75 | let version = get_version(&index_mmap)?; | |
76 | if version != 1 { |
|
76 | if version != 1 { | |
77 | // A proper new version should have had a repo/store requirement. |
|
77 | // A proper new version should have had a repo/store requirement. | |
78 | return Err(HgError::corrupted("corrupted revlog")); |
|
78 | return Err(HgError::corrupted("corrupted revlog")); | |
@@ -179,6 +179,9 b' impl Revlog {' | |||||
179 | /// snapshot to rebuild the final data. |
|
179 | /// snapshot to rebuild the final data. | |
180 | #[timed] |
|
180 | #[timed] | |
181 | pub fn get_rev_data(&self, rev: Revision) -> Result<Vec<u8>, RevlogError> { |
|
181 | pub fn get_rev_data(&self, rev: Revision) -> Result<Vec<u8>, RevlogError> { | |
|
182 | if rev == NULL_REVISION { | |||
|
183 | return Ok(vec![]); | |||
|
184 | }; | |||
182 | // Todo return -> Cow |
|
185 | // Todo return -> Cow | |
183 | let mut entry = self.get_entry(rev)?; |
|
186 | let mut entry = self.get_entry(rev)?; | |
184 | let mut delta_chain = vec![]; |
|
187 | let mut delta_chain = vec![]; | |
@@ -371,8 +374,16 b" impl<'a> RevlogEntry<'a> {" | |||||
371 | } |
|
374 | } | |
372 |
|
375 | |||
373 | /// Format version of the revlog. |
|
376 | /// Format version of the revlog. | |
374 | pub fn get_version(index_bytes: &[u8]) -> u16 { |
|
377 | pub fn get_version(index_bytes: &[u8]) -> Result<u16, HgError> { | |
375 | BigEndian::read_u16(&index_bytes[2..=3]) |
|
378 | if index_bytes.len() == 0 { | |
|
379 | return Ok(1); | |||
|
380 | }; | |||
|
381 | if index_bytes.len() < 4 { | |||
|
382 | return Err(HgError::corrupted( | |||
|
383 | "corrupted revlog: can't read the index format header", | |||
|
384 | )); | |||
|
385 | }; | |||
|
386 | Ok(BigEndian::read_u16(&index_bytes[2..=3])) | |||
376 | } |
|
387 | } | |
377 |
|
388 | |||
378 | /// Calculate the hash of a revision given its data and its parents. |
|
389 | /// Calculate the hash of a revision given its data and its parents. |
General Comments 0
You need to be logged in to leave comments.
Login now