From a4528df515f430f498a83f19016bfa1d0cad8458 Mon Sep 17 00:00:00 2001 From: Wisdom Nwokocha Date: Mon, 26 Aug 2024 17:57:46 +0100 Subject: [PATCH 1/2] added RNS Over and Spec updated content --- .../rif-suite/rns/getting-started/index.md | 8 + docs/01-concepts/rif-suite/rns/index.md | 132 +++++++++ docs/01-concepts/rif-suite/rns/specs/index.md | 66 +++++ .../rif-suite/rns/specs/registrar.md | 88 ++++++ .../rif-suite/rns/specs/registry.md | 68 +++++ .../rif-suite/rns/specs/resolver.md | 252 ++++++++++++++++++ static/img/rif/rns/theStack.png | Bin 0 -> 23577 bytes 7 files changed, 614 insertions(+) create mode 100644 docs/01-concepts/rif-suite/rns/getting-started/index.md create mode 100644 docs/01-concepts/rif-suite/rns/index.md create mode 100644 docs/01-concepts/rif-suite/rns/specs/index.md create mode 100644 docs/01-concepts/rif-suite/rns/specs/registrar.md create mode 100644 docs/01-concepts/rif-suite/rns/specs/registry.md create mode 100644 docs/01-concepts/rif-suite/rns/specs/resolver.md create mode 100644 static/img/rif/rns/theStack.png diff --git a/docs/01-concepts/rif-suite/rns/getting-started/index.md b/docs/01-concepts/rif-suite/rns/getting-started/index.md new file mode 100644 index 00000000..7a815518 --- /dev/null +++ b/docs/01-concepts/rif-suite/rns/getting-started/index.md @@ -0,0 +1,8 @@ +--- +sidebar_label: Getting Started +sidebar_position: 500 +title: "Getting Started" +tags: [rif, rns, Getting Started] +description: "" +--- + diff --git a/docs/01-concepts/rif-suite/rns/index.md b/docs/01-concepts/rif-suite/rns/index.md new file mode 100644 index 00000000..5efde711 --- /dev/null +++ b/docs/01-concepts/rif-suite/rns/index.md @@ -0,0 +1,132 @@ +--- +sidebar_label: RNS +sidebar_position: 400 +title: "RIF RNS: RIF Name Service | Rootstock (RSK)" +tags: [rif, rns, rif-name-service, rsk] +description: "Information about the RIF token, where to obtain it, how to transfer it, and technical details on its token standard" +--- + +RNS provides an architecture which enables the identification of blockchain addresses by human-readable names. + + + +
+
+
+
+ Try the service +
+
+

Register a domain in the Testnet, for free.

+
+
+
+
+ Integrate with RNS +
+
+

Easy guides on how to integrate RNS in your solution.

+
+
+
+
+
+
+ Develop on top of RNS +
+
+

Deploy RNS suite in your local development environment

+
+
+
+
+ Use the libraries +
+
+

Use simple libraries to interact with RNS service.

