From 2f8ebc9ad438dd4a1ccd90f54cba612282f1894f Mon Sep 17 00:00:00 2001 From: Thomas Marchand Date: Fri, 6 Oct 2023 18:48:15 +0700 Subject: [PATCH 01/10] feat: add set_address_to_domain to override reverse resolving --- src/interface/naming.cairo | 2 ++ src/naming/main.cairo | 38 ++++++++++++++++++++++++++++++++++---- 2 files changed, 36 insertions(+), 4 deletions(-) diff --git a/src/interface/naming.cairo b/src/interface/naming.cairo index f28046b..e95d744 100644 --- a/src/interface/naming.cairo +++ b/src/interface/naming.cairo @@ -39,6 +39,8 @@ trait INaming { fn reset_subdomains(ref self: TContractState, domain: Span); + fn set_address_to_domain(ref self: TContractState, domain: Span); + // admin fn set_admin(ref self: TContractState, new_admin: ContractAddress); diff --git a/src/naming/main.cairo b/src/naming/main.cairo index 101d4ad..d3499ca 100644 --- a/src/naming/main.cairo +++ b/src/naming/main.cairo @@ -29,7 +29,8 @@ mod Naming { enum Event { DomainMint: DomainMint, DomainRenewal: DomainRenewal, - DomainToResolver: DomainToResolver, + DomainResolverUpdate: DomainResolverUpdate, + AddressToDomainUpdate: AddressToDomainUpdate, DomainTransfer: DomainTransfer, SubdomainsReset: SubdomainsReset, SaleMetadata: SaleMetadata, @@ -51,12 +52,19 @@ mod Naming { } #[derive(Drop, starknet::Event)] - struct DomainToResolver { + struct DomainResolverUpdate { #[key] domain: Span, resolver: ContractAddress } + #[derive(Drop, starknet::Event)] + struct AddressToDomainUpdate { + #[key] + address: ContractAddress, + domain: Span, + } + #[derive(Drop, starknet::Event)] struct DomainTransfer { #[key] @@ -360,6 +368,14 @@ mod Naming { self.emit(Event::SubdomainsReset(SubdomainsReset { domain: domain, })); } + + fn set_address_to_domain(ref self: ContractState, domain: Span) { + let address = get_caller_address(); + assert(self.domain_to_address(domain) == address, 'domain not pointing back'); + self.emit(Event::AddressToDomainUpdate(AddressToDomainUpdate { address, domain })); + self.set_address_to_domain_util(address, domain); + } + // ADMIN fn set_admin(ref self: ContractState, new_admin: ContractAddress) { @@ -512,6 +528,20 @@ mod Naming { } } + fn set_address_to_domain_util( + ref self: ContractState, address: ContractAddress, mut domain: Span + ) { + match domain.pop_back() { + Option::Some(domain_part) => { + self._address_to_domain.write((address, domain.len()), *domain_part); + return self.set_address_to_domain_util(address, domain); + }, + Option::None => { + return; + } + } + } + fn domain_to_resolver( self: @ContractState, domain: Span, parent_start_id: u32 ) -> (ContractAddress, u32) { @@ -597,8 +627,8 @@ mod Naming { if (resolver.into() != 0) { self .emit( - Event::DomainToResolver( - DomainToResolver { domain: array![domain].span(), resolver } + Event::DomainResolverUpdate( + DomainResolverUpdate { domain: array![domain].span(), resolver } ) ); } From c19c35ec4eee1fa7bba43c67e1adb438a8730f00 Mon Sep 17 00:00:00 2001 From: Thomas Marchand Date: Sat, 7 Oct 2023 15:45:27 +0700 Subject: [PATCH 02/10] feat: add reverse resolving --- src/interface/naming.cairo | 4 +- src/naming/main.cairo | 27 ++++++++---- src/tests/naming/test_usecases.cairo | 65 ++++++++++++++++++++++++++++ 3 files changed, 87 insertions(+), 9 deletions(-) diff --git a/src/interface/naming.cairo b/src/interface/naming.cairo index e95d744..7e6194b 100644 --- a/src/interface/naming.cairo +++ b/src/interface/naming.cairo @@ -12,7 +12,7 @@ trait INaming { fn domain_to_address(self: @TContractState, domain: Span) -> ContractAddress; - fn address_to_domain(self: @TContractState, address: ContractAddress) -> Array; + fn address_to_domain(self: @TContractState, address: ContractAddress) -> Span; // external fn buy( @@ -41,6 +41,8 @@ trait INaming { fn set_address_to_domain(ref self: TContractState, domain: Span); + fn reset_address_to_domain(ref self: TContractState); + // admin fn set_admin(ref self: TContractState, new_admin: ContractAddress); diff --git a/src/naming/main.cairo b/src/naming/main.cairo index d3499ca..2a02efa 100644 --- a/src/naming/main.cairo +++ b/src/naming/main.cairo @@ -215,11 +215,11 @@ mod Naming { } // This function allows to find which domain to use to display an account - fn address_to_domain(self: @ContractState, address: ContractAddress) -> Array { + fn address_to_domain(self: @ContractState, address: ContractAddress) -> Span { let mut domain = ArrayTrait::new(); self._address_to_domain_util(address, ref domain); if domain.len() != 0 && self.domain_to_address(domain.span()) == address { - domain + domain.span() } else { let identity = IIdentityDispatcher { contract_address: self.starknetid_contract.read() @@ -229,9 +229,7 @@ mod Naming { let id_hashed_domain = identity .get_verifier_data(id, 'name', get_contract_address(), 0); let domain = self.unhash_domain(id_hashed_domain); - assert( - self.domain_to_address(domain.span()) == address, 'domain not pointing back' - ); + assert(self.domain_to_address(domain) == address, 'domain not pointing back'); domain } } @@ -369,6 +367,7 @@ mod Naming { } + // will override your main id fn set_address_to_domain(ref self: ContractState, domain: Span) { let address = get_caller_address(); assert(self.domain_to_address(domain) == address, 'domain not pointing back'); @@ -376,6 +375,17 @@ mod Naming { self.set_address_to_domain_util(address, domain); } + fn reset_address_to_domain(ref self: ContractState) { + let address = get_caller_address(); + self + .emit( + Event::AddressToDomainUpdate( + AddressToDomainUpdate { address, domain: array![].span() } + ) + ); + self.set_address_to_domain_util(address, array![0].span()); + } + // ADMIN fn set_admin(ref self: ContractState, new_admin: ContractAddress) { @@ -428,7 +438,7 @@ mod Naming { return hashed_domain; } - fn unhash_domain(self: @ContractState, domain_hash: felt252) -> Array { + fn unhash_domain(self: @ContractState, domain_hash: felt252) -> Span { let mut i = 0; let mut domain = ArrayTrait::new(); loop { @@ -437,8 +447,9 @@ mod Naming { break; } domain.append(domain_part); + i += 1; }; - domain + domain.span() } fn assert_purchase_is_possible( @@ -618,7 +629,7 @@ mod Naming { key: 1, parent_key: 0, }; - self._hash_to_domain.write((hashed_domain, 0), hashed_domain); + self._hash_to_domain.write((hashed_domain, 0), domain); self._domain_data.write(hashed_domain, data); self.emit(Event::DomainMint(DomainMint { domain, owner: id, expiry })); diff --git a/src/tests/naming/test_usecases.cairo b/src/tests/naming/test_usecases.cairo index 7411448..1211cd5 100644 --- a/src/tests/naming/test_usecases.cairo +++ b/src/tests/naming/test_usecases.cairo @@ -202,3 +202,68 @@ fn test_non_owner_can_renew_domain() { eth.approve(naming.contract_address, price); naming.renew(domain_name, 365, ContractAddressZeroable::zero(), 0, 0); } + + +#[test] +#[available_gas(2000000000)] +fn test_set_address_to_domain() { + // setup + let (eth, pricing, identity, naming) = deploy(); + let caller = contract_address_const::<0x123>(); + set_contract_address(caller); + let id1: u128 = 1; + let id2: u128 = 2; + let first_domain_top: felt252 = 82939898252385817; + let second_domain_top: felt252 = 3151716132312299378; + + //we mint the ids + identity.mint(id1); + identity.mint(id2); + + // buy the domains + let (_, price1) = pricing.compute_buy_price(11, 365); + eth.approve(naming.contract_address, price1); + naming + .buy( + id1, + first_domain_top, + 365, + ContractAddressZeroable::zero(), + ContractAddressZeroable::zero(), + 0, + 0 + ); + + let (_, price2) = pricing.compute_buy_price(12, 365); + eth.approve(naming.contract_address, price2); + naming + .buy( + id2, + second_domain_top, + 365, + ContractAddressZeroable::zero(), + ContractAddressZeroable::zero(), + 0, + 0 + ); + + // let's try the resolving + let first_domain = array![first_domain_top].span(); + assert(naming.domain_to_address(first_domain) == caller, 'wrong domain target'); + + // set reverse resolving + identity.set_main_id(id1); + let expect_domain1 = naming.address_to_domain(caller); + assert(expect_domain1 == first_domain, 'wrong rev resolving 1'); + + // override reverse resolving + let second_domain = array![second_domain_top].span(); + naming.set_address_to_domain(second_domain); + let expect_domain2 = naming.address_to_domain(caller); + assert(expect_domain2 == second_domain, 'wrong rev resolving 2'); + + // remove override + naming.reset_address_to_domain(); + let expect_domain1 = naming.address_to_domain(caller); + assert(expect_domain1 == first_domain, 'wrong rev resolving b'); +} From 28f7d174d301db5b49217c3756c56a9efad09fb9 Mon Sep 17 00:00:00 2001 From: Thomas Marchand Date: Sun, 8 Oct 2023 15:44:14 +0700 Subject: [PATCH 03/10] refactor: extract resolve_util to avoid duplicate hash computation --- src/naming/main.cairo | 121 ++++++++++++++++++++++++------------------ 1 file changed, 68 insertions(+), 53 deletions(-) diff --git a/src/naming/main.cairo b/src/naming/main.cairo index 2a02efa..3906685 100644 --- a/src/naming/main.cairo +++ b/src/naming/main.cairo @@ -139,32 +139,8 @@ mod Naming { // naming.resolve(['alice'], 'bitcoin') // Use it with caution in smartcontracts as it can call untrusted contracts fn resolve(self: @ContractState, domain: Span, field: felt252) -> felt252 { - let (resolver, parent_start) = self.domain_to_resolver(domain, 0); - if (resolver != ContractAddressZeroable::zero()) { - IResolverDispatcher { contract_address: resolver } - .resolve(domain.slice(parent_start, domain.len() - parent_start), field) - } else { - let domain_data = self._domain_data.read(self.hash_domain(domain)); - // circuit breaker for root domain - if (domain.len() == 1) { - IIdentityDispatcher { contract_address: self.starknetid_contract.read() } - .get_crosschecked_user_data(domain_data.owner, field) - // handle reset subdomains - } else { - // todo: optimize by changing the hash definition from H(b, a) to H(a, b) - let parent_key = self - ._domain_data - .read(self.hash_domain(domain.slice(1, domain.len() - 1))) - .key; - - if parent_key == domain_data.parent_key { - IIdentityDispatcher { contract_address: self.starknetid_contract.read() } - .get_crosschecked_user_data(domain_data.owner, field) - } else { - 0 - } - } - } + let (_, value) = self.resolve_util(domain, field); + value } // This functions allows to resolve a domain to a native address. Its output is designed @@ -172,12 +148,12 @@ mod Naming { // to a .stark) fn domain_to_address(self: @ContractState, domain: Span) -> ContractAddress { // resolve must be performed first because it calls untrusted resolving contracts - let resolve_result = self.resolve(domain, 'starknet'); - if resolve_result != 0 { - let addr: Option = resolve_result.try_into(); + let (hashed_domain, value) = self.resolve_util(domain, 'starknet'); + if value != 0 { + let addr: Option = value.try_into(); return addr.unwrap(); - } - let data = self._domain_data.read(self.hash_domain(domain)); + }; + let data = self._domain_data.read(hashed_domain); if data.address.into() != 0 { if domain.len() != 1 { let parent_key = self @@ -186,10 +162,10 @@ mod Naming { .key; if parent_key == data.parent_key { return data.address; - } - } + }; + }; return data.address; - } + }; IIdentityDispatcher { contract_address: self.starknetid_contract.read() } .owner_of(self.domain_to_id(domain)) } @@ -209,15 +185,15 @@ mod Naming { .key; if parent_key != data.parent_key { return 0; - } - } + }; + }; data.owner } // This function allows to find which domain to use to display an account fn address_to_domain(self: @ContractState, address: ContractAddress) -> Span { let mut domain = ArrayTrait::new(); - self._address_to_domain_util(address, ref domain); + self.read_address_to_domain(address, ref domain); if domain.len() != 0 && self.domain_to_address(domain.span()) == address { domain.span() } else { @@ -430,7 +406,7 @@ mod Naming { fn hash_domain(self: @ContractState, domain: Span) -> felt252 { if domain.len() == 0 { return 0; - } + }; let new_len = domain.len() - 1; let x = *domain[new_len]; let y = self.hash_domain(domain.slice(0, new_len)); @@ -445,7 +421,7 @@ mod Naming { let domain_part = self._hash_to_domain.read((domain_hash, i)); if domain_part == 0 { break; - } + }; domain.append(domain_part); i += 1; }; @@ -475,14 +451,14 @@ mod Naming { self: @ContractState, domain: Span, account: ContractAddress ) { // 1. account owns the domain - self._assert_is_owner(domain, account); + self.assert_is_owner(domain, account); // 2. check domain expiration let hashed_root_domain = self.hash_domain(domain.slice(domain.len() - 1, 1)); let root_domain_data = self._domain_data.read(hashed_root_domain); assert(get_block_timestamp() <= root_domain_data.expiry, 'this domain has expired'); } - fn _assert_is_owner( + fn assert_is_owner( self: @ContractState, domain: Span, account: ContractAddress ) -> u32 { let hashed_domain = self.hash_domain(domain); @@ -499,18 +475,18 @@ mod Naming { // if caller owns the starknet id, he owns the domain, we return the key if owner == account { return data.key; - } + }; // otherwise, if it is a root domain, he doesn't own it assert(domain.len() != 1 && domain.len() != 0, 'you don\'t own this domain'); // if he doesn't own the starknet id, and doesn't own the domain, he might own the parent domain - let parent_key = self._assert_is_owner(domain.slice(1, domain.len() - 1), account); + let parent_key = self.assert_is_owner(domain.slice(1, domain.len() - 1), account); // we ensure that the key is the same as the parent key // this is to allow to revoke all subdomains in o(1) writes, by juste updating the key of the parent if (data.parent_key != 0) { assert(parent_key == data.parent_key, 'you no longer own this domain'); - } + }; data.key } @@ -527,7 +503,7 @@ mod Naming { ); } - fn _address_to_domain_util( + fn read_address_to_domain( self: @ContractState, address: ContractAddress, ref domain: Array ) -> usize { let subdomain = self._address_to_domain.read((address, domain.len())); @@ -535,7 +511,7 @@ mod Naming { domain.len() } else { domain.append(subdomain); - self._address_to_domain_util(address, ref domain) + self.read_address_to_domain(address, ref domain) } } @@ -545,11 +521,9 @@ mod Naming { match domain.pop_back() { Option::Some(domain_part) => { self._address_to_domain.write((address, domain.len()), *domain_part); - return self.set_address_to_domain_util(address, domain); + self.set_address_to_domain_util(address, domain) }, - Option::None => { - return; - } + Option::None => {} } } @@ -558,7 +532,7 @@ mod Naming { ) -> (ContractAddress, u32) { if parent_start_id == domain.len() { return (ContractAddressZeroable::zero(), 0); - } + }; // hashing parent_domain let hashed_domain = self @@ -648,7 +622,7 @@ mod Naming { fn get_chars_len(self: @ContractState, domain: u256) -> usize { if domain == (u256 { low: 0, high: 0 }) { return 0; - } + }; // 38 = simple_alphabet_size let (p, q, _) = u256_safe_divmod(domain, u256_as_non_zero(u256 { low: 38, high: 0 })); if q == (u256 { low: 37, high: 0 }) { @@ -658,9 +632,50 @@ mod Naming { ); let next = self.get_chars_len(shifted_p); return 1 + next; - } + }; let next = self.get_chars_len(p); 1 + next } + + // returns domain_hash (or zero) and its value for a specific field + fn resolve_util( + self: @ContractState, domain: Span, field: felt252 + ) -> (felt252, felt252) { + let (resolver, parent_start) = self.domain_to_resolver(domain, 0); + if (resolver != ContractAddressZeroable::zero()) { + ( + 0, + IResolverDispatcher { contract_address: resolver } + .resolve(domain.slice(parent_start, domain.len() - parent_start), field) + ) + } else { + let hashed_domain = self.hash_domain(domain); + let domain_data = self._domain_data.read(hashed_domain); + // circuit breaker for root domain + ( + hashed_domain, + if (domain.len() == 1) { + IIdentityDispatcher { contract_address: self.starknetid_contract.read() } + .get_crosschecked_user_data(domain_data.owner, field) + // handle reset subdomains + } else { + // todo: optimize by changing the hash definition from H(b, a) to H(a, b) + let parent_key = self + ._domain_data + .read(self.hash_domain(domain.slice(1, domain.len() - 1))) + .key; + + if parent_key == domain_data.parent_key { + IIdentityDispatcher { + contract_address: self.starknetid_contract.read() + } + .get_crosschecked_user_data(domain_data.owner, field) + } else { + 0 + } + } + ) + } + } } } From 9d6f6ecab3e4eb7aecdebc05845241df5bf398db Mon Sep 17 00:00:00 2001 From: Thomas Marchand Date: Mon, 9 Oct 2023 11:25:31 +0700 Subject: [PATCH 04/10] feat: extract internal to multiple files --- src/naming.cairo | 3 + src/naming/asserts.cairo | 107 ++++++++++ src/naming/internal.cairo | 185 +++++++++++++++++ src/naming/main.cairo | 286 +-------------------------- src/naming/utils.cairo | 48 +++++ src/tests/naming/test_features.cairo | 17 +- 6 files changed, 356 insertions(+), 290 deletions(-) create mode 100644 src/naming/asserts.cairo create mode 100644 src/naming/internal.cairo create mode 100644 src/naming/utils.cairo diff --git a/src/naming.cairo b/src/naming.cairo index d18669a..a967bc9 100644 --- a/src/naming.cairo +++ b/src/naming.cairo @@ -1 +1,4 @@ mod main; +mod internal; +mod asserts; +mod utils; \ No newline at end of file diff --git a/src/naming/asserts.cairo b/src/naming/asserts.cairo new file mode 100644 index 0000000..f058a45 --- /dev/null +++ b/src/naming/asserts.cairo @@ -0,0 +1,107 @@ +use naming::{ + interface::{ + naming::{INaming, INamingDispatcher, INamingDispatcherTrait}, + resolver::{IResolver, IResolverDispatcher, IResolverDispatcherTrait}, + pricing::{IPricing, IPricingDispatcher, IPricingDispatcherTrait}, + referral::{IReferral, IReferralDispatcher, IReferralDispatcherTrait}, + }, + naming::main::{ + Naming, + Naming::{ + EventEmitter, _hash_to_domain, _hash_to_domainContractMemberStateTrait, _domain_data, + _domain_dataContractMemberStateTrait, starknetid_contract, + starknetid_contractContractMemberStateTrait, discounts, + discountsContractMemberStateTrait, _address_to_domain, + _address_to_domainContractMemberStateTrait, _referral_contract, + _referral_contractContractMemberStateTrait, + } + }, +}; +use identity::interface::identity::{IIdentity, IIdentityDispatcher, IIdentityDispatcherTrait}; +use starknet::{ + contract_address::ContractAddressZeroable, ContractAddress, get_caller_address, + get_contract_address, get_block_timestamp +}; +use openzeppelin::token::erc20::interface::{ + IERC20Camel, IERC20CamelDispatcher, IERC20CamelDispatcherTrait +}; +use integer::{u256_safe_divmod, u256_as_non_zero}; +use naming::naming::utils::UtilsTrait; + + +#[generate_trait] +impl AssertionsImpl of AssertionsTrait { + fn assert_purchase_is_possible( + self: @Naming::ContractState, identity: u128, domain: felt252, days: u16 + ) -> (felt252, u64, u64) { + let now = get_block_timestamp(); + + // Verify that the starknet.id doesn't already manage a domain + self.assert_id_availability(identity, now); + + // Verify that the domain is not already taken or expired + let hashed_domain = self.hash_domain(array![domain].span()); + let data = self._domain_data.read(hashed_domain); + assert(data.owner == 0 || data.expiry < now, 'unexpired domain'); + + // Verify expiration range + assert(days < 365 * 25, 'max purchase of 25 years'); + assert(days > 2 * 30, 'min purchase of 2 month'); + return (hashed_domain, now, now + 86400 * days.into()); + } + + fn assert_control_domain( + self: @Naming::ContractState, domain: Span, account: ContractAddress + ) { + // 1. account owns the domain + self.assert_is_owner(domain, account); + // 2. check domain expiration + let hashed_root_domain = self.hash_domain(domain.slice(domain.len() - 1, 1)); + let root_domain_data = self._domain_data.read(hashed_root_domain); + assert(get_block_timestamp() <= root_domain_data.expiry, 'this domain has expired'); + } + + fn assert_is_owner( + self: @Naming::ContractState, domain: Span, account: ContractAddress + ) -> u32 { + let hashed_domain = self.hash_domain(domain); + let data = self._domain_data.read(hashed_domain); + + // because erc721 crashes on zero + let owner = if data.owner == 0 { + ContractAddressZeroable::zero() + } else { + IIdentityDispatcher { contract_address: self.starknetid_contract.read() } + .owner_of(data.owner) + }; + + // if caller owns the starknet id, he owns the domain, we return the key + if owner == account { + return data.key; + }; + + // otherwise, if it is a root domain, he doesn't own it + assert(domain.len() != 1 && domain.len() != 0, 'you don\'t own this domain'); + + // if he doesn't own the starknet id, and doesn't own the domain, he might own the parent domain + let parent_key = self.assert_is_owner(domain.slice(1, domain.len() - 1), account); + // we ensure that the key is the same as the parent key + // this is to allow to revoke all subdomains in o(1) writes, by juste updating the key of the parent + if (data.parent_key != 0) { + assert(parent_key == data.parent_key, 'you no longer own this domain'); + }; + data.key + } + + // this ensures a non expired domain is not already written on this identity + fn assert_id_availability(self: @Naming::ContractState, identity: u128, timestamp: u64) { + let id_hashed_domain = IIdentityDispatcher { + contract_address: self.starknetid_contract.read() + } + .get_verifier_data(identity, 'name', get_contract_address(), 0); + assert( + id_hashed_domain == 0 || self._domain_data.read(id_hashed_domain).expiry < timestamp, + 'this id holds a domain' + ); + } +} diff --git a/src/naming/internal.cairo b/src/naming/internal.cairo new file mode 100644 index 0000000..f4aefd8 --- /dev/null +++ b/src/naming/internal.cairo @@ -0,0 +1,185 @@ +use naming::{ + interface::{ + naming::{INaming, INamingDispatcher, INamingDispatcherTrait}, + resolver::{IResolver, IResolverDispatcher, IResolverDispatcherTrait}, + pricing::{IPricing, IPricingDispatcher, IPricingDispatcherTrait}, + referral::{IReferral, IReferralDispatcher, IReferralDispatcherTrait}, + }, + naming::main::{ + Naming, + Naming::{ + EventEmitter, _hash_to_domain, _hash_to_domainContractMemberStateTrait, _domain_data, + _domain_dataContractMemberStateTrait, starknetid_contract, + starknetid_contractContractMemberStateTrait, discounts, + discountsContractMemberStateTrait, _address_to_domain, + _address_to_domainContractMemberStateTrait, _referral_contract, + _referral_contractContractMemberStateTrait, + } + } +}; +use identity::interface::identity::{IIdentity, IIdentityDispatcher, IIdentityDispatcherTrait}; +use starknet::{ + contract_address::ContractAddressZeroable, ContractAddress, get_caller_address, + get_contract_address, get_block_timestamp +}; +use openzeppelin::token::erc20::interface::{ + IERC20Camel, IERC20CamelDispatcher, IERC20CamelDispatcherTrait +}; +use naming::naming::utils::UtilsTrait; + +#[generate_trait] +impl InternalImpl of InternalTrait { + + fn read_address_to_domain( + self: @Naming::ContractState, address: ContractAddress, ref domain: Array + ) -> usize { + let subdomain = self._address_to_domain.read((address, domain.len())); + if subdomain == 0 { + domain.len() + } else { + domain.append(subdomain); + self.read_address_to_domain(address, ref domain) + } + } + + fn set_address_to_domain_util( + ref self: Naming::ContractState, address: ContractAddress, mut domain: Span + ) { + match domain.pop_back() { + Option::Some(domain_part) => { + self._address_to_domain.write((address, domain.len()), *domain_part); + self.set_address_to_domain_util(address, domain) + }, + Option::None => {} + } + } + + fn domain_to_resolver( + self: @Naming::ContractState, domain: Span, parent_start_id: u32 + ) -> (ContractAddress, u32) { + if parent_start_id == domain.len() { + return (ContractAddressZeroable::zero(), 0); + }; + + // hashing parent_domain + let hashed_domain = self + .hash_domain(domain.slice(parent_start_id, domain.len() - parent_start_id)); + + let domain_data = self._domain_data.read(hashed_domain); + + if domain_data.resolver.into() != 0 { + return (domain_data.resolver, parent_start_id); + } else { + return self.domain_to_resolver(domain, parent_start_id + 1); + } + } + + fn pay_domain( + self: @Naming::ContractState, + domain_len: usize, + erc20: ContractAddress, + price: u256, + now: u64, + days: u16, + domain: felt252, + sponsor: ContractAddress, + discount_id: felt252 + ) -> () { + // check the discount + let discounted_price = if (discount_id == 0) { + price + } else { + let discount = self.discounts.read(discount_id); + let (min, max) = discount.domain_len_range; + assert(min <= domain_len && domain_len <= max, 'invalid length for discount'); + + let (min, max) = discount.days_range; + assert(min <= days && days <= max, 'days out of discount range'); + + let (min, max) = discount.timestamp_range; + assert(min <= now && now <= max, 'time out of discount range'); + // discount.amount won't overflow as it's a value chosen by the admin to be in range (0, 100) + (price * discount.amount) / 100 + }; + + // pay the price + IERC20CamelDispatcher { contract_address: erc20 } + .transferFrom(get_caller_address(), get_contract_address(), discounted_price); + // add sponsor commission if eligible + if sponsor.into() != 0 { + IReferralDispatcher { contract_address: self._referral_contract.read() } + .add_commission(discounted_price, sponsor, sponsored_addr: get_caller_address()); + } + } + + fn mint_domain( + ref self: Naming::ContractState, + expiry: u64, + resolver: ContractAddress, + hashed_domain: felt252, + id: u128, + domain: felt252 + ) { + let data = Naming::DomainData { + owner: id, + resolver, + address: ContractAddressZeroable::zero(), // legacy native address + expiry, + key: 1, + parent_key: 0, + }; + self._hash_to_domain.write((hashed_domain, 0), domain); + self._domain_data.write(hashed_domain, data); + self.emit(Naming::Event::DomainMint(Naming::DomainMint { domain, owner: id, expiry })); + + IIdentityDispatcher { contract_address: self.starknetid_contract.read() } + .set_verifier_data(id, 'name', hashed_domain, 0); + if (resolver.into() != 0) { + self + .emit( + Naming::Event::DomainResolverUpdate( + Naming::DomainResolverUpdate { domain: array![domain].span(), resolver } + ) + ); + } + } + + // returns domain_hash (or zero) and its value for a specific field + fn resolve_util( + self: @Naming::ContractState, domain: Span, field: felt252 + ) -> (felt252, felt252) { + let (resolver, parent_start) = self.domain_to_resolver(domain, 0); + if (resolver != ContractAddressZeroable::zero()) { + ( + 0, + IResolverDispatcher { contract_address: resolver } + .resolve(domain.slice(parent_start, domain.len() - parent_start), field) + ) + } else { + let hashed_domain = self.hash_domain(domain); + let domain_data = self._domain_data.read(hashed_domain); + // circuit breaker for root domain + ( + hashed_domain, + if (domain.len() == 1) { + IIdentityDispatcher { contract_address: self.starknetid_contract.read() } + .get_crosschecked_user_data(domain_data.owner, field) + // handle reset subdomains + } else { + // todo: optimize by changing the hash definition from H(b, a) to H(a, b) + let parent_key = self + ._domain_data + .read(self.hash_domain(domain.slice(1, domain.len() - 1))) + .key; + + if parent_key == domain_data.parent_key { + IIdentityDispatcher { contract_address: self.starknetid_contract.read() } + .get_crosschecked_user_data(domain_data.owner, field) + } else { + 0 + } + } + ) + } + } +} diff --git a/src/naming/main.cairo b/src/naming/main.cairo index 3906685..c1f1545 100644 --- a/src/naming/main.cairo +++ b/src/naming/main.cairo @@ -10,12 +10,14 @@ mod Naming { use starknet::class_hash::ClassHash; use integer::{u256_safe_divmod, u256_as_non_zero}; use core::pedersen; - use naming::interface::{ + use naming::{ + naming::{asserts::AssertionsTrait, internal::InternalTrait, utils::UtilsTrait}, + interface::{ naming::{INaming, INamingDispatcher, INamingDispatcherTrait}, resolver::{IResolver, IResolverDispatcher, IResolverDispatcherTrait}, pricing::{IPricing, IPricingDispatcher, IPricingDispatcherTrait}, referral::{IReferral, IReferralDispatcher, IReferralDispatcherTrait}, - }; + }}; use clone::Clone; use array::ArrayTCloneImpl; use identity::interface::identity::{IIdentity, IIdentityDispatcher, IIdentityDispatcherTrait}; @@ -399,283 +401,5 @@ mod Naming { starknet::replace_class_syscall(new_class_hash).unwrap(); } } - - #[generate_trait] - impl InternalImpl of InternalTrait { - // hash(alpha.bravo.stark) = pedersen(bravo, pedersen(alpha, 0)) - fn hash_domain(self: @ContractState, domain: Span) -> felt252 { - if domain.len() == 0 { - return 0; - }; - let new_len = domain.len() - 1; - let x = *domain[new_len]; - let y = self.hash_domain(domain.slice(0, new_len)); - let hashed_domain = pedersen::pedersen(x, y); - return hashed_domain; - } - - fn unhash_domain(self: @ContractState, domain_hash: felt252) -> Span { - let mut i = 0; - let mut domain = ArrayTrait::new(); - loop { - let domain_part = self._hash_to_domain.read((domain_hash, i)); - if domain_part == 0 { - break; - }; - domain.append(domain_part); - i += 1; - }; - domain.span() - } - - fn assert_purchase_is_possible( - self: @ContractState, identity: u128, domain: felt252, days: u16 - ) -> (felt252, u64, u64) { - let now = get_block_timestamp(); - - // Verify that the starknet.id doesn't already manage a domain - self.assert_id_availability(identity, now); - - // Verify that the domain is not already taken or expired - let hashed_domain = self.hash_domain(array![domain].span()); - let data = self._domain_data.read(hashed_domain); - assert(data.owner == 0 || data.expiry < now, 'unexpired domain'); - - // Verify expiration range - assert(days < 365 * 25, 'max purchase of 25 years'); - assert(days > 2 * 30, 'min purchase of 2 month'); - return (hashed_domain, now, now + 86400 * days.into()); - } - - fn assert_control_domain( - self: @ContractState, domain: Span, account: ContractAddress - ) { - // 1. account owns the domain - self.assert_is_owner(domain, account); - // 2. check domain expiration - let hashed_root_domain = self.hash_domain(domain.slice(domain.len() - 1, 1)); - let root_domain_data = self._domain_data.read(hashed_root_domain); - assert(get_block_timestamp() <= root_domain_data.expiry, 'this domain has expired'); - } - - fn assert_is_owner( - self: @ContractState, domain: Span, account: ContractAddress - ) -> u32 { - let hashed_domain = self.hash_domain(domain); - let data = self._domain_data.read(hashed_domain); - - // because erc721 crashes on zero - let owner = if data.owner == 0 { - ContractAddressZeroable::zero() - } else { - IIdentityDispatcher { contract_address: self.starknetid_contract.read() } - .owner_of(data.owner) - }; - - // if caller owns the starknet id, he owns the domain, we return the key - if owner == account { - return data.key; - }; - - // otherwise, if it is a root domain, he doesn't own it - assert(domain.len() != 1 && domain.len() != 0, 'you don\'t own this domain'); - - // if he doesn't own the starknet id, and doesn't own the domain, he might own the parent domain - let parent_key = self.assert_is_owner(domain.slice(1, domain.len() - 1), account); - // we ensure that the key is the same as the parent key - // this is to allow to revoke all subdomains in o(1) writes, by juste updating the key of the parent - if (data.parent_key != 0) { - assert(parent_key == data.parent_key, 'you no longer own this domain'); - }; - data.key - } - - // this ensures a non expired domain is not already written on this identity - fn assert_id_availability(self: @ContractState, identity: u128, timestamp: u64) { - let id_hashed_domain = IIdentityDispatcher { - contract_address: self.starknetid_contract.read() - } - .get_verifier_data(identity, 'name', get_contract_address(), 0); - assert( - id_hashed_domain == 0 - || self._domain_data.read(id_hashed_domain).expiry < timestamp, - 'this id holds a domain' - ); - } - - fn read_address_to_domain( - self: @ContractState, address: ContractAddress, ref domain: Array - ) -> usize { - let subdomain = self._address_to_domain.read((address, domain.len())); - if subdomain == 0 { - domain.len() - } else { - domain.append(subdomain); - self.read_address_to_domain(address, ref domain) - } - } - - fn set_address_to_domain_util( - ref self: ContractState, address: ContractAddress, mut domain: Span - ) { - match domain.pop_back() { - Option::Some(domain_part) => { - self._address_to_domain.write((address, domain.len()), *domain_part); - self.set_address_to_domain_util(address, domain) - }, - Option::None => {} - } - } - - fn domain_to_resolver( - self: @ContractState, domain: Span, parent_start_id: u32 - ) -> (ContractAddress, u32) { - if parent_start_id == domain.len() { - return (ContractAddressZeroable::zero(), 0); - }; - - // hashing parent_domain - let hashed_domain = self - .hash_domain(domain.slice(parent_start_id, domain.len() - parent_start_id)); - - let domain_data = self._domain_data.read(hashed_domain); - - if domain_data.resolver.into() != 0 { - return (domain_data.resolver, parent_start_id); - } else { - return self.domain_to_resolver(domain, parent_start_id + 1); - } - } - - fn pay_domain( - self: @ContractState, - domain_len: usize, - erc20: ContractAddress, - price: u256, - now: u64, - days: u16, - domain: felt252, - sponsor: ContractAddress, - discount_id: felt252 - ) -> () { - // check the discount - let discounted_price = if (discount_id == 0) { - price - } else { - let discount = self.discounts.read(discount_id); - let (min, max) = discount.domain_len_range; - assert(min <= domain_len && domain_len <= max, 'invalid length for discount'); - - let (min, max) = discount.days_range; - assert(min <= days && days <= max, 'days out of discount range'); - - let (min, max) = discount.timestamp_range; - assert(min <= now && now <= max, 'time out of discount range'); - // discount.amount won't overflow as it's a value chosen by the admin to be in range (0, 100) - (price * discount.amount) / 100 - }; - - // pay the price - IERC20CamelDispatcher { contract_address: erc20 } - .transferFrom(get_caller_address(), get_contract_address(), discounted_price); - // add sponsor commission if eligible - if sponsor.into() != 0 { - IReferralDispatcher { contract_address: self._referral_contract.read() } - .add_commission( - discounted_price, sponsor, sponsored_addr: get_caller_address() - ); - } - } - - fn mint_domain( - ref self: ContractState, - expiry: u64, - resolver: ContractAddress, - hashed_domain: felt252, - id: u128, - domain: felt252 - ) { - let data = DomainData { - owner: id, - resolver, - address: ContractAddressZeroable::zero(), // legacy native address - expiry, - key: 1, - parent_key: 0, - }; - self._hash_to_domain.write((hashed_domain, 0), domain); - self._domain_data.write(hashed_domain, data); - self.emit(Event::DomainMint(DomainMint { domain, owner: id, expiry })); - - IIdentityDispatcher { contract_address: self.starknetid_contract.read() } - .set_verifier_data(id, 'name', hashed_domain, 0); - if (resolver.into() != 0) { - self - .emit( - Event::DomainResolverUpdate( - DomainResolverUpdate { domain: array![domain].span(), resolver } - ) - ); - } - } - - fn get_chars_len(self: @ContractState, domain: u256) -> usize { - if domain == (u256 { low: 0, high: 0 }) { - return 0; - }; - // 38 = simple_alphabet_size - let (p, q, _) = u256_safe_divmod(domain, u256_as_non_zero(u256 { low: 38, high: 0 })); - if q == (u256 { low: 37, high: 0 }) { - // 3 = complex_alphabet_size - let (shifted_p, _, _) = u256_safe_divmod( - p, u256_as_non_zero(u256 { low: 2, high: 0 }) - ); - let next = self.get_chars_len(shifted_p); - return 1 + next; - }; - let next = self.get_chars_len(p); - 1 + next - } - - // returns domain_hash (or zero) and its value for a specific field - fn resolve_util( - self: @ContractState, domain: Span, field: felt252 - ) -> (felt252, felt252) { - let (resolver, parent_start) = self.domain_to_resolver(domain, 0); - if (resolver != ContractAddressZeroable::zero()) { - ( - 0, - IResolverDispatcher { contract_address: resolver } - .resolve(domain.slice(parent_start, domain.len() - parent_start), field) - ) - } else { - let hashed_domain = self.hash_domain(domain); - let domain_data = self._domain_data.read(hashed_domain); - // circuit breaker for root domain - ( - hashed_domain, - if (domain.len() == 1) { - IIdentityDispatcher { contract_address: self.starknetid_contract.read() } - .get_crosschecked_user_data(domain_data.owner, field) - // handle reset subdomains - } else { - // todo: optimize by changing the hash definition from H(b, a) to H(a, b) - let parent_key = self - ._domain_data - .read(self.hash_domain(domain.slice(1, domain.len() - 1))) - .key; - - if parent_key == domain_data.parent_key { - IIdentityDispatcher { - contract_address: self.starknetid_contract.read() - } - .get_crosschecked_user_data(domain_data.owner, field) - } else { - 0 - } - } - ) - } - } - } } + diff --git a/src/naming/utils.cairo b/src/naming/utils.cairo new file mode 100644 index 0000000..5c73168 --- /dev/null +++ b/src/naming/utils.cairo @@ -0,0 +1,48 @@ +use naming::{ + naming::main::{Naming, Naming::{_hash_to_domain, _hash_to_domainContractMemberStateTrait}} +}; +use integer::{u256_safe_divmod, u256_as_non_zero}; + +#[generate_trait] +impl UtilsImpl of UtilsTrait { + fn hash_domain(self: @Naming::ContractState, domain: Span) -> felt252 { + if domain.len() == 0 { + return 0; + }; + let new_len = domain.len() - 1; + let x = *domain[new_len]; + let y = self.hash_domain(domain.slice(0, new_len)); + let hashed_domain = pedersen::pedersen(x, y); + return hashed_domain; + } + + fn unhash_domain(self: @Naming::ContractState, domain_hash: felt252) -> Span { + let mut i = 0; + let mut domain = ArrayTrait::new(); + loop { + let domain_part = self._hash_to_domain.read((domain_hash, i)); + if domain_part == 0 { + break; + }; + domain.append(domain_part); + i += 1; + }; + domain.span() + } + + fn get_chars_len(self: @Naming::ContractState, domain: u256) -> usize { + if domain == (u256 { low: 0, high: 0 }) { + return 0; + }; + // 38 = simple_alphabet_size + let (p, q, _) = u256_safe_divmod(domain, u256_as_non_zero(u256 { low: 38, high: 0 })); + if q == (u256 { low: 37, high: 0 }) { + // 3 = complex_alphabet_size + let (shifted_p, _, _) = u256_safe_divmod(p, u256_as_non_zero(u256 { low: 2, high: 0 })); + let next = self.get_chars_len(shifted_p); + return 1 + next; + }; + let next = self.get_chars_len(p); + 1 + next + } +} diff --git a/src/tests/naming/test_features.cairo b/src/tests/naming/test_features.cairo index 870fdd7..54b3adb 100644 --- a/src/tests/naming/test_features.cairo +++ b/src/tests/naming/test_features.cairo @@ -9,7 +9,6 @@ use starknet::ContractAddress; use starknet::contract_address::ContractAddressZeroable; use starknet::contract_address_const; use starknet::testing::set_contract_address; -use super::super::utils; use openzeppelin::token::erc20::{ erc20::ERC20, interface::{IERC20Camel, IERC20CamelDispatcher, IERC20CamelDispatcherTrait} }; @@ -20,7 +19,9 @@ use naming::interface::naming::{INamingDispatcher, INamingDispatcherTrait}; use naming::interface::pricing::{IPricingDispatcher, IPricingDispatcherTrait}; use naming::naming::main::Naming; use naming::pricing::Pricing; +use naming::naming::utils::UtilsImpl; use super::common::deploy; +use super::super::utils; #[test] @@ -121,25 +122,23 @@ fn test_get_chars_len() { let mut unsafe_state = Naming::unsafe_new_contract_state(); // Should return 0 (empty string) - assert(Naming::InternalImpl::get_chars_len(@unsafe_state, 0) == 0, 'Should return 0'); + assert(UtilsImpl::get_chars_len(@unsafe_state, 0) == 0, 'Should return 0'); // Should return 2 (be) - assert(Naming::InternalImpl::get_chars_len(@unsafe_state, 153) == 2, 'Should return 0'); + assert(UtilsImpl::get_chars_len(@unsafe_state, 153) == 2, 'Should return 0'); // Should return 4 ("toto") - assert(Naming::InternalImpl::get_chars_len(@unsafe_state, 796195) == 4, 'Should return 4'); + assert(UtilsImpl::get_chars_len(@unsafe_state, 796195) == 4, 'Should return 4'); // Should return 5 ("aloha") - assert(Naming::InternalImpl::get_chars_len(@unsafe_state, 77554770) == 5, 'Should return 5'); + assert(UtilsImpl::get_chars_len(@unsafe_state, 77554770) == 5, 'Should return 5'); // Should return 9 ("chocolate") - assert( - Naming::InternalImpl::get_chars_len(@unsafe_state, 19565965532212) == 9, 'Should return 9' - ); + assert(UtilsImpl::get_chars_len(@unsafe_state, 19565965532212) == 9, 'Should return 9'); // Should return 30 ("这来abcdefghijklmopqrstuvwyq1234") assert( - Naming::InternalImpl::get_chars_len( + UtilsImpl::get_chars_len( @unsafe_state, 801855144733576077820330221438165587969903898313 ) == 30, 'Should return 30' From 116833fc1f5f4eeb787cafbd5d26d9d5426dac8b Mon Sep 17 00:00:00 2001 From: Thomas Marchand Date: Mon, 9 Oct 2023 23:09:55 +0700 Subject: [PATCH 05/10] feat: update cairo to 2.3.0 --- Scarb.toml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Scarb.toml b/Scarb.toml index 9e0c8f7..26bc7d3 100644 --- a/Scarb.toml +++ b/Scarb.toml @@ -5,8 +5,8 @@ version = "0.1.0" # See more keys and their definitions at https://docs.swmansion.com/scarb/docs/reference/manifest [dependencies] -starknet = "2.2.0" -openzeppelin = { git = "https://github.com/OpenZeppelin/cairo-contracts.git", branch = "main" } +starknet = "2.3.0-rc0" +openzeppelin = { git = "https://github.com/andrew-fleming/cairo-contracts.git", branch = "component-erc20" } identity = { git = "https://github.com/starknet-id/identity.git", branch = "master" } [[target.starknet-contract]] From ac3cedc82aaa7341873609bd2467000344aae5b4 Mon Sep 17 00:00:00 2001 From: Thomas Marchand Date: Tue, 10 Oct 2023 16:53:44 +0700 Subject: [PATCH 06/10] fix: remove useless line --- src/tests/naming/common.cairo | 1 - 1 file changed, 1 deletion(-) diff --git a/src/tests/naming/common.cairo b/src/tests/naming/common.cairo index 007b7fe..1e42475 100644 --- a/src/tests/naming/common.cairo +++ b/src/tests/naming/common.cairo @@ -23,7 +23,6 @@ use naming::pricing::Pricing; fn deploy() -> (IERC20CamelDispatcher, IPricingDispatcher, IIdentityDispatcher, INamingDispatcher) { //erc20 - let mut calldata = ArrayTrait::::new(); // 0, 1 = low and high of ETH supply let eth = utils::deploy(ERC20::TEST_CLASS_HASH, array!['ether', 'ETH', 0, 1, 0x123]); From d514c2a4badaeaabaac8ea7df75586994b4c73c1 Mon Sep 17 00:00:00 2001 From: Thomas Marchand Date: Tue, 10 Oct 2023 16:54:16 +0700 Subject: [PATCH 07/10] feat: update scarb --- .github/workflows/test.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index a42fa6d..c65dd29 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -3,7 +3,7 @@ name: CI Tests on: [push, pull_request, pull_request_target] env: - SCARB_VERSION: 0.7.0 + SCARB_VERSION: 2.3.0-rc1 jobs: scarb-tests: From 2f127aa8f1261ff9e8e789b1c5b9fe06600af577 Mon Sep 17 00:00:00 2001 From: Thomas Marchand Date: Tue, 24 Oct 2023 14:23:14 +0700 Subject: [PATCH 08/10] feat: update to starknet 2.3.0 --- .github/workflows/test.yml | 2 +- .gitignore | 1 + Scarb.toml | 2 +- 3 files changed, 3 insertions(+), 2 deletions(-) diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index c65dd29..7c17e88 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -3,7 +3,7 @@ name: CI Tests on: [push, pull_request, pull_request_target] env: - SCARB_VERSION: 2.3.0-rc1 + SCARB_VERSION: 2.3.0 jobs: scarb-tests: diff --git a/.gitignore b/.gitignore index 37cebce..f5fc33f 100644 --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,4 @@ +Scarb.lock .env venv target diff --git a/Scarb.toml b/Scarb.toml index 26bc7d3..d42e352 100644 --- a/Scarb.toml +++ b/Scarb.toml @@ -6,7 +6,7 @@ version = "0.1.0" [dependencies] starknet = "2.3.0-rc0" -openzeppelin = { git = "https://github.com/andrew-fleming/cairo-contracts.git", branch = "component-erc20" } +openzeppelin = { git = "https://github.com/Th0rgal/cairo-contracts.git", branch = "main" } identity = { git = "https://github.com/starknet-id/identity.git", branch = "master" } [[target.starknet-contract]] From 3877dc1907817bfee053b8085e3ca7588a859a9a Mon Sep 17 00:00:00 2001 From: Thomas Marchand Date: Sat, 28 Oct 2023 12:14:06 +0700 Subject: [PATCH 09/10] fix: event emitter trait --- src/naming/asserts.cairo | 4 ++-- src/naming/internal.cairo | 5 ++--- 2 files changed, 4 insertions(+), 5 deletions(-) diff --git a/src/naming/asserts.cairo b/src/naming/asserts.cairo index f058a45..a1cbfde 100644 --- a/src/naming/asserts.cairo +++ b/src/naming/asserts.cairo @@ -8,8 +8,8 @@ use naming::{ naming::main::{ Naming, Naming::{ - EventEmitter, _hash_to_domain, _hash_to_domainContractMemberStateTrait, _domain_data, - _domain_dataContractMemberStateTrait, starknetid_contract, + ContractStateEventEmitter, _hash_to_domain, _hash_to_domainContractMemberStateTrait, + _domain_data, _domain_dataContractMemberStateTrait, starknetid_contract, starknetid_contractContractMemberStateTrait, discounts, discountsContractMemberStateTrait, _address_to_domain, _address_to_domainContractMemberStateTrait, _referral_contract, diff --git a/src/naming/internal.cairo b/src/naming/internal.cairo index f4aefd8..a5d2bf1 100644 --- a/src/naming/internal.cairo +++ b/src/naming/internal.cairo @@ -8,8 +8,8 @@ use naming::{ naming::main::{ Naming, Naming::{ - EventEmitter, _hash_to_domain, _hash_to_domainContractMemberStateTrait, _domain_data, - _domain_dataContractMemberStateTrait, starknetid_contract, + ContractStateEventEmitter, _hash_to_domain, _hash_to_domainContractMemberStateTrait, + _domain_data, _domain_dataContractMemberStateTrait, starknetid_contract, starknetid_contractContractMemberStateTrait, discounts, discountsContractMemberStateTrait, _address_to_domain, _address_to_domainContractMemberStateTrait, _referral_contract, @@ -29,7 +29,6 @@ use naming::naming::utils::UtilsTrait; #[generate_trait] impl InternalImpl of InternalTrait { - fn read_address_to_domain( self: @Naming::ContractState, address: ContractAddress, ref domain: Array ) -> usize { From 62ec65474eba824d684b7487f1b4cb104e9962e0 Mon Sep 17 00:00:00 2001 From: Thomas Marchand Date: Sat, 28 Oct 2023 16:31:48 +0700 Subject: [PATCH 10/10] feat: update openzeppelin repo --- Scarb.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Scarb.toml b/Scarb.toml index d42e352..42ac34c 100644 --- a/Scarb.toml +++ b/Scarb.toml @@ -6,7 +6,7 @@ version = "0.1.0" [dependencies] starknet = "2.3.0-rc0" -openzeppelin = { git = "https://github.com/Th0rgal/cairo-contracts.git", branch = "main" } +openzeppelin = { git = "https://github.com/OpenZeppelin/cairo-contracts.git", tag = "v0.8.0-beta.0" } identity = { git = "https://github.com/starknet-id/identity.git", branch = "master" } [[target.starknet-contract]]