tendermint/block/
height.rsuse core::{
fmt::{self, Debug, Display},
str::FromStr,
};
use serde::{de::Error as _, Deserialize, Deserializer, Serialize, Serializer};
use tendermint_proto::Protobuf;
use crate::{error::Error, prelude::*};
#[derive(Copy, Clone, Eq, Hash, PartialEq, PartialOrd, Ord)]
pub struct Height(u64);
impl Protobuf<i64> for Height {}
impl TryFrom<i64> for Height {
type Error = Error;
fn try_from(value: i64) -> Result<Self, Self::Error> {
Ok(Height(value.try_into().map_err(Error::negative_height)?))
}
}
impl From<Height> for i64 {
fn from(value: Height) -> Self {
value.value() as i64 }
}
impl TryFrom<u64> for Height {
type Error = Error;
fn try_from(value: u64) -> Result<Self, Self::Error> {
let _ival: i64 = value.try_into().map_err(Error::integer_overflow)?;
Ok(Height(value))
}
}
impl From<Height> for u64 {
fn from(value: Height) -> Self {
value.value()
}
}
impl From<u32> for Height {
fn from(value: u32) -> Self {
Height(value as u64)
}
}
impl From<u16> for Height {
fn from(value: u16) -> Self {
Height(value as u64)
}
}
impl From<u8> for Height {
fn from(value: u8) -> Self {
Height(value as u64)
}
}
impl Height {
pub fn value(&self) -> u64 {
self.0
}
pub fn increment(self) -> Self {
Height::try_from(self.0.checked_add(1).expect("height overflow")).unwrap()
}
}
impl Debug for Height {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "block::Height({})", self.0)
}
}
impl Default for Height {
fn default() -> Self {
Height(1)
}
}
impl Display for Height {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.0)
}
}
impl FromStr for Height {
type Err = Error;
fn from_str(s: &str) -> Result<Self, Error> {
Height::try_from(
s.parse::<u64>()
.map_err(|e| Error::parse_int(s.to_string(), e))?,
)
}
}
impl<'de> Deserialize<'de> for Height {
fn deserialize<D: Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
Self::from_str(&String::deserialize(deserializer)?)
.map_err(|e| D::Error::custom(format!("{e}")))
}
}
impl Serialize for Height {
fn serialize<S: Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
i64::from(*self).to_string().serialize(serializer)
}
}
pub trait ParseHeight {
fn parse_block_height(&self) -> Result<Height, Error>;
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn increment_by_one() {
assert_eq!(Height::default().increment().value(), 2);
}
#[test]
fn avoid_try_unwrap_dance() {
assert_eq!(
Height::try_from(2_u64).unwrap().value(),
Height::from(2_u32).value()
);
}
}