2022-08-04 12:07:57 +00:00
|
|
|
/// https://github.com/ChainAgnostic/CAIPs/blob/master/CAIPs/caip-2.md
|
|
|
|
use std::str::FromStr;
|
|
|
|
|
|
|
|
use regex::Regex;
|
2022-08-29 19:17:16 +00:00
|
|
|
use serde::{
|
|
|
|
Deserialize,
|
|
|
|
Deserializer,
|
|
|
|
Serialize,
|
|
|
|
Serializer,
|
|
|
|
de::Error as DeserializerError,
|
|
|
|
};
|
2022-08-04 12:07:57 +00:00
|
|
|
|
|
|
|
const CAIP2_RE: &str = r"(?P<namespace>[-a-z0-9]{3,8}):(?P<reference>[-a-zA-Z0-9]{1,32})";
|
2022-08-29 14:33:00 +00:00
|
|
|
const CAIP2_ETHEREUM_NAMESPACE: &str = "eip155";
|
|
|
|
const ETHEREUM_MAINNET_ID: i32 = 1;
|
2022-08-04 12:07:57 +00:00
|
|
|
|
|
|
|
#[derive(Clone, Debug, PartialEq)]
|
|
|
|
pub struct ChainId {
|
|
|
|
pub namespace: String,
|
|
|
|
pub reference: String,
|
|
|
|
}
|
|
|
|
|
2022-08-29 14:33:00 +00:00
|
|
|
impl ChainId {
|
|
|
|
pub fn ethereum_mainnet() -> Self {
|
|
|
|
Self {
|
|
|
|
namespace: CAIP2_ETHEREUM_NAMESPACE.to_string(),
|
|
|
|
reference: ETHEREUM_MAINNET_ID.to_string(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-04 12:07:57 +00:00
|
|
|
#[derive(thiserror::Error, Debug)]
|
|
|
|
#[error("Chain ID parse error")]
|
|
|
|
pub struct ChainIdError;
|
|
|
|
|
|
|
|
impl FromStr for ChainId {
|
|
|
|
type Err = ChainIdError;
|
|
|
|
|
|
|
|
fn from_str(value: &str) -> Result<Self, Self::Err> {
|
|
|
|
let caip2_re = Regex::new(CAIP2_RE).unwrap();
|
|
|
|
let caps = caip2_re.captures(value).ok_or(ChainIdError)?;
|
|
|
|
let chain_id = Self {
|
|
|
|
namespace: caps["namespace"].to_string(),
|
|
|
|
reference: caps["reference"].to_string(),
|
|
|
|
};
|
|
|
|
Ok(chain_id)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-10 11:14:17 +00:00
|
|
|
impl ToString for ChainId {
|
|
|
|
fn to_string(&self) -> String {
|
|
|
|
format!("{}:{}", self.namespace, self.reference)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-29 19:17:16 +00:00
|
|
|
impl Serialize for ChainId {
|
|
|
|
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
|
|
|
where S: Serializer
|
|
|
|
{
|
|
|
|
serializer.serialize_str(&self.to_string())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-10 11:14:17 +00:00
|
|
|
impl<'de> Deserialize<'de> for ChainId {
|
|
|
|
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
|
|
|
|
where D: Deserializer<'de>
|
|
|
|
{
|
|
|
|
String::deserialize(deserializer)?
|
|
|
|
.parse().map_err(DeserializerError::custom)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-04 12:07:57 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_parse_bitcoin_chain_id() {
|
|
|
|
let value = "bip122:000000000019d6689c085ae165831e93";
|
|
|
|
let chain_id = value.parse::<ChainId>().unwrap();
|
|
|
|
assert_eq!(chain_id.namespace, "bip122");
|
|
|
|
assert_eq!(chain_id.reference, "000000000019d6689c085ae165831e93");
|
2022-08-10 11:14:17 +00:00
|
|
|
assert_eq!(chain_id.to_string(), value);
|
2022-08-04 12:07:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_parse_ethereum_chain_id() {
|
|
|
|
let value = "eip155:1";
|
|
|
|
let chain_id = value.parse::<ChainId>().unwrap();
|
|
|
|
assert_eq!(chain_id.namespace, "eip155");
|
|
|
|
assert_eq!(chain_id.reference, "1");
|
2022-08-10 11:14:17 +00:00
|
|
|
assert_eq!(chain_id.to_string(), value);
|
2022-08-04 12:07:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_parse_invalid_chain_id() {
|
|
|
|
let value = "eip155/1/abcde";
|
|
|
|
assert!(value.parse::<ChainId>().is_err());
|
|
|
|
}
|
|
|
|
}
|