use crate::fs::{FileType, ImplFileTypeExt, MetadataExt, Permissions};
use crate::time::SystemTime;
use std::{fs, io};
#[derive(Debug, Clone)]
pub struct Metadata {
pub(crate) file_type: FileType,
pub(crate) len: u64,
pub(crate) permissions: Permissions,
pub(crate) modified: Option<SystemTime>,
pub(crate) accessed: Option<SystemTime>,
pub(crate) created: Option<SystemTime>,
pub(crate) ext: MetadataExt,
}
#[allow(clippy::len_without_is_empty)]
impl Metadata {
#[inline]
pub fn from_file(file: &fs::File) -> io::Result<Self> {
let std = file.metadata()?;
let ext = MetadataExt::from(file, &std)?;
let file_type = ImplFileTypeExt::from(file, &std)?;
Ok(Self::from_parts(std, ext, file_type))
}
#[inline]
pub fn from_just_metadata(std: fs::Metadata) -> Self {
let ext = MetadataExt::from_just_metadata(&std);
let file_type = ImplFileTypeExt::from_just_metadata(&std);
Self::from_parts(std, ext, file_type)
}
#[inline]
fn from_parts(std: fs::Metadata, ext: MetadataExt, file_type: FileType) -> Self {
Self {
file_type,
len: std.len(),
permissions: Permissions::from_std(std.permissions()),
modified: std.modified().ok().map(SystemTime::from_std),
accessed: std.accessed().ok().map(SystemTime::from_std),
created: std.created().ok().map(SystemTime::from_std),
ext,
}
}
#[inline]
pub const fn file_type(&self) -> FileType {
self.file_type
}
#[inline]
pub fn is_dir(&self) -> bool {
self.file_type.is_dir()
}
#[inline]
pub fn is_file(&self) -> bool {
self.file_type.is_file()
}
#[inline]
pub fn is_symlink(&self) -> bool {
self.file_type.is_symlink()
}
#[inline]
pub const fn len(&self) -> u64 {
self.len
}
#[inline]
pub fn permissions(&self) -> Permissions {
self.permissions.clone()
}
#[inline]
pub fn modified(&self) -> io::Result<SystemTime> {
#[cfg(io_error_uncategorized)]
{
self.modified.ok_or_else(|| {
io::Error::new(
io::ErrorKind::Unsupported,
"modified time metadata not available on this platform",
)
})
}
#[cfg(not(io_error_uncategorized))]
{
self.modified.ok_or_else(|| {
io::Error::new(
io::ErrorKind::Other,
"modified time metadata not available on this platform",
)
})
}
}
#[inline]
pub fn accessed(&self) -> io::Result<SystemTime> {
#[cfg(io_error_uncategorized)]
{
self.accessed.ok_or_else(|| {
io::Error::new(
io::ErrorKind::Unsupported,
"accessed time metadata not available on this platform",
)
})
}
#[cfg(not(io_error_uncategorized))]
{
self.accessed.ok_or_else(|| {
io::Error::new(
io::ErrorKind::Other,
"accessed time metadata not available on this platform",
)
})
}
}
#[inline]
pub fn created(&self) -> io::Result<SystemTime> {
#[cfg(io_error_uncategorized)]
{
self.created.ok_or_else(|| {
io::Error::new(
io::ErrorKind::Unsupported,
"created time metadata not available on this platform",
)
})
}
#[cfg(not(io_error_uncategorized))]
{
self.created.ok_or_else(|| {
io::Error::new(
io::ErrorKind::Other,
"created time metadata not available on this platform",
)
})
}
}
#[cfg(any(not(windows), windows_by_handle))]
pub(crate) fn is_same_file(&self, other: &Self) -> bool {
self.ext.is_same_file(&other.ext)
}
#[cfg(windows)]
#[inline]
pub(crate) fn file_attributes(&self) -> u32 {
self.ext.file_attributes()
}
}
#[cfg(unix)]
impl std::os::unix::fs::MetadataExt for Metadata {
#[inline]
fn dev(&self) -> u64 {
self.ext.dev()
}
#[inline]
fn ino(&self) -> u64 {
self.ext.ino()
}
#[inline]
fn mode(&self) -> u32 {
self.ext.mode()
}
#[inline]
fn nlink(&self) -> u64 {
self.ext.nlink()
}
#[inline]
fn uid(&self) -> u32 {
self.ext.uid()
}
#[inline]
fn gid(&self) -> u32 {
self.ext.gid()
}
#[inline]
fn rdev(&self) -> u64 {
self.ext.rdev()
}
#[inline]
fn size(&self) -> u64 {
self.ext.size()
}
#[inline]
fn atime(&self) -> i64 {
self.ext.atime()
}
#[inline]
fn atime_nsec(&self) -> i64 {
self.ext.atime_nsec()
}
#[inline]
fn mtime(&self) -> i64 {
self.ext.mtime()
}
#[inline]
fn mtime_nsec(&self) -> i64 {
self.ext.mtime_nsec()
}
#[inline]
fn ctime(&self) -> i64 {
self.ext.ctime()
}
#[inline]
fn ctime_nsec(&self) -> i64 {
self.ext.ctime_nsec()
}
#[inline]
fn blksize(&self) -> u64 {
self.ext.blksize()
}
#[inline]
fn blocks(&self) -> u64 {
self.ext.blocks()
}
}
#[cfg(target_os = "wasi")]
impl std::os::wasi::fs::MetadataExt for Metadata {
#[inline]
fn dev(&self) -> u64 {
self.ext.dev()
}
#[inline]
fn ino(&self) -> u64 {
self.ext.ino()
}
#[inline]
fn nlink(&self) -> u64 {
self.ext.nlink()
}
#[inline]
fn size(&self) -> u64 {
self.ext.size()
}
#[inline]
fn atim(&self) -> u64 {
self.ext.atim()
}
#[inline]
fn mtim(&self) -> u64 {
self.ext.mtim()
}
#[inline]
fn ctim(&self) -> u64 {
self.ext.ctim()
}
}
#[cfg(target_os = "vxworks")]
impl std::os::vxworks::fs::MetadataExt for Metadata {
#[inline]
fn dev(&self) -> u64 {
self.ext.dev()
}
#[inline]
fn ino(&self) -> u64 {
self.ext.ino()
}
#[inline]
fn mode(&self) -> u32 {
self.ext.mode()
}
#[inline]
fn nlink(&self) -> u64 {
self.ext.nlink()
}
#[inline]
fn uid(&self) -> u32 {
self.ext.uid()
}
#[inline]
fn gid(&self) -> u32 {
self.ext.gid()
}
#[inline]
fn rdev(&self) -> u64 {
self.ext.rdev()
}
#[inline]
fn size(&self) -> u64 {
self.ext.size()
}
#[inline]
fn atime(&self) -> i64 {
self.ext.atime()
}
#[inline]
fn atime_nsec(&self) -> i64 {
self.ext.atime_nsec()
}
#[inline]
fn mtime(&self) -> i64 {
self.ext.mtime()
}
#[inline]
fn mtime_nsec(&self) -> i64 {
self.ext.mtime_nsec()
}
#[inline]
fn ctime(&self) -> i64 {
self.ext.ctime()
}
#[inline]
fn ctime_nsec(&self) -> i64 {
self.ext.ctime_nsec()
}
#[inline]
fn blksize(&self) -> u64 {
self.ext.blksize()
}
#[inline]
fn blocks(&self) -> u64 {
self.ext.blocks()
}
}
#[cfg(all(windows, windows_by_handle))]
impl std::os::windows::fs::MetadataExt for Metadata {
#[inline]
fn file_attributes(&self) -> u32 {
self.ext.file_attributes()
}
#[inline]
fn creation_time(&self) -> u64 {
self.ext.creation_time()
}
#[inline]
fn last_access_time(&self) -> u64 {
self.ext.last_access_time()
}
#[inline]
fn last_write_time(&self) -> u64 {
self.ext.last_write_time()
}
#[inline]
fn file_size(&self) -> u64 {
self.ext.file_size()
}
#[inline]
fn volume_serial_number(&self) -> Option<u32> {
self.ext.volume_serial_number()
}
#[inline]
fn number_of_links(&self) -> Option<u32> {
self.ext.number_of_links()
}
#[inline]
fn file_index(&self) -> Option<u64> {
self.ext.file_index()
}
}
#[cfg(windows)]
#[doc(hidden)]
pub trait _WindowsByHandle {
fn file_attributes(&self) -> u32;
fn volume_serial_number(&self) -> Option<u32>;
fn number_of_links(&self) -> Option<u32>;
fn file_index(&self) -> Option<u64>;
}