use crate::fs::{errors, open, OpenOptions, Permissions};
use posish::{
fs::{fchmod, Mode},
io::Error,
};
#[cfg(unix)]
use std::os::unix::fs::PermissionsExt;
use std::{convert::TryInto, fs, io, path::Path};
pub(crate) fn set_permissions_impl(
start: &fs::File,
path: &Path,
perm: Permissions,
) -> io::Result<()> {
let std_perm = perm.into_std(start)?;
match open(start, path, OpenOptions::new().read(true)) {
Ok(file) => return set_file_permissions(&file, std_perm),
Err(err) => match Error::from_io_error(&err) {
Some(Error::ACCES) => (),
_ => return Err(err),
},
}
match open(start, path, OpenOptions::new().write(true)) {
Ok(file) => return set_file_permissions(&file, std_perm),
Err(err) => match Error::from_io_error(&err) {
Some(Error::ACCES) | Some(Error::ISDIR) => (),
_ => return Err(err),
},
}
Err(Error::NOTSUP.into())
}
pub(crate) fn set_file_permissions(file: &fs::File, perm: fs::Permissions) -> io::Result<()> {
#[allow(clippy::useless_conversion)]
let mode =
Mode::from_bits(perm.mode().try_into().unwrap()).ok_or_else(errors::invalid_flags)?;
Ok(fchmod(file, mode)?)
}