##// END OF EJS Templates
dirstate-v2: don't mmap the data file when on NFS...
Raphaël Gomès -
r51075:be019ac8 stable
parent child Browse files
Show More
@@ -1,567 +1,574
1 use crate::changelog::Changelog;
1 use crate::changelog::Changelog;
2 use crate::config::{Config, ConfigError, ConfigParseError};
2 use crate::config::{Config, ConfigError, ConfigParseError};
3 use crate::dirstate::DirstateParents;
3 use crate::dirstate::DirstateParents;
4 use crate::dirstate_tree::on_disk::Docket as DirstateDocket;
4 use crate::dirstate_tree::on_disk::Docket as DirstateDocket;
5 use crate::dirstate_tree::owning::OwningDirstateMap;
5 use crate::dirstate_tree::owning::OwningDirstateMap;
6 use crate::errors::HgResultExt;
6 use crate::errors::HgResultExt;
7 use crate::errors::{HgError, IoResultExt};
7 use crate::errors::{HgError, IoResultExt};
8 use crate::lock::{try_with_lock_no_wait, LockError};
8 use crate::lock::{try_with_lock_no_wait, LockError};
9 use crate::manifest::{Manifest, Manifestlog};
9 use crate::manifest::{Manifest, Manifestlog};
10 use crate::revlog::filelog::Filelog;
10 use crate::revlog::filelog::Filelog;
11 use crate::revlog::revlog::RevlogError;
11 use crate::revlog::revlog::RevlogError;
12 use crate::utils::files::get_path_from_bytes;
12 use crate::utils::files::get_path_from_bytes;
13 use crate::utils::hg_path::HgPath;
13 use crate::utils::hg_path::HgPath;
14 use crate::utils::SliceExt;
14 use crate::utils::SliceExt;
15 use crate::vfs::{is_dir, is_file, Vfs};
15 use crate::vfs::{is_dir, is_file, Vfs};
16 use crate::{requirements, NodePrefix};
16 use crate::{requirements, NodePrefix};
17 use crate::{DirstateError, Revision};
17 use crate::{DirstateError, Revision};
18 use std::cell::{Ref, RefCell, RefMut};
18 use std::cell::{Ref, RefCell, RefMut};
19 use std::collections::HashSet;
19 use std::collections::HashSet;
20 use std::io::Seek;
20 use std::io::Seek;
21 use std::io::SeekFrom;
21 use std::io::SeekFrom;
22 use std::io::Write as IoWrite;
22 use std::io::Write as IoWrite;
23 use std::path::{Path, PathBuf};
23 use std::path::{Path, PathBuf};
24
24
25 /// A repository on disk
25 /// A repository on disk
26 pub struct Repo {
26 pub struct Repo {
27 working_directory: PathBuf,
27 working_directory: PathBuf,
28 dot_hg: PathBuf,
28 dot_hg: PathBuf,
29 store: PathBuf,
29 store: PathBuf,
30 requirements: HashSet<String>,
30 requirements: HashSet<String>,
31 config: Config,
31 config: Config,
32 dirstate_parents: LazyCell<DirstateParents>,
32 dirstate_parents: LazyCell<DirstateParents>,
33 dirstate_data_file_uuid: LazyCell<Option<Vec<u8>>>,
33 dirstate_data_file_uuid: LazyCell<Option<Vec<u8>>>,
34 dirstate_map: LazyCell<OwningDirstateMap>,
34 dirstate_map: LazyCell<OwningDirstateMap>,
35 changelog: LazyCell<Changelog>,
35 changelog: LazyCell<Changelog>,
36 manifestlog: LazyCell<Manifestlog>,
36 manifestlog: LazyCell<Manifestlog>,
37 }
37 }
38
38
39 #[derive(Debug, derive_more::From)]
39 #[derive(Debug, derive_more::From)]
40 pub enum RepoError {
40 pub enum RepoError {
41 NotFound {
41 NotFound {
42 at: PathBuf,
42 at: PathBuf,
43 },
43 },
44 #[from]
44 #[from]
45 ConfigParseError(ConfigParseError),
45 ConfigParseError(ConfigParseError),
46 #[from]
46 #[from]
47 Other(HgError),
47 Other(HgError),
48 }
48 }
49
49
50 impl From<ConfigError> for RepoError {
50 impl From<ConfigError> for RepoError {
51 fn from(error: ConfigError) -> Self {
51 fn from(error: ConfigError) -> Self {
52 match error {
52 match error {
53 ConfigError::Parse(error) => error.into(),
53 ConfigError::Parse(error) => error.into(),
54 ConfigError::Other(error) => error.into(),
54 ConfigError::Other(error) => error.into(),
55 }
55 }
56 }
56 }
57 }
57 }
58
58
59 impl Repo {
59 impl Repo {
60 /// tries to find nearest repository root in current working directory or
60 /// tries to find nearest repository root in current working directory or
61 /// its ancestors
61 /// its ancestors
62 pub fn find_repo_root() -> Result<PathBuf, RepoError> {
62 pub fn find_repo_root() -> Result<PathBuf, RepoError> {
63 let current_directory = crate::utils::current_dir()?;
63 let current_directory = crate::utils::current_dir()?;
64 // ancestors() is inclusive: it first yields `current_directory`
64 // ancestors() is inclusive: it first yields `current_directory`
65 // as-is.
65 // as-is.
66 for ancestor in current_directory.ancestors() {
66 for ancestor in current_directory.ancestors() {
67 if is_dir(ancestor.join(".hg"))? {
67 if is_dir(ancestor.join(".hg"))? {
68 return Ok(ancestor.to_path_buf());
68 return Ok(ancestor.to_path_buf());
69 }
69 }
70 }
70 }
71 return Err(RepoError::NotFound {
71 return Err(RepoError::NotFound {
72 at: current_directory,
72 at: current_directory,
73 });
73 });
74 }
74 }
75
75
76 /// Find a repository, either at the given path (which must contain a `.hg`
76 /// Find a repository, either at the given path (which must contain a `.hg`
77 /// sub-directory) or by searching the current directory and its
77 /// sub-directory) or by searching the current directory and its
78 /// ancestors.
78 /// ancestors.
79 ///
79 ///
80 /// A method with two very different "modes" like this usually a code smell
80 /// A method with two very different "modes" like this usually a code smell
81 /// to make two methods instead, but in this case an `Option` is what rhg
81 /// to make two methods instead, but in this case an `Option` is what rhg
82 /// sub-commands get from Clap for the `-R` / `--repository` CLI argument.
82 /// sub-commands get from Clap for the `-R` / `--repository` CLI argument.
83 /// Having two methods would just move that `if` to almost all callers.
83 /// Having two methods would just move that `if` to almost all callers.
84 pub fn find(
84 pub fn find(
85 config: &Config,
85 config: &Config,
86 explicit_path: Option<PathBuf>,
86 explicit_path: Option<PathBuf>,
87 ) -> Result<Self, RepoError> {
87 ) -> Result<Self, RepoError> {
88 if let Some(root) = explicit_path {
88 if let Some(root) = explicit_path {
89 if is_dir(root.join(".hg"))? {
89 if is_dir(root.join(".hg"))? {
90 Self::new_at_path(root.to_owned(), config)
90 Self::new_at_path(root.to_owned(), config)
91 } else if is_file(&root)? {
91 } else if is_file(&root)? {
92 Err(HgError::unsupported("bundle repository").into())
92 Err(HgError::unsupported("bundle repository").into())
93 } else {
93 } else {
94 Err(RepoError::NotFound {
94 Err(RepoError::NotFound {
95 at: root.to_owned(),
95 at: root.to_owned(),
96 })
96 })
97 }
97 }
98 } else {
98 } else {
99 let root = Self::find_repo_root()?;
99 let root = Self::find_repo_root()?;
100 Self::new_at_path(root, config)
100 Self::new_at_path(root, config)
101 }
101 }
102 }
102 }
103
103
104 /// To be called after checking that `.hg` is a sub-directory
104 /// To be called after checking that `.hg` is a sub-directory
105 fn new_at_path(
105 fn new_at_path(
106 working_directory: PathBuf,
106 working_directory: PathBuf,
107 config: &Config,
107 config: &Config,
108 ) -> Result<Self, RepoError> {
108 ) -> Result<Self, RepoError> {
109 let dot_hg = working_directory.join(".hg");
109 let dot_hg = working_directory.join(".hg");
110
110
111 let mut repo_config_files = Vec::new();
111 let mut repo_config_files = Vec::new();
112 repo_config_files.push(dot_hg.join("hgrc"));
112 repo_config_files.push(dot_hg.join("hgrc"));
113 repo_config_files.push(dot_hg.join("hgrc-not-shared"));
113 repo_config_files.push(dot_hg.join("hgrc-not-shared"));
114
114
115 let hg_vfs = Vfs { base: &dot_hg };
115 let hg_vfs = Vfs { base: &dot_hg };
116 let mut reqs = requirements::load_if_exists(hg_vfs)?;
116 let mut reqs = requirements::load_if_exists(hg_vfs)?;
117 let relative =
117 let relative =
118 reqs.contains(requirements::RELATIVE_SHARED_REQUIREMENT);
118 reqs.contains(requirements::RELATIVE_SHARED_REQUIREMENT);
119 let shared =
119 let shared =
120 reqs.contains(requirements::SHARED_REQUIREMENT) || relative;
120 reqs.contains(requirements::SHARED_REQUIREMENT) || relative;
121
121
122 // From `mercurial/localrepo.py`:
122 // From `mercurial/localrepo.py`:
123 //
123 //
124 // if .hg/requires contains the sharesafe requirement, it means
124 // if .hg/requires contains the sharesafe requirement, it means
125 // there exists a `.hg/store/requires` too and we should read it
125 // there exists a `.hg/store/requires` too and we should read it
126 // NOTE: presence of SHARESAFE_REQUIREMENT imply that store requirement
126 // NOTE: presence of SHARESAFE_REQUIREMENT imply that store requirement
127 // is present. We never write SHARESAFE_REQUIREMENT for a repo if store
127 // is present. We never write SHARESAFE_REQUIREMENT for a repo if store
128 // is not present, refer checkrequirementscompat() for that
128 // is not present, refer checkrequirementscompat() for that
129 //
129 //
130 // However, if SHARESAFE_REQUIREMENT is not present, it means that the
130 // However, if SHARESAFE_REQUIREMENT is not present, it means that the
131 // repository was shared the old way. We check the share source
131 // repository was shared the old way. We check the share source
132 // .hg/requires for SHARESAFE_REQUIREMENT to detect whether the
132 // .hg/requires for SHARESAFE_REQUIREMENT to detect whether the
133 // current repository needs to be reshared
133 // current repository needs to be reshared
134 let share_safe = reqs.contains(requirements::SHARESAFE_REQUIREMENT);
134 let share_safe = reqs.contains(requirements::SHARESAFE_REQUIREMENT);
135
135
136 let store_path;
136 let store_path;
137 if !shared {
137 if !shared {
138 store_path = dot_hg.join("store");
138 store_path = dot_hg.join("store");
139 } else {
139 } else {
140 let bytes = hg_vfs.read("sharedpath")?;
140 let bytes = hg_vfs.read("sharedpath")?;
141 let mut shared_path =
141 let mut shared_path =
142 get_path_from_bytes(bytes.trim_end_matches(|b| b == b'\n'))
142 get_path_from_bytes(bytes.trim_end_matches(|b| b == b'\n'))
143 .to_owned();
143 .to_owned();
144 if relative {
144 if relative {
145 shared_path = dot_hg.join(shared_path)
145 shared_path = dot_hg.join(shared_path)
146 }
146 }
147 if !is_dir(&shared_path)? {
147 if !is_dir(&shared_path)? {
148 return Err(HgError::corrupted(format!(
148 return Err(HgError::corrupted(format!(
149 ".hg/sharedpath points to nonexistent directory {}",
149 ".hg/sharedpath points to nonexistent directory {}",
150 shared_path.display()
150 shared_path.display()
151 ))
151 ))
152 .into());
152 .into());
153 }
153 }
154
154
155 store_path = shared_path.join("store");
155 store_path = shared_path.join("store");
156
156
157 let source_is_share_safe =
157 let source_is_share_safe =
158 requirements::load(Vfs { base: &shared_path })?
158 requirements::load(Vfs { base: &shared_path })?
159 .contains(requirements::SHARESAFE_REQUIREMENT);
159 .contains(requirements::SHARESAFE_REQUIREMENT);
160
160
161 if share_safe != source_is_share_safe {
161 if share_safe != source_is_share_safe {
162 return Err(HgError::unsupported("share-safe mismatch").into());
162 return Err(HgError::unsupported("share-safe mismatch").into());
163 }
163 }
164
164
165 if share_safe {
165 if share_safe {
166 repo_config_files.insert(0, shared_path.join("hgrc"))
166 repo_config_files.insert(0, shared_path.join("hgrc"))
167 }
167 }
168 }
168 }
169 if share_safe {
169 if share_safe {
170 reqs.extend(requirements::load(Vfs { base: &store_path })?);
170 reqs.extend(requirements::load(Vfs { base: &store_path })?);
171 }
171 }
172
172
173 let repo_config = if std::env::var_os("HGRCSKIPREPO").is_none() {
173 let repo_config = if std::env::var_os("HGRCSKIPREPO").is_none() {
174 config.combine_with_repo(&repo_config_files)?
174 config.combine_with_repo(&repo_config_files)?
175 } else {
175 } else {
176 config.clone()
176 config.clone()
177 };
177 };
178
178
179 let repo = Self {
179 let repo = Self {
180 requirements: reqs,
180 requirements: reqs,
181 working_directory,
181 working_directory,
182 store: store_path,
182 store: store_path,
183 dot_hg,
183 dot_hg,
184 config: repo_config,
184 config: repo_config,
185 dirstate_parents: LazyCell::new(),
185 dirstate_parents: LazyCell::new(),
186 dirstate_data_file_uuid: LazyCell::new(),
186 dirstate_data_file_uuid: LazyCell::new(),
187 dirstate_map: LazyCell::new(),
187 dirstate_map: LazyCell::new(),
188 changelog: LazyCell::new(),
188 changelog: LazyCell::new(),
189 manifestlog: LazyCell::new(),
189 manifestlog: LazyCell::new(),
190 };
190 };
191
191
192 requirements::check(&repo)?;
192 requirements::check(&repo)?;
193
193
194 Ok(repo)
194 Ok(repo)
195 }
195 }
196
196
197 pub fn working_directory_path(&self) -> &Path {
197 pub fn working_directory_path(&self) -> &Path {
198 &self.working_directory
198 &self.working_directory
199 }
199 }
200
200
201 pub fn requirements(&self) -> &HashSet<String> {
201 pub fn requirements(&self) -> &HashSet<String> {
202 &self.requirements
202 &self.requirements
203 }
203 }
204
204
205 pub fn config(&self) -> &Config {
205 pub fn config(&self) -> &Config {
206 &self.config
206 &self.config
207 }
207 }
208
208
209 /// For accessing repository files (in `.hg`), except for the store
209 /// For accessing repository files (in `.hg`), except for the store
210 /// (`.hg/store`).
210 /// (`.hg/store`).
211 pub fn hg_vfs(&self) -> Vfs<'_> {
211 pub fn hg_vfs(&self) -> Vfs<'_> {
212 Vfs { base: &self.dot_hg }
212 Vfs { base: &self.dot_hg }
213 }
213 }
214
214
215 /// For accessing repository store files (in `.hg/store`)
215 /// For accessing repository store files (in `.hg/store`)
216 pub fn store_vfs(&self) -> Vfs<'_> {
216 pub fn store_vfs(&self) -> Vfs<'_> {
217 Vfs { base: &self.store }
217 Vfs { base: &self.store }
218 }
218 }
219
219
220 /// For accessing the working copy
220 /// For accessing the working copy
221 pub fn working_directory_vfs(&self) -> Vfs<'_> {
221 pub fn working_directory_vfs(&self) -> Vfs<'_> {
222 Vfs {
222 Vfs {
223 base: &self.working_directory,
223 base: &self.working_directory,
224 }
224 }
225 }
225 }
226
226
227 pub fn try_with_wlock_no_wait<R>(
227 pub fn try_with_wlock_no_wait<R>(
228 &self,
228 &self,
229 f: impl FnOnce() -> R,
229 f: impl FnOnce() -> R,
230 ) -> Result<R, LockError> {
230 ) -> Result<R, LockError> {
231 try_with_lock_no_wait(self.hg_vfs(), "wlock", f)
231 try_with_lock_no_wait(self.hg_vfs(), "wlock", f)
232 }
232 }
233
233
234 pub fn has_dirstate_v2(&self) -> bool {
234 pub fn has_dirstate_v2(&self) -> bool {
235 self.requirements
235 self.requirements
236 .contains(requirements::DIRSTATE_V2_REQUIREMENT)
236 .contains(requirements::DIRSTATE_V2_REQUIREMENT)
237 }
237 }
238
238
239 pub fn has_sparse(&self) -> bool {
239 pub fn has_sparse(&self) -> bool {
240 self.requirements.contains(requirements::SPARSE_REQUIREMENT)
240 self.requirements.contains(requirements::SPARSE_REQUIREMENT)
241 }
241 }
242
242
243 pub fn has_narrow(&self) -> bool {
243 pub fn has_narrow(&self) -> bool {
244 self.requirements.contains(requirements::NARROW_REQUIREMENT)
244 self.requirements.contains(requirements::NARROW_REQUIREMENT)
245 }
245 }
246
246
247 pub fn has_nodemap(&self) -> bool {
247 pub fn has_nodemap(&self) -> bool {
248 self.requirements
248 self.requirements
249 .contains(requirements::NODEMAP_REQUIREMENT)
249 .contains(requirements::NODEMAP_REQUIREMENT)
250 }
250 }
251
251
252 fn dirstate_file_contents(&self) -> Result<Vec<u8>, HgError> {
252 fn dirstate_file_contents(&self) -> Result<Vec<u8>, HgError> {
253 Ok(self
253 Ok(self
254 .hg_vfs()
254 .hg_vfs()
255 .read("dirstate")
255 .read("dirstate")
256 .io_not_found_as_none()?
256 .io_not_found_as_none()?
257 .unwrap_or(Vec::new()))
257 .unwrap_or(Vec::new()))
258 }
258 }
259
259
260 pub fn dirstate_parents(&self) -> Result<DirstateParents, HgError> {
260 pub fn dirstate_parents(&self) -> Result<DirstateParents, HgError> {
261 Ok(*self
261 Ok(*self
262 .dirstate_parents
262 .dirstate_parents
263 .get_or_init(|| self.read_dirstate_parents())?)
263 .get_or_init(|| self.read_dirstate_parents())?)
264 }
264 }
265
265
266 fn read_dirstate_parents(&self) -> Result<DirstateParents, HgError> {
266 fn read_dirstate_parents(&self) -> Result<DirstateParents, HgError> {
267 let dirstate = self.dirstate_file_contents()?;
267 let dirstate = self.dirstate_file_contents()?;
268 let parents = if dirstate.is_empty() {
268 let parents = if dirstate.is_empty() {
269 if self.has_dirstate_v2() {
269 if self.has_dirstate_v2() {
270 self.dirstate_data_file_uuid.set(None);
270 self.dirstate_data_file_uuid.set(None);
271 }
271 }
272 DirstateParents::NULL
272 DirstateParents::NULL
273 } else if self.has_dirstate_v2() {
273 } else if self.has_dirstate_v2() {
274 let docket =
274 let docket =
275 crate::dirstate_tree::on_disk::read_docket(&dirstate)?;
275 crate::dirstate_tree::on_disk::read_docket(&dirstate)?;
276 self.dirstate_data_file_uuid
276 self.dirstate_data_file_uuid
277 .set(Some(docket.uuid.to_owned()));
277 .set(Some(docket.uuid.to_owned()));
278 docket.parents()
278 docket.parents()
279 } else {
279 } else {
280 crate::dirstate::parsers::parse_dirstate_parents(&dirstate)?
280 crate::dirstate::parsers::parse_dirstate_parents(&dirstate)?
281 .clone()
281 .clone()
282 };
282 };
283 self.dirstate_parents.set(parents);
283 self.dirstate_parents.set(parents);
284 Ok(parents)
284 Ok(parents)
285 }
285 }
286
286
287 fn read_dirstate_data_file_uuid(
287 fn read_dirstate_data_file_uuid(
288 &self,
288 &self,
289 ) -> Result<Option<Vec<u8>>, HgError> {
289 ) -> Result<Option<Vec<u8>>, HgError> {
290 assert!(
290 assert!(
291 self.has_dirstate_v2(),
291 self.has_dirstate_v2(),
292 "accessing dirstate data file ID without dirstate-v2"
292 "accessing dirstate data file ID without dirstate-v2"
293 );
293 );
294 let dirstate = self.dirstate_file_contents()?;
294 let dirstate = self.dirstate_file_contents()?;
295 if dirstate.is_empty() {
295 if dirstate.is_empty() {
296 self.dirstate_parents.set(DirstateParents::NULL);
296 self.dirstate_parents.set(DirstateParents::NULL);
297 Ok(None)
297 Ok(None)
298 } else {
298 } else {
299 let docket =
299 let docket =
300 crate::dirstate_tree::on_disk::read_docket(&dirstate)?;
300 crate::dirstate_tree::on_disk::read_docket(&dirstate)?;
301 self.dirstate_parents.set(docket.parents());
301 self.dirstate_parents.set(docket.parents());
302 Ok(Some(docket.uuid.to_owned()))
302 Ok(Some(docket.uuid.to_owned()))
303 }
303 }
304 }
304 }
305
305
306 fn new_dirstate_map(&self) -> Result<OwningDirstateMap, DirstateError> {
306 fn new_dirstate_map(&self) -> Result<OwningDirstateMap, DirstateError> {
307 let dirstate_file_contents = self.dirstate_file_contents()?;
307 let dirstate_file_contents = self.dirstate_file_contents()?;
308 if dirstate_file_contents.is_empty() {
308 if dirstate_file_contents.is_empty() {
309 self.dirstate_parents.set(DirstateParents::NULL);
309 self.dirstate_parents.set(DirstateParents::NULL);
310 if self.has_dirstate_v2() {
310 if self.has_dirstate_v2() {
311 self.dirstate_data_file_uuid.set(None);
311 self.dirstate_data_file_uuid.set(None);
312 }
312 }
313 Ok(OwningDirstateMap::new_empty(Vec::new()))
313 Ok(OwningDirstateMap::new_empty(Vec::new()))
314 } else if self.has_dirstate_v2() {
314 } else if self.has_dirstate_v2() {
315 let docket = crate::dirstate_tree::on_disk::read_docket(
315 let docket = crate::dirstate_tree::on_disk::read_docket(
316 &dirstate_file_contents,
316 &dirstate_file_contents,
317 )?;
317 )?;
318 self.dirstate_parents.set(docket.parents());
318 self.dirstate_parents.set(docket.parents());
319 self.dirstate_data_file_uuid
319 self.dirstate_data_file_uuid
320 .set(Some(docket.uuid.to_owned()));
320 .set(Some(docket.uuid.to_owned()));
321 let data_size = docket.data_size();
321 let data_size = docket.data_size();
322 let metadata = docket.tree_metadata();
322 let metadata = docket.tree_metadata();
323 if let Some(data_mmap) = self
323 if crate::vfs::is_on_nfs_mount(docket.data_filename()) {
324 // Don't mmap on NFS to prevent `SIGBUS` error on deletion
325 OwningDirstateMap::new_v2(
326 self.hg_vfs().read(docket.data_filename())?,
327 data_size,
328 metadata,
329 )
330 } else if let Some(data_mmap) = self
324 .hg_vfs()
331 .hg_vfs()
325 .mmap_open(docket.data_filename())
332 .mmap_open(docket.data_filename())
326 .io_not_found_as_none()?
333 .io_not_found_as_none()?
327 {
334 {
328 OwningDirstateMap::new_v2(data_mmap, data_size, metadata)
335 OwningDirstateMap::new_v2(data_mmap, data_size, metadata)
329 } else {
336 } else {
330 OwningDirstateMap::new_v2(Vec::new(), data_size, metadata)
337 OwningDirstateMap::new_v2(Vec::new(), data_size, metadata)
331 }
338 }
332 } else {
339 } else {
333 let (map, parents) =
340 let (map, parents) =
334 OwningDirstateMap::new_v1(dirstate_file_contents)?;
341 OwningDirstateMap::new_v1(dirstate_file_contents)?;
335 self.dirstate_parents.set(parents);
342 self.dirstate_parents.set(parents);
336 Ok(map)
343 Ok(map)
337 }
344 }
338 }
345 }
339
346
340 pub fn dirstate_map(
347 pub fn dirstate_map(
341 &self,
348 &self,
342 ) -> Result<Ref<OwningDirstateMap>, DirstateError> {
349 ) -> Result<Ref<OwningDirstateMap>, DirstateError> {
343 self.dirstate_map.get_or_init(|| self.new_dirstate_map())
350 self.dirstate_map.get_or_init(|| self.new_dirstate_map())
344 }
351 }
345
352
346 pub fn dirstate_map_mut(
353 pub fn dirstate_map_mut(
347 &self,
354 &self,
348 ) -> Result<RefMut<OwningDirstateMap>, DirstateError> {
355 ) -> Result<RefMut<OwningDirstateMap>, DirstateError> {
349 self.dirstate_map
356 self.dirstate_map
350 .get_mut_or_init(|| self.new_dirstate_map())
357 .get_mut_or_init(|| self.new_dirstate_map())
351 }
358 }
352
359
353 fn new_changelog(&self) -> Result<Changelog, HgError> {
360 fn new_changelog(&self) -> Result<Changelog, HgError> {
354 Changelog::open(&self.store_vfs(), self.has_nodemap())
361 Changelog::open(&self.store_vfs(), self.has_nodemap())
355 }
362 }
356
363
357 pub fn changelog(&self) -> Result<Ref<Changelog>, HgError> {
364 pub fn changelog(&self) -> Result<Ref<Changelog>, HgError> {
358 self.changelog.get_or_init(|| self.new_changelog())
365 self.changelog.get_or_init(|| self.new_changelog())
359 }
366 }
360
367
361 pub fn changelog_mut(&self) -> Result<RefMut<Changelog>, HgError> {
368 pub fn changelog_mut(&self) -> Result<RefMut<Changelog>, HgError> {
362 self.changelog.get_mut_or_init(|| self.new_changelog())
369 self.changelog.get_mut_or_init(|| self.new_changelog())
363 }
370 }
364
371
365 fn new_manifestlog(&self) -> Result<Manifestlog, HgError> {
372 fn new_manifestlog(&self) -> Result<Manifestlog, HgError> {
366 Manifestlog::open(&self.store_vfs(), self.has_nodemap())
373 Manifestlog::open(&self.store_vfs(), self.has_nodemap())
367 }
374 }
368
375
369 pub fn manifestlog(&self) -> Result<Ref<Manifestlog>, HgError> {
376 pub fn manifestlog(&self) -> Result<Ref<Manifestlog>, HgError> {
370 self.manifestlog.get_or_init(|| self.new_manifestlog())
377 self.manifestlog.get_or_init(|| self.new_manifestlog())
371 }
378 }
372
379
373 pub fn manifestlog_mut(&self) -> Result<RefMut<Manifestlog>, HgError> {
380 pub fn manifestlog_mut(&self) -> Result<RefMut<Manifestlog>, HgError> {
374 self.manifestlog.get_mut_or_init(|| self.new_manifestlog())
381 self.manifestlog.get_mut_or_init(|| self.new_manifestlog())
375 }
382 }
376
383
377 /// Returns the manifest of the *changeset* with the given node ID
384 /// Returns the manifest of the *changeset* with the given node ID
378 pub fn manifest_for_node(
385 pub fn manifest_for_node(
379 &self,
386 &self,
380 node: impl Into<NodePrefix>,
387 node: impl Into<NodePrefix>,
381 ) -> Result<Manifest, RevlogError> {
388 ) -> Result<Manifest, RevlogError> {
382 self.manifestlog()?.data_for_node(
389 self.manifestlog()?.data_for_node(
383 self.changelog()?
390 self.changelog()?
384 .data_for_node(node.into())?
391 .data_for_node(node.into())?
385 .manifest_node()?
392 .manifest_node()?
386 .into(),
393 .into(),
387 )
394 )
388 }
395 }
389
396
390 /// Returns the manifest of the *changeset* with the given revision number
397 /// Returns the manifest of the *changeset* with the given revision number
391 pub fn manifest_for_rev(
398 pub fn manifest_for_rev(
392 &self,
399 &self,
393 revision: Revision,
400 revision: Revision,
394 ) -> Result<Manifest, RevlogError> {
401 ) -> Result<Manifest, RevlogError> {
395 self.manifestlog()?.data_for_node(
402 self.manifestlog()?.data_for_node(
396 self.changelog()?
403 self.changelog()?
397 .data_for_rev(revision)?
404 .data_for_rev(revision)?
398 .manifest_node()?
405 .manifest_node()?
399 .into(),
406 .into(),
400 )
407 )
401 }
408 }
402
409
403 pub fn has_subrepos(&self) -> Result<bool, DirstateError> {
410 pub fn has_subrepos(&self) -> Result<bool, DirstateError> {
404 if let Some(entry) = self.dirstate_map()?.get(HgPath::new(".hgsub"))? {
411 if let Some(entry) = self.dirstate_map()?.get(HgPath::new(".hgsub"))? {
405 Ok(entry.tracked())
412 Ok(entry.tracked())
406 } else {
413 } else {
407 Ok(false)
414 Ok(false)
408 }
415 }
409 }
416 }
410
417
411 pub fn filelog(&self, path: &HgPath) -> Result<Filelog, HgError> {
418 pub fn filelog(&self, path: &HgPath) -> Result<Filelog, HgError> {
412 Filelog::open(self, path)
419 Filelog::open(self, path)
413 }
420 }
414
421
415 /// Write to disk any updates that were made through `dirstate_map_mut`.
422 /// Write to disk any updates that were made through `dirstate_map_mut`.
416 ///
423 ///
417 /// The "wlock" must be held while calling this.
424 /// The "wlock" must be held while calling this.
418 /// See for example `try_with_wlock_no_wait`.
425 /// See for example `try_with_wlock_no_wait`.
419 ///
426 ///
420 /// TODO: have a `WritableRepo` type only accessible while holding the
427 /// TODO: have a `WritableRepo` type only accessible while holding the
421 /// lock?
428 /// lock?
422 pub fn write_dirstate(&self) -> Result<(), DirstateError> {
429 pub fn write_dirstate(&self) -> Result<(), DirstateError> {
423 let map = self.dirstate_map()?;
430 let map = self.dirstate_map()?;
424 // TODO: Maintain a `DirstateMap::dirty` flag, and return early here if
431 // TODO: Maintain a `DirstateMap::dirty` flag, and return early here if
425 // it’s unset
432 // it’s unset
426 let parents = self.dirstate_parents()?;
433 let parents = self.dirstate_parents()?;
427 let (packed_dirstate, old_uuid_to_remove) = if self.has_dirstate_v2() {
434 let (packed_dirstate, old_uuid_to_remove) = if self.has_dirstate_v2() {
428 let uuid_opt = self
435 let uuid_opt = self
429 .dirstate_data_file_uuid
436 .dirstate_data_file_uuid
430 .get_or_init(|| self.read_dirstate_data_file_uuid())?;
437 .get_or_init(|| self.read_dirstate_data_file_uuid())?;
431 let uuid_opt = uuid_opt.as_ref();
438 let uuid_opt = uuid_opt.as_ref();
432 let can_append = uuid_opt.is_some();
439 let can_append = uuid_opt.is_some();
433 let (data, tree_metadata, append, old_data_size) =
440 let (data, tree_metadata, append, old_data_size) =
434 map.pack_v2(can_append)?;
441 map.pack_v2(can_append)?;
435
442
436 // Reuse the uuid, or generate a new one, keeping the old for
443 // Reuse the uuid, or generate a new one, keeping the old for
437 // deletion.
444 // deletion.
438 let (uuid, old_uuid) = match uuid_opt {
445 let (uuid, old_uuid) = match uuid_opt {
439 Some(uuid) => {
446 Some(uuid) => {
440 let as_str = std::str::from_utf8(uuid)
447 let as_str = std::str::from_utf8(uuid)
441 .map_err(|_| {
448 .map_err(|_| {
442 HgError::corrupted(
449 HgError::corrupted(
443 "non-UTF-8 dirstate data file ID",
450 "non-UTF-8 dirstate data file ID",
444 )
451 )
445 })?
452 })?
446 .to_owned();
453 .to_owned();
447 if append {
454 if append {
448 (as_str, None)
455 (as_str, None)
449 } else {
456 } else {
450 (DirstateDocket::new_uid(), Some(as_str))
457 (DirstateDocket::new_uid(), Some(as_str))
451 }
458 }
452 }
459 }
453 None => (DirstateDocket::new_uid(), None),
460 None => (DirstateDocket::new_uid(), None),
454 };
461 };
455
462
456 let data_filename = format!("dirstate.{}", uuid);
463 let data_filename = format!("dirstate.{}", uuid);
457 let data_filename = self.hg_vfs().join(data_filename);
464 let data_filename = self.hg_vfs().join(data_filename);
458 let mut options = std::fs::OpenOptions::new();
465 let mut options = std::fs::OpenOptions::new();
459 options.write(true);
466 options.write(true);
460
467
461 // Why are we not using the O_APPEND flag when appending?
468 // Why are we not using the O_APPEND flag when appending?
462 //
469 //
463 // - O_APPEND makes it trickier to deal with garbage at the end of
470 // - O_APPEND makes it trickier to deal with garbage at the end of
464 // the file, left by a previous uncommitted transaction. By
471 // the file, left by a previous uncommitted transaction. By
465 // starting the write at [old_data_size] we make sure we erase
472 // starting the write at [old_data_size] we make sure we erase
466 // all such garbage.
473 // all such garbage.
467 //
474 //
468 // - O_APPEND requires to special-case 0-byte writes, whereas we
475 // - O_APPEND requires to special-case 0-byte writes, whereas we
469 // don't need that.
476 // don't need that.
470 //
477 //
471 // - Some OSes have bugs in implementation O_APPEND:
478 // - Some OSes have bugs in implementation O_APPEND:
472 // revlog.py talks about a Solaris bug, but we also saw some ZFS
479 // revlog.py talks about a Solaris bug, but we also saw some ZFS
473 // bug: https://github.com/openzfs/zfs/pull/3124,
480 // bug: https://github.com/openzfs/zfs/pull/3124,
474 // https://github.com/openzfs/zfs/issues/13370
481 // https://github.com/openzfs/zfs/issues/13370
475 //
482 //
476 if !append {
483 if !append {
477 options.create_new(true);
484 options.create_new(true);
478 }
485 }
479
486
480 let data_size = (|| {
487 let data_size = (|| {
481 // TODO: loop and try another random ID if !append and this
488 // TODO: loop and try another random ID if !append and this
482 // returns `ErrorKind::AlreadyExists`? Collision chance of two
489 // returns `ErrorKind::AlreadyExists`? Collision chance of two
483 // random IDs is one in 2**32
490 // random IDs is one in 2**32
484 let mut file = options.open(&data_filename)?;
491 let mut file = options.open(&data_filename)?;
485 if append {
492 if append {
486 file.seek(SeekFrom::Start(old_data_size as u64))?;
493 file.seek(SeekFrom::Start(old_data_size as u64))?;
487 }
494 }
488 file.write_all(&data)?;
495 file.write_all(&data)?;
489 file.flush()?;
496 file.flush()?;
490 file.seek(SeekFrom::Current(0))
497 file.seek(SeekFrom::Current(0))
491 })()
498 })()
492 .when_writing_file(&data_filename)?;
499 .when_writing_file(&data_filename)?;
493
500
494 let packed_dirstate = DirstateDocket::serialize(
501 let packed_dirstate = DirstateDocket::serialize(
495 parents,
502 parents,
496 tree_metadata,
503 tree_metadata,
497 data_size,
504 data_size,
498 uuid.as_bytes(),
505 uuid.as_bytes(),
499 )
506 )
500 .map_err(|_: std::num::TryFromIntError| {
507 .map_err(|_: std::num::TryFromIntError| {
501 HgError::corrupted("overflow in dirstate docket serialization")
508 HgError::corrupted("overflow in dirstate docket serialization")
502 })?;
509 })?;
503
510
504 (packed_dirstate, old_uuid)
511 (packed_dirstate, old_uuid)
505 } else {
512 } else {
506 (map.pack_v1(parents)?, None)
513 (map.pack_v1(parents)?, None)
507 };
514 };
508
515
509 let vfs = self.hg_vfs();
516 let vfs = self.hg_vfs();
510 vfs.atomic_write("dirstate", &packed_dirstate)?;
517 vfs.atomic_write("dirstate", &packed_dirstate)?;
511 if let Some(uuid) = old_uuid_to_remove {
518 if let Some(uuid) = old_uuid_to_remove {
512 // Remove the old data file after the new docket pointing to the
519 // Remove the old data file after the new docket pointing to the
513 // new data file was written.
520 // new data file was written.
514 vfs.remove_file(format!("dirstate.{}", uuid))?;
521 vfs.remove_file(format!("dirstate.{}", uuid))?;
515 }
522 }
516 Ok(())
523 Ok(())
517 }
524 }
518 }
525 }
519
526
520 /// Lazily-initialized component of `Repo` with interior mutability
527 /// Lazily-initialized component of `Repo` with interior mutability
521 ///
528 ///
522 /// This differs from `OnceCell` in that the value can still be "deinitialized"
529 /// This differs from `OnceCell` in that the value can still be "deinitialized"
523 /// later by setting its inner `Option` to `None`. It also takes the
530 /// later by setting its inner `Option` to `None`. It also takes the
524 /// initialization function as an argument when the value is requested, not
531 /// initialization function as an argument when the value is requested, not
525 /// when the instance is created.
532 /// when the instance is created.
526 struct LazyCell<T> {
533 struct LazyCell<T> {
527 value: RefCell<Option<T>>,
534 value: RefCell<Option<T>>,
528 }
535 }
529
536
530 impl<T> LazyCell<T> {
537 impl<T> LazyCell<T> {
531 fn new() -> Self {
538 fn new() -> Self {
532 Self {
539 Self {
533 value: RefCell::new(None),
540 value: RefCell::new(None),
534 }
541 }
535 }
542 }
536
543
537 fn set(&self, value: T) {
544 fn set(&self, value: T) {
538 *self.value.borrow_mut() = Some(value)
545 *self.value.borrow_mut() = Some(value)
539 }
546 }
540
547
541 fn get_or_init<E>(
548 fn get_or_init<E>(
542 &self,
549 &self,
543 init: impl Fn() -> Result<T, E>,
550 init: impl Fn() -> Result<T, E>,
544 ) -> Result<Ref<T>, E> {
551 ) -> Result<Ref<T>, E> {
545 let mut borrowed = self.value.borrow();
552 let mut borrowed = self.value.borrow();
546 if borrowed.is_none() {
553 if borrowed.is_none() {
547 drop(borrowed);
554 drop(borrowed);
548 // Only use `borrow_mut` if it is really needed to avoid panic in
555 // Only use `borrow_mut` if it is really needed to avoid panic in
549 // case there is another outstanding borrow but mutation is not
556 // case there is another outstanding borrow but mutation is not
550 // needed.
557 // needed.
551 *self.value.borrow_mut() = Some(init()?);
558 *self.value.borrow_mut() = Some(init()?);
552 borrowed = self.value.borrow()
559 borrowed = self.value.borrow()
553 }
560 }
554 Ok(Ref::map(borrowed, |option| option.as_ref().unwrap()))
561 Ok(Ref::map(borrowed, |option| option.as_ref().unwrap()))
555 }
562 }
556
563
557 fn get_mut_or_init<E>(
564 fn get_mut_or_init<E>(
558 &self,
565 &self,
559 init: impl Fn() -> Result<T, E>,
566 init: impl Fn() -> Result<T, E>,
560 ) -> Result<RefMut<T>, E> {
567 ) -> Result<RefMut<T>, E> {
561 let mut borrowed = self.value.borrow_mut();
568 let mut borrowed = self.value.borrow_mut();
562 if borrowed.is_none() {
569 if borrowed.is_none() {
563 *borrowed = Some(init()?);
570 *borrowed = Some(init()?);
564 }
571 }
565 Ok(RefMut::map(borrowed, |option| option.as_mut().unwrap()))
572 Ok(RefMut::map(borrowed, |option| option.as_mut().unwrap()))
566 }
573 }
567 }
574 }
@@ -1,174 +1,195
1 use crate::errors::{HgError, IoErrorContext, IoResultExt};
1 use crate::errors::{HgError, IoErrorContext, IoResultExt};
2 use memmap2::{Mmap, MmapOptions};
2 use memmap2::{Mmap, MmapOptions};
3 use std::io::{ErrorKind, Write};
3 use std::io::{ErrorKind, Write};
4 use std::path::{Path, PathBuf};
4 use std::path::{Path, PathBuf};
5
5
6 /// Filesystem access abstraction for the contents of a given "base" diretory
6 /// Filesystem access abstraction for the contents of a given "base" diretory
7 #[derive(Clone, Copy)]
7 #[derive(Clone, Copy)]
8 pub struct Vfs<'a> {
8 pub struct Vfs<'a> {
9 pub(crate) base: &'a Path,
9 pub(crate) base: &'a Path,
10 }
10 }
11
11
12 struct FileNotFound(std::io::Error, PathBuf);
12 struct FileNotFound(std::io::Error, PathBuf);
13
13
14 impl Vfs<'_> {
14 impl Vfs<'_> {
15 pub fn join(&self, relative_path: impl AsRef<Path>) -> PathBuf {
15 pub fn join(&self, relative_path: impl AsRef<Path>) -> PathBuf {
16 self.base.join(relative_path)
16 self.base.join(relative_path)
17 }
17 }
18
18
19 pub fn symlink_metadata(
19 pub fn symlink_metadata(
20 &self,
20 &self,
21 relative_path: impl AsRef<Path>,
21 relative_path: impl AsRef<Path>,
22 ) -> Result<std::fs::Metadata, HgError> {
22 ) -> Result<std::fs::Metadata, HgError> {
23 let path = self.join(relative_path);
23 let path = self.join(relative_path);
24 std::fs::symlink_metadata(&path).when_reading_file(&path)
24 std::fs::symlink_metadata(&path).when_reading_file(&path)
25 }
25 }
26
26
27 pub fn read_link(
27 pub fn read_link(
28 &self,
28 &self,
29 relative_path: impl AsRef<Path>,
29 relative_path: impl AsRef<Path>,
30 ) -> Result<PathBuf, HgError> {
30 ) -> Result<PathBuf, HgError> {
31 let path = self.join(relative_path);
31 let path = self.join(relative_path);
32 std::fs::read_link(&path).when_reading_file(&path)
32 std::fs::read_link(&path).when_reading_file(&path)
33 }
33 }
34
34
35 pub fn read(
35 pub fn read(
36 &self,
36 &self,
37 relative_path: impl AsRef<Path>,
37 relative_path: impl AsRef<Path>,
38 ) -> Result<Vec<u8>, HgError> {
38 ) -> Result<Vec<u8>, HgError> {
39 let path = self.join(relative_path);
39 let path = self.join(relative_path);
40 std::fs::read(&path).when_reading_file(&path)
40 std::fs::read(&path).when_reading_file(&path)
41 }
41 }
42
42
43 /// Returns `Ok(None)` if the file does not exist.
43 /// Returns `Ok(None)` if the file does not exist.
44 pub fn try_read(
44 pub fn try_read(
45 &self,
45 &self,
46 relative_path: impl AsRef<Path>,
46 relative_path: impl AsRef<Path>,
47 ) -> Result<Option<Vec<u8>>, HgError> {
47 ) -> Result<Option<Vec<u8>>, HgError> {
48 match self.read(relative_path) {
48 match self.read(relative_path) {
49 Err(e) => match &e {
49 Err(e) => match &e {
50 HgError::IoError { error, .. } => match error.kind() {
50 HgError::IoError { error, .. } => match error.kind() {
51 ErrorKind::NotFound => return Ok(None),
51 ErrorKind::NotFound => return Ok(None),
52 _ => Err(e),
52 _ => Err(e),
53 },
53 },
54 _ => Err(e),
54 _ => Err(e),
55 },
55 },
56 Ok(v) => Ok(Some(v)),
56 Ok(v) => Ok(Some(v)),
57 }
57 }
58 }
58 }
59
59
60 fn mmap_open_gen(
60 fn mmap_open_gen(
61 &self,
61 &self,
62 relative_path: impl AsRef<Path>,
62 relative_path: impl AsRef<Path>,
63 ) -> Result<Result<Mmap, FileNotFound>, HgError> {
63 ) -> Result<Result<Mmap, FileNotFound>, HgError> {
64 let path = self.join(relative_path);
64 let path = self.join(relative_path);
65 let file = match std::fs::File::open(&path) {
65 let file = match std::fs::File::open(&path) {
66 Err(err) => {
66 Err(err) => {
67 if let ErrorKind::NotFound = err.kind() {
67 if let ErrorKind::NotFound = err.kind() {
68 return Ok(Err(FileNotFound(err, path)));
68 return Ok(Err(FileNotFound(err, path)));
69 };
69 };
70 return (Err(err)).when_reading_file(&path);
70 return (Err(err)).when_reading_file(&path);
71 }
71 }
72 Ok(file) => file,
72 Ok(file) => file,
73 };
73 };
74 // TODO: what are the safety requirements here?
74 // TODO: what are the safety requirements here?
75 let mmap = unsafe { MmapOptions::new().map(&file) }
75 let mmap = unsafe { MmapOptions::new().map(&file) }
76 .when_reading_file(&path)?;
76 .when_reading_file(&path)?;
77 Ok(Ok(mmap))
77 Ok(Ok(mmap))
78 }
78 }
79
79
80 pub fn mmap_open_opt(
80 pub fn mmap_open_opt(
81 &self,
81 &self,
82 relative_path: impl AsRef<Path>,
82 relative_path: impl AsRef<Path>,
83 ) -> Result<Option<Mmap>, HgError> {
83 ) -> Result<Option<Mmap>, HgError> {
84 self.mmap_open_gen(relative_path).map(|res| res.ok())
84 self.mmap_open_gen(relative_path).map(|res| res.ok())
85 }
85 }
86
86
87 pub fn mmap_open(
87 pub fn mmap_open(
88 &self,
88 &self,
89 relative_path: impl AsRef<Path>,
89 relative_path: impl AsRef<Path>,
90 ) -> Result<Mmap, HgError> {
90 ) -> Result<Mmap, HgError> {
91 match self.mmap_open_gen(relative_path)? {
91 match self.mmap_open_gen(relative_path)? {
92 Err(FileNotFound(err, path)) => Err(err).when_reading_file(&path),
92 Err(FileNotFound(err, path)) => Err(err).when_reading_file(&path),
93 Ok(res) => Ok(res),
93 Ok(res) => Ok(res),
94 }
94 }
95 }
95 }
96
96
97 pub fn rename(
97 pub fn rename(
98 &self,
98 &self,
99 relative_from: impl AsRef<Path>,
99 relative_from: impl AsRef<Path>,
100 relative_to: impl AsRef<Path>,
100 relative_to: impl AsRef<Path>,
101 ) -> Result<(), HgError> {
101 ) -> Result<(), HgError> {
102 let from = self.join(relative_from);
102 let from = self.join(relative_from);
103 let to = self.join(relative_to);
103 let to = self.join(relative_to);
104 std::fs::rename(&from, &to)
104 std::fs::rename(&from, &to)
105 .with_context(|| IoErrorContext::RenamingFile { from, to })
105 .with_context(|| IoErrorContext::RenamingFile { from, to })
106 }
106 }
107
107
108 pub fn remove_file(
108 pub fn remove_file(
109 &self,
109 &self,
110 relative_path: impl AsRef<Path>,
110 relative_path: impl AsRef<Path>,
111 ) -> Result<(), HgError> {
111 ) -> Result<(), HgError> {
112 let path = self.join(relative_path);
112 let path = self.join(relative_path);
113 std::fs::remove_file(&path)
113 std::fs::remove_file(&path)
114 .with_context(|| IoErrorContext::RemovingFile(path))
114 .with_context(|| IoErrorContext::RemovingFile(path))
115 }
115 }
116
116
117 #[cfg(unix)]
117 #[cfg(unix)]
118 pub fn create_symlink(
118 pub fn create_symlink(
119 &self,
119 &self,
120 relative_link_path: impl AsRef<Path>,
120 relative_link_path: impl AsRef<Path>,
121 target_path: impl AsRef<Path>,
121 target_path: impl AsRef<Path>,
122 ) -> Result<(), HgError> {
122 ) -> Result<(), HgError> {
123 let link_path = self.join(relative_link_path);
123 let link_path = self.join(relative_link_path);
124 std::os::unix::fs::symlink(target_path, &link_path)
124 std::os::unix::fs::symlink(target_path, &link_path)
125 .when_writing_file(&link_path)
125 .when_writing_file(&link_path)
126 }
126 }
127
127
128 /// Write `contents` into a temporary file, then rename to `relative_path`.
128 /// Write `contents` into a temporary file, then rename to `relative_path`.
129 /// This makes writing to a file "atomic": a reader opening that path will
129 /// This makes writing to a file "atomic": a reader opening that path will
130 /// see either the previous contents of the file or the complete new
130 /// see either the previous contents of the file or the complete new
131 /// content, never a partial write.
131 /// content, never a partial write.
132 pub fn atomic_write(
132 pub fn atomic_write(
133 &self,
133 &self,
134 relative_path: impl AsRef<Path>,
134 relative_path: impl AsRef<Path>,
135 contents: &[u8],
135 contents: &[u8],
136 ) -> Result<(), HgError> {
136 ) -> Result<(), HgError> {
137 let mut tmp = tempfile::NamedTempFile::new_in(self.base)
137 let mut tmp = tempfile::NamedTempFile::new_in(self.base)
138 .when_writing_file(self.base)?;
138 .when_writing_file(self.base)?;
139 tmp.write_all(contents)
139 tmp.write_all(contents)
140 .and_then(|()| tmp.flush())
140 .and_then(|()| tmp.flush())
141 .when_writing_file(tmp.path())?;
141 .when_writing_file(tmp.path())?;
142 let path = self.join(relative_path);
142 let path = self.join(relative_path);
143 tmp.persist(&path)
143 tmp.persist(&path)
144 .map_err(|e| e.error)
144 .map_err(|e| e.error)
145 .when_writing_file(&path)?;
145 .when_writing_file(&path)?;
146 Ok(())
146 Ok(())
147 }
147 }
148 }
148 }
149
149
150 fn fs_metadata(
150 fn fs_metadata(
151 path: impl AsRef<Path>,
151 path: impl AsRef<Path>,
152 ) -> Result<Option<std::fs::Metadata>, HgError> {
152 ) -> Result<Option<std::fs::Metadata>, HgError> {
153 let path = path.as_ref();
153 let path = path.as_ref();
154 match std::fs::metadata(path) {
154 match std::fs::metadata(path) {
155 Ok(meta) => Ok(Some(meta)),
155 Ok(meta) => Ok(Some(meta)),
156 Err(error) => match error.kind() {
156 Err(error) => match error.kind() {
157 // TODO: when we require a Rust version where `NotADirectory` is
157 // TODO: when we require a Rust version where `NotADirectory` is
158 // stable, invert this logic and return None for it and `NotFound`
158 // stable, invert this logic and return None for it and `NotFound`
159 // and propagate any other error.
159 // and propagate any other error.
160 ErrorKind::PermissionDenied => Err(error).with_context(|| {
160 ErrorKind::PermissionDenied => Err(error).with_context(|| {
161 IoErrorContext::ReadingMetadata(path.to_owned())
161 IoErrorContext::ReadingMetadata(path.to_owned())
162 }),
162 }),
163 _ => Ok(None),
163 _ => Ok(None),
164 },
164 },
165 }
165 }
166 }
166 }
167
167
168 pub(crate) fn is_dir(path: impl AsRef<Path>) -> Result<bool, HgError> {
168 pub(crate) fn is_dir(path: impl AsRef<Path>) -> Result<bool, HgError> {
169 Ok(fs_metadata(path)?.map_or(false, |meta| meta.is_dir()))
169 Ok(fs_metadata(path)?.map_or(false, |meta| meta.is_dir()))
170 }
170 }
171
171
172 pub(crate) fn is_file(path: impl AsRef<Path>) -> Result<bool, HgError> {
172 pub(crate) fn is_file(path: impl AsRef<Path>) -> Result<bool, HgError> {
173 Ok(fs_metadata(path)?.map_or(false, |meta| meta.is_file()))
173 Ok(fs_metadata(path)?.map_or(false, |meta| meta.is_file()))
174 }
174 }
175
176 /// Returns whether the given `path` is on a network file system.
177 /// Taken from `cargo`'s codebase.
178 #[cfg(target_os = "linux")]
179 pub(crate) fn is_on_nfs_mount(path: impl AsRef<Path>) -> bool {
180 use std::ffi::CString;
181 use std::mem;
182 use std::os::unix::prelude::*;
183
184 let path = match CString::new(path.as_ref().as_os_str().as_bytes()) {
185 Ok(path) => path,
186 Err(_) => return false,
187 };
188
189 unsafe {
190 let mut buf: libc::statfs = mem::zeroed();
191 let r = libc::statfs(path.as_ptr(), &mut buf);
192
193 r == 0 && buf.f_type as u32 == libc::NFS_SUPER_MAGIC as u32
194 }
195 }
General Comments 0
You need to be logged in to leave comments. Login now