use std::{
io,
ops::{Deref, DerefMut},
};
use async_trait::async_trait;
use futures_io::{AsyncBufRead, AsyncRead};
use git_packetline::PacketLineRef;
use crate::{
client::{Error, MessageKind},
Protocol,
};
pub type HandleProgress = Box<dyn FnMut(bool, &[u8])>;
#[async_trait(?Send)]
pub trait ReadlineBufRead: AsyncBufRead {
async fn readline(
&mut self,
) -> Option<io::Result<Result<git_packetline::PacketLineRef<'_>, git_packetline::decode::Error>>>;
}
#[async_trait(?Send)]
pub trait ExtendedBufRead: ReadlineBufRead {
fn set_progress_handler(&mut self, handle_progress: Option<HandleProgress>);
async fn peek_data_line(&mut self) -> Option<io::Result<Result<&[u8], Error>>>;
fn reset(&mut self, version: Protocol);
fn stopped_at(&self) -> Option<MessageKind>;
}
#[async_trait(?Send)]
impl<'a, T: ReadlineBufRead + ?Sized + 'a + Unpin> ReadlineBufRead for Box<T> {
async fn readline(&mut self) -> Option<io::Result<Result<PacketLineRef<'_>, git_packetline::decode::Error>>> {
self.deref_mut().readline().await
}
}
#[async_trait(?Send)]
impl<'a, T: ExtendedBufRead + ?Sized + 'a + Unpin> ExtendedBufRead for Box<T> {
fn set_progress_handler(&mut self, handle_progress: Option<HandleProgress>) {
self.deref_mut().set_progress_handler(handle_progress)
}
async fn peek_data_line(&mut self) -> Option<io::Result<Result<&[u8], Error>>> {
self.deref_mut().peek_data_line().await
}
fn reset(&mut self, version: Protocol) {
self.deref_mut().reset(version)
}
fn stopped_at(&self) -> Option<MessageKind> {
self.deref().stopped_at()
}
}
#[async_trait(?Send)]
impl<T: AsyncRead + Unpin> ReadlineBufRead for git_packetline::read::WithSidebands<'_, T, for<'b> fn(bool, &'b [u8])> {
async fn readline(&mut self) -> Option<io::Result<Result<PacketLineRef<'_>, git_packetline::decode::Error>>> {
self.read_data_line().await
}
}
#[async_trait(?Send)]
impl<'a, T: AsyncRead + Unpin> ReadlineBufRead for git_packetline::read::WithSidebands<'a, T, HandleProgress> {
async fn readline(&mut self) -> Option<io::Result<Result<PacketLineRef<'_>, git_packetline::decode::Error>>> {
self.read_data_line().await
}
}
#[async_trait(?Send)]
impl<'a, T: AsyncRead + Unpin> ExtendedBufRead for git_packetline::read::WithSidebands<'a, T, HandleProgress> {
fn set_progress_handler(&mut self, handle_progress: Option<HandleProgress>) {
self.set_progress_handler(handle_progress)
}
async fn peek_data_line(&mut self) -> Option<io::Result<Result<&[u8], Error>>> {
match self.peek_data_line().await {
Some(Ok(Ok(line))) => Some(Ok(Ok(line))),
Some(Ok(Err(err))) => Some(Ok(Err(err.into()))),
Some(Err(err)) => Some(Err(err)),
None => None,
}
}
fn reset(&mut self, version: Protocol) {
match version {
Protocol::V1 => self.reset_with(&[git_packetline::PacketLineRef::Flush]),
Protocol::V2 => self.reset_with(&[
git_packetline::PacketLineRef::Delimiter,
git_packetline::PacketLineRef::Flush,
]),
}
}
fn stopped_at(&self) -> Option<MessageKind> {
self.stopped_at().map(|l| match l {
git_packetline::PacketLineRef::Flush => MessageKind::Flush,
git_packetline::PacketLineRef::Delimiter => MessageKind::Delimiter,
git_packetline::PacketLineRef::ResponseEnd => MessageKind::ResponseEnd,
git_packetline::PacketLineRef::Data(_) => unreachable!("data cannot be a delimiter"),
})
}
}