io-close 0.3.7

An extension trait for safely dropping I/O writers such as File and BufWriter.
Documentation
#![cfg_attr(docsrs, feature(doc_cfg))]

//! An extension trait for safely dropping [I/O writers](std::io::Write)
//! such as [`File`](std::fs::File) and
//! [`BufWriter`](std::io::BufWriter). Specifically, it is for I/O
//! writers which may contain a resource handle (such as a raw file
//! descriptor), and where the automatic process of closing this handle
//! during drop may generate an unseen I/O error. Using this trait
//! (called [`Close`]) these errors can be seen and dealt with.
//!
//! In the case of Linux [the man page for
//! close(2)](https://linux.die.net/man/2/close) has the following to
//! say:
//!
//! > Not checking the return value of close() is a common but
//! > nevertheless serious programming error. It is quite possible that
//! > errors on a previous write(2) operation are first reported at the
//! > final close(). Not checking the return value when closing the file
//! > may lead to silent loss of data. This can especially be observed
//! > with NFS and with disk quota.
//!
//! Implementations of [`Close`] are provided for most standard library
//! I/O writers and (optionally) for a selection of I/O writers defined
//! in external crates.
//!
//! # BufWriter example
//!
//! ```
//! use std::io::{BufWriter, Result, Write};
//! use io_close::Close;
//!
//! fn main() -> Result<()> {
//!     let data = b"hello world";
//!     let mut buffer = BufWriter::new(tempfile::tempfile()?);
//!     buffer.write_all(data)?;
//!     buffer.close()?; // safely drop buffer and its contained File
//!     Ok(())
//! }
//! ```
//!
//! # Optional implementations
//!
//! Optional implementations of [`Close`] are provided for the following
//! I/O writers defined in external crates, enabled through cargo features:
//!
//! - [`os_pipe::PipeWriter`] (feature: `os_pipe`)

use std::io::{Error, Result, Write};

pub mod fs;

/// An extension trait for safely dropping I/O writers.
pub trait Close: Write {
    /// Drops an I/O writer and closes any resource handle contained
    /// inside (such as a raw file descriptor). Ensures that I/O errors
    /// resulting from closing a handle are not ignored. The writer is
    /// flushed before any handle is closed. If any errors occur during
    /// flushing or closing the first such error is returned.
    fn close(self) -> Result<()>;
}

// MACRO DEFINITIONS

macro_rules! unix_impl_close_raw_fd {
    ($ty:ty, "std" $(,$lt:lifetime)* $(,$id:ident)*) => {
        unix_impl_close_raw_fd!($ty, "unix" $(,$lt)* $(,$id)*);
    };
    ($ty:ty, $ft_fm:literal $(,$lt:lifetime)* $(,$id:ident)*) => {
        #[cfg(unix)]
        #[cfg(any(feature = $ft_fm, target_family = $ft_fm))]
        #[cfg_attr(all(docsrs, feature = $ft_fm), doc(cfg(feature = $ft_fm)))]
        impl<$($lt,)* $($id,)*> Close for $ty {
            /// Drops an I/O writer containing a raw file descriptor.
            fn close(mut self) -> Result<()> {
                use std::io::ErrorKind;
                use std::os::unix::io::IntoRawFd;

                self.flush()?;
                let fd = self.into_raw_fd();
                let rv = unsafe { libc::close(fd) };
                if rv != -1 {
                    Ok(())
                } else {
                    match Error::last_os_error() {
                        e if e.kind() == ErrorKind::Interrupted => Ok(()),
                        e => Err(e),
                    }
                }
            }
        }
    };
}

macro_rules! windows_impl_close_raw_handle {
    ($ty:ty, "std" $(,$lt:lifetime)* $(,$id:ident)*) => {
        windows_impl_close_raw_handle!($ty, "windows" $(,$lt)* $(,$id)*);
    };
    ($ty:ty, $ft_fm:literal $(,$lt:lifetime)* $(,$id:ident)*) => {
        #[cfg(windows)]
        #[cfg(any(feature = $ft_fm, target_family = $ft_fm))]
        #[cfg_attr(all(docsrs, feature = $ft_fm), doc(cfg(feature = $ft_fm)))]
        impl<$($lt,)* $($id,)*> Close for $ty {
            /// Drops an I/O writer containing a raw handle.
            fn close(mut self) -> Result<()> {
                use std::os::windows::io::IntoRawHandle;
                use winapi::um::handleapi;

                self.flush()?;
                let handle = self.into_raw_handle();
                let rv = unsafe { handleapi::CloseHandle(handle) };
                if rv != 0 {
                    Ok(())
                } else {
                    Err(Error::last_os_error())
                }
            }
        }
    };
}

