use tokio::io;
use crate::{MavFrame, MavHeader, MavlinkVersion, Message};
#[cfg(feature = "tcp")]
mod tcp;
#[cfg(feature = "udp")]
mod udp;
mod file;
#[cfg(feature = "signing")]
use crate::SigningConfig;
#[async_trait::async_trait]
pub trait AsyncMavConnection<M: Message + Sync + Send> {
async fn recv(&self) -> Result<(MavHeader, M), crate::error::MessageReadError>;
async fn send(
&self,
header: &MavHeader,
data: &M,
) -> Result<usize, crate::error::MessageWriteError>;
fn set_protocol_version(&mut self, version: MavlinkVersion);
fn get_protocol_version(&self) -> MavlinkVersion;
async fn send_frame(
&self,
frame: &MavFrame<M>,
) -> Result<usize, crate::error::MessageWriteError> {
self.send(&frame.header, &frame.msg).await
}
async fn recv_frame(&self) -> Result<MavFrame<M>, crate::error::MessageReadError> {
let (header, msg) = self.recv().await?;
let protocol_version = self.get_protocol_version();
Ok(MavFrame {
header,
msg,
protocol_version,
})
}
async fn send_default(&self, data: &M) -> Result<usize, crate::error::MessageWriteError> {
let header = MavHeader::default();
self.send(&header, data).await
}
#[cfg(feature = "signing")]
fn setup_signing(&mut self, signing_data: Option<SigningConfig>);
}
pub async fn connect_async<M: Message + Sync + Send>(
address: &str,
) -> io::Result<Box<dyn AsyncMavConnection<M> + Sync + Send>> {
let protocol_err = Err(io::Error::new(
io::ErrorKind::AddrNotAvailable,
"Protocol unsupported",
));
if cfg!(feature = "tcp") && address.starts_with("tcp") {
#[cfg(feature = "tcp")]
{
tcp::select_protocol(address).await
}
#[cfg(not(feature = "tcp"))]
{
protocol_err
}
} else if cfg!(feature = "udp") && address.starts_with("udp") {
#[cfg(feature = "udp")]
{
udp::select_protocol(address).await
}
#[cfg(not(feature = "udp"))]
{
protocol_err
}
} else if address.starts_with("file") {
Ok(Box::new(file::open(&address["file:".len()..]).await?))
} else {
protocol_err
}
}
pub(crate) fn get_socket_addr<T: std::net::ToSocketAddrs>(
address: T,
) -> Result<std::net::SocketAddr, io::Error> {
let addr = match address.to_socket_addrs()?.next() {
Some(addr) => addr,
None => {
return Err(io::Error::new(
io::ErrorKind::Other,
"Host address lookup failed",
));
}
};
Ok(addr)
}