+
+
+
+
+ +## The stack + +![image](/img/rif/rns/theStack.png) + +## Motivation + +By adding a name resolution service, also known as “alias”, the probability of errors is significantly reduced. In addition, centralizing the access to multiple resources associated with a human-readable name improves the blockchain platform user experience. As resource names may change over time, the system needs to be flexible to support frequent changes. + +Currently over the World Wide Web, the Domain Name System (DNS) is responsible for mapping human-readable names to IP addresses. RNS is a decentralized and secure service that works over RSK's blockchain. + +Here’s a refined version of your text, maintaining the same tone and voice: + + +## Design + +RNS is a hierarchical namespace inspired by DNS, where the hierarchy roughly reflects organizational structure, with levels separated by the "." character. + +The design of the RIF Name Service is shaped by specific goals: + +- The primary objective is to establish a consistent namespace for referencing resources. +- Each piece of data associated with a name is tagged with a type, allowing queries to be limited to a specific type. +- To ensure the namespace is adaptable across different networks and applications, RNS supports the use of the same namespace with various protocol families or management systems. Data in RNS is tagged with both a class and a type, enabling the parallel use of different formats for data of type "address." +- There may be trade-offs between data acquisition costs, update speed, and cache accuracy. The domain owner, as the data source, should consider these trade-offs and decide what to store and how to cache it. + +> [RNS specs](./specs) + + +## Elements of the RNS + +RNS has three major components: + +- **The RNS Registry**, which is specification for a tree structured name space and data associated with the names. + + Conceptually, each node and leaf of the domain name space tree names a set of information, and query operations are attempts to extract specific types of information from a particular set. A query names the domain name of interest and describes the type of resource information that is desired. + + [Specs](./specs/registry) + +- **RNS Resolvers** are contracts that provide information from a name in response to client requests. + + Resolvers must be able to answer a query directly, or pursue the query using referrals to other resolvers. A resolver will typically be a contract's public function that is directly accessible to user programs or other contracts; hence no protocol is necessary between the resolver and the user program. + + [Specs](./specs/resolver) + +- **RNS Registrar** is a critical component within the RIF Name Service, responsible for managing the registration of `.rsk` domain names. This contract is granted the authority to register names in the RSK Owner contract, ensuring that new domain registrations are handled securely and efficiently. +[Specs](./specs/registrar) + +These three components roughly correspond to the three layers or views of the domain system: +- From the user's point of view, the domain system is accessed through a simple resolution operation. The domain space consists of a single tree and the user can request information from any section of the tree. +- From the resolver's point of view, the domain system is composed of an unknown number of names. Each name has a corresponding resolver that provides information for a set of resolution types directly. +- From the registry's point of view, the domain system consists of a hierarchical tree where each leaf has an owner (contract or account) and an associated resolver that provides information of the name. + + +## Guidelines on use + +Before RNS can be used to hold naming information for some kind of object, two needs must be met: +- A convention for mapping between object names and domain names. This describes how information about an object is accessed. Find specs [here](specs#name-mapping-convention) +- Resource record types and data formats for describing the object. Find specs. + +The guideline for finding a specific record for a name is as follows: +1. Calculate the name identifier with [`namehash` function](specs#name-mapping-convention). +2. Get the name's resolver address via [`resolver(bytes32)`](specs/registry#AcessFunctions). +3. Determine if resolver supports desired resource record via [ERC-165 interface detection](https://eips.ethereum.org/EIPS/eip-165). +4. Get the desired resource record. Find currently standardized [resolvers](./specs/resolver). + +> Guidelines on integration + +### Resource records + +A domain name identifies a node. Each node has a set of resource information, which may be empty. The set of resource information associated with a particular name is composed of separate resource records (RRs). The order of RRs in a set is not significant. Resource records associated with a name are found in the domain's resolver diff --git a/docs/01-concepts/rif-suite/rns/specs/index.md b/docs/01-concepts/rif-suite/rns/specs/index.md new file mode 100644 index 00000000..85aa59ba --- /dev/null +++ b/docs/01-concepts/rif-suite/rns/specs/index.md @@ -0,0 +1,66 @@ +--- +sidebar_label: Spec +sidebar_position: 400 +title: "Specification" +tags: [rif, rns, spec] +description: "The domain name space is a tree structure. Each node and leaf on the tree corresponds to a resource set (which may be empty)." +--- + +The domain name space is a tree structure. Each node and leaf on the tree corresponds to a resource set (which may be empty). + +The domain system makes no distinctions between the uses of the interior nodes and leaves, and this memo uses the term "node" to refer to both. + +The domain name of a node is the list of the labels on the path from the node to the root of the tree. By convention, the labels that compose a domain name are printed or read left to right, from the most specific (lowest, farthest from the root) to the least specific (highest, closest to the root). + +When a user needs to type a domain name, the length of each label is omitted and the labels are separated by dots `(.)`. + +Since a complete domain name ends with the root label, this leads to a printed form which ends in a dot, that is omitted. For example, a valid domain name is `wallet.alice.rsk`. + +## Valid names +For simplicity, a valid RNS domain is defined as follows: + +- TLD is one of the predefined TLDs, which currently may only be `rsk` +- The first label is compulsory +- Any second and subsequent labels are optional +- All labels must be alphanumeric and lower case +- All labels and the TLD are delimited by `.` + +The reference implementation for RNS domain validation can be found in `rns.js` + +- AVAILABLE_TLDS in src/constants.ts +- isValidDomain and isValidLabel in src/utils.ts + + +Example #1: uno2tres.rsk is valid: + +✔️ TLD is rsk +✔️ First label is uno2tres; which is present and lowercase alphanumeric +✔️ Second and subsequent labels are not present + +Example #2: rss.website.alice.rsk is valid: + +- ✔️ TLD is rsk +- ✔️ First label is alice; which is present and lowercase alphanumeric +- ✔️ Second label is website; which is present and lowercase alphanumeric +- ✔️ Third label is rss; which is present and lowercase alphanumeric + +Example #3: my_illegal_domain.com is invalid: + +- ❌ TLD is com +- ❌ First label is my_illegal_domain; which is present, however contains non-alphanumeric characters +- ✔️ Second and subsequent labels are not present + +## Name mapping convention +Each domain name has a node identifier in the RNS Registry, that is obtained via `namehash` functions, that is as follows: + +```python +def namehash(name): + if name == '': + return '\0' * 32 + else: + label, _, remainder = name.partition('.') + return sha3(namehash(remainder) + sha3(label)) +``` + +Given a `node`, a `subnode` can be identified by using its label: `sha3(namehash(node) + sha3(label))` + diff --git a/docs/01-concepts/rif-suite/rns/specs/registrar.md b/docs/01-concepts/rif-suite/rns/specs/registrar.md new file mode 100644 index 00000000..56092afd --- /dev/null +++ b/docs/01-concepts/rif-suite/rns/specs/registrar.md @@ -0,0 +1,88 @@ +--- +sidebar_label: Registrar Specs +sidebar_position: 400 +title: "Registry Specs" +tags: [rif, rns, rif-name-service, Registrar Specs] +description: "The registry contract provides a simple mapping between a domain and its resolver. " +--- + +### FIFSRegistrarBase.sol Contract Specification + +The `FIFSRegistrarBase` contract implements a First-In-First-Served (FIFS) registration mechanism for domain names within the RNS system. The contract follows a structured process to ensure secure and fair registration of domain names. Below is a detailed breakdown of the registration process and key functions: + +#### Registration Steps + +1. **Calculate Commitment Hash (Off-chain)** + - The first step in registering a domain name is to calculate a commitment hash off-chain using the `makeCommitment` function. This hash ensures that the registration process remains secure by preventing front-running attacks. + - **Function: `makeCommitment`** + - **Parameters**: + - `label`: The keccak256 hash of the domain name to be registered. + - `nameOwner`: The address of the owner of the domain name. + - `secret`: A secret value to protect the name registration. + - **Returns**: The commitment hash (`bytes32`). + - **Usage Note**: This function should be used off-chain and not on-chain when committing. + + ```solidity + function makeCommitment(bytes32 label, address nameOwner, bytes32 secret) public pure returns (bytes32) { + return keccak256(abi.encodePacked(label, nameOwner, secret)); + } + ``` + +2. **Commit the Calculated Hash** + - Once the commitment hash is calculated, it must be submitted to the contract to initiate the registration process. + - **Function: `commit`** + - **Parameters**: + - `commitment`: The valid commitment hash obtained from `makeCommitment`. + - **Usage Note**: The commitment must be unique, and the function ensures that no duplicate commitments are made. + + ```solidity + function commit(bytes32 commitment) external { + require(commitmentRevealTime[commitment] < 1, "Existent commitment"); + commitmentRevealTime[commitment] = now.add(minCommitmentAge); + } + ``` + +3. **Wait for the Commitment to Mature** + - After committing, there is a mandatory waiting period (`minCommitmentAge`) to ensure the commitment is valid and secure. During this time, the commitment cannot be revealed or used to register the domain. + - **Function: `canReveal`** + - **Parameters**: + - `commitment`: The commitment hash to be queried. + - **Returns**: `true` if the commitment can be revealed, `false` otherwise. + + ```solidity + function canReveal(bytes32 commitment) public view returns (bool) { + uint revealTime = commitmentRevealTime[commitment]; + return 0 < revealTime && revealTime <= now; + } + ``` + +4. **Execute Registration** + - Once the commitment is ready to be revealed, the actual registration of the domain can be performed. The `FIFSRegistrarBase` contract supports multiple ways to execute this registration, such as using ERC-20 or ERC-677 tokens. + - **Function: `register`** + - **Parameters**: + - `name`: The domain name to register. + - `nameOwner`: The owner of the domain. + - `secret`: The secret used in the commitment process. + - `duration`: The registration duration in years. + - **Usage Note**: The registration cost is calculated based on the domain name and is transferred using RIF tokens. + + ```solidity + function register(string calldata name, address nameOwner, bytes32 secret, uint duration) external { + uint cost = executeRegistration(name, nameOwner, secret, duration); + require(rif.transferFrom(msg.sender, pool, cost), "Token transfer failed"); + } + ``` + + - **Function: `tokenFallback`** + - **Parameters**: + - `from`: The address sending the tokens. + - `value`: The amount of tokens sent. + - `data`: Additional data for the registration process. + - **Returns**: `true` if the transaction is successful. + + ```solidity + function tokenFallback(address from, uint value, bytes calldata data) external returns (bool) { + require(msg.sender == address(rif), "Only RIF token"); + require(data.length > 88, "Invalid data"); + } + ``` \ No newline at end of file diff --git a/docs/01-concepts/rif-suite/rns/specs/registry.md b/docs/01-concepts/rif-suite/rns/specs/registry.md new file mode 100644 index 00000000..1a2b5ba9 --- /dev/null +++ b/docs/01-concepts/rif-suite/rns/specs/registry.md @@ -0,0 +1,68 @@ +--- +sidebar_label: Registry Specs +sidebar_position: 400 +title: "Registry Specs" +tags: [rif, rns, rif-name-service, Registry Specs] +description: "The registry contract provides a simple mapping between a domain and its resolver. " +--- + +The registry contract provides a simple mapping between a domain and its resolver. + +This contract manages all aspects of domain ownership, including transferring ownership and creating subdomains. + +Each entry in the registry points to a resolver, which handles the resolution between the name domain and the desired resource. + +The RNS Registry contract provides both data access and modification capabilities through the following functions: + + +## Access Functions + +- Ownership + ``` + function owner(bytes32 node) constant returns (address); + ``` + + Retrieves the owner (registrar) of the specified node. + +- Resolution + ``` + function resolver(bytes32 node) constant returns (address); + ``` + + Returns the resolver for the specified node. + +- Caching + ``` + function ttl(bytes32 node) constant returns (uint64); + ``` + Retrieves the time-to-live (TTL) of the specified node. The TTL defines the maximum period during which the node's information can be cached. + +## Modify Functions +The RNS Registry contract also allows for the modification of node data through the following functions: + +- Ownership + ``` + function setOwner(bytes32 node, address owner); + ``` + + Transfers ownership of a node to another registrar. This function may only be called by the current owner of node. A successful call to this function logs the `Transfer(bytes32 indexed, address)` event. + + ``` + function setSubnodeOwner(bytes32 node, bytes32 label, address owner); + ``` + + Creates a new node `label.node` and sets its owner to owner, or updates the node with a new owner if it already exists. This function may only be called by the current owner of node. A successful call to this function logs the `NewOwner(bytes32 indexed, bytes32 indexed, address)` event. + +- Resolution + ``` + function setResolver(bytes32 node, address resolver); + ``` + + Sets the Resolver address for node, the contract that handles the desired resolutions. This function may only be called by the owner of node. A successful call to this function logs the `NewResolver(bytes32 indexed, address)` event. + +- Caching + ``` + function setTTL(bytes32 node, uint64 ttl); + ``` + Sets the TTL for a node. A node's TTL applies to the 'owner' and 'resolver' records in the Registry, as well as to any information returned by the associated resolver. + diff --git a/docs/01-concepts/rif-suite/rns/specs/resolver.md b/docs/01-concepts/rif-suite/rns/specs/resolver.md new file mode 100644 index 00000000..ad0a9514 --- /dev/null +++ b/docs/01-concepts/rif-suite/rns/specs/resolver.md @@ -0,0 +1,252 @@ +--- +sidebar_label: Resolver Specs +sidebar_position: 400 +title: "Resolver Specs" +tags: [rif, rns, rif-name-service, Resolver Specs] +description: "Resolvers may implement any subset of the record types specified here. Where a record types specification requires a resolver to provide multiple functions." +--- + +Resolvers may implement any subset of the record types specified here. Where a record types specification requires a resolver to provide multiple functions, the resolver MUST implement either all or none of them. + +Resolvers MUST specify a fallback function that throws. + +Resolvers must implement ERC-165 interface detection standard. supportsInterface method must return if the interfaceID queried is simply equal to the signature hash of the function that resolves the desired resource record. + +Currently standardized resolver interfaces are specified below. + +> Check out [definitive resolver](/rif/rns/architecture/definitive-resolver) for implementation details. + + +### Contract address + +Provides the contract address for the specified domain. + +```solidity +function addr(bytes32 node) returns (address); +``` + +- `node`: the namehash of the domain to query for. +- Returns the contract address of the specified domain. A zero address is returned if the node has no address specified. + +When updated emits + +```solidity +event AddrChanged(bytes32 indexed node, address a); +``` + +Interface ID: `0x3b3b57de` + +Specification: [EIP-137](https://eips.ethereum.org/EIPS/eip-137#resolver-specification) + +Resolution protocol: + +1. Query the resolver address to the registry. +2. Query `addr` to the resolver contract. +3. Check for non zero-address response. + +> No modifications to the adoption of this protocol. + +### Multicoin address + +Add multiple `addr` fields for resolvers, which permit resolution of addresses across multiple blockchains. + +```solidity +function addr(bytes32 node, uint coinType) external view returns(bytes memory); +``` + +- `node`: The name hash of the domain to query for. +- `coinType`: The cryptocurrency coin type index from [SLIP44](https://github.com/satoshilabs/slips/blob/master/slip-0044.md). +- Returns the cryptocurency address in its native binary format. A zero-length string is returned if the specified coin ID does not exist on the specified node. + +> Detailed descriptions of the binary encodings for several popular chains are provided in the [Address Encoding section](https://eips.ethereum.org/EIPS/eip-2304#address-encoding) of EIP-2304. + +When updated emits + +```solidity +event AddressChanged(bytes32 indexed node, uint coinType, bytes newAddress); +``` + +Specification: [EIP-2304](https://eips.ethereum.org/EIPS/eip-2304) + +Interface ID: `0xf1cb7e06` + +Deprecates: [RNSIP-03](https://github.com/rnsdomains/RNSIPs/blob/master/IPs/RNSIP03.md) + +Resolution protocol: + +1. Query the resolver address to the registry. +2. Query `addr` to the resolver contract with the specified coin type. +3. Check for non zero-length string response. + +Resolvers implementing this interface may utilise a fallback strategy regarding backward compatibility. + +1. Query the resolver address to the registry. +2. Query `addr` to the resolver contract with the specified coin type. +3. If zero-length string is returned, query `chainAddr` to the resolver contract. +4. Check for non zero-length string response. + +### Contenthash + +System of mapping names to network and content addresses. + +```solidity +function contenthash(bytes32 node) external view returns (bytes memory); +``` + +- `node`: the namehash of the domain to query for. +- Returns a machine-readable multicodec representation of the content address. The format is specified as follows: ``. + +> `protoCode`s and their meanings are specified in the [multiformats/multicodec](https://github.com/multiformats/multicodec) repository. + +When updated emits: + +```solidity +event ContenthashChanged(bytes32 indexed node, bytes hash); +``` + +Interface ID: `0xbc1c58d1` + +Specification: [EIP-1577](https://eips.ethereum.org/EIPS/eip-1577) + +Deprecates: `content` and [`multihash`](https://eips.ethereum.org/EIPS/eip-1577) fields. + +Resolution protocol: + +1. Query the resolver address to the registry. +2. Query `contenthash` to the resolver contract. +3. Check for non zero-length string response. + +Resolvers implementing this interface may utilise a fallback strategy regarding backward compatibility. + +1. Query the resolver address to the registry. +2. Query `contenthash` to the resolver contract. +3. If zero-length string is returned, query `multihash` to the resolver contract. +4. If zero-length string is returned, query `content` to the resolver contract. +5. Check for non zero-length string response. + + +### Contract ABI + +Easy lookup of contract interfaces by callers + +```solidity +function ABI(bytes32 node, uint256 contentType) constant returns (uint256, bytes memory); +``` + +- `node`: the namehash of the domain to query for. +- `contentType`: a bitwise OR of the ABI formats accepted by the caller. +- Returns the content type of the return value and the ABI data. `(0, "")` is returned if the domain has no ABI specified, or does not support any of the requested formats. + +> The currently recognised encodings are described in [ABI encodings section](https://eips.ethereum.org/EIPS/eip-205#abi-encodings) EIP-205. + +When updated emits: + +```solidity +event ABIChanged(bytes32 indexed node, uint256 indexed contentType); +``` + +> Notice that the updated content is not emitted. It requires an extra query to get the resource record when logged, but saves gas setting it. + +Interface ID: `0x2203ab56` + +Specification: [EIP-205](https://eips.ethereum.org/EIPS/eip-205) + +Resolution protocol: + +1. Query the resolver address to the registry. +2. Query `ABI` to the resolver contract with a bitwise OR of the ABI formats accepted by the caller. +3. Check for a response that is not `(0, "")`. + +### SECP256k1 public keys + +Permits the lookup of SECP256k1 public keys. + +```solidity +function pubkey(bytes32 node) constant returns (bytes32 x, bytes32 y); +``` + +- `node`: the namehash of the domain to query for. +- Returns the coordinates of an uncompressed SECP256k1 curve point comprising the public key. A `(0, 0)` value is returned if the domain has no pubkey specified. + +When updated emits: + +```solidity +event PubkeyChanged(bytes32 indexed node, bytes32 x, bytes32 y); +``` + +Interface ID: `0xc8690233` + +Specification: [EIP-619](https://github.com/ethereum/EIPs/pull/619) + +Resolution protocol: + +1. Query the resolver address to the registry. +2. Query `pubkey` to the resolver contract. +3. Check for non `(0, 0)` response. + +### Text records + +Permits the lookup of arbitrary key-value text data. + +```solidity +function text(bytes32 node, string key) constant returns (string text); +``` + +- `node`: the namehash of the domain to query for. +- `key`: the text data key to query. +- Returns the associated text data. If the key is not present, the zero-length string is returned. + +> For supported keys refer to the [initial recommended keys section of EIP-634](https://eips.ethereum.org/EIPS/eip-634#initial-recommended-keys). + +When updated emits: + +```solidity +event TextChanged(bytes32 indexed node, string indexed indexedKey, string key); +``` + +Interface ID: `0x59d1d43c` + +Specification: [EIP-634](https://eips.ethereum.org/EIPS/eip-634) + + +Resolution protocol: + +1. Query the resolver address to the registry. +2. Query `text` to the resolver contract with the specified key. +3. Check for non zero-length string response. + +### Interface discovery + +Defines a method of associating contract interfaces with domains and addresses, and of discovering those interfaces. + +```solidity +function interfaceImplementer(bytes32 node, bytes4 interfaceID) external view returns (address); +``` + +- `node`: the namehash of the domain to query for. +- `interfaceID`: The EIP 165 interface ID to query for. +- Returns the address of a contract that implements the specified interface for the given domain. A zero-address value is returned if there is no interface matching that interface ID for that node. + +When updated emits: + +```solidity +event InterfaceChanged(bytes32 indexed node, bytes4 indexed interfaceID, address implementer); +``` + +Interface ID: `0xb8f2bbb4` + +Specifitcation: [EIP-1844](https://eips.ethereum.org/EIPS/eip-1844) + +Resolution protocol: + + +1. Query the resolver address to the registry. +2. Query `interfaceImplementer` to the resolver contract with the specified interface ID. +4. Check for non zero-address response. + +Resolvers implementing this interface may utilise a fallback strategy. This strategy has nothing to do with backward compatibility. + +1. Query the resolver address to the registry. +2. Query `interfaceImplementer` to the resolver contract with the specified interface ID. +3. If zero-address string is returned, query `addr` to the resolver contract and query `supportsInterface` to the given contract address. +4. Check for non zero-address response. diff --git a/static/img/rif/rns/theStack.png b/static/img/rif/rns/theStack.png new file mode 100644 index 0000000000000000000000000000000000000000..69a7c48a3c786ac1d276d78fe812f520574b396b GIT binary patch literal 23577 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 literal 0 HcmV?d00001 From 7996b64b2c51582d8da14168c2dfd9334371669d Mon Sep 17 00:00:00 2001 From: Wisdom Nwokocha Date: Fri, 4 Oct 2024 15:38:49 +0100 Subject: [PATCH 2/2] Added a new component I added a new component called renewal which helsop to renew a domain --- docs/01-concepts/rif-suite/rns/index.md | 25 +++++++------------ docs/01-concepts/rif-suite/rns/specs/index.md | 2 +- 2 files changed, 10 insertions(+), 17 deletions(-) diff --git a/docs/01-concepts/rif-suite/rns/index.md b/docs/01-concepts/rif-suite/rns/index.md index 5efde711..9aca5a8b 100644 --- a/docs/01-concepts/rif-suite/rns/index.md +++ b/docs/01-concepts/rif-suite/rns/index.md @@ -90,27 +90,20 @@ The design of the RIF Name Service is shaped by specific goals: ## Elements of the RNS -RNS has three major components: +RNS has four major components: -- **The RNS Registry**, which is specification for a tree structured name space and data associated with the names. +| **Component** | **Description** | **Specs** | +|------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-----------| +| **RNS Registry** | The RNS Registry is a specification for a tree-structured namespace and the data associated with the names. Conceptually, each node and leaf in the domain name space tree represents a set of information. Query operations attempt to extract specific types of information from a particular set. A query specifies the domain name of interest and the type of resource information desired. | [Specs](./specs/registry) | +| **RNS Resolvers** | RNS Resolvers are contracts that provide information from a name in response to client requests. Resolvers must answer a query directly or use referrals to other resolvers. Typically, a resolver is a contract's public function that is directly accessible to user programs or other contracts. No specific protocol is required between the resolver and the user program. | [Specs](./specs/resolver) | +| **RNS Registrar** | The RNS Registrar is a critical component within the RIF Name Service, managing the registration of `.rsk` domain names. This contract has the authority to register names in the RSK Owner contract, ensuring that new domain registrations are handled securely and efficiently. | [Specs](./specs/registrar) | +| **Renewer** | The Renewer is a contract designed to facilitate the renewal of names registered in the Node Owner. It is equipped with permissions to renew these names and provides flexibility in how the renewal is executed. - Conceptually, each node and leaf of the domain name space tree names a set of information, and query operations are attempts to extract specific types of information from a particular set. A query names the domain name of interest and describes the type of resource information that is desired. - - [Specs](./specs/registry) - -- **RNS Resolvers** are contracts that provide information from a name in response to client requests. - - Resolvers must be able to answer a query directly, or pursue the query using referrals to other resolvers. A resolver will typically be a contract's public function that is directly accessible to user programs or other contracts; hence no protocol is necessary between the resolver and the user program. - - [Specs](./specs/resolver) - -- **RNS Registrar** is a critical component within the RIF Name Service, responsible for managing the registration of `.rsk` domain names. This contract is granted the authority to register names in the RSK Owner contract, ensuring that new domain registrations are handled securely and efficiently. -[Specs](./specs/registrar) - -These three components roughly correspond to the three layers or views of the domain system: +These fours components roughly correspond to the four layers or views of the domain system: - From the user's point of view, the domain system is accessed through a simple resolution operation. The domain space consists of a single tree and the user can request information from any section of the tree. - From the resolver's point of view, the domain system is composed of an unknown number of names. Each name has a corresponding resolver that provides information for a set of resolution types directly. - From the registry's point of view, the domain system consists of a hierarchical tree where each leaf has an owner (contract or account) and an associated resolver that provides information of the name. +- From the **renewal's point of view**, the domain system ensures continued ownership through renewal processes, facilitating the payment of fees for name renewals via supported methods like ERC-20 or ERC-677. ## Guidelines on use diff --git a/docs/01-concepts/rif-suite/rns/specs/index.md b/docs/01-concepts/rif-suite/rns/specs/index.md index 85aa59ba..34f349b3 100644 --- a/docs/01-concepts/rif-suite/rns/specs/index.md +++ b/docs/01-concepts/rif-suite/rns/specs/index.md @@ -25,7 +25,7 @@ For simplicity, a valid RNS domain is defined as follows: - All labels must be alphanumeric and lower case - All labels and the TLD are delimited by `.` -The reference implementation for RNS domain validation can be found in `rns.js` +The reference implementation for RNS domain validation can be found in [RNS SDK](https://www.npmjs.com/package/@rsksmart/rns-sdk) - AVAILABLE_TLDS in src/constants.ts - isValidDomain and isValidLabel in src/utils.ts