devela 0.27.0

A development layer of coherence.
Documentation
// devela::sys::io::no_std_define::error
//
//! Defines `IoError`, `IoResult`.
//

#[cfg(all(doc, any(feature = "std", feature = "io")))]
use crate::IoSeek;
use crate::{Debug, From, Result, impl_trait};
#[cfg(doc)]
use crate::{IoRead, IoWrite};

#[doc = crate::_tags!(io result)]
/// A specialized [`Result`] type for I/O operations.
///
/// See <https://doc.rust-lang.org/std/io/struct.Result.html>.
pub type IoResult<T> = Result<T, IoError>;

#[doc = crate::_tags!(io error_composite)]
/// Error type for [`IoRead`], [`IoWrite`], [`IoSeek`] operations.
///
/// See <https://doc.rust-lang.org/std/io/struct.Error.html>.
// #[derive(Clone, Copy)] // std::io::Error derives no Clone, Copy or PartialEq…
pub struct IoError {
    repr: Repr,
}
impl_trait![fmt::Debug for IoError |self, f| Debug::fmt(&self.repr, f)];
impl_trait![fmt::Display+Error for IoError |self, f|
    match self.repr {
        Repr::Custom(ref c) => Debug::fmt(&c, f),
        Repr::Simple(kind) => write!(f, "{}", kind.as_str()),
    }
];

#[derive(Clone, Copy, PartialEq)]
enum Repr {
    Custom(Custom),
    Simple(IoErrorKind),
}
impl_trait![fmt::Debug for Repr |self, f|
    match *self {
        Repr::Custom(ref c) => Debug::fmt(&c, f),
        Repr::Simple(kind) => f.debug_tuple("Kind").field(&kind).finish(),
    }
];

#[derive(Clone, Copy, Debug, PartialEq)]
struct Custom {
    kind: IoErrorKind,
    error: &'static str,
}

#[doc = crate::_tags!(io error_composite)]
/// A list specifying general categories of I/O error.
///
/// This list is intended to grow over time and it is not recommended to
/// exhaustively match against it.
///
/// It is used with the [`IoError`] type.
#[derive(Clone, Copy, Debug, Eq, Hash, Ord, PartialEq, PartialOrd)]
#[non_exhaustive]
pub enum IoErrorKind {
    /// An entity was not found, often a file.
    NotFound,

    /// The operation lacked the necessary privileges to complete.
    PermissionDenied,

    /// The connection was refused by the remote server.
    ConnectionRefused,

    /// The connection was reset by the remote server.
    ConnectionReset,

    /// The remote host is not reachable.
    HostUnreachable,

    /// The network containing the remote host is not reachable.
    NetworkUnreachable,

    /// The connection was aborted (terminated) by the remote server.
    ConnectionAborted,

    /// The network operation failed because it was not connected yet.
    NotConnected,

    /// A socket address could not be bound because the address is already in use elsewhere.
    AddrInUse,

    /// A nonexistent interface was requested or the requested address was not local.
    AddrNotAvailable,

    /// The system's networking is down.
    NetworkDown,

    /// The operation failed because a pipe was closed.
    BrokenPipe,

    /// An entity already exists, often a file.
    AlreadyExists,

    /// The operation needs to block to complete,
    /// but the blocking operation was requested to not occur.
    WouldBlock,

    /// A filesystem object is, unexpectedly, not a directory.
    ///
    /// For example, a filesystem path was specified where one of the intermediate directory
    /// components was, in fact, a plain file.
    NotADirectory,

    /// The filesystem object is, unexpectedly, a directory.
    ///
    /// A directory was specified when a non-directory was expected.
    IsADirectory,

    /// A non-empty directory was specified where an empty directory was expected.
    DirectoryNotEmpty,

    /// The filesystem or storage medium is read-only, but a write operation was attempted.
    ReadOnlyFilesystem,

    /// Loop in the filesystem or IO subsystem; often, too many levels of symbolic links.
    ///
    /// There was a loop (or excessively long chain) resolving a filesystem object
    /// or file IO object.
    ///
    /// On Unix this is usually the result of a symbolic link loop;
    /// or, of exceeding the system-specific limit on the depth of symlink traversal.
    // WAIT:1.?? [io_error_more](https://github.com/rust-lang/rust/issues/86442)
    FilesystemLoop,

