#![cfg_attr(feature = "strict", deny(warnings))]
#![deny(missing_docs, missing_debug_implementations, missing_copy_implementations, trivial_casts,
trivial_numeric_casts, unsafe_code, unstable_features, unused_import_braces,
unused_qualifications)]
extern crate base58;
extern crate base64;
extern crate crypto;
extern crate duniter_crypto;
#[macro_use]
extern crate lazy_static;
extern crate linked_hash_map;
extern crate regex;
use std::fmt::{Debug, Display, Error, Formatter};
use duniter_crypto::keys::BaseConvertionError;
pub mod blockchain;
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
pub struct BlockId(pub u32);
impl Display for BlockId {
fn fmt(&self, f: &mut Formatter) -> Result<(), Error> {
write!(f, "{}", self.0)
}
}
#[derive(Copy, Clone, PartialEq, Eq, Hash)]
pub struct Hash(pub [u8; 32]);
impl Display for Hash {
fn fmt(&self, f: &mut Formatter) -> Result<(), Error> {
write!(f, "{}", self.to_hex())
}
}
impl Debug for Hash {
fn fmt(&self, f: &mut Formatter) -> Result<(), Error> {
write!(f, "Hash({})", self)
}
}
impl Default for Hash {
fn default() -> Hash {
let default: [u8; 32] = [0; 32];
Hash(default)
}
}
impl Hash {
pub fn to_hex(&self) -> String {
let strings: Vec<String> = self.0.iter().map(|b| format!("{:02X}", b)).collect();
strings.join("")
}
pub fn from_hex(text: &str) -> Result<Hash, BaseConvertionError> {
if text.len() != 64 {
Err(BaseConvertionError::InvalidKeyLendth(text.len(), 64))
} else {
let mut hash = Hash([0u8; 32]);
let chars: Vec<char> = text.chars().collect();
for i in 0..64 {
if i % 2 != 0 {
continue;
}
let byte1 = chars[i].to_digit(16);
let byte2 = chars[i + 1].to_digit(16);
if byte1.is_none() {
return Err(BaseConvertionError::InvalidCharacter(chars[i], i));
} else if byte2.is_none() {
return Err(BaseConvertionError::InvalidCharacter(chars[i + 1], i + 1));
}
let byte1 = byte1.unwrap() as u8;
let byte2 = byte2.unwrap() as u8;
let byte = (byte1 << 4) | byte2;
hash.0[i / 2] = byte;
}
Ok(hash)
}
}
}
#[derive(Copy, Clone, PartialEq, Eq, Hash)]
pub struct BlockHash(Hash);
impl Display for BlockHash {
fn fmt(&self, f: &mut Formatter) -> Result<(), Error> {
write!(f, "{}", self.0.to_hex())
}
}
impl Debug for BlockHash {
fn fmt(&self, f: &mut Formatter) -> Result<(), Error> {
write!(f, "BlockHash({})", self)
}
}
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
pub enum BlockUIdParseError {
InvalidFormat(),
InvalidBlockId(),
InvalidBlockHash(),
}
#[derive(Copy, Clone, PartialEq, Eq, Hash)]
pub struct Blockstamp {
pub id: BlockId,
pub hash: BlockHash,
}
impl Display for Blockstamp {
fn fmt(&self, f: &mut Formatter) -> Result<(), Error> {
write!(f, "{}-{}", self.id, self.hash)
}
}
impl Debug for Blockstamp {
fn fmt(&self, f: &mut Formatter) -> Result<(), Error> {
write!(f, "BlockUId({})", self)
}
}
impl Default for Blockstamp {
fn default() -> Blockstamp {
Blockstamp {
id: BlockId(0),
hash: BlockHash(Hash::default()),
}
}
}
impl Blockstamp {
pub fn from_string(src: &str) -> Result<Blockstamp, BlockUIdParseError> {
let mut split = src.split('-');
if split.clone().count() != 2 {
Err(BlockUIdParseError::InvalidFormat())
} else {
let id = split.next().unwrap().parse::<u32>();
let hash = Hash::from_hex(split.next().unwrap());
if id.is_err() {
Err(BlockUIdParseError::InvalidBlockId())
} else if hash.is_err() {
Err(BlockUIdParseError::InvalidBlockHash())
} else {
Ok(Blockstamp {
id: BlockId(id.unwrap()),
hash: BlockHash(hash.unwrap()),
})
}
}
}
pub fn to_string(&self) -> String {
format!("{}", self)
}
}