The purpose of this crate is to make it a bit more ergonomic for portable
applications that need to work with the platform level RawFd
and
RawHandle
types.
Rather than conditionally using RawFd
and RawHandle
, the FileDescriptor
type can be used to manage ownership, duplicate, read and write.
This is a bit of a contrived example, but demonstrates how to avoid
the conditional code that would otherwise be required to deal with
calling as_raw_fd
and as_raw_handle
:
``` use filedescriptor::{FileDescriptor, FromRawFileDescriptor}; use failure::Fallible; use std::io::Write;
fn get_stdout() -> Fallible
fn printsomething() -> Fallible<()> { getstdout()?.write(b"hello")?; Ok(()) } ```
The Pipe
type makes it more convenient to create a pipe and manage
the lifetime of both the read and write ends of that pipe.
``` use filedescriptor::Pipe; use std::io::{Read, Write}; use failure::Error;
let mut pipe = Pipe::new()?; pipe.write.write(b"hello")?; drop(pipe.write);
let mut s = String::new(); pipe.read.readtostring(&mut s)?; assert_eq!(s, "hello"); ```
The socketpair
function returns a pair of connected SOCK_STREAM
sockets and functions both on posix and windows systems.
``` use std::io::{Read, Write}; use failure::Error;
let (mut a, mut b) = filedescriptor::socketpair()?; a.write(b"hello")?; drop(a);
let mut s = String::new(); b.readtostring(&mut s)?; assert_eq!(s, "hello"); ```
The mio
crate offers powerful and scalable IO multiplexing, but there
are some situations where mio
doesn't fit. The filedescriptor
crate
offers a poll(2)
compatible interface suitable for testing the readiness
of a set of file descriptors. On unix systems this is a very thin wrapper
around poll(2)
, except on macOS where it is actually a wrapper around
the select(2)
interface. On Windows systems the winsock WSAPoll
function is used instead.
``` use filedescriptor::*; use failure::Error; use std::time::Duration; use std::io::{Read, Write};
let (mut a, mut b) = filedescriptor::socketpair()?;
let mut pollarray = [pollfd {
fd: a.assocketdescriptor(),
events: POLLIN,
revents: 0
}];
// sleeps for 20 milliseconds because a
is not yet ready
asserteq!(poll(&mut pollarray, Some(Duration::frommillis(20)))?, 0);
b.write(b"hello")?;
// Now a is ready for read asserteq!(poll(&mut pollarray, Some(Duration::from_millis(20)))?, 1);
```