a10 0.4.2

This library is meant as a low-level library safely exposing different OS's abilities to perform non-blocking I/O.
Documentation
use a10::fd::{self, AsyncFd};
#[cfg(any(target_os = "android", target_os = "linux"))]
use a10::fd::{ToDirect, ToFd};
#[cfg(any(target_os = "android", target_os = "linux"))]
use a10::fs::OpenOptions;

#[cfg(any(target_os = "android", target_os = "linux"))]
use crate::util::{LOREM_IPSUM_5, Waker, expect_io_error_kind, test_queue};
use crate::util::{is_send, is_sync};

#[test]
fn async_fd_size() {
    #[cfg(any(target_os = "android", target_os = "linux"))]
    const SIZE: usize = 16;
    #[cfg(any(
        target_os = "dragonfly",
        target_os = "freebsd",
        target_os = "ios",
        target_os = "macos",
        target_os = "netbsd",
        target_os = "openbsd",
        target_os = "tvos",
        target_os = "visionos",
        target_os = "watchos",
    ))]
    const SIZE: usize = 24;
    assert_eq!(std::mem::size_of::<AsyncFd>(), SIZE);
    assert_eq!(std::mem::size_of::<Option<AsyncFd>>(), SIZE);
}

#[test]
fn async_fd_is_send_and_sync() {
    is_send::<AsyncFd>();
    is_sync::<AsyncFd>();
}

#[test]
fn kind_is_send_and_sync() {
    is_send::<fd::Kind>();
    is_sync::<fd::Kind>();
}

#[test]
#[cfg(any(target_os = "android", target_os = "linux"))]
fn to_fd_is_send_and_sync() {
    is_send::<ToFd>();
    is_sync::<ToFd>();
}

#[test]
#[cfg(any(target_os = "android", target_os = "linux"))]
fn to_direct_is_send_and_sync() {
    is_send::<ToDirect>();
    is_sync::<ToDirect>();
}

#[test]
#[cfg(any(target_os = "android", target_os = "linux"))]
fn to_direct_descriptor() {
    test_change_descriptor_type(fd::Kind::File);
}

#[test]
#[cfg(any(target_os = "android", target_os = "linux"))]
fn to_file_descriptor() {
    test_change_descriptor_type(fd::Kind::Direct);
}

#[cfg(any(target_os = "android", target_os = "linux"))]
fn test_change_descriptor_type(fd_kind: fd::Kind) {
    let sq = test_queue();
    let waker = Waker::new();

    let open_file = OpenOptions::new()
        .kind(fd_kind)
        .open(sq, LOREM_IPSUM_5.path.into());
    let fd = waker.block_on(open_file).unwrap();

    let (regular_fd, direct_fd) = if let fd::Kind::File = fd_kind {
        let regular_fd = fd;
        let direct_fd = waker.block_on(regular_fd.to_direct_descriptor()).unwrap();
        (regular_fd, direct_fd)
    } else {
        let direct_fd = fd;
        let regular_fd = waker.block_on(direct_fd.to_file_descriptor()).unwrap();
        (regular_fd, direct_fd)
    };

    let mut buf = Vec::with_capacity(LOREM_IPSUM_5.content.len() + 1);

    // Regular file descriptor.
    let read = regular_fd.read(buf).from(0);
    buf = waker.block_on(read).unwrap();
    if buf.is_empty() {
        panic!("read zero bytes");
    }
    assert!(
        buf == &LOREM_IPSUM_5.content[..buf.len()],
        "read content is different"
    );

    // Direct descriptor.
    buf.clear();
    let read = direct_fd.read(buf).from(0);
    buf = waker.block_on(read).unwrap();
    if buf.is_empty() {
        panic!("read zero bytes");
    }
    assert!(
        buf == &LOREM_IPSUM_5.content[..buf.len()],
        "read content is different"
    );

    waker.block_on(regular_fd.close()).unwrap();
    waker.block_on(direct_fd.close()).unwrap();
}

#[test]
#[cfg_attr(
    debug_assertions,
    should_panic = "can't covert a direct descriptor to a different direct descriptor"
)]
#[cfg(any(target_os = "android", target_os = "linux"))]
fn direct_to_direct_descriptor() {
    let sq = test_queue();
    let waker = Waker::new();

    let open_file = OpenOptions::new()
        .kind(fd::Kind::Direct)
        .open(sq, LOREM_IPSUM_5.path.into());
    let direct_fd = waker.block_on(open_file).unwrap();
    assert_eq!(direct_fd.kind(), fd::Kind::Direct);
    // This should panic.
    let res = waker.block_on(direct_fd.to_direct_descriptor());
    expect_io_error_kind(res, std::io::ErrorKind::Unsupported);
}

#[test]
#[cfg_attr(
    debug_assertions,
    should_panic = "can't covert a file descriptor to a different file descriptor"
)]
#[cfg(any(target_os = "android", target_os = "linux"))]
fn file_to_file_descriptor() {
    let sq = test_queue();
    let waker = Waker::new();

    let open_file = OpenOptions::new().open(sq, LOREM_IPSUM_5.path.into());
    let regular_fd = waker.block_on(open_file).unwrap();
    assert_eq!(regular_fd.kind(), fd::Kind::File);
    // This should panic.
    let res = waker.block_on(regular_fd.to_file_descriptor());
    expect_io_error_kind(res, std::io::ErrorKind::Unsupported);
}