use std::ops::DerefMut;
use async_trait::async_trait;
use bstr::BString;
use futures_lite::io::AsyncWriteExt;
use crate::{
client::{Capabilities, Error, ExtendedBufRead, MessageKind, TransportWithoutIO, WriteMode},
Protocol, Service,
};
pub struct SetServiceResponse<'a> {
pub actual_protocol: Protocol,
pub capabilities: Capabilities,
pub refs: Option<Box<dyn crate::client::ReadlineBufRead + Unpin + 'a>>,
}
#[async_trait(?Send)]
pub trait Transport: TransportWithoutIO {
async fn handshake<'a>(
&mut self,
service: Service,
extra_parameters: &'a [(&'a str, Option<&'a str>)],
) -> Result<SetServiceResponse<'_>, Error>;
}
#[async_trait(?Send)]
impl<T: Transport + ?Sized> Transport for Box<T> {
async fn handshake<'a>(
&mut self,
service: Service,
extra_parameters: &'a [(&'a str, Option<&'a str>)],
) -> Result<SetServiceResponse<'_>, Error> {
self.deref_mut().handshake(service, extra_parameters).await
}
}
#[async_trait(?Send)]
impl<T: Transport + ?Sized> Transport for &mut T {
async fn handshake<'a>(
&mut self,
service: Service,
extra_parameters: &'a [(&'a str, Option<&'a str>)],
) -> Result<SetServiceResponse<'_>, Error> {
self.deref_mut().handshake(service, extra_parameters).await
}
}
#[async_trait(?Send)]
pub trait TransportV2Ext {
async fn invoke<'a>(
&mut self,
command: &str,
capabilities: impl Iterator<Item = (&'a str, Option<impl AsRef<str>>)> + 'a,
arguments: Option<impl Iterator<Item = bstr::BString> + 'a>,
) -> Result<Box<dyn ExtendedBufRead + Unpin + '_>, Error>;
}
#[async_trait(?Send)]
impl<T: Transport> TransportV2Ext for T {
async fn invoke<'a>(
&mut self,
command: &str,
capabilities: impl Iterator<Item = (&'a str, Option<impl AsRef<str>>)> + 'a,
arguments: Option<impl Iterator<Item = BString> + 'a>,
) -> Result<Box<dyn ExtendedBufRead + Unpin + '_>, Error> {
let mut writer = self.request(WriteMode::OneLfTerminatedLinePerWriteCall, MessageKind::Flush)?;
writer.write_all(format!("command={}", command).as_bytes()).await?;
for (name, value) in capabilities {
match value {
Some(value) => {
writer
.write_all(format!("{}={}", name, value.as_ref()).as_bytes())
.await
}
None => writer.write_all(name.as_bytes()).await,
}?;
}
if let Some(arguments) = arguments {
writer.write_message(MessageKind::Delimiter).await?;
for argument in arguments {
writer.write_all(argument.as_ref()).await?;
}
}
Ok(writer.into_read().await?)
}
}