feeless 0.1.11

A Nano (cryptocurrency) node and utilities such as nano addresses, hashing blocks, signing, etc.
Documentation
use anyhow::anyhow;

use std::fmt::Debug;

/// A wrapper around a u8 slice which incrementally slices the data.
#[derive(Debug)]
pub struct Bytes<'a> {
    bytes: &'a [u8],
    offset: usize,
}

impl<'a> Bytes<'a> {
    pub fn new(bytes: &'a [u8]) -> Self {
        Self { bytes, offset: 0 }
    }

    pub fn eof(&self) -> bool {
        self.offset >= self.bytes.len()
    }

    pub fn offset(&self) -> usize {
        self.offset
    }

    pub fn remain(&self) -> usize {
        self.len() - self.offset()
    }

    pub fn seek(&mut self, amount: i64) -> anyhow::Result<()> {
        self.bounds_check(amount)?;
        self.offset = (self.offset as i64 + amount) as usize;
        Ok(())
    }

    pub fn len(&self) -> usize {
        self.bytes.len()
    }

    pub fn slice(&mut self, size: usize) -> anyhow::Result<&[u8]> {
        // TODO: make this safer--maybe use replace usize with u32 so it's always smaller than i64.
        self.bounds_check(size as i64)?;
        let bytes = &self.bytes[self.offset..self.offset + size];
        self.offset += size;
        Ok(bytes)
    }

    pub fn u8(&mut self) -> anyhow::Result<u8> {
        self.bounds_check(1)?;
        let b = self.bytes[self.offset];
        self.offset += 1;
        Ok(b)
    }

    fn bounds_check(&mut self, size: i64) -> anyhow::Result<()> {
        if (self.offset as i64 + size) as usize > self.bytes.len() {
            Err(anyhow!(
                "Slice extended past end. Offset: {} Requested size: {} Bytes len: {} Remain: {}",
                self.offset,
                size,
                self.bytes.len(),
                self.remain(),
            ))
        } else {
            Ok(())
        }
    }
}