From 08da7d8af11d4354eec4c7f303f80990c14f6e02 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Adam=20Chuda=C5=9B?= Date: Thu, 2 Nov 2023 10:24:44 +0100 Subject: [PATCH 1/3] Add AccountType, ETH-implicit accounts --- src/account_id_ref.rs | 127 ++++++++++++++++++++++++++++++++++++++---- src/lib.rs | 2 +- 2 files changed, 117 insertions(+), 12 deletions(-) diff --git a/src/account_id_ref.rs b/src/account_id_ref.rs index be440d9..05eaeed 100644 --- a/src/account_id_ref.rs +++ b/src/account_id_ref.rs @@ -29,6 +29,28 @@ use crate::{AccountId, ParseAccountError}; #[cfg_attr(feature = "abi", derive(schemars::JsonSchema, BorshSchema))] pub struct AccountIdRef(pub(crate) str); +/// Enum representing possible types of accounts. +#[derive(PartialEq)] +pub enum AccountType { + /// Any valid account, that is neither NEAR-implicit nor ETH-implicit. + NamedAccount, + /// An account with 64 characters long hexadecimal address. + NearImplicitAccount, + /// An account which address starts with '0x', followed by 40 hex characters. + EthImplicitAccount, +} + +impl AccountType { + pub fn is_implicit(&self) -> bool { + match &self { + Self::NearImplicitAccount => true, + // TODO(eth-implicit) change to true later, see https://github.com/near/nearcore/issues/10018 + Self::EthImplicitAccount => false, + Self::NamedAccount => false, + } + } +} + impl AccountIdRef { /// Shortest valid length for a NEAR Account ID. pub const MIN_LEN: usize = crate::validation::MIN_LEN; @@ -140,6 +162,29 @@ impl AccountIdRef { .map_or(false, |s| !s.contains('.')) } + /// Returns `true` if the `AccountId` is a 40 characters long hexadecimal prefixed with '0x'. + /// + /// See [Implicit-Accounts](https://docs.near.org/docs/concepts/account#implicit-accounts). + /// + /// ## Examples + /// + /// ``` + /// use near_account_id::AccountId; + /// + /// let alice: AccountId = "alice.near".parse().unwrap(); + /// assert!(!alice.is_eth_implicit()); + /// + /// let rando = "0xb794f5ea0ba39494ce839613fffba74279579268" + /// .parse::() + /// .unwrap(); + /// assert!(rando.is_eth_implicit()); + /// ``` + pub fn is_eth_implicit(&self) -> bool { + self.len() == 42 + && self.0.starts_with("0x") + && self.0[2..].as_bytes().iter().all(|b| matches!(b, b'a'..=b'f' | b'0'..=b'9')) + } + /// Returns `true` if the `AccountId` is a 64 characters long hexadecimal. /// /// See [Implicit-Accounts](https://docs.near.org/docs/concepts/account#implicit-accounts). @@ -150,14 +195,14 @@ impl AccountIdRef { /// use near_account_id::AccountId; /// /// let alice: AccountId = "alice.near".parse().unwrap(); - /// assert!(!alice.is_implicit()); + /// assert!(!alice.is_near_implicit()); /// /// let rando = "98793cd91a3f870fb126f66285808c7e094afcfc4eda8a970f6648cdf0dbd6de" /// .parse::() /// .unwrap(); - /// assert!(rando.is_implicit()); + /// assert!(rando.is_near_implicit()); /// ``` - pub fn is_implicit(&self) -> bool { + pub fn is_near_implicit(&self) -> bool { self.0.len() == 64 && self .as_bytes() @@ -165,6 +210,16 @@ impl AccountIdRef { .all(|b| matches!(b, b'a'..=b'f' | b'0'..=b'9')) } + pub fn get_account_type(&self) -> AccountType { + if self.is_eth_implicit() { + return AccountType::EthImplicitAccount; + } + if self.is_near_implicit() { + return AccountType::NearImplicitAccount; + } + AccountType::NamedAccount + } + /// Returns `true` if this `AccountId` is the system account. /// /// See [System account](https://nomicon.io/DataStructures/Account.html?highlight=system#system-account). @@ -457,6 +512,9 @@ mod tests { "alex-skidanov", "b-o_w_e-n", "no_lols", + // ETH-implicit account + "0xb794f5ea0ba39494ce839613fffba74279579268", + // NEAR-implicit account "0123456789012345678901234567890123456789012345678901234567890123", ]; for account_id in ok_top_level_account_ids { @@ -581,6 +639,11 @@ mod tests { "123456789012345678901234567890123456789012345678901234567890", "1234567890.123456789012345678901234567890123456789012345678901234567890", ), + ( + "b794f5ea0ba39494ce839613fffba74279579268", + // ETH-implicit account + "0xb794f5ea0ba39494ce839613fffba74279579268", + ), ("aa", "ъ@aa"), ("aa", "ъ.aa"), ]; @@ -598,26 +661,26 @@ mod tests { } #[test] - fn test_is_account_id_64_len_hex() { - let valid_64_len_hex_account_ids = &[ + fn test_is_account_id_near_implicit() { + let valid_near_implicit_account_ids = &[ "0000000000000000000000000000000000000000000000000000000000000000", "6174617461746174617461746174617461746174617461746174617461746174", "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef", "ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff", "20782e20662e64666420482123494b6b6c677573646b6c66676a646b6c736667", ]; - for valid_account_id in valid_64_len_hex_account_ids { + for valid_account_id in valid_near_implicit_account_ids { assert!( matches!( AccountIdRef::new(valid_account_id), - Ok(account_id) if account_id.is_implicit() + Ok(account_id) if account_id.get_account_type() == AccountType::NearImplicitAccount ), "Account ID {} should be valid 64-len hex", valid_account_id ); } - let invalid_64_len_hex_account_ids = &[ + let invalid_near_implicit_account_ids = &[ "000000000000000000000000000000000000000000000000000000000000000", "6.74617461746174617461746174617461746174617461746174617461746174", "012-456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef", @@ -625,13 +688,55 @@ mod tests { "oooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooo", "00000000000000000000000000000000000000000000000000000000000000", ]; - for invalid_account_id in invalid_64_len_hex_account_ids { + for invalid_account_id in invalid_near_implicit_account_ids { assert!( !matches!( AccountIdRef::new(invalid_account_id), - Ok(account_id) if account_id.is_implicit() + Ok(account_id) if account_id.get_account_type() == AccountType::NearImplicitAccount + ), + "Account ID {} is not a NEAR-implicit account", + invalid_account_id + ); + } + } + + #[test] + fn test_is_account_id_eth_implicit() { + let valid_eth_implicit_account_ids = &[ + "0x0000000000000000000000000000000000000000", + "0x6174617461746174617461746174617461746174", + "0x0123456789abcdef0123456789abcdef01234567", + "0xffffffffffffffffffffffffffffffffffffffff", + "0x20782e20662e64666420482123494b6b6c677573", + ]; + for valid_account_id in valid_eth_implicit_account_ids { + assert!( + matches!( + valid_account_id.parse::(), + Ok(account_id) if account_id.get_account_type() == AccountType::EthImplicitAccount + ), + "Account ID {} should be valid 42-len hex, starting with 0x", + valid_account_id + ); + } + + let invalid_eth_implicit_account_ids = &[ + "04b794f5ea0ba39494ce839613fffba74279579268", + "0x000000000000000000000000000000000000000", + "0x6.74617461746174617461746174617461746174", + "0x012-456789abcdef0123456789abcdef01234567", + "0xfffff_ffffffffffffffffffffffffffffffffff", + "0xoooooooooooooooooooooooooooooooooooooooo", + "0x00000000000000000000000000000000000000000", + "0000000000000000000000000000000000000000000000000000000000000000", + ]; + for invalid_account_id in invalid_eth_implicit_account_ids { + assert!( + !matches!( + invalid_account_id.parse::(), + Ok(account_id) if account_id.get_account_type() == AccountType::EthImplicitAccount ), - "Account ID {} is not an implicit account", + "Account ID {} is not an ETH-implicit account", invalid_account_id ); } diff --git a/src/lib.rs b/src/lib.rs index 9d8cb46..4db5a5b 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -48,5 +48,5 @@ mod test_data; mod validation; pub use account_id::AccountId; -pub use account_id_ref::AccountIdRef; +pub use account_id_ref::{AccountIdRef, AccountType}; pub use errors::{ParseAccountError, ParseErrorKind}; From d3b440ea958af51cc46d8becf9e7a18284a97d5f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Adam=20Chuda=C5=9B?= Date: Fri, 3 Nov 2023 14:45:35 +0100 Subject: [PATCH 2/3] Move is_near_implicit and is_eth_implicit to validation.rs --- src/account_id_ref.rs | 51 ++++++++++++++----------------------------- src/validation.rs | 14 ++++++++++++ 2 files changed, 30 insertions(+), 35 deletions(-) diff --git a/src/account_id_ref.rs b/src/account_id_ref.rs index 05eaeed..0d6fd7b 100644 --- a/src/account_id_ref.rs +++ b/src/account_id_ref.rs @@ -30,6 +30,11 @@ use crate::{AccountId, ParseAccountError}; pub struct AccountIdRef(pub(crate) str); /// Enum representing possible types of accounts. +/// This `enum` is returned by the [`get_account_type`] method on [`AccountIdRef`]. +/// See its documentation for more. +/// +/// [`get_account_type`]: AccountIdRef::get_account_type +/// [`AccountIdRef`]: struct.AccountIdRef.html #[derive(PartialEq)] pub enum AccountType { /// Any valid account, that is neither NEAR-implicit nor ETH-implicit. @@ -162,59 +167,35 @@ impl AccountIdRef { .map_or(false, |s| !s.contains('.')) } - /// Returns `true` if the `AccountId` is a 40 characters long hexadecimal prefixed with '0x'. + /// Returns `AccountType::EthImplicitAccount` if the `AccountId` is a 40 characters long hexadecimal prefixed with '0x'. + /// Returns `AccountType::NearImplicitAccount` if the `AccountId` is a 64 characters long hexadecimal. + /// Otherwise, returns `AccountType::NamedAccount`. /// /// See [Implicit-Accounts](https://docs.near.org/docs/concepts/account#implicit-accounts). /// /// ## Examples /// /// ``` - /// use near_account_id::AccountId; + /// use near_account_id::{AccountId, AccountType}; /// /// let alice: AccountId = "alice.near".parse().unwrap(); - /// assert!(!alice.is_eth_implicit()); + /// assert!(alice.get_account_type() == AccountType::NamedAccount); /// - /// let rando = "0xb794f5ea0ba39494ce839613fffba74279579268" + /// let eth_rando = "0xb794f5ea0ba39494ce839613fffba74279579268" /// .parse::() /// .unwrap(); - /// assert!(rando.is_eth_implicit()); - /// ``` - pub fn is_eth_implicit(&self) -> bool { - self.len() == 42 - && self.0.starts_with("0x") - && self.0[2..].as_bytes().iter().all(|b| matches!(b, b'a'..=b'f' | b'0'..=b'9')) - } - - /// Returns `true` if the `AccountId` is a 64 characters long hexadecimal. - /// - /// See [Implicit-Accounts](https://docs.near.org/docs/concepts/account#implicit-accounts). - /// - /// ## Examples - /// - /// ``` - /// use near_account_id::AccountId; + /// assert!(eth_rando.get_account_type() == AccountType::EthImplicitAccount); /// - /// let alice: AccountId = "alice.near".parse().unwrap(); - /// assert!(!alice.is_near_implicit()); - /// - /// let rando = "98793cd91a3f870fb126f66285808c7e094afcfc4eda8a970f6648cdf0dbd6de" + /// let near_rando = "98793cd91a3f870fb126f66285808c7e094afcfc4eda8a970f6648cdf0dbd6de" /// .parse::() /// .unwrap(); - /// assert!(rando.is_near_implicit()); + /// assert!(near_rando.get_account_type() == AccountType::NearImplicitAccount); /// ``` - pub fn is_near_implicit(&self) -> bool { - self.0.len() == 64 - && self - .as_bytes() - .iter() - .all(|b| matches!(b, b'a'..=b'f' | b'0'..=b'9')) - } - pub fn get_account_type(&self) -> AccountType { - if self.is_eth_implicit() { + if crate::validation::is_eth_implicit(self.as_str()) { return AccountType::EthImplicitAccount; } - if self.is_near_implicit() { + if crate::validation::is_near_implicit(self.as_str()) { return AccountType::NearImplicitAccount; } AccountType::NamedAccount diff --git a/src/validation.rs b/src/validation.rs index d7f1eca..3c985f5 100644 --- a/src/validation.rs +++ b/src/validation.rs @@ -93,6 +93,20 @@ pub fn validate(account_id: &str) -> Result<(), ParseAccountError> { } } +pub fn is_eth_implicit(account_id: &str) -> bool { + account_id.len() == 42 + && account_id.starts_with("0x") + && account_id[2..].as_bytes().iter().all(|b| matches!(b, b'a'..=b'f' | b'0'..=b'9')) +} + +pub fn is_near_implicit(account_id: &str) -> bool { + account_id.len() == 64 + && account_id + .as_bytes() + .iter() + .all(|b| matches!(b, b'a'..=b'f' | b'0'..=b'9')) +} + #[cfg(test)] mod tests { use super::*; From 07bf40f374f13d86b85339cab3a646a78154bb4f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Adam=20Chuda=C5=9B?= Date: Fri, 3 Nov 2023 21:19:37 +0100 Subject: [PATCH 3/3] Treat EthImplicit as implicit --- src/account_id_ref.rs | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/account_id_ref.rs b/src/account_id_ref.rs index 0d6fd7b..69951bb 100644 --- a/src/account_id_ref.rs +++ b/src/account_id_ref.rs @@ -49,8 +49,7 @@ impl AccountType { pub fn is_implicit(&self) -> bool { match &self { Self::NearImplicitAccount => true, - // TODO(eth-implicit) change to true later, see https://github.com/near/nearcore/issues/10018 - Self::EthImplicitAccount => false, + Self::EthImplicitAccount => true, Self::NamedAccount => false, } }