use std::fs;
use std::io;
use std::path::Path;
pub(crate) fn ensure_private_dir(path: &Path) -> io::Result<()> {
fs::create_dir_all(path)?;
set_private_permissions(path)
}
pub(crate) fn private_dir_permissions_are_private(path: &Path) -> io::Result<bool> {
platform_private_dir_permissions_are_private(path)
}
#[cfg(unix)]
fn set_private_permissions(path: &Path) -> io::Result<()> {
use std::os::unix::fs::PermissionsExt;
let mut perms = fs::metadata(path)?.permissions();
perms.set_mode(0o700);
fs::set_permissions(path, perms)
}
#[cfg(unix)]
fn platform_private_dir_permissions_are_private(path: &Path) -> io::Result<bool> {
use std::os::unix::fs::PermissionsExt;
let mode = fs::metadata(path)?.permissions().mode();
Ok(mode & 0o077 == 0)
}
#[cfg(windows)]
fn set_private_permissions(path: &Path) -> io::Result<()> {
use std::os::windows::ffi::OsStrExt;
use windows_sys::Win32::Foundation::ERROR_SUCCESS;
use windows_sys::Win32::Security::Authorization::{SetNamedSecurityInfoW, SE_FILE_OBJECT};
use windows_sys::Win32::Security::{
GetSecurityDescriptorDacl, DACL_SECURITY_INFORMATION, PROTECTED_DACL_SECURITY_INFORMATION,
};
let sd = LocalSecurityDescriptor::from_sddl("D:P(A;;FA;;;OW)")?;
let mut present = 0;
let mut defaulted = 0;
let mut dacl = std::ptr::null_mut();
let ok = unsafe { GetSecurityDescriptorDacl(sd.0, &mut present, &mut dacl, &mut defaulted) };
if ok == 0 || present == 0 || dacl.is_null() {
return Err(io::Error::last_os_error());
}
let wide: Vec<u16> = path
.as_os_str()
.encode_wide()
.chain(std::iter::once(0))
.collect();
let status = unsafe {
SetNamedSecurityInfoW(
wide.as_ptr(),
SE_FILE_OBJECT,
DACL_SECURITY_INFORMATION | PROTECTED_DACL_SECURITY_INFORMATION,
std::ptr::null_mut(),
std::ptr::null_mut(),
dacl,
std::ptr::null_mut(),
)
};
if status != ERROR_SUCCESS {
Err(io::Error::from_raw_os_error(status as i32))
} else {
Ok(())
}
}
#[cfg(windows)]
fn platform_private_dir_permissions_are_private(path: &Path) -> io::Result<bool> {
let sddl = dir_dacl_sddl(path)?;
let ace_count = sddl.matches("(A;;").count();
Ok(sddl.starts_with("D:P")
&& ace_count == 1
&& (sddl.contains("(A;;FA;;;OW)") || sddl.contains("(A;;0x1f01ff;;;OW)")))
}
#[cfg(windows)]
fn dir_dacl_sddl(path: &Path) -> io::Result<String> {
use std::os::windows::ffi::OsStrExt;
use windows_sys::Win32::Foundation::ERROR_SUCCESS;
use windows_sys::Win32::Security::Authorization::{
ConvertSecurityDescriptorToStringSecurityDescriptorW, GetNamedSecurityInfoW,
SDDL_REVISION_1, SE_FILE_OBJECT,
};
use windows_sys::Win32::Security::{DACL_SECURITY_INFORMATION, PSECURITY_DESCRIPTOR};
let wide: Vec<u16> = path
.as_os_str()
.encode_wide()
.chain(std::iter::once(0))
.collect();
let mut sd: PSECURITY_DESCRIPTOR = std::ptr::null_mut();
let status = unsafe {
GetNamedSecurityInfoW(
wide.as_ptr(),
SE_FILE_OBJECT,
DACL_SECURITY_INFORMATION,
std::ptr::null_mut(),
std::ptr::null_mut(),
std::ptr::null_mut(),
std::ptr::null_mut(),
&mut sd,
)
};
if status != ERROR_SUCCESS {
return Err(io::Error::from_raw_os_error(status as i32));
}
let sd = LocalSecurityDescriptor(sd);
let mut sddl = std::ptr::null_mut();
let ok = unsafe {
ConvertSecurityDescriptorToStringSecurityDescriptorW(
sd.0,
SDDL_REVISION_1,
DACL_SECURITY_INFORMATION,
&mut sddl,
std::ptr::null_mut(),
)
};
if ok == 0 || sddl.is_null() {
return Err(io::Error::last_os_error());
}
let _sddl_guard = LocalWideString(sddl);
let mut len = 0;
unsafe {
while *sddl.add(len) != 0 {
len += 1;
}
}
Ok(String::from_utf16_lossy(unsafe {
std::slice::from_raw_parts(sddl, len)
}))
}
#[cfg(windows)]
struct LocalSecurityDescriptor(windows_sys::Win32::Security::PSECURITY_DESCRIPTOR);
#[cfg(windows)]
impl LocalSecurityDescriptor {
fn from_sddl(sddl: &str) -> io::Result<Self> {
use windows_sys::Win32::Security::Authorization::{
ConvertStringSecurityDescriptorToSecurityDescriptorW, SDDL_REVISION_1,
};
let wide: Vec<u16> = sddl.encode_utf16().chain(std::iter::once(0)).collect();
let mut sd = std::ptr::null_mut();
let ok = unsafe {
ConvertStringSecurityDescriptorToSecurityDescriptorW(
wide.as_ptr(),
SDDL_REVISION_1,
&mut sd,
std::ptr::null_mut(),
)
};
if ok == 0 || sd.is_null() {
Err(io::Error::last_os_error())
} else {
Ok(Self(sd))
}
}
}
#[cfg(windows)]
impl Drop for LocalSecurityDescriptor {
fn drop(&mut self) {
unsafe {
windows_sys::Win32::Foundation::LocalFree(self.0.cast());
}
}
}
#[cfg(windows)]
struct LocalWideString(windows_sys::core::PWSTR);
#[cfg(windows)]
impl Drop for LocalWideString {
fn drop(&mut self) {
unsafe {
windows_sys::Win32::Foundation::LocalFree(self.0.cast());
}
}
}