use crate::bindings::clocks::wall_clock;
use crate::bindings::filesystem::preopens;
use crate::bindings::filesystem::types::{
self, ErrorCode, HostDescriptor, HostDirectoryEntryStream,
};
use crate::bindings::io::streams::{InputStream, OutputStream};
use crate::filesystem::{
Descriptor, Dir, File, FileInputStream, FileOutputStream, OpenMode, ReaddirIterator,
};
use crate::{DirPerms, FilePerms, FsError, FsResult, WasiView};
use anyhow::Context;
use wasmtime::component::Resource;
mod sync;
impl<T: WasiView> preopens::Host for T {
fn get_directories(
&mut self,
) -> Result<Vec<(Resource<types::Descriptor>, String)>, anyhow::Error> {
let mut results = Vec::new();
for (dir, name) in self.ctx().preopens.clone() {
let fd = self
.table()
.push(Descriptor::Dir(dir))
.with_context(|| format!("failed to push preopen {name}"))?;
results.push((fd, name));
}
Ok(results)
}
}
#[async_trait::async_trait]
impl<T: WasiView> types::Host for T {
fn convert_error_code(&mut self, err: FsError) -> anyhow::Result<ErrorCode> {
err.downcast()
}
fn filesystem_error_code(
&mut self,
err: Resource<anyhow::Error>,
) -> anyhow::Result<Option<ErrorCode>> {
let err = self.table().get(&err)?;
if let Some(err) = err.downcast_ref::<std::io::Error>() {
return Ok(Some(ErrorCode::from(err)));
}
Ok(None)
}
}
#[async_trait::async_trait]
impl<T: WasiView> HostDescriptor for T {
async fn advise(
&mut self,
fd: Resource<types::Descriptor>,
offset: types::Filesize,
len: types::Filesize,
advice: types::Advice,
) -> FsResult<()> {
use system_interface::fs::{Advice as A, FileIoExt};
use types::Advice;
let advice = match advice {
Advice::Normal => A::Normal,
Advice::Sequential => A::Sequential,
Advice::Random => A::Random,
Advice::WillNeed => A::WillNeed,
Advice::DontNeed => A::DontNeed,
Advice::NoReuse => A::NoReuse,
};
let f = self.table().get(&fd)?.file()?;
f.spawn_blocking(move |f| f.advise(offset, len, advice))
.await?;
Ok(())
}
async fn sync_data(&mut self, fd: Resource<types::Descriptor>) -> FsResult<()> {
let descriptor = self.table().get(&fd)?;
match descriptor {
Descriptor::File(f) => {
match f.spawn_blocking(|f| f.sync_data()).await {
Ok(()) => Ok(()),
#[cfg(windows)]
Err(e)
if e.raw_os_error()
== Some(windows_sys::Win32::Foundation::ERROR_ACCESS_DENIED as _) =>
{
Ok(())
}
Err(e) => Err(e.into()),
}
}
Descriptor::Dir(d) => {
d.spawn_blocking(|d| Ok(d.open(std::path::Component::CurDir)?.sync_data()?))
.await
}
}
}
async fn get_flags(
&mut self,
fd: Resource<types::Descriptor>,
) -> FsResult<types::DescriptorFlags> {
use system_interface::fs::{FdFlags, GetSetFdFlags};
use types::DescriptorFlags;
fn get_from_fdflags(flags: FdFlags) -> DescriptorFlags {
let mut out = DescriptorFlags::empty();
if flags.contains(FdFlags::DSYNC) {
out |= DescriptorFlags::REQUESTED_WRITE_SYNC;
}
if flags.contains(FdFlags::RSYNC) {
out |= DescriptorFlags::DATA_INTEGRITY_SYNC;
}
if flags.contains(FdFlags::SYNC) {
out |= DescriptorFlags::FILE_INTEGRITY_SYNC;
}
out
}
let descriptor = self.table().get(&fd)?;
match descriptor {
Descriptor::File(f) => {
let flags = f.spawn_blocking(|f| f.get_fd_flags()).await?;
let mut flags = get_from_fdflags(flags);
if f.open_mode.contains(OpenMode::READ) {
flags |= DescriptorFlags::READ;
}
if f.open_mode.contains(OpenMode::WRITE) {
flags |= DescriptorFlags::WRITE;
}
Ok(flags)
}
Descriptor::Dir(d) => {
let flags = d.spawn_blocking(|d| d.get_fd_flags()).await?;
let mut flags = get_from_fdflags(flags);
if d.open_mode.contains(OpenMode::READ) {
flags |= DescriptorFlags::READ;
}
if d.open_mode.contains(OpenMode::WRITE) {
flags |= DescriptorFlags::MUTATE_DIRECTORY;
}
Ok(flags)
}
}
}
async fn get_type(
&mut self,
fd: Resource<types::Descriptor>,
) -> FsResult<types::DescriptorType> {
let descriptor = self.table().get(&fd)?;
match descriptor {
Descriptor::File(f) => {
let meta = f.spawn_blocking(|f| f.metadata()).await?;
Ok(descriptortype_from(meta.file_type()))
}
Descriptor::Dir(_) => Ok(types::DescriptorType::Directory),
}
}
async fn set_size(
&mut self,
fd: Resource<types::Descriptor>,
size: types::Filesize,
) -> FsResult<()> {
let f = self.table().get(&fd)?.file()?;
if !f.perms.contains(FilePerms::WRITE) {
Err(ErrorCode::NotPermitted)?;
}
f.spawn_blocking(move |f| f.set_len(size)).await?;
Ok(())
}
async fn set_times(
&mut self,
fd: Resource<types::Descriptor>,
atim: types::NewTimestamp,
mtim: types::NewTimestamp,
) -> FsResult<()> {
use fs_set_times::SetTimes;
let descriptor = self.table().get(&fd)?;
match descriptor {
Descriptor::File(f) => {
if !f.perms.contains(FilePerms::WRITE) {
return Err(ErrorCode::NotPermitted.into());
}
let atim = systemtimespec_from(atim)?;
let mtim = systemtimespec_from(mtim)?;
f.spawn_blocking(|f| f.set_times(atim, mtim)).await?;
Ok(())
}
Descriptor::Dir(d) => {
if !d.perms.contains(DirPerms::MUTATE) {
return Err(ErrorCode::NotPermitted.into());
}
let atim = systemtimespec_from(atim)?;
let mtim = systemtimespec_from(mtim)?;
d.spawn_blocking(|d| d.set_times(atim, mtim)).await?;
Ok(())
}
}
}
async fn read(
&mut self,
fd: Resource<types::Descriptor>,
len: types::Filesize,
offset: types::Filesize,
) -> FsResult<(Vec<u8>, bool)> {
use std::io::IoSliceMut;
use system_interface::fs::FileIoExt;
let table = self.table();
let f = table.get(&fd)?.file()?;
if !f.perms.contains(FilePerms::READ) {
return Err(ErrorCode::NotPermitted.into());
}
let (mut buffer, r) = f
.spawn_blocking(move |f| {
let mut buffer = vec![0; len.try_into().unwrap_or(usize::MAX)];
let r = f.read_vectored_at(&mut [IoSliceMut::new(&mut buffer)], offset);
(buffer, r)
})
.await;
let (bytes_read, state) = match r? {
0 => (0, true),
n => (n, false),
};
buffer.truncate(
bytes_read
.try_into()
.expect("bytes read into memory as u64 fits in usize"),
);
Ok((buffer, state))
}
async fn write(
&mut self,
fd: Resource<types::Descriptor>,
buf: Vec<u8>,
offset: types::Filesize,
) -> FsResult<types::Filesize> {
use std::io::IoSlice;
use system_interface::fs::FileIoExt;
let table = self.table();
let f = table.get(&fd)?.file()?;
if !f.perms.contains(FilePerms::WRITE) {
return Err(ErrorCode::NotPermitted.into());
}
let bytes_written = f
.spawn_blocking(move |f| f.write_vectored_at(&[IoSlice::new(&buf)], offset))
.await?;
Ok(types::Filesize::try_from(bytes_written).expect("usize fits in Filesize"))
}
async fn read_directory(
&mut self,
fd: Resource<types::Descriptor>,
) -> FsResult<Resource<types::DirectoryEntryStream>> {
let table = self.table();
let d = table.get(&fd)?.dir()?;
if !d.perms.contains(DirPerms::READ) {
return Err(ErrorCode::NotPermitted.into());
}
enum ReaddirError {
Io(std::io::Error),
IllegalSequence,
}
impl From<std::io::Error> for ReaddirError {
fn from(e: std::io::Error) -> ReaddirError {
ReaddirError::Io(e)
}
}
let entries = d
.spawn_blocking(|d| {
Ok::<_, std::io::Error>(
d.entries()?
.map(|entry| {
let entry = entry?;
let meta = entry.metadata()?;
let type_ = descriptortype_from(meta.file_type());
let name = entry
.file_name()
.into_string()
.map_err(|_| ReaddirError::IllegalSequence)?;
Ok(types::DirectoryEntry { type_, name })
})
.collect::<Vec<Result<types::DirectoryEntry, ReaddirError>>>(),
)
})
.await?
.into_iter();
#[cfg(windows)]
let entries = entries.filter(|entry| {
use windows_sys::Win32::Foundation::{ERROR_ACCESS_DENIED, ERROR_SHARING_VIOLATION};
if let Err(ReaddirError::Io(err)) = entry {
if err.raw_os_error() == Some(ERROR_SHARING_VIOLATION as i32)
|| err.raw_os_error() == Some(ERROR_ACCESS_DENIED as i32)
{
return false;
}
}
true
});
let entries = entries.map(|r| match r {
Ok(r) => Ok(r),
Err(ReaddirError::Io(e)) => Err(e.into()),
Err(ReaddirError::IllegalSequence) => Err(ErrorCode::IllegalByteSequence.into()),
});
Ok(table.push(ReaddirIterator::new(entries))?)
}
async fn sync(&mut self, fd: Resource<types::Descriptor>) -> FsResult<()> {
let descriptor = self.table().get(&fd)?;
match descriptor {
Descriptor::File(f) => {
match f.spawn_blocking(|f| f.sync_all()).await {
Ok(()) => Ok(()),
#[cfg(windows)]
Err(e)
if e.raw_os_error()
== Some(windows_sys::Win32::Foundation::ERROR_ACCESS_DENIED as _) =>
{
Ok(())
}
Err(e) => Err(e.into()),
}
}
Descriptor::Dir(d) => {
d.spawn_blocking(|d| Ok(d.open(std::path::Component::CurDir)?.sync_all()?))
.await
}
}
}
async fn create_directory_at(
&mut self,
fd: Resource<types::Descriptor>,
path: String,
) -> FsResult<()> {
let table = self.table();
let d = table.get(&fd)?.dir()?;
if !d.perms.contains(DirPerms::MUTATE) {
return Err(ErrorCode::NotPermitted.into());
}
d.spawn_blocking(move |d| d.create_dir(&path)).await?;
Ok(())
}
async fn stat(&mut self, fd: Resource<types::Descriptor>) -> FsResult<types::DescriptorStat> {
let descriptor = self.table().get(&fd)?;
match descriptor {
Descriptor::File(f) => {
let meta = f.spawn_blocking(|f| f.metadata()).await?;
Ok(descriptorstat_from(meta))
}
Descriptor::Dir(d) => {
let meta = d.spawn_blocking(|d| d.dir_metadata()).await?;
Ok(descriptorstat_from(meta))
}
}
}
async fn stat_at(
&mut self,
fd: Resource<types::Descriptor>,
path_flags: types::PathFlags,
path: String,
) -> FsResult<types::DescriptorStat> {
let table = self.table();
let d = table.get(&fd)?.dir()?;
if !d.perms.contains(DirPerms::READ) {
return Err(ErrorCode::NotPermitted.into());
}
let meta = if symlink_follow(path_flags) {
d.spawn_blocking(move |d| d.metadata(&path)).await?
} else {
d.spawn_blocking(move |d| d.symlink_metadata(&path)).await?
};
Ok(descriptorstat_from(meta))
}
async fn set_times_at(
&mut self,
fd: Resource<types::Descriptor>,
path_flags: types::PathFlags,
path: String,
atim: types::NewTimestamp,
mtim: types::NewTimestamp,
) -> FsResult<()> {
use cap_fs_ext::DirExt;
let table = self.table();
let d = table.get(&fd)?.dir()?;
if !d.perms.contains(DirPerms::MUTATE) {
return Err(ErrorCode::NotPermitted.into());
}
let atim = systemtimespec_from(atim)?;
let mtim = systemtimespec_from(mtim)?;
if symlink_follow(path_flags) {
d.spawn_blocking(move |d| {
d.set_times(
&path,
atim.map(cap_fs_ext::SystemTimeSpec::from_std),
mtim.map(cap_fs_ext::SystemTimeSpec::from_std),
)
})
.await?;
} else {
d.spawn_blocking(move |d| {
d.set_symlink_times(
&path,
atim.map(cap_fs_ext::SystemTimeSpec::from_std),
mtim.map(cap_fs_ext::SystemTimeSpec::from_std),
)
})
.await?;
}
Ok(())
}
async fn link_at(
&mut self,
fd: Resource<types::Descriptor>,
old_path_flags: types::PathFlags,
old_path: String,
new_descriptor: Resource<types::Descriptor>,
new_path: String,
) -> FsResult<()> {
let table = self.table();
let old_dir = table.get(&fd)?.dir()?;
if !old_dir.perms.contains(DirPerms::MUTATE) {
return Err(ErrorCode::NotPermitted.into());
}
let new_dir = table.get(&new_descriptor)?.dir()?;
if !new_dir.perms.contains(DirPerms::MUTATE) {
return Err(ErrorCode::NotPermitted.into());
}
if symlink_follow(old_path_flags) {
return Err(ErrorCode::Invalid.into());
}
let new_dir_handle = std::sync::Arc::clone(&new_dir.dir);
old_dir
.spawn_blocking(move |d| d.hard_link(&old_path, &new_dir_handle, &new_path))
.await?;
Ok(())
}
async fn open_at(
&mut self,
fd: Resource<types::Descriptor>,
path_flags: types::PathFlags,
path: String,
oflags: types::OpenFlags,
flags: types::DescriptorFlags,
) -> FsResult<Resource<types::Descriptor>> {
use cap_fs_ext::{FollowSymlinks, OpenOptionsFollowExt, OpenOptionsMaybeDirExt};
use system_interface::fs::{FdFlags, GetSetFdFlags};
use types::{DescriptorFlags, OpenFlags};
let allow_blocking_current_thread = self.ctx().allow_blocking_current_thread;
let table = self.table();
let d = table.get(&fd)?.dir()?;
if !d.perms.contains(DirPerms::READ) {
Err(ErrorCode::NotPermitted)?;
}
if !d.perms.contains(DirPerms::MUTATE) {
if oflags.contains(OpenFlags::CREATE) || oflags.contains(OpenFlags::TRUNCATE) {
Err(ErrorCode::NotPermitted)?;
}
if flags.contains(DescriptorFlags::WRITE) {
Err(ErrorCode::NotPermitted)?;
}
}
let mut create = false;
let mut open_mode = OpenMode::empty();
let mut opts = cap_std::fs::OpenOptions::new();
opts.maybe_dir(true);
if oflags.contains(OpenFlags::CREATE) {
if oflags.contains(OpenFlags::EXCLUSIVE) {
opts.create_new(true);
} else {
opts.create(true);
}
create = true;
opts.write(true);
open_mode |= OpenMode::WRITE;
}
if oflags.contains(OpenFlags::TRUNCATE) {
opts.truncate(true);
}
if flags.contains(DescriptorFlags::READ) {
opts.read(true);
open_mode |= OpenMode::READ;
}
if flags.contains(DescriptorFlags::WRITE) {
opts.write(true);
open_mode |= OpenMode::WRITE;
} else {
opts.read(true);
open_mode |= OpenMode::READ;
}
if symlink_follow(path_flags) {
opts.follow(FollowSymlinks::Yes);
} else {
opts.follow(FollowSymlinks::No);
}
if flags.contains(DescriptorFlags::FILE_INTEGRITY_SYNC)
|| flags.contains(DescriptorFlags::DATA_INTEGRITY_SYNC)
|| flags.contains(DescriptorFlags::REQUESTED_WRITE_SYNC)
{
Err(ErrorCode::Unsupported)?;
}
if oflags.contains(OpenFlags::DIRECTORY) {
if oflags.contains(OpenFlags::CREATE)
|| oflags.contains(OpenFlags::EXCLUSIVE)
|| oflags.contains(OpenFlags::TRUNCATE)
{
Err(ErrorCode::Invalid)?;
}
}
if !d.perms.contains(DirPerms::MUTATE) && create {
Err(ErrorCode::NotPermitted)?;
}
if !d.file_perms.contains(FilePerms::WRITE) && open_mode.contains(OpenMode::WRITE) {
Err(ErrorCode::NotPermitted)?;
}
enum OpenResult {
Dir(cap_std::fs::Dir),
File(cap_std::fs::File),
NotDir,
}
let opened = d
.spawn_blocking::<_, std::io::Result<OpenResult>>(move |d| {
let mut opened = d.open_with(&path, &opts)?;
if opened.metadata()?.is_dir() {
Ok(OpenResult::Dir(cap_std::fs::Dir::from_std_file(
opened.into_std(),
)))
} else if oflags.contains(OpenFlags::DIRECTORY) {
Ok(OpenResult::NotDir)
} else {
let set_fd_flags = opened.new_set_fd_flags(FdFlags::NONBLOCK)?;
opened.set_fd_flags(set_fd_flags)?;
Ok(OpenResult::File(opened))
}
})
.await?;
match opened {
OpenResult::Dir(dir) => Ok(table.push(Descriptor::Dir(Dir::new(
dir,
d.perms,
d.file_perms,
open_mode,
allow_blocking_current_thread,
)))?),
OpenResult::File(file) => Ok(table.push(Descriptor::File(File::new(
file,
d.file_perms,
open_mode,
allow_blocking_current_thread,
)))?),
OpenResult::NotDir => Err(ErrorCode::NotDirectory.into()),
}
}
fn drop(&mut self, fd: Resource<types::Descriptor>) -> anyhow::Result<()> {
let table = self.table();
table.delete(fd)?;
Ok(())
}
async fn readlink_at(
&mut self,
fd: Resource<types::Descriptor>,
path: String,
) -> FsResult<String> {
let table = self.table();
let d = table.get(&fd)?.dir()?;
if !d.perms.contains(DirPerms::READ) {
return Err(ErrorCode::NotPermitted.into());
}
let link = d.spawn_blocking(move |d| d.read_link(&path)).await?;
Ok(link
.into_os_string()
.into_string()
.map_err(|_| ErrorCode::IllegalByteSequence)?)
}
async fn remove_directory_at(
&mut self,
fd: Resource<types::Descriptor>,
path: String,
) -> FsResult<()> {
let table = self.table();
let d = table.get(&fd)?.dir()?;
if !d.perms.contains(DirPerms::MUTATE) {
return Err(ErrorCode::NotPermitted.into());
}
Ok(d.spawn_blocking(move |d| d.remove_dir(&path)).await?)
}
async fn rename_at(
&mut self,
fd: Resource<types::Descriptor>,
old_path: String,
new_fd: Resource<types::Descriptor>,
new_path: String,
) -> FsResult<()> {
let table = self.table();
let old_dir = table.get(&fd)?.dir()?;
if !old_dir.perms.contains(DirPerms::MUTATE) {
return Err(ErrorCode::NotPermitted.into());
}
let new_dir = table.get(&new_fd)?.dir()?;
if !new_dir.perms.contains(DirPerms::MUTATE) {
return Err(ErrorCode::NotPermitted.into());
}
let new_dir_handle = std::sync::Arc::clone(&new_dir.dir);
Ok(old_dir
.spawn_blocking(move |d| d.rename(&old_path, &new_dir_handle, &new_path))
.await?)
}
async fn symlink_at(
&mut self,
fd: Resource<types::Descriptor>,
src_path: String,
dest_path: String,
) -> FsResult<()> {
#[cfg(windows)]
use cap_fs_ext::DirExt;
let table = self.table();
let d = table.get(&fd)?.dir()?;
if !d.perms.contains(DirPerms::MUTATE) {
return Err(ErrorCode::NotPermitted.into());
}
Ok(d.spawn_blocking(move |d| d.symlink(&src_path, &dest_path))
.await?)
}
async fn unlink_file_at(
&mut self,
fd: Resource<types::Descriptor>,
path: String,
) -> FsResult<()> {
use cap_fs_ext::DirExt;
let table = self.table();
let d = table.get(&fd)?.dir()?;
if !d.perms.contains(DirPerms::MUTATE) {
return Err(ErrorCode::NotPermitted.into());
}
Ok(d.spawn_blocking(move |d| d.remove_file_or_symlink(&path))
.await?)
}
fn read_via_stream(
&mut self,
fd: Resource<types::Descriptor>,
offset: types::Filesize,
) -> FsResult<Resource<InputStream>> {
let f = self.table().get(&fd)?.file()?;
if !f.perms.contains(FilePerms::READ) {
Err(types::ErrorCode::BadDescriptor)?;
}
let reader = FileInputStream::new(f, offset);
let index = self.table().push(InputStream::File(reader))?;
Ok(index)
}
fn write_via_stream(
&mut self,
fd: Resource<types::Descriptor>,
offset: types::Filesize,
) -> FsResult<Resource<OutputStream>> {
let f = self.table().get(&fd)?.file()?;
if !f.perms.contains(FilePerms::WRITE) {
Err(types::ErrorCode::BadDescriptor)?;
}
let writer = FileOutputStream::write_at(f, offset);
let writer: OutputStream = Box::new(writer);
let index = self.table().push(writer)?;
Ok(index)
}
fn append_via_stream(
&mut self,
fd: Resource<types::Descriptor>,
) -> FsResult<Resource<OutputStream>> {
let f = self.table().get(&fd)?.file()?;
if !f.perms.contains(FilePerms::WRITE) {
Err(types::ErrorCode::BadDescriptor)?;
}
let appender = FileOutputStream::append(f);
let appender: OutputStream = Box::new(appender);
let index = self.table().push(appender)?;
Ok(index)
}
async fn is_same_object(
&mut self,
a: Resource<types::Descriptor>,
b: Resource<types::Descriptor>,
) -> anyhow::Result<bool> {
use cap_fs_ext::MetadataExt;
let descriptor_a = self.table().get(&a)?;
let meta_a = get_descriptor_metadata(descriptor_a).await?;
let descriptor_b = self.table().get(&b)?;
let meta_b = get_descriptor_metadata(descriptor_b).await?;
if meta_a.dev() == meta_b.dev() && meta_a.ino() == meta_b.ino() {
debug_assert_eq!(
calculate_metadata_hash(&meta_a).upper,
calculate_metadata_hash(&meta_b).upper
);
debug_assert_eq!(
calculate_metadata_hash(&meta_a).lower,
calculate_metadata_hash(&meta_b).lower
);
Ok(true)
} else {
Ok(false)
}
}
async fn metadata_hash(
&mut self,
fd: Resource<types::Descriptor>,
) -> FsResult<types::MetadataHashValue> {
let descriptor_a = self.table().get(&fd)?;
let meta = get_descriptor_metadata(descriptor_a).await?;
Ok(calculate_metadata_hash(&meta))
}
async fn metadata_hash_at(
&mut self,
fd: Resource<types::Descriptor>,
path_flags: types::PathFlags,
path: String,
) -> FsResult<types::MetadataHashValue> {
let table = self.table();
let d = table.get(&fd)?.dir()?;
let meta = d
.spawn_blocking(move |d| {
if symlink_follow(path_flags) {
d.metadata(path)
} else {
d.symlink_metadata(path)
}
})
.await?;
Ok(calculate_metadata_hash(&meta))
}
}
#[async_trait::async_trait]
impl<T: WasiView> HostDirectoryEntryStream for T {
async fn read_directory_entry(
&mut self,
stream: Resource<types::DirectoryEntryStream>,
) -> FsResult<Option<types::DirectoryEntry>> {
let table = self.table();
let readdir = table.get(&stream)?;
readdir.next()
}
fn drop(&mut self, stream: Resource<types::DirectoryEntryStream>) -> anyhow::Result<()> {
self.table().delete(stream)?;
Ok(())
}
}
async fn get_descriptor_metadata(fd: &types::Descriptor) -> FsResult<cap_std::fs::Metadata> {
match fd {
Descriptor::File(f) => {
Ok(f.spawn_blocking(|f| f.metadata()).await?)
}
Descriptor::Dir(d) => {
Ok(d.spawn_blocking(|d| d.dir_metadata()).await?)
}
}
}
fn calculate_metadata_hash(meta: &cap_std::fs::Metadata) -> types::MetadataHashValue {
use cap_fs_ext::MetadataExt;
use std::hash::Hasher;
let mut hasher = std::collections::hash_map::DefaultHasher::new();
hasher.write_u64(meta.dev());
hasher.write_u64(meta.ino());
let lower = hasher.finish();
let upper = lower ^ 4614256656552045848u64;
types::MetadataHashValue { lower, upper }
}
#[cfg(unix)]
fn from_raw_os_error(err: Option<i32>) -> Option<ErrorCode> {
use rustix::io::Errno as RustixErrno;
if err.is_none() {
return None;
}
Some(match RustixErrno::from_raw_os_error(err.unwrap()) {
RustixErrno::PIPE => ErrorCode::Pipe,
RustixErrno::PERM => ErrorCode::NotPermitted,
RustixErrno::NOENT => ErrorCode::NoEntry,
RustixErrno::NOMEM => ErrorCode::InsufficientMemory,
RustixErrno::IO => ErrorCode::Io,
RustixErrno::BADF => ErrorCode::BadDescriptor,
RustixErrno::BUSY => ErrorCode::Busy,
RustixErrno::ACCESS => ErrorCode::Access,
RustixErrno::NOTDIR => ErrorCode::NotDirectory,
RustixErrno::ISDIR => ErrorCode::IsDirectory,
RustixErrno::INVAL => ErrorCode::Invalid,
RustixErrno::EXIST => ErrorCode::Exist,
RustixErrno::FBIG => ErrorCode::FileTooLarge,
RustixErrno::NOSPC => ErrorCode::InsufficientSpace,
RustixErrno::SPIPE => ErrorCode::InvalidSeek,
RustixErrno::MLINK => ErrorCode::TooManyLinks,
RustixErrno::NAMETOOLONG => ErrorCode::NameTooLong,
RustixErrno::NOTEMPTY => ErrorCode::NotEmpty,
RustixErrno::LOOP => ErrorCode::Loop,
RustixErrno::OVERFLOW => ErrorCode::Overflow,
RustixErrno::ILSEQ => ErrorCode::IllegalByteSequence,
RustixErrno::NOTSUP => ErrorCode::Unsupported,
RustixErrno::ALREADY => ErrorCode::Already,
RustixErrno::INPROGRESS => ErrorCode::InProgress,
RustixErrno::INTR => ErrorCode::Interrupted,
#[allow(unreachable_patterns)]
RustixErrno::OPNOTSUPP => ErrorCode::Unsupported,
_ => return None,
})
}
#[cfg(windows)]
fn from_raw_os_error(raw_os_error: Option<i32>) -> Option<ErrorCode> {
use windows_sys::Win32::Foundation;
Some(match raw_os_error.map(|code| code as u32) {
Some(Foundation::ERROR_FILE_NOT_FOUND) => ErrorCode::NoEntry,
Some(Foundation::ERROR_PATH_NOT_FOUND) => ErrorCode::NoEntry,
Some(Foundation::ERROR_ACCESS_DENIED) => ErrorCode::Access,
Some(Foundation::ERROR_SHARING_VIOLATION) => ErrorCode::Access,
Some(Foundation::ERROR_PRIVILEGE_NOT_HELD) => ErrorCode::NotPermitted,
Some(Foundation::ERROR_INVALID_HANDLE) => ErrorCode::BadDescriptor,
Some(Foundation::ERROR_INVALID_NAME) => ErrorCode::NoEntry,
Some(Foundation::ERROR_NOT_ENOUGH_MEMORY) => ErrorCode::InsufficientMemory,
Some(Foundation::ERROR_OUTOFMEMORY) => ErrorCode::InsufficientMemory,
Some(Foundation::ERROR_DIR_NOT_EMPTY) => ErrorCode::NotEmpty,
Some(Foundation::ERROR_NOT_READY) => ErrorCode::Busy,
Some(Foundation::ERROR_BUSY) => ErrorCode::Busy,
Some(Foundation::ERROR_NOT_SUPPORTED) => ErrorCode::Unsupported,
Some(Foundation::ERROR_FILE_EXISTS) => ErrorCode::Exist,
Some(Foundation::ERROR_BROKEN_PIPE) => ErrorCode::Pipe,
Some(Foundation::ERROR_BUFFER_OVERFLOW) => ErrorCode::NameTooLong,
Some(Foundation::ERROR_NOT_A_REPARSE_POINT) => ErrorCode::Invalid,
Some(Foundation::ERROR_NEGATIVE_SEEK) => ErrorCode::Invalid,
Some(Foundation::ERROR_DIRECTORY) => ErrorCode::NotDirectory,
Some(Foundation::ERROR_ALREADY_EXISTS) => ErrorCode::Exist,
Some(Foundation::ERROR_STOPPED_ON_SYMLINK) => ErrorCode::Loop,
Some(Foundation::ERROR_DIRECTORY_NOT_SUPPORTED) => ErrorCode::IsDirectory,
_ => return None,
})
}
impl From<std::io::Error> for ErrorCode {
fn from(err: std::io::Error) -> ErrorCode {
ErrorCode::from(&err)
}
}
impl<'a> From<&'a std::io::Error> for ErrorCode {
fn from(err: &'a std::io::Error) -> ErrorCode {
match from_raw_os_error(err.raw_os_error()) {
Some(errno) => errno,
None => {
tracing::debug!("unknown raw os error: {err}");
match err.kind() {
std::io::ErrorKind::NotFound => ErrorCode::NoEntry,
std::io::ErrorKind::PermissionDenied => ErrorCode::NotPermitted,
std::io::ErrorKind::AlreadyExists => ErrorCode::Exist,
std::io::ErrorKind::InvalidInput => ErrorCode::Invalid,
_ => ErrorCode::Io,
}
}
}
}
}
impl From<cap_rand::Error> for ErrorCode {
fn from(err: cap_rand::Error) -> ErrorCode {
from_raw_os_error(err.raw_os_error()).unwrap_or(ErrorCode::Io)
}
}
impl From<std::num::TryFromIntError> for ErrorCode {
fn from(_err: std::num::TryFromIntError) -> ErrorCode {
ErrorCode::Overflow
}
}
fn descriptortype_from(ft: cap_std::fs::FileType) -> types::DescriptorType {
use cap_fs_ext::FileTypeExt;
use types::DescriptorType;
if ft.is_dir() {
DescriptorType::Directory
} else if ft.is_symlink() {
DescriptorType::SymbolicLink
} else if ft.is_block_device() {
DescriptorType::BlockDevice
} else if ft.is_char_device() {
DescriptorType::CharacterDevice
} else if ft.is_file() {
DescriptorType::RegularFile
} else {
DescriptorType::Unknown
}
}
fn systemtimespec_from(t: types::NewTimestamp) -> FsResult<Option<fs_set_times::SystemTimeSpec>> {
use fs_set_times::SystemTimeSpec;
use types::NewTimestamp;
match t {
NewTimestamp::NoChange => Ok(None),
NewTimestamp::Now => Ok(Some(SystemTimeSpec::SymbolicNow)),
NewTimestamp::Timestamp(st) => Ok(Some(SystemTimeSpec::Absolute(systemtime_from(st)?))),
}
}
fn systemtime_from(t: wall_clock::Datetime) -> FsResult<std::time::SystemTime> {
use std::time::{Duration, SystemTime};
SystemTime::UNIX_EPOCH
.checked_add(Duration::new(t.seconds, t.nanoseconds))
.ok_or_else(|| ErrorCode::Overflow.into())
}
fn datetime_from(t: std::time::SystemTime) -> wall_clock::Datetime {
wall_clock::Datetime::try_from(cap_std::time::SystemTime::from_std(t)).unwrap()
}
fn descriptorstat_from(meta: cap_std::fs::Metadata) -> types::DescriptorStat {
use cap_fs_ext::MetadataExt;
types::DescriptorStat {
type_: descriptortype_from(meta.file_type()),
link_count: meta.nlink(),
size: meta.len(),
data_access_timestamp: meta.accessed().map(|t| datetime_from(t.into_std())).ok(),
data_modification_timestamp: meta.modified().map(|t| datetime_from(t.into_std())).ok(),
status_change_timestamp: meta.created().map(|t| datetime_from(t.into_std())).ok(),
}
}
fn symlink_follow(path_flags: types::PathFlags) -> bool {
path_flags.contains(types::PathFlags::SYMLINK_FOLLOW)
}
#[cfg(test)]
mod test {
use super::*;
use wasmtime::component::ResourceTable;
#[test]
fn table_readdir_works() {
let mut table = ResourceTable::new();
let ix = table
.push(ReaddirIterator::new(std::iter::empty()))
.unwrap();
let _ = table.get(&ix).unwrap();
table.delete(ix).unwrap();
}
}