Show More
@@ -0,0 +1,145 | |||
|
1 | // list_tracked_files.rs | |
|
2 | // | |
|
3 | // Copyright 2020 Antoine Cezar <antoine.cezar@octobus.net> | |
|
4 | // | |
|
5 | // This software may be used and distributed according to the terms of the | |
|
6 | // GNU General Public License version 2 or any later version. | |
|
7 | ||
|
8 | use std::convert::From; | |
|
9 | use std::path::PathBuf; | |
|
10 | ||
|
11 | use crate::revlog::changelog::Changelog; | |
|
12 | use crate::revlog::manifest::{Manifest, ManifestEntry}; | |
|
13 | use crate::revlog::path_encode::path_encode; | |
|
14 | use crate::revlog::revlog::Revlog; | |
|
15 | use crate::revlog::revlog::RevlogError; | |
|
16 | use crate::revlog::Revision; | |
|
17 | use crate::utils::hg_path::HgPathBuf; | |
|
18 | ||
|
19 | /// Kind of error encountered by `CatRev` | |
|
20 | #[derive(Debug)] | |
|
21 | pub enum CatRevErrorKind { | |
|
22 | /// Error when reading a `revlog` file. | |
|
23 | IoError(std::io::Error), | |
|
24 | /// The revision has not been found. | |
|
25 | InvalidRevision, | |
|
26 | /// A `revlog` file is corrupted. | |
|
27 | CorruptedRevlog, | |
|
28 | /// The `revlog` format version is not supported. | |
|
29 | UnsuportedRevlogVersion(u16), | |
|
30 | /// The `revlog` data format is not supported. | |
|
31 | UnknowRevlogDataFormat(u8), | |
|
32 | } | |
|
33 | ||
|
34 | /// A `CatRev` error | |
|
35 | #[derive(Debug)] | |
|
36 | pub struct CatRevError { | |
|
37 | /// Kind of error encountered by `CatRev` | |
|
38 | pub kind: CatRevErrorKind, | |
|
39 | } | |
|
40 | ||
|
41 | impl From<CatRevErrorKind> for CatRevError { | |
|
42 | fn from(kind: CatRevErrorKind) -> Self { | |
|
43 | CatRevError { kind } | |
|
44 | } | |
|
45 | } | |
|
46 | ||
|
47 | impl From<RevlogError> for CatRevError { | |
|
48 | fn from(err: RevlogError) -> Self { | |
|
49 | match err { | |
|
50 | RevlogError::IoError(err) => CatRevErrorKind::IoError(err), | |
|
51 | RevlogError::UnsuportedVersion(version) => { | |
|
52 | CatRevErrorKind::UnsuportedRevlogVersion(version) | |
|
53 | } | |
|
54 | RevlogError::InvalidRevision => CatRevErrorKind::InvalidRevision, | |
|
55 | RevlogError::Corrupted => CatRevErrorKind::CorruptedRevlog, | |
|
56 | RevlogError::UnknowDataFormat(format) => { | |
|
57 | CatRevErrorKind::UnknowRevlogDataFormat(format) | |
|
58 | } | |
|
59 | } | |
|
60 | .into() | |
|
61 | } | |
|
62 | } | |
|
63 | ||
|
64 | /// List files under Mercurial control at a given revision. | |
|
65 | pub struct CatRev<'a> { | |
|
66 | root: &'a PathBuf, | |
|
67 | /// The revision to cat the files from. | |
|
68 | rev: &'a str, | |
|
69 | /// The files to output. | |
|
70 | files: &'a [HgPathBuf], | |
|
71 | /// The changelog file | |
|
72 | changelog: Changelog, | |
|
73 | /// The manifest file | |
|
74 | manifest: Manifest, | |
|
75 | /// The manifest entry corresponding to the revision. | |
|
76 | /// | |
|
77 | /// Used to hold the owner of the returned references. | |
|
78 | manifest_entry: Option<ManifestEntry>, | |
|
79 | } | |
|
80 | ||
|
81 | impl<'a> CatRev<'a> { | |
|
82 | pub fn new( | |
|
83 | root: &'a PathBuf, | |
|
84 | rev: &'a str, | |
|
85 | files: &'a [HgPathBuf], | |
|
86 | ) -> Result<Self, CatRevError> { | |
|
87 | let changelog = Changelog::open(&root)?; | |
|
88 | let manifest = Manifest::open(&root)?; | |
|
89 | let manifest_entry = None; | |
|
90 | ||
|
91 | Ok(Self { | |
|
92 | root, | |
|
93 | rev, | |
|
94 | files, | |
|
95 | changelog, | |
|
96 | manifest, | |
|
97 | manifest_entry, | |
|
98 | }) | |
|
99 | } | |
|
100 | ||
|
101 | pub fn run(&mut self) -> Result<Vec<u8>, CatRevError> { | |
|
102 | let changelog_entry = match self.rev.parse::<Revision>() { | |
|
103 | Ok(rev) => self.changelog.get_rev(rev)?, | |
|
104 | _ => { | |
|
105 | let changelog_node = hex::decode(&self.rev) | |
|
106 | .map_err(|_| CatRevErrorKind::InvalidRevision)?; | |
|
107 | self.changelog.get_node(&changelog_node)? | |
|
108 | } | |
|
109 | }; | |
|
110 | let manifest_node = hex::decode(&changelog_entry.manifest_node()?) | |
|
111 | .map_err(|_| CatRevErrorKind::CorruptedRevlog)?; | |
|
112 | ||
|
113 | self.manifest_entry = Some(self.manifest.get_node(&manifest_node)?); | |
|
114 | if let Some(ref manifest_entry) = self.manifest_entry { | |
|
115 | let mut bytes = vec![]; | |
|
116 | ||
|
117 | for (manifest_file, node_bytes) in | |
|
118 | manifest_entry.files_with_nodes() | |
|
119 | { | |
|
120 | for cat_file in self.files.iter() { | |
|
121 | if cat_file.as_bytes() == manifest_file.as_bytes() { | |
|
122 | let encoded_bytes = | |
|
123 | path_encode(manifest_file.as_bytes()); | |
|
124 | let revlog_index_string = format!( | |
|
125 | ".hg/store/data/{}.i", | |
|
126 | String::from_utf8_lossy(&encoded_bytes), | |
|
127 | ); | |
|
128 | let revlog_index_path = | |
|
129 | self.root.join(&revlog_index_string); | |
|
130 | let file_log = Revlog::open(&revlog_index_path)?; | |
|
131 | let file_node = hex::decode(&node_bytes) | |
|
132 | .map_err(|_| CatRevErrorKind::CorruptedRevlog)?; | |
|
133 | let file_rev = file_log.get_node_rev(&file_node)?; | |
|
134 | let data = file_log.get_rev_data(file_rev)?; | |
|
135 | bytes.extend(data); | |
|
136 | } | |
|
137 | } | |
|
138 | } | |
|
139 | ||
|
140 | Ok(bytes) | |
|
141 | } else { | |
|
142 | unreachable!("manifest_entry should have been stored"); | |
|
143 | } | |
|
144 | } | |
|
145 | } |
@@ -1,26 +1,28 | |||
|
1 | 1 | //! A distinction is made between operations and commands. |
|
2 | 2 | //! An operation is what can be done whereas a command is what is exposed by |
|
3 | 3 | //! the cli. A single command can use several operations to achieve its goal. |
|
4 | 4 | |
|
5 | mod cat; | |
|
5 | 6 | mod debugdata; |
|
6 | 7 | mod dirstate_status; |
|
7 | 8 | mod find_root; |
|
8 | 9 | mod list_tracked_files; |
|
10 | pub use cat::{CatRev, CatRevError, CatRevErrorKind}; | |
|
9 | 11 | pub use debugdata::{ |
|
10 | 12 | DebugData, DebugDataError, DebugDataErrorKind, DebugDataKind, |
|
11 | 13 | }; |
|
12 | 14 | pub use find_root::{FindRoot, FindRootError, FindRootErrorKind}; |
|
13 | 15 | pub use list_tracked_files::{ |
|
14 | 16 | ListDirstateTrackedFiles, ListDirstateTrackedFilesError, |
|
15 | 17 | ListDirstateTrackedFilesErrorKind, |
|
16 | 18 | }; |
|
17 | 19 | pub use list_tracked_files::{ |
|
18 | 20 | ListRevTrackedFiles, ListRevTrackedFilesError, |
|
19 | 21 | ListRevTrackedFilesErrorKind, |
|
20 | 22 | }; |
|
21 | 23 | |
|
22 | 24 | // TODO add an `Operation` trait when GAT have landed (rust #44265): |
|
23 | 25 | // there is no way to currently define a trait which can both return |
|
24 | 26 | // references to `self` and to passed data, which is what we would need. |
|
25 | 27 | // Generic Associated Types may fix this and allow us to have a unified |
|
26 | 28 | // interface. |
General Comments 0
You need to be logged in to leave comments.
Login now