use std::{
cmp::{Eq, Ord, Ordering, PartialEq, PartialOrd},
ffi::CString,
hash::{Hash, Hasher},
num::NonZeroU32,
};
use crate::{
error::{Error, Result},
socket::Socket,
};
#[derive(Clone, Copy, Debug)]
pub struct Listener
{
handle: nng_sys::nng_listener,
}
impl Listener
{
pub fn new(socket: &Socket, url: &str) -> Result<Self>
{
let addr = CString::new(url).map_err(|_| Error::AddressInvalid)?;
let mut handle = nng_sys::nng_listener::NNG_LISTENER_INITIALIZER;
let rv = unsafe {
nng_sys::nng_listen(socket.handle(), addr.as_ptr(), &mut handle as *mut _, 0)
};
rv2res!(rv, Listener { handle })
}
#[allow(clippy::missing_panics_doc)]
pub fn close(self)
{
let rv = unsafe { nng_sys::nng_listener_close(self.handle) };
assert!(
rv == 0 || rv == nng_sys::NNG_ECLOSED as i32,
"Unexpected error code while closing listener ({})",
rv
);
}
pub(crate) fn from_nng_sys(handle: nng_sys::nng_listener) -> Self
{
assert!(
unsafe { nng_sys::nng_listener_id(handle) > 0 },
"Listener handle is not initialized"
);
Listener { handle }
}
}
#[cfg(feature = "ffi-module")]
impl Listener
{
pub fn nng_listener(self) -> nng_sys::nng_listener { self.handle }
}
impl PartialEq for Listener
{
fn eq(&self, other: &Listener) -> bool
{
unsafe { nng_sys::nng_listener_id(self.handle) == nng_sys::nng_listener_id(other.handle) }
}
}
impl Eq for Listener {}
impl PartialOrd for Listener
{
fn partial_cmp(&self, other: &Listener) -> Option<Ordering> { Some(self.cmp(other)) }
}
impl Ord for Listener
{
fn cmp(&self, other: &Listener) -> Ordering
{
unsafe {
let us = nng_sys::nng_listener_id(self.handle);
let them = nng_sys::nng_listener_id(other.handle);
us.cmp(&them)
}
}
}
impl Hash for Listener
{
fn hash<H: Hasher>(&self, state: &mut H)
{
let id = unsafe { nng_sys::nng_listener_id(self.handle) };
id.hash(state);
}
}
#[rustfmt::skip]
expose_options!{
Listener :: handle -> nng_sys::nng_listener;
GETOPT_BOOL = nng_sys::nng_listener_get_bool;
GETOPT_INT = nng_sys::nng_listener_get_int;
GETOPT_MS = nng_sys::nng_listener_get_ms;
GETOPT_SIZE = nng_sys::nng_listener_get_size;
GETOPT_SOCKADDR = nng_sys::nng_listener_get_addr;
GETOPT_STRING = nng_sys::nng_listener_get_string;
GETOPT_UINT64 = nng_sys::nng_listener_get_uint64;
SETOPT = nng_sys::nng_listener_set;
SETOPT_BOOL = nng_sys::nng_listener_set_bool;
SETOPT_INT = nng_sys::nng_listener_set_int;
SETOPT_MS = nng_sys::nng_listener_set_ms;
SETOPT_PTR = nng_sys::nng_listener_set_ptr;
SETOPT_SIZE = nng_sys::nng_listener_set_size;
SETOPT_STRING = nng_sys::nng_listener_set_string;
Gets -> [LocalAddr, Raw, RecvBufferSize,
RecvTimeout, SendBufferSize, Url,
SendTimeout, SocketName, MaxTtl,
protocol::reqrep::ResendTime,
protocol::survey::SurveyTime,
transport::tcp::NoDelay,
transport::tcp::KeepAlive,
transport::tcp::BoundPort,
transport::websocket::Protocol];
Sets -> [];
}
#[derive(Debug)]
pub struct ListenerBuilder
{
handle: nng_sys::nng_listener,
}
impl ListenerBuilder
{
pub fn new(socket: &Socket, url: &str) -> Result<Self>
{
let addr = CString::new(url).map_err(|_| Error::AddressInvalid)?;
let mut handle = nng_sys::nng_listener::NNG_LISTENER_INITIALIZER;
let rv = unsafe {
nng_sys::nng_listener_create(&mut handle as *mut _, socket.handle(), addr.as_ptr())
};
rv2res!(rv, ListenerBuilder { handle })
}
pub fn start(self) -> std::result::Result<Listener, (Self, Error)>
{
let rv = unsafe { nng_sys::nng_listener_start(self.handle, 0) };
if let Some(e) = NonZeroU32::new(rv as u32) {
Err((self, Error::from(e)))
}
else {
let handle = Listener { handle: self.handle };
std::mem::forget(self);
Ok(handle)
}
}
}
#[cfg(feature = "ffi-module")]
impl ListenerBuilder
{
pub fn nng_listener(&self) -> nng_sys::nng_listener { self.handle }
}
#[rustfmt::skip]
expose_options!{
ListenerBuilder :: handle -> nng_sys::nng_listener;
GETOPT_BOOL = nng_sys::nng_listener_get_bool;
GETOPT_INT = nng_sys::nng_listener_get_int;
GETOPT_MS = nng_sys::nng_listener_get_ms;
GETOPT_SIZE = nng_sys::nng_listener_get_size;
GETOPT_SOCKADDR = nng_sys::nng_listener_get_addr;
GETOPT_STRING = nng_sys::nng_listener_get_string;
GETOPT_UINT64 = nng_sys::nng_listener_get_uint64;
SETOPT = nng_sys::nng_listener_setopt;
SETOPT_BOOL = nng_sys::nng_listener_setopt_bool;
SETOPT_INT = nng_sys::nng_listener_setopt_int;
SETOPT_MS = nng_sys::nng_listener_setopt_ms;
SETOPT_PTR = nng_sys::nng_listener_setopt_ptr;
SETOPT_SIZE = nng_sys::nng_listener_setopt_size;
SETOPT_STRING = nng_sys::nng_listener_setopt_string;
Gets -> [LocalAddr, Raw, RecvBufferSize,
RecvTimeout, SendBufferSize, Url,
SendTimeout, SocketName, MaxTtl,
protocol::reqrep::ResendTime,
protocol::survey::SurveyTime,
transport::tcp::NoDelay,
transport::tcp::KeepAlive,
transport::websocket::Protocol];
Sets -> [RecvMaxSize, transport::tcp::NoDelay,
transport::tcp::KeepAlive,
transport::tls::CaFile,
transport::tls::CertKeyFile,
transport::websocket::ResponseHeaders,
transport::websocket::Protocol];
}
#[cfg(unix)]
mod unix_impls
{
use super::*;
use crate::options::transport::ipc;
impl crate::options::SetOpt<ipc::Permissions> for ListenerBuilder {}
}
impl Drop for ListenerBuilder
{
fn drop(&mut self)
{
let rv = unsafe { nng_sys::nng_listener_close(self.handle) };
assert!(
rv == 0 || rv == nng_sys::NNG_ECLOSED as i32,
"Unexpected error code while closing listener ({})",
rv
);
}
}