##// END OF EJS Templates
rhg: strip copied files metadata from `cat` output...
Antoine cezar -
r46395:056aeed4 default draft
parent child Browse files
Show More
@@ -1,145 +1,158 b''
1 1 // list_tracked_files.rs
2 2 //
3 3 // Copyright 2020 Antoine Cezar <antoine.cezar@octobus.net>
4 4 //
5 5 // This software may be used and distributed according to the terms of the
6 6 // GNU General Public License version 2 or any later version.
7 7
8 8 use std::convert::From;
9 9 use std::path::PathBuf;
10 10
11 11 use crate::revlog::changelog::Changelog;
12 12 use crate::revlog::manifest::{Manifest, ManifestEntry};
13 13 use crate::revlog::path_encode::path_encode;
14 14 use crate::revlog::revlog::Revlog;
15 15 use crate::revlog::revlog::RevlogError;
16 16 use crate::revlog::Revision;
17 17 use crate::utils::hg_path::HgPathBuf;
18 18
19 const METADATA_DELIMITER: [u8; 2] = [b'\x01', b'\n'];
20
19 21 /// Kind of error encountered by `CatRev`
20 22 #[derive(Debug)]
21 23 pub enum CatRevErrorKind {
22 24 /// Error when reading a `revlog` file.
23 25 IoError(std::io::Error),
24 26 /// The revision has not been found.
25 27 InvalidRevision,
26 28 /// A `revlog` file is corrupted.
27 29 CorruptedRevlog,
28 30 /// The `revlog` format version is not supported.
29 31 UnsuportedRevlogVersion(u16),
30 32 /// The `revlog` data format is not supported.
31 33 UnknowRevlogDataFormat(u8),
32 34 }
33 35
34 36 /// A `CatRev` error
35 37 #[derive(Debug)]
36 38 pub struct CatRevError {
37 39 /// Kind of error encountered by `CatRev`
38 40 pub kind: CatRevErrorKind,
39 41 }
40 42
41 43 impl From<CatRevErrorKind> for CatRevError {
42 44 fn from(kind: CatRevErrorKind) -> Self {
43 45 CatRevError { kind }
44 46 }
45 47 }
46 48
47 49 impl From<RevlogError> for CatRevError {
48 50 fn from(err: RevlogError) -> Self {
49 51 match err {
50 52 RevlogError::IoError(err) => CatRevErrorKind::IoError(err),
51 53 RevlogError::UnsuportedVersion(version) => {
52 54 CatRevErrorKind::UnsuportedRevlogVersion(version)
53 55 }
54 56 RevlogError::InvalidRevision => CatRevErrorKind::InvalidRevision,
55 57 RevlogError::Corrupted => CatRevErrorKind::CorruptedRevlog,
56 58 RevlogError::UnknowDataFormat(format) => {
57 59 CatRevErrorKind::UnknowRevlogDataFormat(format)
58 60 }
59 61 }
60 62 .into()
61 63 }
62 64 }
63 65
64 66 /// List files under Mercurial control at a given revision.
65 67 pub struct CatRev<'a> {
66 68 root: &'a PathBuf,
67 69 /// The revision to cat the files from.
68 70 rev: &'a str,
69 71 /// The files to output.
70 72 files: &'a [HgPathBuf],
71 73 /// The changelog file
72 74 changelog: Changelog,
73 75 /// The manifest file
74 76 manifest: Manifest,
75 77 /// The manifest entry corresponding to the revision.
76 78 ///
77 79 /// Used to hold the owner of the returned references.
78 80 manifest_entry: Option<ManifestEntry>,
79 81 }
80 82
81 83 impl<'a> CatRev<'a> {
82 84 pub fn new(
83 85 root: &'a PathBuf,
84 86 rev: &'a str,
85 87 files: &'a [HgPathBuf],
86 88 ) -> Result<Self, CatRevError> {
87 89 let changelog = Changelog::open(&root)?;
88 90 let manifest = Manifest::open(&root)?;
89 91 let manifest_entry = None;
90 92
91 93 Ok(Self {
92 94 root,
93 95 rev,
94 96 files,
95 97 changelog,
96 98 manifest,
97 99 manifest_entry,
98 100 })
99 101 }
100 102
101 103 pub fn run(&mut self) -> Result<Vec<u8>, CatRevError> {
102 104 let changelog_entry = match self.rev.parse::<Revision>() {
103 105 Ok(rev) => self.changelog.get_rev(rev)?,
104 106 _ => {
105 107 let changelog_node = hex::decode(&self.rev)
106 108 .map_err(|_| CatRevErrorKind::InvalidRevision)?;
107 109 self.changelog.get_node(&changelog_node)?
108 110 }
109 111 };
110 112 let manifest_node = hex::decode(&changelog_entry.manifest_node()?)
111 113 .map_err(|_| CatRevErrorKind::CorruptedRevlog)?;
112 114
113 115 self.manifest_entry = Some(self.manifest.get_node(&manifest_node)?);
114 116 if let Some(ref manifest_entry) = self.manifest_entry {
115 117 let mut bytes = vec![];
116 118
117 119 for (manifest_file, node_bytes) in
118 120 manifest_entry.files_with_nodes()
119 121 {
120 122 for cat_file in self.files.iter() {
121 123 if cat_file.as_bytes() == manifest_file.as_bytes() {
122 124 let encoded_bytes =
123 125 path_encode(manifest_file.as_bytes());
124 126 let revlog_index_string = format!(
125 127 ".hg/store/data/{}.i",
126 128 String::from_utf8_lossy(&encoded_bytes),
127 129 );
128 130 let revlog_index_path =
129 131 self.root.join(&revlog_index_string);
130 132 let file_log = Revlog::open(&revlog_index_path)?;
131 133 let file_node = hex::decode(&node_bytes)
132 134 .map_err(|_| CatRevErrorKind::CorruptedRevlog)?;
133 135 let file_rev = file_log.get_node_rev(&file_node)?;
134 136 let data = file_log.get_rev_data(file_rev)?;
137 if data.starts_with(&METADATA_DELIMITER) {
138 let end_delimiter_position = data
139 [METADATA_DELIMITER.len()..]
140 .windows(METADATA_DELIMITER.len())
141 .position(|bytes| bytes == METADATA_DELIMITER);
142 if let Some(position) = end_delimiter_position {
143 let offset = METADATA_DELIMITER.len() * 2;
144 bytes.extend(data[position + offset..].iter());
145 }
146 } else {
135 147 bytes.extend(data);
136 148 }
137 149 }
138 150 }
151 }
139 152
140 153 Ok(bytes)
141 154 } else {
142 155 unreachable!("manifest_entry should have been stored");
143 156 }
144 157 }
145 158 }
@@ -1,92 +1,108 b''
1 1 #require rust
2 2
3 3 Define an rhg function that will only run if rhg exists
4 4 $ rhg() {
5 5 > if [ -f "$RUNTESTDIR/../rust/target/debug/rhg" ]; then
6 6 > "$RUNTESTDIR/../rust/target/debug/rhg" "$@"
7 7 > else
8 8 > echo "skipped: Cannot find rhg. Try to run cargo build in rust/rhg."
9 9 > exit 80
10 10 > fi
11 11 > }
12 12
13 13 Unimplemented command
14 14 $ rhg unimplemented-command
15 15 error: Found argument 'unimplemented-command' which wasn't expected, or isn't valid in this context
16 16
17 17 USAGE:
18 18 rhg <SUBCOMMAND>
19 19
20 20 For more information try --help
21 21 [252]
22 22
23 23 Finding root
24 24 $ rhg root
25 25 abort: no repository found in '$TESTTMP' (.hg not found)!
26 26 [255]
27 27
28 28 $ hg init repository
29 29 $ cd repository
30 30 $ rhg root
31 31 $TESTTMP/repository
32 32
33 33 Unwritable file descriptor
34 34 $ rhg root > /dev/full
35 35 abort: No space left on device (os error 28)
36 36 [255]
37 37
38 38 Deleted repository
39 39 $ rm -rf `pwd`
40 40 $ rhg root
41 41 abort: error getting current working directory: $ENOENT$
42 42 [255]
43 43
44 44 Listing tracked files
45 45 $ cd $TESTTMP
46 46 $ hg init repository
47 47 $ cd repository
48 48 $ for i in 1 2 3; do
49 49 > echo $i >> file$i
50 50 > hg add file$i
51 51 > done
52 52 > hg commit -m "commit $i" -q
53 53
54 54 Listing tracked files from root
55 55 $ rhg files
56 56 file1
57 57 file2
58 58 file3
59 59
60 60 Listing tracked files from subdirectory
61 61 $ mkdir -p path/to/directory
62 62 $ cd path/to/directory
63 63 $ rhg files
64 64 ../../../file1
65 65 ../../../file2
66 66 ../../../file3
67 67
68 68 Listing tracked files through broken pipe
69 69 $ rhg files | head -n 1
70 70 ../../../file1
71 71
72 72 Debuging data in inline index
73 73 $ cd $TESTTMP
74 74 $ rm -rf repository
75 75 $ hg init repository
76 76 $ cd repository
77 77 $ for i in 1 2 3; do
78 78 > echo $i >> file$i
79 79 > hg add file$i
80 80 > hg commit -m "commit $i" -q
81 81 > done
82 82 $ rhg debugdata -c 2
83 83 e36fa63d37a576b27a69057598351db6ee5746bd
84 84 test
85 85 0 0
86 86 file3
87 87
88 88 commit 3 (no-eol)
89 89 $ rhg debugdata -m 2
90 90 file1\x00b8e02f6433738021a065f94175c7cd23db5f05be (esc)
91 91 file2\x005d9299349fc01ddd25d0070d149b124d8f10411e (esc)
92 92 file3\x002661d26c649684b482d10f91960cc3db683c38b4 (esc)
93
94 Cat files
95 $ cd $TESTTMP
96 $ rm -rf repository
97 $ hg init repository
98 $ cd repository
99 $ echo "original content" > original
100 $ hg add original
101 $ hg commit -m "add original" original
102 $ rhg cat -r 0 original
103 original content
104 Cat copied file should not display copy metadata
105 $ hg copy original copy_of_original
106 $ hg commit -m "add copy of original"
107 $ rhg cat -r 1 copy_of_original
108 original content
General Comments 0
You need to be logged in to leave comments. Login now