1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
use std::io::{Cursor, Read, Result as IoResult, Seek, SeekFrom};

#[cfg(feature="partial_state")]
use rand;

use crate::error::{parse_io, TychoResult, TychoStatus};
use crate::partial::element::{PartialElement, read_partial_element};

/// A reader with an inner pointer and state management for reading tycho partially.
///
/// PartialReader can take two types of reader:
/// - sync (`std::io::Read` + `std::io::Seek`)
/// - async (`tokio::io:AsyncRead` + `tokio::io::AsyncSeek`)
///
/// When using async, functions are suffixed with a `_async`.
/// For example `element()` would become `element_async().await`
///
/// ### Creating
/// You can create a partial reader by giving any sized type.
/// ```
/// use std::io::Cursor;
/// use tycho::partial::PartialReader;
/// let cursor = Cursor::new(vec![ 0 ]);
///
/// let mut reader = PartialReader::from(cursor);
/// ```
///
/// You can also give a vec of bytes:
///
/// ```
/// use tycho::partial::PartialReader;
/// let mut reader = PartialReader::from_vec(vec![ 0 ]);
/// ```
///
/// ### Reading
///
/// You can get the inital/root element of the reader by calling `.element()`
/// ```
/// use tycho::partial::PartialReader;
/// let mut reader = PartialReader::from_vec(vec![ 0 ]);
///
/// let element = reader.element().unwrap();
/// ```
pub struct PartialReader<R> {
    pub(crate) reader: R,
    pub(crate) pointer: u64,

    #[cfg(feature="partial_state")]
    pub(crate) ident: u16,
}

impl PartialReader<Cursor<Vec<u8>>> {
    pub fn from_vec(reader: Vec<u8>) -> Self {
        PartialReader {
            reader: Cursor::new(reader),
            pointer: 0,

            #[cfg(feature="partial_state")]
            ident: rand::random()
        }
    }
}

// sync implementation
impl<R: Read + Seek> PartialReader<R> {

    /// Jump to a pointer location.
    pub fn jump(&mut self, to: &u64) -> TychoStatus {
        parse_io(self.reader.seek(SeekFrom::Current((*to as i64) - (self.pointer as i64))))?;
        self.pointer = *to;
        Ok(())
    }

    /// Get the next element of the reader.
    pub fn element(&mut self) -> TychoResult<PartialElement> {
        read_partial_element(self)

    }

}

/*
impl<R: Read + Seek + Write> PartialReader<R> {
    pub fn snip(&mut self, pointer: PartialPointer) -> TychoResult<Vec<u8>> {
        #[cfg(feature="partial_state")]
        if self.pointer.ident != reader.ident {
            return Err(TychoError::OutdatedPointer)
        }

        self.jump(&pointer.pos)?;

        self.reader.rem2
    }
}*/

impl<R> PartialReader<R> {
    pub fn from(reader: R) -> PartialReader<R> {
        PartialReader {
            reader,
            pointer: 0,

            #[cfg(feature = "partial_state")]
            ident: rand::random()
        }
    }

    pub(crate) fn pointer(&self, pos: u64, size: u64) -> PartialPointer {
        PartialPointer {
            pos,
            size,
            #[cfg(feature="partial_state")]
            ident: self.ident.clone()
        }
    }

    pub(crate) fn empty_pointer(&self) -> PartialPointer {
        PartialPointer {
            pos: 0,
            size: 0,
            #[cfg(feature="partial_state")]
            ident: self.ident.clone()
        }
    }
}

impl<R: Read + Seek> Read for PartialReader<R> {
    fn read(&mut self, buf: &mut [u8]) -> IoResult<usize> {
        let size = self.reader.read(buf)?;
        self.pointer += size.clone() as u64;
        Ok(size)
    }
}


#[derive(Debug, Clone)]
/// A pointer, referring to a block of data in a partial reader.
pub struct PartialPointer {
    pub(crate) pos: u64,
    pub(crate) size: u64,

    #[cfg(feature="partial_state")]
    pub(crate) ident: u16,
}