mod errors;
mod network;
mod payload;
mod protocol;
use std::borrow::Cow;
use std::fmt;
use std::hash::Hash;
use std::str::FromStr;
use data_encoding::Encoding;
use data_encoding_macro::new_encoding;
use fvm_ipld_encoding::{Cbor, serde_bytes};
use serde::{Deserialize, Deserializer, Serialize, Serializer, de};
pub use self::errors::Error;
pub use self::network::Network;
pub use self::payload::Payload;
pub use self::protocol::Protocol;
use crate::ActorID;
const ADDRESS_ENCODER: Encoding = new_encoding! {
symbols: "abcdefghijklmnopqrstuvwxyz234567",
padding: None,
};
pub const PAYLOAD_HASH_LEN: usize = 20;
pub const SECP_PUB_LEN: usize = 65;
pub const BLS_PUB_LEN: usize = 48;
pub const FIRST_NON_SINGLETON_ADDR: ActorID = 100;
lazy_static::lazy_static! {
static ref BLS_ZERO_ADDR_BYTES: [u8; BLS_PUB_LEN] = {
let bz_addr = Address::from_str("f3yaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaby2smx7a");
if let Ok(Address {payload: Payload::BLS(pubkey), ..}) = bz_addr {
pubkey
} else {
panic!("failed to parse BLS address from provided BLS_ZERO_ADDR string")
}
};
}
pub const CHECKSUM_HASH_LEN: usize = 4;
const MAX_ADDRESS_LEN: usize = 84 + 2;
const MAINNET_PREFIX: &str = "f";
const TESTNET_PREFIX: &str = "t";
pub const NETWORK_DEFAULT: Network = Network::Mainnet;
#[derive(PartialEq, Eq, Clone, Debug, Hash, Copy)]
#[cfg_attr(feature = "testing", derive(Default))]
#[cfg_attr(feature = "arb", derive(arbitrary::Arbitrary))]
pub struct Address {
network: Network,
payload: Payload,
}
impl Cbor for Address {}
impl Address {
fn new(network: Network, protocol: Protocol, bz: &[u8]) -> Result<Self, Error> {
Ok(Self {
network,
payload: Payload::new(protocol, bz)?,
})
}
pub fn from_bytes(bz: &[u8]) -> Result<Self, Error> {
if bz.len() < 2 {
Err(Error::InvalidLength)
} else {
let protocol = Protocol::from_byte(bz[0]).ok_or(Error::UnknownProtocol)?;
Self::new(NETWORK_DEFAULT, protocol, &bz[1..])
}
}
pub const fn new_id(id: u64) -> Self {
Self {
network: NETWORK_DEFAULT,
payload: Payload::ID(id),
}
}
pub fn new_secp256k1(pubkey: &[u8]) -> Result<Self, Error> {
if pubkey.len() != 65 {
return Err(Error::InvalidSECPLength(pubkey.len()));
}
Ok(Self {
network: NETWORK_DEFAULT,
payload: Payload::Secp256k1(address_hash(pubkey)),
})
}
pub fn new_actor(data: &[u8]) -> Self {
Self {
network: NETWORK_DEFAULT,
payload: Payload::Actor(address_hash(data)),
}
}
pub fn new_bls(pubkey: &[u8]) -> Result<Self, Error> {
if pubkey.len() != BLS_PUB_LEN {
return Err(Error::InvalidBLSLength(pubkey.len()));
}
let mut key = [0u8; BLS_PUB_LEN];
key.copy_from_slice(pubkey);
Ok(Self {
network: NETWORK_DEFAULT,
payload: Payload::BLS(key),
})
}
pub fn is_bls_zero_address(&self) -> bool {
match self.payload {
Payload::BLS(payload_bytes) => payload_bytes == *BLS_ZERO_ADDR_BYTES,
_ => false,
}
}
pub fn protocol(&self) -> Protocol {
Protocol::from(self.payload)
}
pub fn payload(&self) -> &Payload {
&self.payload
}
pub fn into_payload(self) -> Payload {
self.payload
}
pub fn payload_bytes(&self) -> Vec<u8> {
self.payload.to_raw_bytes()
}
pub fn network(&self) -> Network {
self.network
}
pub fn set_network(&mut self, network: Network) -> &mut Self {
self.network = network;
self
}
pub fn to_bytes(self) -> Vec<u8> {
self.payload.to_bytes()
}
pub fn id(&self) -> Result<u64, Error> {
match self.payload {
Payload::ID(id) => Ok(id),
_ => Err(Error::NonIDAddress),
}
}
}
impl fmt::Display for Address {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", encode(self))
}
}
impl FromStr for Address {
type Err = Error;
fn from_str(addr: &str) -> Result<Self, Error> {
if addr.len() > MAX_ADDRESS_LEN || addr.len() < 3 {
return Err(Error::InvalidLength);
}
let network: Network = match addr.get(0..1).ok_or(Error::UnknownNetwork)? {
TESTNET_PREFIX => Network::Testnet,
MAINNET_PREFIX => Network::Mainnet,
_ => {
return Err(Error::UnknownNetwork);
}
};
let protocol: Protocol = match addr.get(1..2).ok_or(Error::UnknownProtocol)? {
"0" => Protocol::ID,
"1" => Protocol::Secp256k1,
"2" => Protocol::Actor,
"3" => Protocol::BLS,
_ => {
return Err(Error::UnknownProtocol);
}
};
let raw = addr.get(2..).ok_or(Error::InvalidPayload)?;
if protocol == Protocol::ID {
if raw.len() > 20 {
return Err(Error::InvalidLength);
}
let id = raw.parse::<u64>()?;
return Ok(Address {
network,
payload: Payload::ID(id),
});
}
let mut payload = ADDRESS_ENCODER.decode(raw.as_bytes())?;
let cksm = payload.split_off(payload.len() - CHECKSUM_HASH_LEN);
if (protocol == Protocol::Secp256k1 || protocol == Protocol::Actor)
&& payload.len() != PAYLOAD_HASH_LEN
{
return Err(Error::InvalidPayload);
}
if protocol == Protocol::BLS && payload.len() != BLS_PUB_LEN {
return Err(Error::InvalidPayload);
}
let mut ingest = payload.clone();
ingest.insert(0, protocol as u8);
if !validate_checksum(&ingest, cksm) {
return Err(Error::InvalidChecksum);
}
Address::new(network, protocol, &payload)
}
}
impl Serialize for Address {
fn serialize<S>(&self, s: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
let address_bytes = self.to_bytes();
serde_bytes::Serialize::serialize(&address_bytes, s)
}
}
impl<'de> Deserialize<'de> for Address {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: Deserializer<'de>,
{
let bz: Cow<'de, [u8]> = serde_bytes::Deserialize::deserialize(deserializer)?;
Address::from_bytes(&bz).map_err(de::Error::custom)
}
}
fn encode(addr: &Address) -> String {
match addr.protocol() {
Protocol::Secp256k1 | Protocol::Actor | Protocol::BLS => {
let ingest = addr.to_bytes();
let mut bz = addr.payload_bytes();
bz.extend(checksum(&ingest));
format!(
"{}{}{}",
addr.network.to_prefix(),
addr.protocol(),
ADDRESS_ENCODER.encode(bz.as_mut()),
)
}
Protocol::ID => format!(
"{}{}{}",
addr.network.to_prefix(),
addr.protocol(),
from_leb_bytes(&addr.payload_bytes()).expect("should read encoded bytes"),
),
}
}
pub(crate) fn to_leb_bytes(id: u64) -> Result<Vec<u8>, Error> {
Ok(unsigned_varint::encode::u64(id, &mut unsigned_varint::encode::u64_buffer()).into())
}
pub(crate) fn from_leb_bytes(bz: &[u8]) -> Result<u64, Error> {
let (id, remaining) = unsigned_varint::decode::u64(bz)?;
if !remaining.is_empty() {
return Err(Error::InvalidPayload);
}
Ok(id)
}
pub fn checksum(ingest: &[u8]) -> Vec<u8> {
blake2b_simd::Params::new()
.hash_length(CHECKSUM_HASH_LEN)
.to_state()
.update(ingest)
.finalize()
.as_bytes()
.to_vec()
}
pub fn validate_checksum(ingest: &[u8], expect: Vec<u8>) -> bool {
let digest = checksum(ingest);
digest == expect
}
fn address_hash(ingest: &[u8]) -> [u8; 20] {
let digest = blake2b_simd::Params::new()
.hash_length(PAYLOAD_HASH_LEN)
.to_state()
.update(ingest)
.finalize();
let mut hash = [0u8; 20];
hash.copy_from_slice(digest.as_bytes());
hash
}
#[cfg(test)]
mod tests {
use crate::address::errors::Error;
use crate::address::{from_leb_bytes, to_leb_bytes};
#[test]
fn test_from_leb_bytes_passing() {
let passing = vec![67];
assert_eq!(
to_leb_bytes(from_leb_bytes(&passing).unwrap()),
Ok(vec![67])
);
}
#[test]
fn test_from_leb_bytes_extra_bytes() {
let extra_bytes = vec![67, 0, 1, 2];
match from_leb_bytes(&extra_bytes) {
Ok(id) => {
println!(
"Successfully decoded bytes when it was not supposed to. Result was: {:?}",
&to_leb_bytes(id).unwrap()
);
panic!();
}
Err(e) => {
assert_eq!(e, Error::InvalidPayload);
}
}
}
#[test]
fn test_from_leb_bytes_minimal_encoding() {
let minimal_encoding = vec![67, 0, 130, 0];
match from_leb_bytes(&minimal_encoding) {
Ok(id) => {
println!(
"Successfully decoded bytes when it was not supposed to. Result was: {:?}",
&to_leb_bytes(id).unwrap()
);
panic!();
}
Err(e) => {
assert_eq!(e, Error::InvalidPayload);
}
}
}
}