macro_rules! windows_impl_close_raw_socket {
    ($ty:ty, "std" $(,$lt:lifetime)* $(,$id:ident)*) => {
        windows_impl_close_raw_socket!($ty, "windows" $(,$lt)* $(,$id)*);
    };
    ($ty:ty, $ft_fm:literal $(,$lt:lifetime)* $(,$id:ident)*) => {
        #[cfg(windows)]
        #[cfg(any(feature = $ft_fm, target_family = $ft_fm))]
        #[cfg_attr(all(docsrs, feature = $ft_fm), doc(cfg(feature = $ft_fm)))]
        impl<$($lt,)* $($id,)*> Close for $ty {
            /// Drops an I/O writer containing a raw socket.
            fn close(mut self) -> Result<()> {
                use std::convert::TryInto;
                use std::os::windows::io::IntoRawSocket;
                use winapi::um::winsock2;

                self.flush()?;
                let socket = self.into_raw_socket().try_into().unwrap();
                let rv = unsafe { winsock2::closesocket(socket) };
                if rv == 0 {
                    Ok(())
                } else {
                    Err(Error::from_raw_os_error(unsafe {
                        winsock2::WSAGetLastError()
                    }))
                }
            }
        }
    };
}

macro_rules! impl_close_no_error_no_flush {
    ($ty:ty, "std" $(,$lt:lifetime)* $(,$id:ident)*) => {
        #[cfg(unix)]
        impl_close_no_error_no_flush!($ty, "unix" $(,$lt)* $(,$id)*);
        #[cfg(windows)]
        impl_close_no_error_no_flush!($ty, "windows" $(,$lt)* $(,$id)*);
    };
    ($ty:ty, $ft_fm:literal $(,$lt:lifetime)* $(,$id:ident)*) => {
        #[cfg(any(unix, windows))]
        #[cfg(any(feature = $ft_fm, target_family = $ft_fm))]
        #[cfg_attr(all(docsrs, feature = $ft_fm), doc(cfg(feature = $ft_fm)))]
        impl<$($lt,)* $($id,)*> Close for $ty {
            /// Drops an I/O writer for which `close()` never produces
            /// an error, and for which flushing is unnecessary.
            #[inline]
            fn close(self) -> Result<()> {
                Ok(())
            }
        }
    };
}

macro_rules! impl_close_into_inner {
    ($ty:ty, "std" $(,$lt:lifetime)* $(,$id:ident)*) => {
        #[cfg(unix)]
        impl_close_into_inner!($ty, "unix" $(,$lt)* $(,$id)*);
        #[cfg(windows)]
        impl_close_into_inner!($ty, "windows" $(,$lt)* $(,$id)*);
    };
    ($ty:ty, $ft_fm:literal $(,$lt:lifetime)* $(,$id:ident)*) => {
        #[cfg(any(unix, windows))]
        #[cfg(any(feature = $ft_fm, target_family = $ft_fm))]
        #[cfg_attr(all(docsrs, feature = $ft_fm), doc(cfg(feature = $ft_fm)))]
        impl<$($lt,)* W: Close, $($id,)*> Close for $ty {
            /// Drops an I/O writer which can be unwrapped using
            /// `into_inner()` to return an underlying writer.
            fn close(self) -> Result<()> {
                self.into_inner()?.close()
            }
        }
    };
}

// IMPLEMENTATIONS
//
// In the below macro implementations for Close the macro parameters use
// the following system:
//
//     1st paramater: the type that Close is being implemented for
//     2nd parameter: either "std", or a feature name
//     3rd, 4th, etc. parameters: additional generic arguments for type
//
// If the 2nd parameter is a feature name then the implementation will
// be conditionally compiled only when that feature is present.

unix_impl_close_raw_fd!(std::fs::File, "std");
unix_impl_close_raw_fd!(std::net::TcpStream, "std");
unix_impl_close_raw_fd!(std::os::unix::net::UnixStream, "std");
unix_impl_close_raw_fd!(std::process::ChildStdin, "std");
unix_impl_close_raw_fd!(os_pipe::PipeWriter, "os_pipe");

windows_impl_close_raw_handle!(std::fs::File, "std");
windows_impl_close_raw_socket!(std::net::TcpStream, "std");
windows_impl_close_raw_handle!(std::process::ChildStdin, "std");
windows_impl_close_raw_handle!(os_pipe::PipeWriter, "os_pipe");

impl_close_no_error_no_flush!(&mut [u8], "std");
impl_close_no_error_no_flush!(std::io::Cursor<&mut Vec<u8>>, "std");
impl_close_no_error_no_flush!(std::io::Cursor<&mut [u8]>, "std");
impl_close_no_error_no_flush!(std::io::Cursor<Box<[u8]>>, "std");
impl_close_no_error_no_flush!(std::io::Cursor<Vec<u8>>, "std");
impl_close_no_error_no_flush!(std::io::Sink, "std");
impl_close_no_error_no_flush!(Vec<u8>, "std");

impl_close_into_inner!(std::io::BufWriter<W>, "std");
impl_close_into_inner!(std::io::LineWriter<W>, "std");