This is the module for access control of the whole ZNS system. Every contract inherits the AccessControlled
abstract contract that implements:
accessController
as a state variable to be present in every child- a getter for accessController
- a setter declaration for
accessController
to make sure every child contract implements it locally - an event to fire every time
accessController
is changes in state onlyAdmin
andonlyRegistrar
modifiers since they are used in most child contracts
Upon inheriting the AccessControlled
every contract is connected to the overall access control of the system through ZNSAccessController
address that has to be passed at initialization time in order to prevent non-admin accounts from setting crucial values in storage.
ZNSAccessController
itself inherits AccessControl
from Open Zeppelin and adds some specific logic to provide ZNS contracts with necessary functions to limit system access to crucial functions.
- Provide a template for every contract in the system on how to add access control to its logic (via
AccessControlled
) - Make sure important functions are implemented in children so that
accessController
variable can not be lost and can be reset on a live system - Provide constants for all active roles in the system that will be used in production to validate callers
- Provide functionality to set good role structure for system safety
- Provide functionality to set and remove roles from accounts
- Provide easy role checking functionality that all contracts using this module can access within their logic
Key storage contract of the system responsible for basic crucial data about every domain/subdomain registered. The below data is mapped to each domain’s hash in ZNSRegistry.records
.
owner
(address) - owner of a domain. This address is necessary for the system to keep track of owners of domains and to provide secure access to domain management functionality. Only domain owner can modify and control a domain. Can be any arbitrary Ethereum address: smart contract, EOA. In cases when subdomain distribution (or other functionality) is being managed by the arbitrary contract, this contract has to be the owner of the parent domain.resolver
(address) - address of a contract responsible for binding each domain name to a content of a certain type (e.g. ethereum address). For more info see the "Resolvers" section.
Every domain registered has to settle on this contract with its crucial data (above) written into a mapping keyed by domain name hash (bytes32).
ZNSRegistry
also has a mapping of operators
in its state to manage operators for each owner. An operator gets access to all domains of the owner and can perform any operation on them. This is a crucial functionality for the system to be able to provide a way for owners to delegate domain management to other accounts. An operator is not able to change the owner of the domain in ZNSRegistry
.
- Serves as the last step of a domain registration process by saving the domain data in its storage
- Official final reference of an existence of a certain domain - we call
ZNSRegistry
to verify that a new domain being minted has not been minted already - Is the first step in domain source discovery - we call
ZNSRegistry.getDomainResolver()
to find the Resolver that will tell us what this domain was created for (contract or wallet address, string, metadata hash, etc.). Currently ONLYZNSAddressResolver
is implemented, but more resolvers to come in the future for more data types. - Reference for crucial domain related data (owner, resolver).
- Provide a way to install operators for any owner to allow them access to changing resolvers without the presence of the owner
A single token contract (ERC-721) responsible for tokenizing every domain/subdomain in the system, providing standard token functionality (e.g. transfer, mint, burn, etc.) for the ease of domain management along with ZNS specific functionality added on top of the ERC-721 ABI.
- Mint a new token every time a domain is registered atomically within the register transaction (i.e.
ZNSRegistrar.registerRootDomain()
->ZNSDomainToken.register()
) - Burn the respective token every time a domain is revoked atomically within the revoke transaction (
ZNSRegistrar.revokeDomain()
->ZNSDomainToken.revoke()
) - Determine and check owner of any given domain token by the tokenId
- Transfer/sell domain token to change the owner of the Token
- Serve as a standard ERC-721 token with all the functionality provided so the token can be traded and managed by other applications.
Resolvers and ZNSAddressResolver
The zNS system is expected to have multiple Resolver contracts, each being responsible for resolutions to their own supported types for domain sources. Zero will deploy a certain amount of them to support different data types in the future, but resolvers can potentially be developed, deployed and managed by any parent domain owner to provide more data type resolutions for their subdomains.
A resolver is structured to be a simple contract, having a mapping of a domain namehash to the specific source type (e.g. bytes32 => address
OR bytes32 => bytes
, etc.). Each Resolver can support one data type at a time or can be a combined one, supporting multiple.
The only resolver currently implemented is ZNSAddressResolver
. It supports only address data type and has a simple mapping of bytes32 domainHash => address contentAddress
.
- Provide a straightforward binging of a domain namehash to specific domain source data (e.g.
hash("cooldomainname") => 0x1bc5d767ff…
) - Provide a simple and straightforward resolution from a name to domain source
- Provide the way for any domain owner to change their domain source at any point in time along with simple access control which will not allow anyone else, other than the domain owner, to change this data
- Interface checking logic (ERC-165) to provide easy type checking on supported types for the current Resolver.
The zNS system supports using multiple different contracts as "pricers". A Pricer contract is a contract that encompases a certain mathematical model to price the registrations of its subdomains. Every time a new subdomain is minted one of the pricer contracts is used to determine how much a subdomain buyer needs to pay its parent.
Currently, 2 different pricers are implemented and supported:
ZNSCurvePricer
is one of the options a domain owner can use to price their subdomains. It is based on a curve formula that takes the length of the label string and calculates the price compared to the maximum price the owner of the parent domain has set.
The curve has 3 main parts:
- the flat beginning, until
baseLength
, where a fixedmaxPrice
is returned; - a curve part where the price declines the longer the domain label is;
- the flat end, from
maxLength
, where a fixedminPrice
is returned.
This contract also includes the possibility to set a certain feePercentage
that will be applied to the calculated price and will be sent to the domain owner.
The feePercentage
is set by the domain owner and can be changed at any point in time, but not be taken into account unless the distribution type of the parent domain is set to STAKE
.
Another feature of the contract is the ability to set precisionMultiplier
that allows a calculated price to be truncated to a certain precision.
The precisionMultiplier
is set by the domain owner and can be changed at any point in time.
Zer0 uses
ZNSCurvePricer
as the default pricer for all World domains.
To learn more about the formula, fees and precisions, please refer to the contract documentation - ZNSCurvePricer.
- Provides a deterministic formula for domain name price calculations based on the length of the label string, where a price declines the longer the label is
- Provides a way to set and get a calculated fee value based on the price of the domain and the fee percentage set by the domain owner
- Provides ways to reconfigure pricing and fees on a live system based on the set of variables through their setters
ZNSFixedPricer
is another, more straightforward contract a domain owner can use to price their subdomains. It is based on a fixed price that is set by the owner of the parent domain.
If a parent domain is using ZNSFixedPricer
, the prices returned for subdomains will be fixed and always the same, until the owner changes the price on the contract.
This contract also includes the possibility to set a certain feePercentage
that will be applied to the calculated price and will be sent to the domain owner.
The feePercentage
is set by the domain owner and can be changed at any point in time, but will ONLY be taken into account if the distribution type of the parent domain is STAKE
.
- Provides a more straightforward way to set a fixed price that will be used for all subdomains
- Provides a way to set and get a calculated fee value based on the price of the domain and the fee percentage set by the domain owner
- Provides ways to reconfigure pricing and fees on a live system based on the set of variables through their setters
This contract is responsible for all payments in the system, regardless of the domain depth and currently supports 2 types of payments:
DIRECT
Payment Type - the most straighforward way whereZNSTreasury
withdraws funds from the buyer's wallet and sends them to the parent domain owner'sbeneficiary
.STAKE
Payment Type - here the buyer is required to stake the parent'sprice
taken from one of the Pricer contracts ontoZNSTreasury
contract itself. In this case, a parent is allowed to set a certainfeePercentage
that will be applied to the calculated price and will be sent to the domain owner. That payment must also include an additional fee charged by Zer0's protocol. This is taken on top of the above both on invest and devest of the stake. TheSTAKE
payment will be refunded back upon domain revocation minus the fees. The treasury itself performs the actual stake transfers and stores their amounts along with the token address used, mapped bydomainHash
. These amounts are used to unstake upon domain revocation later.
The treasury will send any generated protocol fees to the Zer0 Vault address specified by the zNS Admin.
ZNSTreasury
, besides stake data, also stores the PaymentConfigs
for each domain, set by the domain owner, which is a struct that contains the following data:
token
- the token to be used for paymentsbeneficiary
- the address that will receive the funds upon domain registration, ifSTAKE
is used aspaymentType
onZNSSubRegistrar
thebeneficiary
will ONLY receive thestakeFee
set by the domain owner, ifDIRECT
is used, thebeneficiary
will receive the wholeprice
The above configs are used to determine how to charge the buyer and where to send funds.
- Perform stake transfers based on the prices and fees acquired from the pricer contract as a part of the register transaction
- Perform stake withdrawals for users who are removing their domains from zNS as a part of the revoke transaction
- Perform direct transfers of price and all fees for registration
- Store staked amount and token used in state, tied to the domain hash and not a specific owner to be more adaptable to owner changes without the need to perform extra logic
This contract is the entry point for 3 main operations:
- Registering a new Root Domain
- Reclaiming a Domain of any level
- Revoking a Domain of any level
ZNSRootRegistrar
is connected to all other contracts and combines multiple operations into larger logical, atomic transactions. The root registrar routes the logic to other contracts inside zNS to achieve proper results for domain operations. In doing this, proper sequence is kept in ordering certain operations with different contracts.
ZNSRegistrar
is the only contract in the system that has a proper role assigned to it. It is given the REGISTRAR_ROLE
in ZNSAccessController
to use it
as a way to protect other contracts to be called by any other address. This protects the data in the system and ensures proper operation execution.
Many other contract functions have onlyRegistrar()
modifiers which make sure that the caller can only be a contract with that specific role.
- Store all system crucial modules in state to be able to call required contracts for any flow.
- Provide the main function for registering Root Domains, make sure that operations, especially ones using other contracts are performed properly and in proper sequence. Combine logic of other contracts in one atomic transaction. Check name existence, take the stake from the user, mint a domain token, set the data in
ZNSRegistry
. - Provide the main function for reclaiming full domain ownership for owners of the specific Domain Token. Ensure that only the owner of the actual ERC-721 token owner can do this, call
ZNSRegistry
to update the owner in records. - Provide the main function for revoking domains. Make sure that the only address allowed to revoke owns both the name, in the domain record held by the
ZNSRegistry
and the token, held byZNSDomainToken
. This will callZNSRegistry
to delete record,ZNSDomainToken
to burn the token, andZNSTreasury
to unstake if needed.
Contract responsible for the registration of subdomains and distribution configs for domains of all levels that are used to set and determine how subdomains of any domain will be distributed.
This contract is only connected to ZNSRegistry
and ZNSRootRegistrar
and uses its common functions to finalize subdomain registration. It stores DistributionConfig
s for each domain in state, mapped by domain hash.
It also manages mintlists for each domain, where the owner of the domain can add and remove addresses that will be allowed or disallowed to mint subdomains of this domain.
This contract is also assigned the REGISTRAR_ROLE
at deploy time to be able to call other contracts with onlyRegistrar()
modifiers.
All the state variables on this contract are able to be reset by either the domain owner (for domain related setters) or Zer0 Admin for the setters of system variables.
For more information on Distribution Configs and Mintlists, please refer to the contract docs ZNSSubRegistrar.
- Provide a way to register subdomains of any level.
- Provide a way to set and get Distribution Rules (configs) for any domain.
- Provide a way to set and get Mintlists for any domain.
- Work together with
ZNSRootRegistrar
to finalize registration of subdomains.
For any additional contract specific documentation, see the "docs/contracts" folder