A ENS Subdomain Decentralized Autonomous Organization (SDAO) is a DAO organized around a ENS domain name (e.g sismo.eth
), which is owned by the DAO.
SDAO members are all owners of a subdomain (e.g ziki.sismo.eth
). You can read about Sismo's specific SDAO here.
This repository publishes a set of smart contracts that can be used to kickstart a SDAO. Please feel free to create issues or to connect with its maintainer: @sismo_eth on twitter or contact at sismo.io
If using npm
npm install @sismo-core/ens-sdao
If using yarn
yarn add @sismo-core/ens-sdao
Once the package installed, the contracts are available using regular solidity imports.
Your Subdomain DAO contract can be constructed by using the available extensions.
As an example, here is the Sismo Subdomain DAO contract used for one of our releases
pragma solidity >=0.8.4;
import {PublicResolver} from '@ensdomains/ens-contracts/contracts/resolvers/PublicResolver.sol';
import {ENS} from '@ensdomains/ens-contracts/contracts/registry/ENS.sol';
import {SDaoRegistrar} from '@sismo-core/ens-sdao/contracts/sdao/SDaoRegistrar.sol';
import {SDaoRegistrarLimited} from '@sismo-core/ens-sdao/contracts/sdao/extensions/SDaoRegistrarLimited.sol';
import {SDaoRegistrarReserved} from '@sismo-core/ens-sdao/contracts/sdao/extensions/SDaoRegistrarReserved.sol';
import {SDaoRegistrarERC721Generator, IERC721Minter} from '@sismo-core/ens-sdao/contracts/sdao/extensions/SDaoRegistrarERC721Generator.sol';
import {SDaoRegistrarCodeAccessible} from '@sismo-core/ens-sdao/contracts/sdao/extensions/SDaoRegistrarCodeAccessible.sol';
contract SismoSDaoRegistrar is
SDaoRegistrar,
SDaoRegistrarLimited,
SDaoRegistrarReserved,
SDaoRegistrarERC721Generator,
SDaoRegistrarCodeAccessible
{
uint256 public _groupId;
event GroupIdUpdated(uint256 groupId);
constructor(
ENS ensAddr,
PublicResolver resolver,
IERC721Minter erc721Token,
bytes32 node,
address owner,
uint256 reservationDuration,
uint256 registrationLimit,
uint256 groupId,
address codeSigner
)
SDaoRegistrarCodeAccessible('Sismo', '1.0', codeSigner)
SDaoRegistrarERC721Generator(erc721Token)
SDaoRegistrarLimited(registrationLimit)
SDaoRegistrarReserved(reservationDuration)
SDaoRegistrar(ensAddr, resolver, node, owner)
{
_groupId = groupId;
}
function _beforeRegistration(address account, bytes32 labelHash)
internal
virtual
override(
SDaoRegistrar,
SDaoRegistrarReserved,
SDaoRegistrarLimited,
SDaoRegistrarERC721Generator
)
{
super._beforeRegistration(account, labelHash);
}
function _afterRegistration(address account, bytes32 labelHash)
internal
virtual
override(SDaoRegistrar, SDaoRegistrarLimited, SDaoRegistrarERC721Generator)
{
super._afterRegistration(account, labelHash);
}
function _getCurrentGroupId() internal view override returns (uint256) {
return _groupId;
}
function updateGroupId(uint256 groupId) external onlyOwner {
_groupId = groupId;
emit GroupIdUpdated(groupId);
}
}
The core contract SDaoRegistrar
distributes subdomains to registrants. This contract must be set as the owner of the DAO domain name (e.g domain.eth
). It contains the registration logic.
The organisation of the code is heavily inspired from the OpenZeppelin contracts packages. We built modular extensions extending contracts and presets. (see the ERC20 approach).
Extensions are for developers to choose what additional features or restrictions should be applied to their SDaoRegistrar.
A list of presets is also available. They are pre-configured contracts, ready to be deployed. Each of them uses a different set of extensions.
The code of this repository resolves around ENS, it is advised to be familiar with Ethereum Name Service notions.
The contract allows first-come first-served (FCFS) registration of a subdomain, e.g. label.domain.eth
through the register
method.
The ownership of the subdomain is given to the registrant. The newly created subdomain resolves to the registrant.
function register(string memory label)
public
virtual
override
onlyUnrestricted
{
_register(_msgSender(), label);
}
The internal registration method _register
is exposed internally for extensions of the SDaoRegistrar
. Two hooks can be used: _beforeRegistration
and _afterRegistration
function _register(address account, string memory label) internal {
bytes32 labelHash = keccak256(bytes(label));
_beforeRegistration(account, labelHash);
// [...]: subdomain created, subdomain resolves towards accounts, subdomain owner is account
_afterRegistration(account, labelHash);
}
constructor(
ENS ensAddr, // ENS Registry
PublicResolver resolver, // Resolver to be used
bytes32 node, // nameHash('domain.eth')
address owner // owner of the registrar
)
The node corresponds to the namehash of the domain of the SDAO, e.g. domain.eth
.
See contracts/sdao/ISDaoRegistrar.sol
and contracts/sdao/SDaoRegistrar.sol
for the exact interface and implementation.
An extension is an abstract contract which inherits the SDaoRegistrar
core contract.
It may add other public methods for registration using the internal registration method or/and implements the beforeRegistration
and afterRegistration
hooks.
extensions are easy to read: /contracts/sdao/extensions/*.sol
function _beforeRegistration(address account, bytes32 labelHash)
internal
virtual
override
{
super._beforeRegistration(account, labelHash);
if (block.timestamp - DAO_BIRTH_DATE <= _reservationDuration) {
address dotEthSubdomainOwner = ENS_REGISTRY.owner(
keccak256(abi.encodePacked(ETH_NODE, labelHash))
);
require(
dotEthSubdomainOwner == address(0x0) ||
dotEthSubdomainOwner == _msgSender(),
'SDAO_REGISTRAR_RESERVED: SUBDOMAIN_RESERVED'
);
}
}
A reservation period is introduced during which registration of a subdomain subdomain.domain.eth
is blocked if the related subdomain.eth
is owned by someone else than the registrant. The reservation period can be updated by the owner of the contract.
See contracts/sdao/extensions/SDaoRegistrarReserved.sol
for the implementation.
function _beforeRegistration(address account, bytes32 labelHash)
internal
virtual
override
{
super._beforeRegistration(account, labelHash);
require(
_counter < _registrationLimit,
'SDAO_REGISTRAR_LIMITED: REGISTRATION_LIMIT_REACHED'
);
}
function _afterRegistration(address account, bytes32 labelHash)
internal
virtual
override
{
super._afterRegistration(account, labelHash);
_counter += 1;
}
A counter for the number of registered subdomains and a registration limit number are added. If the counter reaches the registration limit, registration is blocked. The registration limit can be updated by the owner of the contract.
See contracts/sdao/extensions/SDaoRegistrarLimited.sol
for the implementation.
function _afterRegistration(address account, bytes32 labelHash)
internal
virtual
override
{
super._afterRegistration(account, labelHash);
bytes32 childNode = keccak256(abi.encodePacked(ROOT_NODE, labelHash));
ERC721_TOKEN.mintTo(account, uint256(childNode));
}
An ERC721 is minted and the registration is blocked if the balance of the registrant is not zero.
See contracts/sdao/extensions/SDaoRegistrarERC721Generator.sol
for the implementation.
function _afterRegistration(address account, bytes32 labelHash)
internal
virtual
override
{
super._afterRegistration(account, labelHash);
(uint256 id, bytes memory data) = _getToken(account, labelHash);
ERC1155_MINTER.mint(account, id, 1, data);
}
An ERC1155 token is minted for the registrant after each registration. The ERC1155 token ID and data are left free to be implemented by the developer.
The registration is blocked if the balance of the registrant is not zero, based on a balanceOf
method to be implemented by the developer.
See contracts/sdao/extensions/SDaoRegistrarERC1155Generator.sol
for the implementation.
function registerWithAccessCode(
string memory label,
address recipient,
bytes memory accessCode
) external override {
// [...]
require(
!_consumed[digest],
'SDAO_REGISTRAR_LIMITED_CODE_ACCESSIBLE: ACCESS_CODE_ALREADY_CONSUMED'
);
require(
digest.recover(accessCode) == _codeSigner,
'SDAO_REGISTRAR_LIMITED_CODE_ACCESSIBLE: INVALID_ACCESS_CODE OR INVALID_SENDER'
);
// [...]
}
A new public method of registration registerWithAccessCode
is added. It allows to register a subdomain if a valid access code is given.
An access code is a signed message according to the EIP712 specification. The message contains domain specific information set at the deployment of the contract and the data of the message. The data contains the recipient address, which will receive the subdomain, and a number called group ID
.
The message must be signed by a specific code signer
address in order to be valid. The code signer
value is managed by the owner of the contract. The group ID
is retrieved using an internal method that needs to be implemented in the final contract.
An access code can only be consumed once.
See contracts/sdao/extensions/SDaoRegistrarCodeAccessible.sol
for the implementation.
function claim(string memory label, address account) public override {
bytes32 labelHash = keccak256(bytes(label));
address bookingAddress = ENS_LABEL_BOOKER.getBooking(labelHash);
require(
bookingAddress != address(0),
'SDAO_REGISTRAR_CLAIMABLE: LABEL_NOT_BOOKED'
);
require(
bookingAddress == _msgSender() || owner() == _msgSender(),
'SDAO_REGISTRAR_CLAIMABLE: SENDER_NOT_ALLOWED'
);
_register(account, label);
ENS_LABEL_BOOKER.deleteBooking(labelHash);
}
An address of an ENS Label Booker is added. The latter manages a mapping of booking formed by a link between a subdomain and a booking address.
A new public method of registration claim
is added. It allows to register a subdomain if the sender of the message is the booking address associated to the subdomain.
The FCFS served registration is blocked if the subdomain is already booked.
See contracts/sdao/extensions/SDaoRegistrarClaimable.sol
and contracts/ens-label-booker/ENSLabelBooker.sol
for the implementation of the extension and the Label Booker contracts.
A preset is an SDAO Registrar contract extended with a set of extensions. It may be considered as ready to use contracts or examples of the extensions usage.
- Limited Reserved ERC1155 Preset
- hardhat task
deploy-sdao-preset-erc1155
) - implements
SDaoRegistrarLimited
,SDaoRegistrarReserved
andSDaoRegistrarERC1155Generator
extensions.
- hardhat task
- Limited Reserved ERC721 Preset
- hardhat task:
deploy-sdao-preset-erc721
- implements
SDaoRegistrarLimited
,SDaoRegistrarReserved
andSDaoRegistrarERC721Generator
extensions.
- hardhat task:
- Code Accessible Preset
- hardhat task:
deploy-sdao-preset-code-accessible
- implements
SDaoRegistrarCodeAccessible
extension
- hardhat task:
- Claimable Preset
- hardhat task:
deploy-sdao-preset-claimable
- implements
SDaoRegistrarClaimable
extension
- hardhat task:
See contracts/sdao/presets/*.sol
for the implementations.
Note: Most of the scripts are used for development purposes. Deployment scripts for the various presets are available but it is strongly advised to understand them before using them.
const deployedENS: DeployedEns = await HRE.run('deploy-ens-full', {sDao: true, log: true});
// with
export type DeployedEns = {
ensDeployer: ENSDeployer;
registry: ENSRegistry;
registrar: EthRegistrar;
reverseRegistrar: ReverseRegistrar;
publicResolver: PublicResolver;
};
Deploys a suite of ENS contracts. The deployed .eth Registrar is simplified in order to allow direct registration instead of the usual two steps registration process.
The boolean flag sDao
allows to additionally deploy a SDAO Registrar.
See tasks/deploy-ens-full.ts
for the script and the full list of parameters.
Deploys a Label Booker contract.
The name
of the targeted domain is given as parameter, i.e. <name>.eth
.
See tasks/deploy-label-booker.ts
for the script and the full list of parameters.
Install the dependencies.
Compile the smart contracts and generate the associated TypeScript types.
Launch all the tests.
Individual scripts are available for each test suite.
The contracts/test-utils
repository contains various helpful contracts for the tests, in particular the ENSDeployer
that can be used in order to redeploy a suite of ENS contracts with a modified Registrar.
The code of this repository is released under the MIT License.