#[repr(u8)]
#[non_exhaustive]
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum ParseErrorKind {
BlockSizeIsEmpty,
BlockSizeIsInvalid,
BlockSizeIsTooLarge,
BlockHashIsTooLong,
UnexpectedCharacter,
UnexpectedEndOfString,
}
impl core::fmt::Display for ParseErrorKind {
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
f.write_str(match self {
ParseErrorKind::BlockHashIsTooLong => "block hash is too long.",
ParseErrorKind::BlockSizeIsEmpty => "block size field is empty.",
ParseErrorKind::BlockSizeIsInvalid => "block size is not valid.",
ParseErrorKind::BlockSizeIsTooLarge => "block size is too large.",
ParseErrorKind::UnexpectedCharacter => "an unexpected character is encountered.",
ParseErrorKind::UnexpectedEndOfString => "end-of-string is not expected.",
})
}
}
#[repr(u8)]
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum ParseErrorOrigin {
BlockSize,
BlockHash1,
BlockHash2,
}
impl core::fmt::Display for ParseErrorOrigin {
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
f.write_str(match self {
ParseErrorOrigin::BlockSize => "block size",
ParseErrorOrigin::BlockHash1 => "block hash 1",
ParseErrorOrigin::BlockHash2 => "block hash 2",
})
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub struct ParseError(
pub(crate) ParseErrorKind,
pub(crate) ParseErrorOrigin,
pub(crate) usize
);
pub trait ParseErrorInfo {
fn kind(&self) -> ParseErrorKind;
fn origin(&self) -> ParseErrorOrigin;
fn offset(&self) -> usize;
}
impl ParseErrorInfo for ParseError {
fn kind(&self) -> ParseErrorKind { self.0 }
fn origin(&self) -> ParseErrorOrigin { self.1 }
fn offset(&self) -> usize { self.2 }
}
impl core::fmt::Display for ParseError {
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
write!(f, "Error occurred while parsing a fuzzy hash ({1}, at byte offset {2}): {0}",
self.kind(),
self.origin(),
self.offset()
)
}
}
#[cfg(feature = "std")]
impl std::error::Error for ParseError {}
#[repr(u32)]
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub(crate) enum BlockHashParseState {
MetEndOfString,
MetComma,
MetColon,
OverflowError,
Base64Error,
}
#[cfg(test)]
mod tests {
use super::*;
#[cfg(feature = "alloc")]
use alloc::format;
#[cfg(feature = "alloc")]
use crate::test_utils::test_auto_debug_for_enum;
use crate::test_utils::test_auto_clone;
#[test]
fn test_parse_error_enums_clone() {
test_auto_clone::<ParseErrorKind>(&ParseErrorKind::BlockHashIsTooLong);
test_auto_clone::<ParseErrorOrigin>(&ParseErrorOrigin::BlockSize);
}
#[cfg(feature = "alloc")]
#[test]
fn test_parse_error_enums_display_and_debug() {
assert_eq!(format!("{}", ParseErrorKind::BlockHashIsTooLong), "block hash is too long.");
assert_eq!(format!("{}", ParseErrorKind::BlockSizeIsEmpty), "block size field is empty.");
assert_eq!(format!("{}", ParseErrorKind::BlockSizeIsInvalid), "block size is not valid.");
assert_eq!(format!("{}", ParseErrorKind::BlockSizeIsTooLarge), "block size is too large.");
assert_eq!(format!("{}", ParseErrorKind::UnexpectedCharacter), "an unexpected character is encountered.");
assert_eq!(format!("{}", ParseErrorKind::UnexpectedEndOfString), "end-of-string is not expected.");
assert_eq!(format!("{}", ParseErrorOrigin::BlockSize), "block size");
assert_eq!(format!("{}", ParseErrorOrigin::BlockHash1), "block hash 1");
assert_eq!(format!("{}", ParseErrorOrigin::BlockHash2), "block hash 2");
test_auto_debug_for_enum!(
ParseErrorKind,
[
BlockSizeIsEmpty,
BlockSizeIsInvalid,
BlockSizeIsTooLarge,
BlockHashIsTooLong,
UnexpectedCharacter,
UnexpectedEndOfString,
]
);
test_auto_debug_for_enum!(
ParseErrorOrigin,
[
BlockSize,
BlockHash1,
BlockHash2,
]
);
}
#[test]
fn test_block_hash_parse_state_clone() {
test_auto_clone::<BlockHashParseState>(&BlockHashParseState::MetEndOfString);
}
#[cfg(feature = "alloc")]
#[test]
fn test_block_hash_parse_state_debug() {
test_auto_debug_for_enum!(
BlockHashParseState,
[
MetEndOfString,
MetComma,
MetColon,
OverflowError,
Base64Error,
]
);
}
#[test]
fn test_parse_error_basic() {
let err = ParseError(ParseErrorKind::UnexpectedEndOfString, ParseErrorOrigin::BlockHash1, 2);
test_auto_clone::<ParseError>(&err);
assert_eq!(err.kind(), ParseErrorKind::UnexpectedEndOfString);
assert_eq!(err.origin(), ParseErrorOrigin::BlockHash1);
assert_eq!(err.offset(), 2);
}
#[cfg(feature = "alloc")]
#[test]
fn test_parse_error_display_and_debug() {
let err_empty = ParseError(ParseErrorKind::UnexpectedEndOfString, ParseErrorOrigin::BlockSize, 0);
let err_eos_bh1 = ParseError(ParseErrorKind::UnexpectedEndOfString, ParseErrorOrigin::BlockHash1, 2);
let err_bs_invalid = ParseError(ParseErrorKind::BlockSizeIsInvalid, ParseErrorOrigin::BlockSize, 0);
assert_eq!(format!("{}", err_empty),
"Error occurred while parsing a fuzzy hash (block size, at byte offset 0): end-of-string is not expected.");
assert_eq!(format!("{}", err_eos_bh1),
"Error occurred while parsing a fuzzy hash (block hash 1, at byte offset 2): end-of-string is not expected.");
assert_eq!(format!("{}", err_bs_invalid),
"Error occurred while parsing a fuzzy hash (block size, at byte offset 0): block size is not valid.");
assert_eq!(format!("{:?}", err_empty), "ParseError(UnexpectedEndOfString, BlockSize, 0)");
assert_eq!(format!("{:?}", err_eos_bh1), "ParseError(UnexpectedEndOfString, BlockHash1, 2)");
assert_eq!(format!("{:?}", err_bs_invalid), "ParseError(BlockSizeIsInvalid, BlockSize, 0)");
}
}