    /// Stale network file handle.
    ///
    /// With some network filesystems, notably NFS,
    /// an open file (or directory) can be invalidated by problems with the network or server.
    StaleNetworkFileHandle,

    /// A parameter was incorrect.
    InvalidInput,

    /// Data not valid for the operation were encountered.
    ///
    /// Unlike [`InvalidInput`], this typically means that the operation parameters were valid,
    /// however the error was caused by malformed input data.
    ///
    /// For example, a function that reads a file into a string will error with
    /// `InvalidData` if the file's contents are not valid UTF-8.
    ///
    /// [`InvalidInput`]: IoErrorKind::InvalidInput
    InvalidData,

    /// The I/O operation's timeout expired, causing it to be canceled.
    TimedOut,

    /// An error returned when an operation could not be completed because a
    /// call to [`write`] returned [`Ok(0)`].
    ///
    /// This typically means that an operation could only succeed if it wrote a
    /// particular number of bytes but only a smaller number of bytes could be written.
    ///
    /// [`write`]: crate::sys::io::IoWrite::write
    /// [`Ok(0)`]: Ok
    WriteZero,

    /// The underlying storage (typically, a filesystem) is full.
    ///
    /// This does not include out of quota errors.
    StorageFull,

    /// Seek on unseekable file.
    ///
    /// Seeking was attempted on an open file handle which is not suitable for seeking.
    /// For example, on Unix, a named pipe opened with `File::open`.
    NotSeekable,

    /// Filesystem quota was exceeded.
    FilesystemQuotaExceeded,

    /// File larger than allowed or supported.
    ///
    /// This might arise from a hard limit of the underlying filesystem or file access API,
    /// or from an administratively imposed resource limitation.
    /// Simple disk full, and out of quota, have their own errors.
    FileTooLarge,

    /// Resource is busy.
    ResourceBusy,

    /// Executable file is busy.
    ///
    /// An attempt was made to write to a file which is also in use as a running program.
    /// (Not all operating systems detect this situation.)
    ExecutableFileBusy,

    /// Deadlock (avoided).
    ///
    /// A file locking operation would result in deadlock.
    /// This situation is typically detected, if at all, on a best-effort basis.
    Deadlock,

    /// Cross-device or cross-filesystem (hard) link or rename.
    CrossesDevices,

    /// Too many (hard) links to the same filesystem object.
    ///
    /// The filesystem does not support making so many hardlinks to the same file.
    TooManyLinks,

    /// A filename was invalid.
    ///
    /// This error can also cause if it exceeded the filename length limit.
    // WAIT:1.87 [io_error_more](https://github.com/rust-lang/rust/pull/134076)
    InvalidFilename,

    /// Program argument list too long.
    ///
    /// When trying to run an external program, a system or process limit on the size of the
    /// arguments would have been exceeded.
    ArgumentListTooLong,

    /// This operation was interrupted.
    ///
    /// Interrupted operations can typically be retried.
    Interrupted,

    /// This operation is unsupported on this platform.
    ///
    /// This means that the operation can never succeed.
    Unsupported,

    /// An error returned when an operation could not be completed because an
    /// "end of file" was reached prematurely.
    ///
    /// This typically means that an operation could only succeed if it read a
    /// particular number of bytes but only a smaller number of bytes could be
    /// read.
    UnexpectedEof,

    /// An operation could not be completed, because it failed to allocate enough memory.
    OutOfMemory,

    /// The operation was partially successful and needs to be checked later on due to not blocking.
    // WAIT:1.?? [io_error_inprogress](https://github.com/rust-lang/rust/issues/130840)
    InProgress,

    /// Any I/O error not part of this list.
    ///
    /// Errors that are `Other` now may move to a different or a new
    /// [`IoErrorKind`] variant in the future. It is not recommended to match
    /// an error against `Other` and to expect any additional characteristics,
    /// e.g., a specific [`Error::raw_os_error`] return value.
    Other,
}

