From cb98423aeb49f60c6b858c71e49e46215ce23ad1 Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Wed, 7 Dec 2022 22:12:42 -0500 Subject: [PATCH 01/16] Adds reverse lookup support (primary name) --- core/sources/domains.move | 82 ++++++++++++++++++++++++++++++++------- 1 file changed, 69 insertions(+), 13 deletions(-) diff --git a/core/sources/domains.move b/core/sources/domains.move index 95f13c98..9172497c 100644 --- a/core/sources/domains.move +++ b/core/sources/domains.move @@ -71,6 +71,14 @@ module aptos_names::domains { registry: Table, } + struct ReverseLookupRegistryV1 has key, store { + registry: Table + } + + struct SetReverseLookupEventsV1 has key, store { + set_reverse_lookup_events: event::EventHandle, + } + /// Holder for `SetNameAddressEventV1` events struct SetNameAddressEventsV1 has key, store { set_name_events: event::EventHandle, @@ -81,6 +89,9 @@ module aptos_names::domains { register_name_events: event::EventHandle, } + struct SetReverseLookupEventV1 has drop, store { + } + /// A name (potentially subdomain) has had it's address changed /// This could be to a new address, or it could have been cleared struct SetNameAddressEventV1 has drop, store { @@ -135,7 +146,7 @@ module aptos_names::domains { token_helper::initialize(account); } - fun register_domain_generic(sign: &signer, domain_name: String, num_years: u8) acquires NameRegistryV1, RegisterNameEventsV1, SetNameAddressEventsV1 { + fun register_domain_generic(sign: &signer, domain_name: String, num_years: u8) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { assert!(config::is_enabled(), error::unavailable(ENOT_ENABLED)); assert!(num_years > 0 && num_years <= config::max_number_of_years_registered(), error::out_of_range(EINVALID_NUMBER_YEARS)); @@ -155,18 +166,16 @@ module aptos_names::domains { coin::transfer(sign, config::fund_destination_address(), price); register_name_internal(sign, subdomain_name, domain_name, registration_duration_secs, price); - // Automatically set the name to point to the sender's address - set_name_address_internal(subdomain_name, domain_name, signer::address_of(sign)); } /// A wrapper around `register_name` as an entry function. /// Option is not currently serializable, so we have these convenience methods - public entry fun register_domain(sign: &signer, domain_name: String, num_years: u8) acquires NameRegistryV1, RegisterNameEventsV1, SetNameAddressEventsV1 { + public entry fun register_domain(sign: &signer, domain_name: String, num_years: u8) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { assert!(config::unrestricted_mint_enabled(), error::permission_denied(EVALID_SIGNATURE_REQUIRED)); register_domain_generic(sign, domain_name, num_years); } - public entry fun register_domain_with_signature(sign: &signer, domain_name: String, num_years: u8, signature: vector) acquires NameRegistryV1, RegisterNameEventsV1, SetNameAddressEventsV1 { + public entry fun register_domain_with_signature(sign: &signer, domain_name: String, num_years: u8, signature: vector) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { let account_address = signer::address_of(sign); verify::assert_register_domain_signature_verifies(signature, account_address, domain_name); register_domain_generic(sign, domain_name, num_years); @@ -175,7 +184,7 @@ module aptos_names::domains { /// A wrapper around `register_name` as an entry function. /// Option is not currently serializable, so we have these convenience method /// `expiration_time_sec` is the timestamp, in seconds, when the name expires - public entry fun register_subdomain(sign: &signer, subdomain_name: String, domain_name: String, expiration_time_sec: u64) acquires NameRegistryV1, RegisterNameEventsV1 { + public entry fun register_subdomain(sign: &signer, subdomain_name: String, domain_name: String, expiration_time_sec: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { assert!(config::is_enabled(), error::unavailable(ENOT_ENABLED)); assert!(name_is_registerable(option::some(subdomain_name), domain_name), error::invalid_state(ENAME_NOT_AVAILABLE)); @@ -203,7 +212,7 @@ module aptos_names::domains { /// For domains, the registration duration is only allowed to be in increments of 1 year, for now /// Since the owner of the domain is the only one that can create the subdomain, we allow them to decide how long they want the underlying registration to be /// The maximum subdomain registration duration is limited to the duration of its parent domain registration - fun register_name_internal(sign: &signer, subdomain_name: Option, domain_name: String, registration_duration_secs: u64, price: u64) acquires NameRegistryV1, RegisterNameEventsV1 { + fun register_name_internal(sign: &signer, subdomain_name: Option, domain_name: String, registration_duration_secs: u64, price: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { let aptos_names = borrow_global_mut(@aptos_names); let name_expiration_time_secs = timestamp::now_seconds() + registration_duration_secs; @@ -232,6 +241,16 @@ module aptos_names::domains { table::upsert(&mut aptos_names.registry, name_record_key, name_record); + let account_addr = signer::address_of(sign); + let reverse_lookup_result = get_reverse_lookup(account_addr); + if (option::is_none(&reverse_lookup_result)) { + // If the user has no reverse lookup set, set the user's reverse lookup. This also points the address to the sender. + set_reverse_lookup(sign, &NameRecordKeyV1 { subdomain_name: option::none(), domain_name }); + } else { + // Automatically set the name to point to the sender's address + set_name_address_internal(subdomain_name, domain_name, account_addr); + }; + event::emit_event( &mut borrow_global_mut(@aptos_names).register_name_events, RegisterNameEventV1 { @@ -265,15 +284,15 @@ module aptos_names::domains { /// The `registration_duration_secs` parameter is the number of seconds to register the domain for, but is not limited to the maximum set in the config for domains registered normally. /// This allows, for example, to create a domain for the system address for 100 years so we don't need to worry about expiry /// Or for moderation purposes, it allows us to seize a racist/harassing domain for 100 years, and park it somewhere safe - public entry fun force_create_or_seize_domain_name(sign: &signer, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1 { + public entry fun force_create_or_seize_domain_name(sign: &signer, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { force_create_or_seize_name(sign, option::none(), domain_name, registration_duration_secs); } - public entry fun force_create_or_seize_subdomain_name(sign: &signer, subdomain_name: String, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1 { + public entry fun force_create_or_seize_subdomain_name(sign: &signer, subdomain_name: String, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { force_create_or_seize_name(sign, option::some(subdomain_name), domain_name, registration_duration_secs); } - public fun force_create_or_seize_name(sign: &signer, subdomain_name: Option, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1 { + public fun force_create_or_seize_name(sign: &signer, subdomain_name: Option, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { config::assert_signer_is_admin(sign); register_name_internal(sign, subdomain_name, domain_name, registration_duration_secs, 0); } @@ -361,16 +380,16 @@ module aptos_names::domains { } } - public entry fun set_domain_address(sign: &signer, domain_name: String, new_address: address) acquires NameRegistryV1, SetNameAddressEventsV1 { + public entry fun set_domain_address(sign: &signer, domain_name: String, new_address: address) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { set_name_address(sign, option::none(), domain_name, new_address); } - public entry fun set_subdomain_address(sign: &signer, subdomain_name: String, domain_name: String, new_address: address) acquires NameRegistryV1, SetNameAddressEventsV1 { + public entry fun set_subdomain_address(sign: &signer, subdomain_name: String, domain_name: String, new_address: address) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { set_name_address(sign, option::some(subdomain_name), domain_name, new_address); } - public fun set_name_address(sign: &signer, subdomain_name: Option, domain_name: String, new_address: address) acquires NameRegistryV1, SetNameAddressEventsV1 { + public fun set_name_address(sign: &signer, subdomain_name: Option, domain_name: String, new_address: address) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { let signer_addr = signer::address_of(sign); let (is_owner, token_id) = is_owner_of_name(signer_addr, subdomain_name, domain_name); assert!(is_owner, error::permission_denied(ENOT_OWNER_OF_NAME)); @@ -379,6 +398,22 @@ module aptos_names::domains { let (_property_version, expiration_time_sec, _target_address) = get_name_record_v1_props(&name_record); let (property_keys, property_values, property_types) = get_name_property_map(subdomain_name, expiration_time_sec); token_helper::set_token_props(signer_addr, property_keys, property_values, property_types, token_id); + + // If the signer's reverse lookup is the domain, and the new address is not the signer, clear the signer's reverse lookup. + // Example: + // The current state is bob.apt points to @a and the reverse lookup of @a points to bob.apt. + // The owner wants to set bob.apt to point to @b. + // The new state should be bob.apt points to @b, and the reverse lookup of @a should be none. + let maybe_reverse_lookup = get_reverse_lookup(signer_addr); + if (option::is_none(&maybe_reverse_lookup)) { + return + }; + let current_reverse_lookup = option::borrow(&maybe_reverse_lookup); + let key = NameRecordKeyV1 { subdomain_name, domain_name }; + if (*current_reverse_lookup == key && signer_addr != new_address) { + let registry = &mut borrow_global_mut(@aptos_names).registry; + table::remove(registry, signer_addr); + }; } fun set_name_address_internal(subdomain_name: Option, domain_name: String, new_address: address): NameRecordV1 acquires NameRegistryV1, SetNameAddressEventsV1 { @@ -438,6 +473,27 @@ module aptos_names::domains { }; } + public fun set_reverse_lookup(account: &signer, key: &NameRecordKeyV1) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { + let account_addr = signer::address_of(account); + let (maybe_subdomain_name, domain_name) = get_name_record_key_v1_props(key); + let (is_owner, _) = is_owner_of_name(account_addr, maybe_subdomain_name, domain_name); + assert!(is_owner, ENOT_AUTHORIZED); + + let registry = &mut borrow_global_mut(@aptos_names).registry; + table::upsert(registry, account_addr, *key); + + set_name_address(account, maybe_subdomain_name, domain_name, account_addr); + } + + public fun get_reverse_lookup(account_addr: address): Option acquires ReverseLookupRegistryV1 { + let registry = &borrow_global_mut(@aptos_names).registry; + if (table::contains(registry, account_addr)) { + option::some(*table::borrow(registry, account_addr)) + } else { + option::none() + } + } + fun emit_set_name_address_event_v1(subdomain_name: Option, domain_name: String, property_version: u64, expiration_time_secs: u64, new_address: Option
) acquires SetNameAddressEventsV1 { let event = SetNameAddressEventV1 { subdomain_name, From e0187e67e66e572f73ef31be9ff88a99f572447c Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Thu, 8 Dec 2022 13:42:28 -0500 Subject: [PATCH 02/16] Refactor & test setup --- core/sources/domains.move | 35 +++++++++++++++++++++++++---------- core/sources/test_helper.move | 22 +++++++++++++++++----- 2 files changed, 42 insertions(+), 15 deletions(-) diff --git a/core/sources/domains.move b/core/sources/domains.move index 9172497c..22e9648b 100644 --- a/core/sources/domains.move +++ b/core/sources/domains.move @@ -146,6 +146,14 @@ module aptos_names::domains { token_helper::initialize(account); } + public fun init_reverse_lookup_registry_v1(account: &signer) { + assert!(signer::address_of(account) == @aptos_names, ENOT_AUTHORIZED); + + move_to(account, ReverseLookupRegistryV1 { + registry: table::new() + }); + } + fun register_domain_generic(sign: &signer, domain_name: String, num_years: u8) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { assert!(config::is_enabled(), error::unavailable(ENOT_ENABLED)); assert!(num_years > 0 && num_years <= config::max_number_of_years_registered(), error::out_of_range(EINVALID_NUMBER_YEARS)); @@ -244,11 +252,11 @@ module aptos_names::domains { let account_addr = signer::address_of(sign); let reverse_lookup_result = get_reverse_lookup(account_addr); if (option::is_none(&reverse_lookup_result)) { - // If the user has no reverse lookup set, set the user's reverse lookup. This also points the address to the sender. - set_reverse_lookup(sign, &NameRecordKeyV1 { subdomain_name: option::none(), domain_name }); - } else { + // If the user has no reverse lookup set, set the user's reverse lookup. + set_reverse_lookup(sign, &NameRecordKeyV1 { subdomain_name, domain_name }); + } else if (option::is_none(&subdomain_name)) { // Automatically set the name to point to the sender's address - set_name_address_internal(subdomain_name, domain_name, account_addr); + set_name_address_internal(subdomain_name, domain_name, signer::address_of(sign)); }; event::emit_event( @@ -474,14 +482,10 @@ module aptos_names::domains { } public fun set_reverse_lookup(account: &signer, key: &NameRecordKeyV1) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { + set_reverse_lookup_internal(account, key); + let account_addr = signer::address_of(account); let (maybe_subdomain_name, domain_name) = get_name_record_key_v1_props(key); - let (is_owner, _) = is_owner_of_name(account_addr, maybe_subdomain_name, domain_name); - assert!(is_owner, ENOT_AUTHORIZED); - - let registry = &mut borrow_global_mut(@aptos_names).registry; - table::upsert(registry, account_addr, *key); - set_name_address(account, maybe_subdomain_name, domain_name, account_addr); } @@ -494,6 +498,16 @@ module aptos_names::domains { } } + fun set_reverse_lookup_internal(account: &signer, key: &NameRecordKeyV1) acquires ReverseLookupRegistryV1 { + let account_addr = signer::address_of(account); + let (maybe_subdomain_name, domain_name) = get_name_record_key_v1_props(key); + let (is_owner, _) = is_owner_of_name(account_addr, maybe_subdomain_name, domain_name); + assert!(is_owner, ENOT_AUTHORIZED); + + let registry = &mut borrow_global_mut(@aptos_names).registry; + table::upsert(registry, account_addr, *key); + } + fun emit_set_name_address_event_v1(subdomain_name: Option, domain_name: String, property_version: u64, expiration_time_secs: u64, new_address: Option
) acquires SetNameAddressEventsV1 { let event = SetNameAddressEventV1 { subdomain_name, @@ -555,6 +569,7 @@ module aptos_names::domains { #[test_only] public fun init_module_for_test(account: &signer) { init_module(account); + init_reverse_lookup_registry_v1(account); } #[test_only] diff --git a/core/sources/test_helper.move b/core/sources/test_helper.move index c76a75a7..99d15f64 100644 --- a/core/sources/test_helper.move +++ b/core/sources/test_helper.move @@ -66,6 +66,7 @@ module aptos_names::test_helper { let user_balance_before = coin::balance(user_addr); let register_name_event_v1_event_count_before = domains::get_register_name_event_v1_count(); let set_name_address_event_v1_event_count_before = domains::get_set_name_address_event_v1_count(); + let user_reverse_lookup_before = domains::get_reverse_lookup(user_addr); let years = (time_helper::seconds_to_years(registration_duration_secs) as u8); if (option::is_none(&subdomain_name)) { @@ -113,8 +114,13 @@ module aptos_names::test_helper { assert!(time_helper::seconds_to_days(expiration_time_sec - timestamp::now_seconds()) == 365, 10); if (is_subdomain) { - // We haven't set a target address yet! - assert!(target_address == option::none(), 11); + if (option::is_none(&user_reverse_lookup_before)) { + // Should automatically point to the users address + assert!(target_address == option::some(user_addr), 11); + } else { + // We haven't set a target address yet! + assert!(target_address == option::none(), 11); + } } else { // Should automatically point to the users address assert!(target_address == option::some(user_addr), 11); @@ -139,9 +145,15 @@ module aptos_names::test_helper { assert!(register_name_event_v1_num_emitted == 1, register_name_event_v1_num_emitted); if (is_subdomain) { - // We haven't set a target address yet! - test_utils::print_actual_expected(b"set_name_address_event_v1_num_emitted: ", set_name_address_event_v1_num_emitted, 0, false); - assert!(set_name_address_event_v1_num_emitted == 0, set_name_address_event_v1_num_emitted); + if (option::is_none(&user_reverse_lookup_before)) { + // Should automatically point to the users address + test_utils::print_actual_expected(b"set_name_address_event_v1_num_emitted: ", set_name_address_event_v1_num_emitted, 1, false); + assert!(set_name_address_event_v1_num_emitted == 1, set_name_address_event_v1_num_emitted); + } else { + // We haven't set a target address yet! + test_utils::print_actual_expected(b"set_name_address_event_v1_num_emitted: ", set_name_address_event_v1_num_emitted, 0, false); + assert!(set_name_address_event_v1_num_emitted == 0, set_name_address_event_v1_num_emitted); + } } else { // Should automatically point to the users address test_utils::print_actual_expected(b"set_name_address_event_v1_num_emitted: ", set_name_address_event_v1_num_emitted, 1, false); From 2cf7df4fdb0440ad61591d10c5754a87147751fd Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Thu, 15 Dec 2022 09:59:19 -0500 Subject: [PATCH 03/16] Add events --- core/sources/domains.move | 46 ++++++++++++++++++++++++++++++--------- 1 file changed, 36 insertions(+), 10 deletions(-) diff --git a/core/sources/domains.move b/core/sources/domains.move index 22e9648b..4ad9a679 100644 --- a/core/sources/domains.move +++ b/core/sources/domains.move @@ -90,6 +90,9 @@ module aptos_names::domains { } struct SetReverseLookupEventV1 has drop, store { + subdomain_name: Option, + domain_name: String, + target_address: address, } /// A name (potentially subdomain) has had it's address changed @@ -152,9 +155,13 @@ module aptos_names::domains { move_to(account, ReverseLookupRegistryV1 { registry: table::new() }); + + move_to(account, SetReverseLookupEventsV1 { + set_reverse_lookup_events: account::new_event_handle(account), + }); } - fun register_domain_generic(sign: &signer, domain_name: String, num_years: u8) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { + fun register_domain_generic(sign: &signer, domain_name: String, num_years: u8) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { assert!(config::is_enabled(), error::unavailable(ENOT_ENABLED)); assert!(num_years > 0 && num_years <= config::max_number_of_years_registered(), error::out_of_range(EINVALID_NUMBER_YEARS)); @@ -178,12 +185,12 @@ module aptos_names::domains { /// A wrapper around `register_name` as an entry function. /// Option is not currently serializable, so we have these convenience methods - public entry fun register_domain(sign: &signer, domain_name: String, num_years: u8) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { + public entry fun register_domain(sign: &signer, domain_name: String, num_years: u8) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { assert!(config::unrestricted_mint_enabled(), error::permission_denied(EVALID_SIGNATURE_REQUIRED)); register_domain_generic(sign, domain_name, num_years); } - public entry fun register_domain_with_signature(sign: &signer, domain_name: String, num_years: u8, signature: vector) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { + public entry fun register_domain_with_signature(sign: &signer, domain_name: String, num_years: u8, signature: vector) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { let account_address = signer::address_of(sign); verify::assert_register_domain_signature_verifies(signature, account_address, domain_name); register_domain_generic(sign, domain_name, num_years); @@ -192,7 +199,7 @@ module aptos_names::domains { /// A wrapper around `register_name` as an entry function. /// Option is not currently serializable, so we have these convenience method /// `expiration_time_sec` is the timestamp, in seconds, when the name expires - public entry fun register_subdomain(sign: &signer, subdomain_name: String, domain_name: String, expiration_time_sec: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { + public entry fun register_subdomain(sign: &signer, subdomain_name: String, domain_name: String, expiration_time_sec: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { assert!(config::is_enabled(), error::unavailable(ENOT_ENABLED)); assert!(name_is_registerable(option::some(subdomain_name), domain_name), error::invalid_state(ENAME_NOT_AVAILABLE)); @@ -220,7 +227,7 @@ module aptos_names::domains { /// For domains, the registration duration is only allowed to be in increments of 1 year, for now /// Since the owner of the domain is the only one that can create the subdomain, we allow them to decide how long they want the underlying registration to be /// The maximum subdomain registration duration is limited to the duration of its parent domain registration - fun register_name_internal(sign: &signer, subdomain_name: Option, domain_name: String, registration_duration_secs: u64, price: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { + fun register_name_internal(sign: &signer, subdomain_name: Option, domain_name: String, registration_duration_secs: u64, price: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { let aptos_names = borrow_global_mut(@aptos_names); let name_expiration_time_secs = timestamp::now_seconds() + registration_duration_secs; @@ -292,15 +299,15 @@ module aptos_names::domains { /// The `registration_duration_secs` parameter is the number of seconds to register the domain for, but is not limited to the maximum set in the config for domains registered normally. /// This allows, for example, to create a domain for the system address for 100 years so we don't need to worry about expiry /// Or for moderation purposes, it allows us to seize a racist/harassing domain for 100 years, and park it somewhere safe - public entry fun force_create_or_seize_domain_name(sign: &signer, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { + public entry fun force_create_or_seize_domain_name(sign: &signer, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { force_create_or_seize_name(sign, option::none(), domain_name, registration_duration_secs); } - public entry fun force_create_or_seize_subdomain_name(sign: &signer, subdomain_name: String, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { + public entry fun force_create_or_seize_subdomain_name(sign: &signer, subdomain_name: String, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { force_create_or_seize_name(sign, option::some(subdomain_name), domain_name, registration_duration_secs); } - public fun force_create_or_seize_name(sign: &signer, subdomain_name: Option, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { + public fun force_create_or_seize_name(sign: &signer, subdomain_name: Option, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { config::assert_signer_is_admin(sign); register_name_internal(sign, subdomain_name, domain_name, registration_duration_secs, 0); } @@ -481,7 +488,7 @@ module aptos_names::domains { }; } - public fun set_reverse_lookup(account: &signer, key: &NameRecordKeyV1) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { + public fun set_reverse_lookup(account: &signer, key: &NameRecordKeyV1) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { set_reverse_lookup_internal(account, key); let account_addr = signer::address_of(account); @@ -498,7 +505,7 @@ module aptos_names::domains { } } - fun set_reverse_lookup_internal(account: &signer, key: &NameRecordKeyV1) acquires ReverseLookupRegistryV1 { + fun set_reverse_lookup_internal(account: &signer, key: &NameRecordKeyV1) acquires ReverseLookupRegistryV1, SetReverseLookupEventsV1 { let account_addr = signer::address_of(account); let (maybe_subdomain_name, domain_name) = get_name_record_key_v1_props(key); let (is_owner, _) = is_owner_of_name(account_addr, maybe_subdomain_name, domain_name); @@ -506,6 +513,12 @@ module aptos_names::domains { let registry = &mut borrow_global_mut(@aptos_names).registry; table::upsert(registry, account_addr, *key); + + emit_set_reverse_lookup_event_v1( + maybe_subdomain_name, + domain_name, + account_addr + ); } fun emit_set_name_address_event_v1(subdomain_name: Option, domain_name: String, property_version: u64, expiration_time_secs: u64, new_address: Option
) acquires SetNameAddressEventsV1 { @@ -523,6 +536,19 @@ module aptos_names::domains { ); } + fun emit_set_reverse_lookup_event_v1(subdomain_name: Option, domain_name: String, target_address: address) acquires SetReverseLookupEventsV1 { + let event = SetReverseLookupEventV1 { + subdomain_name, + domain_name, + target_address, + }; + + event::emit_event( + &mut borrow_global_mut(@aptos_names).set_reverse_lookup_events, + event, + ); + } + public fun get_name_property_map(subdomain_name: Option, expiration_time_sec: u64): (vector, vector>, vector) { let type; if (option::is_some(&subdomain_name)) { From 23c0512bf33e69a33e1f25eae6b2b779313d7204 Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Thu, 15 Dec 2022 10:07:23 -0500 Subject: [PATCH 04/16] Adds tests for set reverse lookup event --- core/sources/domains.move | 5 +++++ core/sources/test_helper.move | 11 ++++++++++- 2 files changed, 15 insertions(+), 1 deletion(-) diff --git a/core/sources/domains.move b/core/sources/domains.move index 4ad9a679..f67fe101 100644 --- a/core/sources/domains.move +++ b/core/sources/domains.move @@ -608,6 +608,11 @@ module aptos_names::domains { event::counter(&borrow_global(@aptos_names).register_name_events) } + #[test_only] + public fun get_set_reverse_lookup_event_v1_count(): u64 acquires SetReverseLookupEventsV1 { + event::counter(&borrow_global(@aptos_names).set_reverse_lookup_events) + } + #[test(aptos = @0x1)] fun test_time_is_expired(aptos: &signer) { timestamp::set_time_has_started_for_testing(aptos); diff --git a/core/sources/test_helper.move b/core/sources/test_helper.move index 99d15f64..657b15e8 100644 --- a/core/sources/test_helper.move +++ b/core/sources/test_helper.move @@ -64,9 +64,10 @@ module aptos_names::test_helper { let is_subdomain = option::is_some(&subdomain_name); let user_balance_before = coin::balance(user_addr); + let user_reverse_lookup_before = domains::get_reverse_lookup(user_addr); let register_name_event_v1_event_count_before = domains::get_register_name_event_v1_count(); let set_name_address_event_v1_event_count_before = domains::get_set_name_address_event_v1_count(); - let user_reverse_lookup_before = domains::get_reverse_lookup(user_addr); + let set_reverse_lookup_event_v1_event_count_before = domains::get_set_reverse_lookup_event_v1_count(); let years = (time_helper::seconds_to_years(registration_duration_secs) as u8); if (option::is_none(&subdomain_name)) { @@ -140,10 +141,18 @@ module aptos_names::test_helper { // Assert events have been correctly emmitted let register_name_event_v1_num_emitted = domains::get_register_name_event_v1_count() - register_name_event_v1_event_count_before; let set_name_address_event_v1_num_emitted = domains::get_set_name_address_event_v1_count() - set_name_address_event_v1_event_count_before; + let set_reverse_lookup_event_v1_num_emitted = domains::get_set_reverse_lookup_event_v1_count() - set_reverse_lookup_event_v1_event_count_before; test_utils::print_actual_expected(b"register_name_event_v1_num_emitted: ", register_name_event_v1_num_emitted, 1, false); assert!(register_name_event_v1_num_emitted == 1, register_name_event_v1_num_emitted); + // Reverse lookup should be set if user did not have one before + if (option::is_none(&user_reverse_lookup_before)) { + assert!(set_reverse_lookup_event_v1_num_emitted == 1, set_reverse_lookup_event_v1_num_emitted); + } else { + assert!(set_reverse_lookup_event_v1_num_emitted == 0, set_reverse_lookup_event_v1_num_emitted); + }; + if (is_subdomain) { if (option::is_none(&user_reverse_lookup_before)) { // Should automatically point to the users address From 2841de3b8e02e1dcd5fb91ffd256888fa7226da3 Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Thu, 15 Dec 2022 10:21:40 -0500 Subject: [PATCH 05/16] More tests --- core/sources/domains.move | 17 +++++++++++------ core/sources/test_helper.move | 21 ++++++++++++++++++++- 2 files changed, 31 insertions(+), 7 deletions(-) diff --git a/core/sources/domains.move b/core/sources/domains.move index f67fe101..6f1a6832 100644 --- a/core/sources/domains.move +++ b/core/sources/domains.move @@ -92,7 +92,7 @@ module aptos_names::domains { struct SetReverseLookupEventV1 has drop, store { subdomain_name: Option, domain_name: String, - target_address: address, + target_address: Option
, } /// A name (potentially subdomain) has had it's address changed @@ -395,16 +395,16 @@ module aptos_names::domains { } } - public entry fun set_domain_address(sign: &signer, domain_name: String, new_address: address) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { + public entry fun set_domain_address(sign: &signer, domain_name: String, new_address: address) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { set_name_address(sign, option::none(), domain_name, new_address); } - public entry fun set_subdomain_address(sign: &signer, subdomain_name: String, domain_name: String, new_address: address) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { + public entry fun set_subdomain_address(sign: &signer, subdomain_name: String, domain_name: String, new_address: address) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { set_name_address(sign, option::some(subdomain_name), domain_name, new_address); } - public fun set_name_address(sign: &signer, subdomain_name: Option, domain_name: String, new_address: address) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1 { + public fun set_name_address(sign: &signer, subdomain_name: Option, domain_name: String, new_address: address) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { let signer_addr = signer::address_of(sign); let (is_owner, token_id) = is_owner_of_name(signer_addr, subdomain_name, domain_name); assert!(is_owner, error::permission_denied(ENOT_OWNER_OF_NAME)); @@ -428,6 +428,11 @@ module aptos_names::domains { if (*current_reverse_lookup == key && signer_addr != new_address) { let registry = &mut borrow_global_mut(@aptos_names).registry; table::remove(registry, signer_addr); + emit_set_reverse_lookup_event_v1( + subdomain_name, + domain_name, + option::none() + ); }; } @@ -517,7 +522,7 @@ module aptos_names::domains { emit_set_reverse_lookup_event_v1( maybe_subdomain_name, domain_name, - account_addr + option::some(account_addr) ); } @@ -536,7 +541,7 @@ module aptos_names::domains { ); } - fun emit_set_reverse_lookup_event_v1(subdomain_name: Option, domain_name: String, target_address: address) acquires SetReverseLookupEventsV1 { + fun emit_set_reverse_lookup_event_v1(subdomain_name: Option, domain_name: String, target_address: Option
) acquires SetReverseLookupEventsV1 { let event = SetReverseLookupEventV1 { subdomain_name, domain_name, diff --git a/core/sources/test_helper.move b/core/sources/test_helper.move index 657b15e8..9abb5925 100644 --- a/core/sources/test_helper.move +++ b/core/sources/test_helper.move @@ -101,7 +101,7 @@ module aptos_names::test_helper { if (is_subdomain) { // If it's a subdomain, we only charge a nomincal fee expected_user_balance_after = user_balance_before - price_model::price_for_subdomain_v1(registration_duration_secs); - }else { + } else { let domain_price = price_model::price_for_domain_v1(string::length(&domain_name), years); assert!(domain_price / config::octas() == 40, domain_price / config::octas()); expected_user_balance_after = user_balance_before - domain_price; @@ -172,23 +172,42 @@ module aptos_names::test_helper { /// Set the domain address, and verify the address was set correctly public fun set_name_address(user: &signer, subdomain_name: Option, domain_name: String, expected_target_address: address) { + let user_addr = signer::address_of(user); + let register_name_event_v1_event_count_before = domains::get_register_name_event_v1_count(); let set_name_address_event_v1_event_count_before = domains::get_set_name_address_event_v1_count(); + let set_reverse_lookup_event_v1_event_count_before = domains::get_set_reverse_lookup_event_v1_count(); + let maybe_reverse_lookup_before = domains::get_reverse_lookup(user_addr); domains::set_name_address(user, subdomain_name, domain_name, expected_target_address); let (_property_version, _expiration_time_sec, target_address) = domains::get_name_record_v1_props_for_name(subdomain_name, domain_name); test_utils::print_actual_expected(b"set_domain_address: ", target_address, option::some(expected_target_address), false); assert!(target_address == option::some(expected_target_address), 33); + // When setting the target address to an address that is *not* the owner's, the reverse lookup should also be cleared + if (signer::address_of(user) != expected_target_address) { + let maybe_reverse_lookup = domains::get_reverse_lookup(user_addr); + assert!(option::is_none(&maybe_reverse_lookup), 33); + }; + // Assert events have been correctly emmitted let register_name_event_v1_num_emitted = domains::get_register_name_event_v1_count() - register_name_event_v1_event_count_before; let set_name_address_event_v1_num_emitted = domains::get_set_name_address_event_v1_count() - set_name_address_event_v1_event_count_before; + let set_reverse_lookup_event_v1_num_emitted = domains::get_set_reverse_lookup_event_v1_count() - set_reverse_lookup_event_v1_event_count_before; test_utils::print_actual_expected(b"register_name_event_v1_num_emitted: ", register_name_event_v1_num_emitted, 0, false); assert!(register_name_event_v1_num_emitted == 0, register_name_event_v1_num_emitted); test_utils::print_actual_expected(b"set_name_address_event_v1_num_emitted: ", set_name_address_event_v1_num_emitted, 1, false); assert!(set_name_address_event_v1_num_emitted == 1, set_name_address_event_v1_num_emitted); + + // If the signer had a reverse lookup before, and set his reverse lookup name to a different address, it should be cleared + if (option::is_some(&maybe_reverse_lookup_before)) { + let (maybe_reverse_subdomain, reverse_domain) = domains::get_name_record_key_v1_props(option::borrow(&maybe_reverse_lookup_before)); + if (maybe_reverse_subdomain == subdomain_name && reverse_domain == domain_name && signer::address_of(user) != expected_target_address) { + assert!(set_reverse_lookup_event_v1_num_emitted == 1, set_reverse_lookup_event_v1_num_emitted); + }; + }; } /// Clear the domain address, and verify the address was cleared From 4b4029b4925ae2f821a30289c56956dad3ab278a Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Thu, 15 Dec 2022 10:24:25 -0500 Subject: [PATCH 06/16] Uses mainnet branch --- core/Move.toml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/core/Move.toml b/core/Move.toml index 6c2a4eae..db15d9c9 100644 --- a/core/Move.toml +++ b/core/Move.toml @@ -9,10 +9,10 @@ aptos_names_funds = "0x78ee3915e67ef5d19fa91d1e05e60ae08751efd12ce58e23fc1109de8 [dependencies.AptosFramework] git = 'https://github.com/aptos-labs/aptos-core.git' -rev = 'main' +rev = 'mainnet' subdir = 'aptos-move/framework/aptos-framework' [dependencies.AptosToken] git = 'https://github.com/aptos-labs/aptos-core.git' -rev = 'main' +rev = 'mainnet' subdir = 'aptos-move/framework/aptos-token' From ec036e7a3c87c3e7ca0ac804963657e213d7f31d Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Fri, 16 Dec 2022 09:32:35 -0500 Subject: [PATCH 07/16] Adds entry fn --- core/sources/domains.move | 21 ++++++++++++++++++--- 1 file changed, 18 insertions(+), 3 deletions(-) diff --git a/core/sources/domains.move b/core/sources/domains.move index 6f1a6832..4611cb7b 100644 --- a/core/sources/domains.move +++ b/core/sources/domains.move @@ -17,7 +17,7 @@ module aptos_names::domains { use std::error; use std::option::{Self, Option}; use std::signer; - use std::string::String; + use std::string::{Self, String}; /// The Naming Service contract is not enabled const ENOT_ENABLED: u64 = 1; @@ -150,7 +150,7 @@ module aptos_names::domains { } public fun init_reverse_lookup_registry_v1(account: &signer) { - assert!(signer::address_of(account) == @aptos_names, ENOT_AUTHORIZED); + assert!(signer::address_of(account) == @aptos_names, error::permission_denied(ENOT_AUTHORIZED)); move_to(account, ReverseLookupRegistryV1 { registry: table::new() @@ -493,6 +493,21 @@ module aptos_names::domains { }; } + /// Entry function for |set_reverse_lookup|. + public entry fun set_reverse_lookup_entry(account: &signer, subdomain_name: String, domain_name: String) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { + let key = NameRecordKeyV1 { + subdomain_name: if (string::length(&subdomain_name) > 0) { + option::some(subdomain_name) + } else { + option::none() + }, + domain_name + }; + set_reverse_lookup(account, &key); + } + + /// Sets the |account|'s reverse lookup, aka "primary name". This allows a user to specify which of their Aptos Names + /// is their "primary", so that dapps can display the user's primary name rather than their address. public fun set_reverse_lookup(account: &signer, key: &NameRecordKeyV1) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { set_reverse_lookup_internal(account, key); @@ -514,7 +529,7 @@ module aptos_names::domains { let account_addr = signer::address_of(account); let (maybe_subdomain_name, domain_name) = get_name_record_key_v1_props(key); let (is_owner, _) = is_owner_of_name(account_addr, maybe_subdomain_name, domain_name); - assert!(is_owner, ENOT_AUTHORIZED); + assert!(is_owner, error::permission_denied(ENOT_AUTHORIZED)); let registry = &mut borrow_global_mut(@aptos_names).registry; table::upsert(registry, account_addr, *key); From 9fe81ec1f4a20977149960c7d1759e1ef5c5b871 Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Fri, 16 Dec 2022 09:36:23 -0500 Subject: [PATCH 08/16] Adds more docstring --- core/sources/domains.move | 1 + 1 file changed, 1 insertion(+) diff --git a/core/sources/domains.move b/core/sources/domains.move index 4611cb7b..b2c58af9 100644 --- a/core/sources/domains.move +++ b/core/sources/domains.move @@ -516,6 +516,7 @@ module aptos_names::domains { set_name_address(account, maybe_subdomain_name, domain_name, account_addr); } + /// Returns the reverse lookup for an address if any. public fun get_reverse_lookup(account_addr: address): Option acquires ReverseLookupRegistryV1 { let registry = &borrow_global_mut(@aptos_names).registry; if (table::contains(registry, account_addr)) { From b1bf4a35cc2a4de1b7dab8a654ad69ea39dfd11c Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Mon, 19 Dec 2022 16:20:05 -0500 Subject: [PATCH 09/16] Correctly clear primary name when clearing target address --- core/sources/domains.move | 68 ++++++++++++++++++++++++++++------- core/sources/test_helper.move | 14 ++++++++ 2 files changed, 70 insertions(+), 12 deletions(-) diff --git a/core/sources/domains.move b/core/sources/domains.move index b2c58af9..4a4c6e8b 100644 --- a/core/sources/domains.move +++ b/core/sources/domains.move @@ -71,10 +71,15 @@ module aptos_names::domains { registry: Table, } + /// The registry for reverse lookups (aka primary names), which maps addresses to their primary name + /// - Users can set their primary name to a name that they own. This also forces the name to target their own address too. + /// - If you point your primary name to an address that isn't your own, it is no longer your primary name. + /// - If you mint a name while you don't have a primary name, your minted name is auto-set to be your primary name. (including minting subdomains) struct ReverseLookupRegistryV1 has key, store { registry: Table } + /// Holder for `SetReverseLookupEventV1` events struct SetReverseLookupEventsV1 has key, store { set_reverse_lookup_events: event::EventHandle, } @@ -89,6 +94,9 @@ module aptos_names::domains { register_name_events: event::EventHandle, } + /// A name has been set as the reverse lookup for an address, or + /// the reverse lookup has been cleared (in which case |target_address| + /// will be none) struct SetReverseLookupEventV1 has drop, store { subdomain_name: Option, domain_name: String, @@ -223,7 +231,7 @@ module aptos_names::domains { register_name_internal(sign, option::some(subdomain_name), domain_name, registration_duration_secs, price); } - /// Register a nane. Accepts an optional subdomain name, a required domain name, and a registration duration in seconds. + /// Register a name. Accepts an optional subdomain name, a required domain name, and a registration duration in seconds. /// For domains, the registration duration is only allowed to be in increments of 1 year, for now /// Since the owner of the domain is the only one that can create the subdomain, we allow them to decide how long they want the underlying registration to be /// The maximum subdomain registration duration is limited to the duration of its parent domain registration @@ -309,6 +317,18 @@ module aptos_names::domains { public fun force_create_or_seize_name(sign: &signer, subdomain_name: Option, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { config::assert_signer_is_admin(sign); + + // If the name is a primary name, clear it + let maybe_target_address = name_resolved_address(subdomain_name, domain_name); + if (option::is_some(&maybe_target_address)) { + let target_address = option::borrow(&maybe_target_address); + let maybe_reverse_lookup = get_reverse_lookup(*target_address); + if (option::is_some(&maybe_reverse_lookup) && NameRecordKeyV1 { subdomain_name, domain_name } == *option::borrow(&maybe_reverse_lookup)) { + clear_reverse_lookup_internal(*target_address); + }; + }; + + // Register the name register_name_internal(sign, subdomain_name, domain_name, registration_duration_secs, 0); } @@ -399,7 +419,6 @@ module aptos_names::domains { set_name_address(sign, option::none(), domain_name, new_address); } - public entry fun set_subdomain_address(sign: &signer, subdomain_name: String, domain_name: String, new_address: address) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { set_name_address(sign, option::some(subdomain_name), domain_name, new_address); } @@ -426,13 +445,7 @@ module aptos_names::domains { let current_reverse_lookup = option::borrow(&maybe_reverse_lookup); let key = NameRecordKeyV1 { subdomain_name, domain_name }; if (*current_reverse_lookup == key && signer_addr != new_address) { - let registry = &mut borrow_global_mut(@aptos_names).registry; - table::remove(registry, signer_addr); - emit_set_reverse_lookup_event_v1( - subdomain_name, - domain_name, - option::none() - ); + clear_reverse_lookup(sign); }; } @@ -453,20 +466,30 @@ module aptos_names::domains { *name_record } - public entry fun clear_domain_address(sign: &signer, domain_name: String) acquires NameRegistryV1, SetNameAddressEventsV1 { + public entry fun clear_domain_address(sign: &signer, domain_name: String) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { clear_name_address(sign, option::none(), domain_name); } - public entry fun clear_subdomain_address(sign: &signer, subdomain_name: String, domain_name: String) acquires NameRegistryV1, SetNameAddressEventsV1 { + public entry fun clear_subdomain_address(sign: &signer, subdomain_name: String, domain_name: String) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { clear_name_address(sign, option::some(subdomain_name), domain_name); } /// This is a shared entry point for clearing the address of a domain or subdomain /// It enforces owner permissions - fun clear_name_address(sign: &signer, subdomain_name: Option, domain_name: String) acquires NameRegistryV1, SetNameAddressEventsV1 { + fun clear_name_address(sign: &signer, subdomain_name: Option, domain_name: String) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { assert!(name_is_registered(subdomain_name, domain_name), error::not_found(ENAME_NOT_EXIST)); let signer_addr = signer::address_of(sign); + + // Clear the reverse lookup if this name is the signer's reverse lookup + let maybe_reverse_lookup = get_reverse_lookup(signer_addr); + if (option::is_some(&maybe_reverse_lookup)) { + let reverse_lookup = option::borrow(&maybe_reverse_lookup); + if (NameRecordKeyV1 { subdomain_name, domain_name } == *reverse_lookup) { + clear_reverse_lookup_internal(signer_addr); + }; + }; + // Only the owner or the registered address can clear the address let (is_owner, token_id) = is_owner_of_name(signer_addr, subdomain_name, domain_name); let is_name_resolved_address = name_resolved_address(subdomain_name, domain_name) == option::some
(signer_addr); @@ -516,6 +539,12 @@ module aptos_names::domains { set_name_address(account, maybe_subdomain_name, domain_name, account_addr); } + /// Clears the user's reverse lookup. + public fun clear_reverse_lookup(account: &signer) acquires ReverseLookupRegistryV1, SetReverseLookupEventsV1 { + let account_addr = signer::address_of(account); + clear_reverse_lookup_internal(account_addr); + } + /// Returns the reverse lookup for an address if any. public fun get_reverse_lookup(account_addr: address): Option acquires ReverseLookupRegistryV1 { let registry = &borrow_global_mut(@aptos_names).registry; @@ -542,6 +571,21 @@ module aptos_names::domains { ); } + fun clear_reverse_lookup_internal(account_addr: address) acquires ReverseLookupRegistryV1, SetReverseLookupEventsV1 { + let maybe_reverse_lookup = get_reverse_lookup(account_addr); + if (option::is_none(&maybe_reverse_lookup)) { + return + }; + let NameRecordKeyV1 { subdomain_name, domain_name } = option::borrow(&maybe_reverse_lookup); + let registry = &mut borrow_global_mut(@aptos_names).registry; + table::remove(registry, account_addr); + emit_set_reverse_lookup_event_v1( + *subdomain_name, + *domain_name, + option::none() + ); + } + fun emit_set_name_address_event_v1(subdomain_name: Option, domain_name: String, property_version: u64, expiration_time_secs: u64, new_address: Option
) acquires SetNameAddressEventsV1 { let event = SetNameAddressEventV1 { subdomain_name, diff --git a/core/sources/test_helper.move b/core/sources/test_helper.move index 9abb5925..47060159 100644 --- a/core/sources/test_helper.move +++ b/core/sources/test_helper.move @@ -212,8 +212,11 @@ module aptos_names::test_helper { /// Clear the domain address, and verify the address was cleared public fun clear_name_address(user: &signer, subdomain_name: Option, domain_name: String) { + let user_addr = signer::address_of(user); let register_name_event_v1_event_count_before = domains::get_register_name_event_v1_count(); let set_name_address_event_v1_event_count_before = domains::get_set_name_address_event_v1_count(); + let set_reverse_lookup_event_v1_event_count_before = domains::get_set_reverse_lookup_event_v1_count(); + let maybe_reverse_lookup_before = domains::get_reverse_lookup(user_addr); // And also can clear if is registered address, but not owner if (option::is_none(&subdomain_name)) { @@ -225,6 +228,17 @@ module aptos_names::test_helper { test_utils::print_actual_expected(b"clear_domain_address: ", target_address, option::none(), false); assert!(target_address == option::none(), 32); + if (option::is_some(&maybe_reverse_lookup_before)) { + let reverse_lookup_before = option::borrow(&maybe_reverse_lookup_before); + if (*reverse_lookup_before == domains::create_name_record_key_v1(subdomain_name, domain_name)) { + let reverse_lookup_after = domains::get_reverse_lookup(user_addr); + assert!(option::is_none(&reverse_lookup_after), 35); + + let set_reverse_lookup_event_v1_num_emitted = domains::get_set_reverse_lookup_event_v1_count() - set_reverse_lookup_event_v1_event_count_before; + assert!(set_reverse_lookup_event_v1_num_emitted == 1, set_reverse_lookup_event_v1_num_emitted); + }; + }; + // Assert events have been correctly emmitted let register_name_event_v1_num_emitted = domains::get_register_name_event_v1_count() - register_name_event_v1_event_count_before; let set_name_address_event_v1_num_emitted = domains::get_set_name_address_event_v1_count() - set_name_address_event_v1_event_count_before; From 8dcfd843976e07924f8b7e6c8c077e5ff2e85bc0 Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Mon, 19 Dec 2022 16:36:25 -0500 Subject: [PATCH 10/16] Adds more tests --- core/sources/domain_e2e_tests.move | 15 +++++++++++++++ core/sources/test_helper.move | 1 + 2 files changed, 16 insertions(+) diff --git a/core/sources/domain_e2e_tests.move b/core/sources/domain_e2e_tests.move index 6db3bc5b..19f0b5f8 100644 --- a/core/sources/domain_e2e_tests.move +++ b/core/sources/domain_e2e_tests.move @@ -296,4 +296,19 @@ module aptos_names::domain_e2e_tests { // Take the domain name for much longer than users are allowed to register it for domains::force_create_or_seize_name(rando, option::none(), test_helper::domain_name(), test_helper::two_hundred_year_secs()); } + + #[test(myself = @aptos_names, user = @0x077, aptos = @0x1, rando = @0x266f, foundation = @0xf01d)] + fun clear_name_happy_path_e2e_test(myself: &signer, user: signer, aptos: signer, rando: signer, foundation: signer) { + let users = test_helper::e2e_test_setup(myself, user, &aptos, rando, &foundation); + let user = vector::borrow(&users, 0); + let user_addr = signer::address_of(user); + + // Register the domain + test_helper::register_name(user, option::none(), test_helper::domain_name(), test_helper::one_year_secs(), test_helper::fq_domain_name(), 1, vector::empty()); + + // Clear my reverse lookup. + domains::clear_reverse_lookup(user); + + assert!(option::is_none(&domains::get_reverse_lookup(user_addr)), 1); + } } diff --git a/core/sources/test_helper.move b/core/sources/test_helper.move index 47060159..d79b41f8 100644 --- a/core/sources/test_helper.move +++ b/core/sources/test_helper.move @@ -148,6 +148,7 @@ module aptos_names::test_helper { // Reverse lookup should be set if user did not have one before if (option::is_none(&user_reverse_lookup_before)) { + assert!(option::is_some(&domains::get_reverse_lookup(user_addr)), 36); assert!(set_reverse_lookup_event_v1_num_emitted == 1, set_reverse_lookup_event_v1_num_emitted); } else { assert!(set_reverse_lookup_event_v1_num_emitted == 0, set_reverse_lookup_event_v1_num_emitted); From f7ad20ff721230015c22df8fc13c50ca8c18d0cb Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Tue, 20 Dec 2022 11:04:06 -0500 Subject: [PATCH 11/16] Switches to framework-mainnet for core --- core/Move.toml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/core/Move.toml b/core/Move.toml index db15d9c9..f45778c2 100644 --- a/core/Move.toml +++ b/core/Move.toml @@ -9,10 +9,10 @@ aptos_names_funds = "0x78ee3915e67ef5d19fa91d1e05e60ae08751efd12ce58e23fc1109de8 [dependencies.AptosFramework] git = 'https://github.com/aptos-labs/aptos-core.git' -rev = 'mainnet' +rev = 'framework-mainnet' subdir = 'aptos-move/framework/aptos-framework' [dependencies.AptosToken] git = 'https://github.com/aptos-labs/aptos-core.git' -rev = 'mainnet' +rev = 'framework-mainnet' subdir = 'aptos-move/framework/aptos-token' From 7ee8289a9f8bbe64106745171af8822e33ec07f7 Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Tue, 20 Dec 2022 11:16:42 -0500 Subject: [PATCH 12/16] Adds some more tests for seizing domain name --- core/sources/domain_e2e_tests.move | 38 ++++++++++++++++++++++++++++++ core/sources/test_helper.move | 9 ++++++- 2 files changed, 46 insertions(+), 1 deletion(-) diff --git a/core/sources/domain_e2e_tests.move b/core/sources/domain_e2e_tests.move index 19f0b5f8..c4a32777 100644 --- a/core/sources/domain_e2e_tests.move +++ b/core/sources/domain_e2e_tests.move @@ -9,6 +9,7 @@ module aptos_names::domain_e2e_tests { use aptos_names::test_utils; use std::option; use std::signer; + use std::string; use std::vector; #[test(myself = @aptos_names, user = @0x077, aptos = @0x1, rando = @0x266f, foundation = @0xf01d)] @@ -230,6 +231,7 @@ module aptos_names::domain_e2e_tests { fun admin_can_force_seize_domain_name_e2e_test(myself: &signer, user: signer, aptos: signer, rando: signer, foundation: signer) { let users = test_helper::e2e_test_setup(myself, user, &aptos, rando, &foundation); let user = vector::borrow(&users, 0); + let user_addr = signer::address_of(user); // Register the domain test_helper::register_name(user, option::none(), test_helper::domain_name(), test_helper::one_year_secs(), test_helper::fq_domain_name(), 1, vector::empty()); @@ -244,6 +246,42 @@ module aptos_names::domain_e2e_tests { // Ensure the expiration_time_sec is set to the new far future value let (_, expiration_time_sec, _) = domains::get_name_record_v1_props_for_name(option::none(), test_helper::domain_name()); assert!(time_helper::seconds_to_years(expiration_time_sec) == 200, time_helper::seconds_to_years(expiration_time_sec)); + + // Ensure that the user's primary name is no longer set. + assert!(option::is_none(&domains::get_reverse_lookup(user_addr)), 1); + } + + #[test(myself = @aptos_names, user = @0x077, aptos = @0x1, rando = @0x266f, foundation = @0xf01d)] + fun admin_force_seize_domain_name_doesnt_clear_unrelated_primary_name_e2e_test(myself: &signer, user: signer, aptos: signer, rando: signer, foundation: signer) { + let users = test_helper::e2e_test_setup(myself, user, &aptos, rando, &foundation); + let user = vector::borrow(&users, 0); + let user_addr = signer::address_of(user); + + // Register the domain. This will be the user's reverse lookup + { + test_helper::register_name(user, option::none(), test_helper::domain_name(), test_helper::one_year_secs(), test_helper::fq_domain_name(), 1, vector::empty()); + let (is_owner, _token_id) = domains::is_owner_of_name(signer::address_of(user), option::none(), test_helper::domain_name()); + assert!(is_owner, 1); + }; + + // Register another domain. This will **not** be the user's reverse lookup + let domain_name = string::utf8(b"sets"); + let fq_domain_name = string::utf8(b"sets.apt"); + test_helper::register_name(user, option::none(), domain_name, test_helper::one_year_secs(), fq_domain_name, 1, vector::empty()); + let (is_owner, _token_id) = domains::is_owner_of_name(signer::address_of(user), option::none(), domain_name); + assert!(is_owner, 1); + + // Take the domain name for much longer than users are allowed to register it for + domains::force_create_or_seize_name(myself, option::none(), domain_name, test_helper::two_hundred_year_secs()); + let (is_owner, _token_id) = domains::is_owner_of_name(signer::address_of(myself), option::none(), domain_name); + assert!(is_owner, 2); + + // Ensure the expiration_time_sec is set to the new far future value + let (_, expiration_time_sec, _) = domains::get_name_record_v1_props_for_name(option::none(), domain_name); + assert!(time_helper::seconds_to_years(expiration_time_sec) == 200, time_helper::seconds_to_years(expiration_time_sec)); + + // Ensure that the user's primary name is still set. + assert!(option::is_some(&domains::get_reverse_lookup(user_addr)), 1); } #[test(myself = @aptos_names, user = @0x077, aptos = @0x1, rando = @0x266f, foundation = @0xf01d)] diff --git a/core/sources/test_helper.move b/core/sources/test_helper.move index d79b41f8..5b62d10e 100644 --- a/core/sources/test_helper.move +++ b/core/sources/test_helper.move @@ -148,7 +148,14 @@ module aptos_names::test_helper { // Reverse lookup should be set if user did not have one before if (option::is_none(&user_reverse_lookup_before)) { - assert!(option::is_some(&domains::get_reverse_lookup(user_addr)), 36); + let maybe_reverse_lookup_after = domains::get_reverse_lookup(user_addr); + if (option::is_some(&maybe_reverse_lookup_after)) { + let reverse_lookup_after = option::borrow(&maybe_reverse_lookup_after); + assert!(*reverse_lookup_after == domains::create_name_record_key_v1(subdomain_name, domain_name), 36); + } else { + // Reverse lookup is not set, even though user did not have a reverse lookup before. + assert!(false, 37); + }; assert!(set_reverse_lookup_event_v1_num_emitted == 1, set_reverse_lookup_event_v1_num_emitted); } else { assert!(set_reverse_lookup_event_v1_num_emitted == 0, set_reverse_lookup_event_v1_num_emitted); From 42ae4408fc4c5ff2bbe92d4f7377389be7dc8774 Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Mon, 9 Jan 2023 14:41:33 -0500 Subject: [PATCH 13/16] Clears reverse lookup for name when setting reverse lookup --- core/sources/domain_e2e_tests.move | 28 ++++++++++++++++++++++++++++ core/sources/domains.move | 27 ++++++++++++++++----------- 2 files changed, 44 insertions(+), 11 deletions(-) diff --git a/core/sources/domain_e2e_tests.move b/core/sources/domain_e2e_tests.move index c4a32777..cc26dc25 100644 --- a/core/sources/domain_e2e_tests.move +++ b/core/sources/domain_e2e_tests.move @@ -7,6 +7,7 @@ module aptos_names::domain_e2e_tests { use aptos_names::time_helper; use aptos_names::test_helper; use aptos_names::test_utils; + use aptos_token::token; use std::option; use std::signer; use std::string; @@ -349,4 +350,31 @@ module aptos_names::domain_e2e_tests { assert!(option::is_none(&domains::get_reverse_lookup(user_addr)), 1); } + + #[test(myself = @aptos_names, user = @0x077, aptos = @0x1, rando = @0x266f, foundation = @0xf01d)] + fun set_primary_name_clears_old_primary_name_e2e_test(myself: &signer, user: signer, aptos: signer, rando: signer, foundation: signer) { + let users = test_helper::e2e_test_setup(myself, user, &aptos, rando, &foundation); + let user = vector::borrow(&users, 0); + let user_addr = signer::address_of(user); + let rando = vector::borrow(&users, 1); + + // Register the domain + test_helper::register_name(user, option::none(), test_helper::domain_name(), test_helper::one_year_secs(), test_helper::fq_domain_name(), 1, vector::empty()); + let (is_owner, token_id) = domains::is_owner_of_name(user_addr, option::none(), test_helper::domain_name()); + assert!(is_owner, 1); + + // Transfer the domain to rando + token::direct_transfer(user, rando, token_id, 1); + + // Verify primary name for |user| hasn't changed + assert!(option::is_some(&domains::get_reverse_lookup(user_addr)), 1); + + // |rando| sets his primary name + let subdomain_name_str = string::utf8(b""); + let domain_name_str = string::utf8(b"test"); + domains::set_reverse_lookup_entry(rando, subdomain_name_str, domain_name_str); + + // |user|'s primary name should be none. + assert!(option::is_none(&domains::get_reverse_lookup(user_addr)), 1); + } } diff --git a/core/sources/domains.move b/core/sources/domains.move index 4a4c6e8b..c7099279 100644 --- a/core/sources/domains.move +++ b/core/sources/domains.move @@ -318,15 +318,7 @@ module aptos_names::domains { public fun force_create_or_seize_name(sign: &signer, subdomain_name: Option, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { config::assert_signer_is_admin(sign); - // If the name is a primary name, clear it - let maybe_target_address = name_resolved_address(subdomain_name, domain_name); - if (option::is_some(&maybe_target_address)) { - let target_address = option::borrow(&maybe_target_address); - let maybe_reverse_lookup = get_reverse_lookup(*target_address); - if (option::is_some(&maybe_reverse_lookup) && NameRecordKeyV1 { subdomain_name, domain_name } == *option::borrow(&maybe_reverse_lookup)) { - clear_reverse_lookup_internal(*target_address); - }; - }; + clear_reverse_lookup_for_name(subdomain_name, domain_name); // Register the name register_name_internal(sign, subdomain_name, domain_name, registration_duration_secs, 0); @@ -532,10 +524,11 @@ module aptos_names::domains { /// Sets the |account|'s reverse lookup, aka "primary name". This allows a user to specify which of their Aptos Names /// is their "primary", so that dapps can display the user's primary name rather than their address. public fun set_reverse_lookup(account: &signer, key: &NameRecordKeyV1) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { - set_reverse_lookup_internal(account, key); - let account_addr = signer::address_of(account); let (maybe_subdomain_name, domain_name) = get_name_record_key_v1_props(key); + + clear_reverse_lookup_for_name(maybe_subdomain_name, domain_name); + set_reverse_lookup_internal(account, key); set_name_address(account, maybe_subdomain_name, domain_name, account_addr); } @@ -586,6 +579,18 @@ module aptos_names::domains { ); } + fun clear_reverse_lookup_for_name(subdomain_name: Option, domain_name: String) acquires NameRegistryV1, ReverseLookupRegistryV1, SetReverseLookupEventsV1 { + // If the name is a primary name, clear it + let maybe_target_address = name_resolved_address(subdomain_name, domain_name); + if (option::is_some(&maybe_target_address)) { + let target_address = option::borrow(&maybe_target_address); + let maybe_reverse_lookup = get_reverse_lookup(*target_address); + if (option::is_some(&maybe_reverse_lookup) && NameRecordKeyV1 { subdomain_name, domain_name } == *option::borrow(&maybe_reverse_lookup)) { + clear_reverse_lookup_internal(*target_address); + }; + }; + } + fun emit_set_name_address_event_v1(subdomain_name: Option, domain_name: String, property_version: u64, expiration_time_secs: u64, new_address: Option
) acquires SetNameAddressEventsV1 { let event = SetNameAddressEventV1 { subdomain_name, From c6aefc247deedd2ade54193597203fd4fdee357d Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Wed, 11 Jan 2023 14:05:51 -0500 Subject: [PATCH 14/16] Clear reverse lookup for expired names on registration --- core/sources/domain_e2e_tests.move | 3 +++ core/sources/domains.move | 7 ++++--- core/sources/test_helper.move | 31 ++++++++++++++++++++++++++++-- 3 files changed, 36 insertions(+), 5 deletions(-) diff --git a/core/sources/domain_e2e_tests.move b/core/sources/domain_e2e_tests.move index cc26dc25..e678f8a4 100644 --- a/core/sources/domain_e2e_tests.move +++ b/core/sources/domain_e2e_tests.move @@ -127,6 +127,9 @@ module aptos_names::domain_e2e_tests { // Lets try to register it again, now that it is expired test_helper::register_name(rando, option::none(), test_helper::domain_name(), test_helper::one_year_secs(), test_helper::fq_domain_name(), 2, vector::empty()); + // Reverse lookup for |user| should be none. + assert!(option::is_none(&domains::get_reverse_lookup(signer::address_of(user))), 85); + // And again! let (_, expiration_time_sec, _) = domains::get_name_record_v1_props_for_name(option::none(), test_helper::domain_name()); timestamp::update_global_time_for_test_secs(expiration_time_sec + 5); diff --git a/core/sources/domains.move b/core/sources/domains.move index c7099279..d974d4c2 100644 --- a/core/sources/domains.move +++ b/core/sources/domains.move @@ -236,6 +236,10 @@ module aptos_names::domains { /// Since the owner of the domain is the only one that can create the subdomain, we allow them to decide how long they want the underlying registration to be /// The maximum subdomain registration duration is limited to the duration of its parent domain registration fun register_name_internal(sign: &signer, subdomain_name: Option, domain_name: String, registration_duration_secs: u64, price: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { + // If we're registering a name that exists but is expired, and the expired name is a primary name, + // it should get removed from being a primary name. + clear_reverse_lookup_for_name(subdomain_name, domain_name); + let aptos_names = borrow_global_mut(@aptos_names); let name_expiration_time_secs = timestamp::now_seconds() + registration_duration_secs; @@ -317,9 +321,6 @@ module aptos_names::domains { public fun force_create_or_seize_name(sign: &signer, subdomain_name: Option, domain_name: String, registration_duration_secs: u64) acquires NameRegistryV1, RegisterNameEventsV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { config::assert_signer_is_admin(sign); - - clear_reverse_lookup_for_name(subdomain_name, domain_name); - // Register the name register_name_internal(sign, subdomain_name, domain_name, registration_duration_secs, 0); } diff --git a/core/sources/test_helper.move b/core/sources/test_helper.move index 5b62d10e..f0d6d328 100644 --- a/core/sources/test_helper.move +++ b/core/sources/test_helper.move @@ -65,6 +65,13 @@ module aptos_names::test_helper { let user_balance_before = coin::balance(user_addr); let user_reverse_lookup_before = domains::get_reverse_lookup(user_addr); + let maybe_target_address = domains::name_resolved_address(subdomain_name, domain_name); + let name_reverse_lookup_before = if (option::is_some(&maybe_target_address)) { + domains::get_reverse_lookup(*option::borrow(&maybe_target_address)) + } else { + option::none() + }; + let is_expired_before = domains::name_is_registered(subdomain_name, domain_name) && domains::name_is_expired(subdomain_name, domain_name); let register_name_event_v1_event_count_before = domains::get_register_name_event_v1_count(); let set_name_address_event_v1_event_count_before = domains::get_set_name_address_event_v1_count(); let set_reverse_lookup_event_v1_event_count_before = domains::get_set_reverse_lookup_event_v1_count(); @@ -156,9 +163,29 @@ module aptos_names::test_helper { // Reverse lookup is not set, even though user did not have a reverse lookup before. assert!(false, 37); }; - assert!(set_reverse_lookup_event_v1_num_emitted == 1, set_reverse_lookup_event_v1_num_emitted); + // If we are registering over a name that is already registered but expired and was a primary name, + // that name should be removed from being a primary name. + if (option::is_some(&name_reverse_lookup_before) && is_expired_before) { + assert!(set_reverse_lookup_event_v1_num_emitted == 2, set_reverse_lookup_event_v1_num_emitted); + } else { + assert!(set_reverse_lookup_event_v1_num_emitted == 1, set_reverse_lookup_event_v1_num_emitted); + } } else { - assert!(set_reverse_lookup_event_v1_num_emitted == 0, set_reverse_lookup_event_v1_num_emitted); + // If we are registering over a name that is already registered but expired and was the user's primary name, + // that name should be removed from being a primary name and the new one should be set. + if (option::is_some(&name_reverse_lookup_before) + && option::is_some(&user_reverse_lookup_before) + && *option::borrow(&name_reverse_lookup_before) == *option::borrow(&user_reverse_lookup_before) + && is_expired_before + ) { + assert!(set_reverse_lookup_event_v1_num_emitted == 2, set_reverse_lookup_event_v1_num_emitted); + } else if (option::is_some(&name_reverse_lookup_before) && is_expired_before) { + // If we are registering over a name that is already registered but expired and was a primary name, + // that name should be removed from being a primary name. + assert!(set_reverse_lookup_event_v1_num_emitted == 1, set_reverse_lookup_event_v1_num_emitted); + } else { + assert!(set_reverse_lookup_event_v1_num_emitted == 0, set_reverse_lookup_event_v1_num_emitted); + } }; if (is_subdomain) { From 916c363ac193e5a2097f251dca3ce80f0de54cbc Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Wed, 11 Jan 2023 15:06:41 -0500 Subject: [PATCH 15/16] Maintain primary name invariants when setting target address --- core/sources/domain_e2e_tests.move | 34 +++++++++++++++++++++++++++++- core/sources/domains.move | 7 +++--- 2 files changed, 37 insertions(+), 4 deletions(-) diff --git a/core/sources/domain_e2e_tests.move b/core/sources/domain_e2e_tests.move index e678f8a4..66498a5e 100644 --- a/core/sources/domain_e2e_tests.move +++ b/core/sources/domain_e2e_tests.move @@ -355,11 +355,12 @@ module aptos_names::domain_e2e_tests { } #[test(myself = @aptos_names, user = @0x077, aptos = @0x1, rando = @0x266f, foundation = @0xf01d)] - fun set_primary_name_clears_old_primary_name_e2e_test(myself: &signer, user: signer, aptos: signer, rando: signer, foundation: signer) { + fun set_primary_name_after_transfer_clears_old_primary_name_e2e_test(myself: &signer, user: signer, aptos: signer, rando: signer, foundation: signer) { let users = test_helper::e2e_test_setup(myself, user, &aptos, rando, &foundation); let user = vector::borrow(&users, 0); let user_addr = signer::address_of(user); let rando = vector::borrow(&users, 1); + let rando_addr = signer::address_of(rando); // Register the domain test_helper::register_name(user, option::none(), test_helper::domain_name(), test_helper::one_year_secs(), test_helper::fq_domain_name(), 1, vector::empty()); @@ -371,6 +372,7 @@ module aptos_names::domain_e2e_tests { // Verify primary name for |user| hasn't changed assert!(option::is_some(&domains::get_reverse_lookup(user_addr)), 1); + assert!(*option::borrow(&domains::name_resolved_address(option::none(), test_helper::domain_name())) == user_addr, 1); // |rando| sets his primary name let subdomain_name_str = string::utf8(b""); @@ -379,5 +381,35 @@ module aptos_names::domain_e2e_tests { // |user|'s primary name should be none. assert!(option::is_none(&domains::get_reverse_lookup(user_addr)), 1); + assert!(*option::borrow(&domains::name_resolved_address(option::none(), test_helper::domain_name())) == rando_addr, 1); + } + + #[test(myself = @aptos_names, user = @0x077, aptos = @0x1, rando = @0x266f, foundation = @0xf01d)] + fun set_target_address_after_transfer_clears_old_primary_name_e2e_test(myself: &signer, user: signer, aptos: signer, rando: signer, foundation: signer) { + let users = test_helper::e2e_test_setup(myself, user, &aptos, rando, &foundation); + let user = vector::borrow(&users, 0); + let user_addr = signer::address_of(user); + let rando = vector::borrow(&users, 1); + let rando_addr = signer::address_of(rando); + + // Register the domain + test_helper::register_name(user, option::none(), test_helper::domain_name(), test_helper::one_year_secs(), test_helper::fq_domain_name(), 1, vector::empty()); + let (is_owner, token_id) = domains::is_owner_of_name(user_addr, option::none(), test_helper::domain_name()); + assert!(is_owner, 1); + + // Transfer the domain to rando + token::direct_transfer(user, rando, token_id, 1); + + // Verify primary name for |user| hasn't changed + assert!(option::is_some(&domains::get_reverse_lookup(user_addr)), 1); + assert!(*option::borrow(&domains::name_resolved_address(option::none(), test_helper::domain_name())) == user_addr, 1); + + // |rando| sets target address + let domain_name_str = string::utf8(b"test"); + domains::set_domain_address(rando, domain_name_str, rando_addr); + + // |user|'s primary name should be none. + assert!(option::is_none(&domains::get_reverse_lookup(user_addr)), 1); + assert!(*option::borrow(&domains::name_resolved_address(option::none(), test_helper::domain_name())) == rando_addr, 1); } } diff --git a/core/sources/domains.move b/core/sources/domains.move index d974d4c2..f255ae6a 100644 --- a/core/sources/domains.move +++ b/core/sources/domains.move @@ -417,6 +417,9 @@ module aptos_names::domains { } public fun set_name_address(sign: &signer, subdomain_name: Option, domain_name: String, new_address: address) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { + // If the domain name is a primary name, clear it. + clear_reverse_lookup_for_name(subdomain_name, domain_name); + let signer_addr = signer::address_of(sign); let (is_owner, token_id) = is_owner_of_name(signer_addr, subdomain_name, domain_name); assert!(is_owner, error::permission_denied(ENOT_OWNER_OF_NAME)); @@ -527,10 +530,8 @@ module aptos_names::domains { public fun set_reverse_lookup(account: &signer, key: &NameRecordKeyV1) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { let account_addr = signer::address_of(account); let (maybe_subdomain_name, domain_name) = get_name_record_key_v1_props(key); - - clear_reverse_lookup_for_name(maybe_subdomain_name, domain_name); - set_reverse_lookup_internal(account, key); set_name_address(account, maybe_subdomain_name, domain_name, account_addr); + set_reverse_lookup_internal(account, key); } /// Clears the user's reverse lookup. From 316dd5e9aabedfdd2245e5886385f74662c7bf7e Mon Sep 17 00:00:00 2001 From: 0xChucky Date: Tue, 17 Jan 2023 14:07:22 -0500 Subject: [PATCH 16/16] Clears reverse lookup in force_set_name_address --- core/sources/domains.move | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/core/sources/domains.move b/core/sources/domains.move index f255ae6a..fc91729a 100644 --- a/core/sources/domains.move +++ b/core/sources/domains.move @@ -293,16 +293,18 @@ module aptos_names::domains { /// Forcefully set the name of a domain. /// This is a privileged operation, used via governance, to forcefully set a domain address /// This can be used, for example, to forcefully set the domain for a system address domain - public entry fun force_set_domain_address(sign: &signer, domain_name: String, new_owner: address) acquires NameRegistryV1, SetNameAddressEventsV1 { + public entry fun force_set_domain_address(sign: &signer, domain_name: String, new_owner: address) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { force_set_name_address(sign, option::none(), domain_name, new_owner); } - public entry fun force_set_subdomain_address(sign: &signer, subdomain_name: String, domain_name: String, new_owner: address) acquires NameRegistryV1, SetNameAddressEventsV1 { + public entry fun force_set_subdomain_address(sign: &signer, subdomain_name: String, domain_name: String, new_owner: address) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { force_set_name_address(sign, option::some(subdomain_name), domain_name, new_owner); } - fun force_set_name_address(sign: &signer, subdomain_name: Option, domain_name: String, new_owner: address) acquires NameRegistryV1, SetNameAddressEventsV1 { + fun force_set_name_address(sign: &signer, subdomain_name: Option, domain_name: String, new_owner: address) acquires NameRegistryV1, ReverseLookupRegistryV1, SetNameAddressEventsV1, SetReverseLookupEventsV1 { config::assert_signer_is_admin(sign); + // If the domain name is a primary name, clear it. + clear_reverse_lookup_for_name(subdomain_name, domain_name); set_name_address_internal(subdomain_name, domain_name, new_owner); }