1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147
//! Asynchronous signal handling for Tokio //! //! This crate implements asynchronous signal handling for Tokio, an //! asynchronous I/O framework in Rust. The primary type exported from this //! crate, `unix::Signal`, allows listening for arbitrary signals on Unix //! platforms, receiving them in an asynchronous fashion. //! //! Note that signal handling is in general a very tricky topic and should be //! used with great care. This crate attempts to implement 'best practice' for //! signal handling, but it should be evaluated for your own applications' needs //! to see if it's suitable. //! //! The are some fundamental limitations of this crate documented on the //! `Signal` structure as well. //! //! # Examples //! //! Print out all ctrl-C notifications received //! //! ```rust,no_run //! extern crate futures; //! extern crate tokio_core; //! extern crate tokio_signal; //! //! use tokio_core::reactor::Core; //! use futures::{Future, Stream}; //! //! fn main() { //! let mut core = Core::new().unwrap(); //! //! // Create an infinite stream of "Ctrl+C" notifications. Each item received //! // on this stream may represent multiple ctrl-c signals. //! let ctrl_c = tokio_signal::ctrl_c().flatten_stream(); //! //! // Process each ctrl-c as it comes in //! let prog = ctrl_c.for_each(|()| { //! println!("ctrl-c received!"); //! Ok(()) //! }); //! //! core.run(prog).unwrap(); //! } //! ``` //! //! Wait for SIGHUP on Unix //! //! ```rust,no_run //! # extern crate futures; //! # extern crate tokio_core; //! # extern crate tokio_signal; //! # #[cfg(unix)] //! # mod foo { //! # //! extern crate futures; //! extern crate tokio_core; //! extern crate tokio_signal; //! //! use tokio_core::reactor::Core; //! use futures::{Future, Stream}; //! use tokio_signal::unix::{Signal, SIGHUP}; //! //! fn main() { //! let mut core = Core::new().unwrap(); //! //! // Like the previous example, this is an infinite stream of signals //! // being received, and signals may be coalesced while pending. //! let stream = Signal::new(SIGHUP).flatten_stream(); //! //! // Convert out stream into a future and block the program //! core.run(stream.into_future()).ok().unwrap(); //! } //! # } //! # fn main() {} //! ``` #![doc(html_root_url = "https://docs.rs/tokio-signal/0.1")] #![deny(missing_docs)] extern crate futures; extern crate mio; extern crate tokio_executor; extern crate tokio_io; extern crate tokio_reactor; use std::io; use futures::stream::Stream; use futures::{future, Future}; use tokio_reactor::Handle; pub mod unix; pub mod windows; /// A future whose error is `io::Error` pub type IoFuture<T> = Box<Future<Item = T, Error = io::Error> + Send>; /// A stream whose error is `io::Error` pub type IoStream<T> = Box<Stream<Item = T, Error = io::Error> + Send>; /// Creates a stream which receives "ctrl-c" notifications sent to a process. /// /// In general signals are handled very differently across Unix and Windows, but /// this is somewhat cross platform in terms of how it can be handled. A ctrl-c /// event to a console process can be represented as a stream for both Windows /// and Unix. /// /// This function binds to the default event loop. Note that /// there are a number of caveats listening for signals, and you may wish to /// read up on the documentation in the `unix` or `windows` module to take a /// peek. pub fn ctrl_c() -> IoFuture<IoStream<()>> { ctrl_c_handle(&Handle::current()) } /// Creates a stream which receives "ctrl-c" notifications sent to a process. /// /// In general signals are handled very differently across Unix and Windows, but /// this is somewhat cross platform in terms of how it can be handled. A ctrl-c /// event to a console process can be represented as a stream for both Windows /// and Unix. /// /// This function receives a `Handle` to an event loop and returns a future /// which when resolves yields a stream receiving all signal events. Note that /// there are a number of caveats listening for signals, and you may wish to /// read up on the documentation in the `unix` or `windows` module to take a /// peek. pub fn ctrl_c_handle(handle: &Handle) -> IoFuture<IoStream<()>> { return ctrl_c_imp(handle); #[cfg(unix)] fn ctrl_c_imp(handle: &Handle) -> IoFuture<IoStream<()>> { let handle = handle.clone(); Box::new(future::lazy(move || { unix::Signal::with_handle(unix::libc::SIGINT, &handle) .map(|x| Box::new(x.map(|_| ())) as Box<Stream<Item = _, Error = _> + Send>) })) } #[cfg(windows)] fn ctrl_c_imp(handle: &Handle) -> IoFuture<IoStream<()>> { let handle = handle.clone(); // Use lazy to ensure that `ctrl_c` gets called while on an event loop Box::new(future::lazy(move || { windows::Event::ctrl_c_handle(&handle) .map(|x| Box::new(x) as Box<Stream<Item = _, Error = _> + Send>) })) } }