impl IoErrorKind {
    pub(crate) const fn as_str(self) -> &'static str {
        use IoErrorKind as E;
        // See: https://doc.rust-lang.org/src/std/io/error.rs.html
        match self {
            E::AddrInUse => "address in use",
            E::AddrNotAvailable => "address not available",
            E::AlreadyExists => "entity already exists",
            E::ArgumentListTooLong => "argument list too long",
            E::BrokenPipe => "broken pipe",
            E::ConnectionAborted => "connection aborted",
            E::ConnectionRefused => "connection refused",
            E::ConnectionReset => "connection reset",
            E::CrossesDevices => "cross-device link or rename",
            E::Deadlock => "deadlock",
            E::DirectoryNotEmpty => "directory not empty",
            E::ExecutableFileBusy => "executable file busy",
            E::FileTooLarge => "file too large",
            E::FilesystemLoop => "filesystem loop or indirection limit (e.g. symlink loop)",
            E::FilesystemQuotaExceeded => "filesystem quota exceeded",
            E::HostUnreachable => "host unreachable",
            E::Interrupted => "operation interrupted",
            E::InProgress => "in progress",
            E::InvalidData => "invalid data",
            E::InvalidFilename => "invalid filename",
            E::InvalidInput => "invalid input parameter",
            E::IsADirectory => "is a directory",
            E::NetworkDown => "network down",
            E::NetworkUnreachable => "network unreachable",
            E::NotADirectory => "not a directory",
            E::NotConnected => "not connected",
            E::NotFound => "entity not found",
            E::NotSeekable => "seek on unseekable file",
            E::Other => "other error",
            E::OutOfMemory => "out of memory",
            E::PermissionDenied => "permission denied",
            E::ReadOnlyFilesystem => "read-only filesystem or storage medium",
            E::ResourceBusy => "resource busy",
            E::StaleNetworkFileHandle => "stale network file handle",
            E::StorageFull => "no storage space",
            E::TimedOut => "timed out",
            E::TooManyLinks => "too many links",
            E::UnexpectedEof => "unexpected end of file",
            E::Unsupported => "unsupported",
            E::WouldBlock => "operation would block",
            E::WriteZero => "write zero",
        }
    }
}

/// Intended for use for errors not exposed to the user, where allocating onto
/// the heap (for normal construction via Error::new) is too costly.
impl From<IoErrorKind> for IoError {
    /// Converts an [`IoErrorKind`] into an [`Error`].
    ///
    /// See <https://doc.rust-lang.org/std/io/struct.Error.html#impl-From%3CErrorKind%3E-for-Error>.
    fn from(kind: IoErrorKind) -> IoError {
        IoError { repr: Repr::Simple(kind) }
    }
}

impl IoError {
    /// Creates a new I/O error from a known kind of error as well as an arbitrary error payload.
    ///
    /// Compare with <https://doc.rust-lang.org/std/io/struct.Error.html#method.new>.
    pub const fn new(kind: IoErrorKind, error: &'static str) -> IoError {
        IoError { repr: Repr::Custom(Custom { kind, error }) }
    }

    /// Creates a new I/O error from a static string (no `alloc` needed).
    pub const fn other(msg: &'static str) -> Self {
        IoError {
            repr: Repr::Custom(Custom { kind: IoErrorKind::Other, error: msg }),
        }
    }

    /// Returns a reference to the inner error wrapped by this error (if any).
    pub const fn get_ref(&self) -> Option<&&'static str> {
        match self.repr {
            Repr::Custom(ref c) => Some(&c.error),
            Repr::Simple(..) => None,
        }
    }

    /// Consumes the `IoError`, returning its inner error (if any).
    pub const fn into_inner(self) -> Option<&'static str> {
        match self.repr {
            Repr::Custom(c) => Some(c.error),
            Repr::Simple(..) => None,
        }
    }

    /// Returns the corresponding [`IoErrorKind`] for this error.
    pub const fn kind(&self) -> IoErrorKind {
        match self.repr {
            Repr::Custom(ref c) => c.kind,
            Repr::Simple(kind) => kind,
        }
    }
}

// IMPROVE
const fn _assert_error_is_sync_send() {
    const fn _is_sync_send<T: Sync + Send>() {}
    _is_sync_send::<IoError>();
}