use std::ffi::OsStr;
use std::io::{stdin, stdout, Cursor, Read, Seek, SeekFrom};
use std::rc::Rc;
use super::{InputFeatures, InputHandle, InputOrigin, IoProvider, OpenResult, OutputHandle};
use crate::errors::Result;
use crate::status::StatusBackend;
#[derive(Clone, Copy, Debug, Default, Eq, PartialEq)]
pub struct GenuineStdoutIo {}
impl GenuineStdoutIo {
pub fn new() -> GenuineStdoutIo {
Default::default()
}
}
impl IoProvider for GenuineStdoutIo {
fn output_open_stdout(&mut self) -> OpenResult<OutputHandle> {
OpenResult::Ok(OutputHandle::new(OsStr::new(""), stdout()))
}
}
#[derive(Clone, Debug, Eq, PartialEq)]
struct SharedByteBuffer(Rc<[u8]>);
impl SharedByteBuffer {
fn new(data: Vec<u8>) -> SharedByteBuffer {
SharedByteBuffer(data.into())
}
}
impl AsRef<[u8]> for SharedByteBuffer {
fn as_ref(&self) -> &[u8] {
&self.0
}
}
impl InputFeatures for Cursor<SharedByteBuffer> {
fn get_size(&mut self) -> Result<usize> {
Ok(self.get_ref().0.len())
}
fn get_unix_mtime(&mut self) -> Result<Option<i64>> {
Ok(None)
}
fn try_seek(&mut self, pos: SeekFrom) -> Result<u64> {
Ok(self.seek(pos)?)
}
}
#[derive(Clone, Debug, Eq, PartialEq)]
pub struct BufferedPrimaryIo {
buffer: SharedByteBuffer,
}
impl BufferedPrimaryIo {
pub fn from_stream<T: Read>(stream: &mut T) -> Result<Self> {
let mut buf = [0u8; 8192];
let mut alldata = Vec::<u8>::new();
loop {
let nbytes = stream.read(&mut buf)?;
if nbytes == 0 {
break;
}
alldata.extend_from_slice(&buf[..nbytes]);
}
Ok(BufferedPrimaryIo {
buffer: SharedByteBuffer::new(alldata),
})
}
pub fn from_stdin() -> Result<Self> {
Self::from_stream(&mut stdin())
}
pub fn from_text<T: AsRef<str>>(text: T) -> Self {
BufferedPrimaryIo {
buffer: SharedByteBuffer::new(text.as_ref().as_bytes().to_owned()),
}
}
pub fn from_buffer(buf: Vec<u8>) -> Self {
BufferedPrimaryIo {
buffer: SharedByteBuffer::new(buf),
}
}
}
impl IoProvider for BufferedPrimaryIo {
fn input_open_primary(&mut self, _status: &mut dyn StatusBackend) -> OpenResult<InputHandle> {
OpenResult::Ok(InputHandle::new(
OsStr::new(""),
Cursor::new(self.buffer.clone()),
InputOrigin::Other,
))
}
}