use futures::sync::mpsc;
use futures::{self,Async};
use std::os::raw::{c_void,c_char};
use std::io;
use tokio_core::reactor::{Handle,Remote};
use cstr;
use error::Error;
use evented::EventedDNSService;
use ffi;
use interface::Interface;
use raw;
use remote::GetRemote;
use stream::ServiceStream;
#[derive(Clone,Copy,PartialEq,Eq,PartialOrd,Ord,Hash,Debug)]
pub enum Enumerate {
BrowseDomains,
RegistrationDomains,
}
impl Into<ffi::DNSServiceFlags> for Enumerate {
fn into(self) -> ffi::DNSServiceFlags {
match self {
Enumerate::BrowseDomains => ffi::FLAGS_BROWSE_DOMAINS,
Enumerate::RegistrationDomains => ffi::FLAGS_REGISTRATION_DOMAINS,
}
}
}
#[derive(Clone,Copy,PartialEq,Eq,PartialOrd,Ord,Hash)]
pub struct EnumeratedFlags(u8);
#[derive(Clone,Copy,PartialEq,Eq,PartialOrd,Ord,Hash,Debug)]
#[repr(u8)]
pub enum EnumeratedFlag {
MoreComing = 0,
Add,
Default,
}
flags_ops!{EnumeratedFlags: u8: EnumeratedFlag:
MoreComing,
Add,
Default,
}
flag_mapping!{EnumeratedFlags: EnumeratedFlag => ffi::DNSServiceFlags:
MoreComing => ffi::FLAGS_MORE_COMING,
Add => ffi::FLAGS_ADD,
Default => ffi::FLAGS_DEFAULT,
}
pub struct EnumerateDomains(ServiceStream<EnumerateResult>);
impl futures::Stream for EnumerateDomains {
type Item = EnumerateResult;
type Error = io::Error;
fn poll(&mut self) -> Result<Async<Option<Self::Item>>, Self::Error> {
self.0.poll()
}
}
impl GetRemote for EnumerateDomains {
fn remote(&self) -> &Remote {
self.0.remote()
}
}
#[derive(Clone,PartialEq,Eq,PartialOrd,Ord,Hash,Debug)]
pub struct EnumerateResult{
pub flags: EnumeratedFlags,
pub interface: Interface,
pub domain: String,
}
extern "C" fn enumerate_callback(
_sd_ref: ffi::DNSServiceRef,
flags: ffi::DNSServiceFlags,
interface_index: u32,
error_code: ffi::DNSServiceErrorType,
reply_domain: *const c_char,
context: *mut c_void
) {
let sender = context as *mut mpsc::UnboundedSender<io::Result<EnumerateResult>>;
let sender : &mpsc::UnboundedSender<io::Result<EnumerateResult>> = unsafe { &*sender };
let data = Error::from(error_code).map_err(io::Error::from).and_then(|_| {
let reply_domain = unsafe { cstr::from_cstr(reply_domain) }?;
Ok(EnumerateResult{
flags: EnumeratedFlags::from(flags),
interface: Interface::from_raw(interface_index),
domain: reply_domain.to_string(),
})
});
sender.send(data).unwrap();
}
pub fn enumerate_domains(enumerate: Enumerate, interface: Interface, handle: &Handle) -> io::Result<EnumerateDomains> {
Ok(EnumerateDomains(ServiceStream::new(move |sender|
EventedDNSService::new(
raw::DNSService::enumerate_domains(
enumerate.into(),
interface.into_raw(),
Some(enumerate_callback),
sender as *mut c_void,
)?,
handle
)
)?))
}