ens-usernames/contracts/registry/ENSSubdomainRegistry.sol
2018-08-30 09:44:22 -04:00

465 lines
16 KiB
Solidity

pragma solidity ^0.4.23;
import "../common/MerkleProof.sol";
import "../common/Controlled.sol";
import "../token/ERC20Token.sol";
import "../ens/ENS.sol";
import "../ens/PublicResolver.sol";
/**
* @author Ricardo Guilherme Schmidt (Status Research & Development GmbH)
* @notice Sell ENS subdomains of owned domains.
*/
contract ENSSubdomainRegistry is Controlled {
ERC20Token public token;
ENS public ens;
PublicResolver public resolver;
address public parentRegistry;
uint256 public releaseDelay = 365 days;
mapping (bytes32 => Domain) public domains;
mapping (bytes32 => Account) public accounts;
//slashing conditions
uint256 public subdomainMinLenght;
bytes32[] public reservedSubdomainsMerkleRoots;
event DomainPrice(bytes32 indexed namehash, uint256 price);
event DomainMoved(bytes32 indexed namehash, address newRegistry);
event SubdomainOwner(bytes32 subdomainHash, address accountOwner);
enum NodeState { Free, Owned, Moved }
struct Domain {
NodeState state;
uint256 price;
}
struct Account {
uint256 tokenBalance;
uint256 creationTime;
address accountOwner;
}
modifier onlyParentRegistry {
require(msg.sender == parentRegistry, "Migration only.");
_;
}
/**
* @notice Initializes a UserRegistry contract
* @param _token fee token base
* @param _ens Ethereum Name Service root address
* @param _resolver Default resolver to use in initial settings
* @param _subdomainMinLenght Minimum length of usernames
* @param _reservedSubdomainsMerkleRoots Merkle Roots of reserved subdomains
* @param _parentRegistry Address of old registry (if any) for account migration.
*/
constructor(
ERC20Token _token,
ENS _ens,
PublicResolver _resolver,
uint256 _subdomainMinLenght,
bytes32[] _reservedSubdomainsMerkleRoots,
address _parentRegistry
)
public
{
token = _token;
ens = _ens;
resolver = _resolver;
subdomainMinLenght = _subdomainMinLenght;
reservedSubdomainsMerkleRoots = _reservedSubdomainsMerkleRoots;
parentRegistry = _parentRegistry;
}
/**
* @notice Registers `_userHash` subdomain to `_domainHash` setting msg.sender as owner.
* @param _userHash choosen unowned subdomain hash
* @param _domainHash choosen contract owned domain hash
* @param _account optional address to set at public resolver
* @param _pubkeyA optional pubkey part A to set at public resolver
* @param _pubkeyB optional pubkey part B to set at public resolver
*/
function register(
bytes32 _userHash,
bytes32 _domainHash,
address _account,
bytes32 _pubkeyA,
bytes32 _pubkeyB
)
external
returns(bytes32 subdomainHash)
{
Domain memory domain = domains[_domainHash];
require(domain.state == NodeState.Owned, "Domain unavailable.");
subdomainHash = keccak256(abi.encodePacked(_domainHash, _userHash));
require(ens.owner(subdomainHash) == address(0), "ENS node already owned.");
require(accounts[subdomainHash].creationTime == 0, "Username already registered.");
accounts[subdomainHash] = Account(domain.price, block.timestamp, msg.sender);
if(domain.price > 0) {
require(token.allowance(msg.sender, address(this)) >= domain.price, "Unallowed to spend.");
require(
token.transferFrom(
address(msg.sender),
address(this),
domain.price
),
"Transfer failed"
);
}
bool resolvePubkey = _pubkeyA != 0 || _pubkeyB != 0;
bool resolveAccount = _account != address(0);
if (resolvePubkey || resolveAccount) {
//set to self the ownship to setup initial resolver
ens.setSubnodeOwner(_domainHash, _userHash, address(this));
ens.setResolver(subdomainHash, resolver); //default resolver
if (resolveAccount) {
resolver.setAddr(subdomainHash, _account);
}
if (resolvePubkey) {
resolver.setPubkey(subdomainHash, _pubkeyA, _pubkeyB);
}
ens.setOwner(subdomainHash, msg.sender);
}else {
//transfer ownship of subdone directly to registrant
ens.setSubnodeOwner(_domainHash, _userHash, msg.sender);
}
emit SubdomainOwner(subdomainHash, msg.sender);
}
/**
* @notice release subdomain and retrieve locked fee, needs to be called after `releasePeriod` from creation time.
* @param _userHash `msg.sender` owned subdomain hash
* @param _domainHash choosen contract owned domain hash
*/
function release(
bytes32 _userHash,
bytes32 _domainHash
)
external
{
bool isDomainController = ens.owner(_domainHash) == address(this);
bytes32 subdomainHash = keccak256(abi.encodePacked(_domainHash, _userHash));
Account memory account = accounts[subdomainHash];
require(account.creationTime > 0, "Username not registered.");
if (isDomainController) {
require(msg.sender == ens.owner(subdomainHash), "Not owner of ENS node.");
require(block.timestamp > account.creationTime + releaseDelay, "Release period not reached.");
ens.setSubnodeOwner(_domainHash, _userHash, address(this));
ens.setResolver(subdomainHash, address(0));
ens.setOwner(subdomainHash, address(0));
} else {
require(msg.sender == account.accountOwner, "Not the former account owner.");
}
delete accounts[subdomainHash];
if (account.tokenBalance > 0) {
require(token.transfer(msg.sender, account.tokenBalance), "Transfer failed");
}
emit SubdomainOwner(subdomainHash, address(0));
}
/**
* @notice updates funds owner, useful to move subdomain account to new registry.
* @param _userHash `msg.sender` owned subdomain hash
* @param _domainHash choosen contract owned domain hash
**/
function updateAccountOwner(
bytes32 _userHash,
bytes32 _domainHash
)
external
{
bytes32 subdomainHash = keccak256(abi.encodePacked(_domainHash, _userHash));
require(accounts[subdomainHash].creationTime > 0, "Username not registered.");
require(msg.sender == ens.owner(subdomainHash), "Caller not owner of ENS node.");
require(ens.owner(_domainHash) == address(this), "Registry not owner of domain.");
accounts[subdomainHash].accountOwner = msg.sender;
emit SubdomainOwner(subdomainHash, msg.sender);
}
/**
* @notice slash account due too length restriction
* @param _subdomain raw value of offending subdomain
* @param _domainHash domain hash
*/
function slashSmallSubdomain(
bytes _subdomain,
bytes32 _domainHash
)
external
{
require(_subdomain.length < subdomainMinLenght, "Not a small subdomain.");
slashSubdomain(_subdomain, _domainHash);
}
/**
* @notice slash account due look like an address
* @param _subdomain raw value of offending subdomain
* @param _domainHash domain hash
*/
function slashAddressLikeSubdomain(
string _subdomain,
bytes32 _domainHash
)
external
{
bytes memory subdomain = bytes(_subdomain);
require(subdomain.length > 12, "Too small to look like an address.");
require(subdomain[0] == byte("0"), "First character need to be 0");
require(subdomain[1] == byte("x"), "Second character need to be x");
slashSubdomain(subdomain, _domainHash);
}
/**
* @notice slash account due reserved name
* @param _subdomain raw value of offending subdomain
* @param _domainHash domain hash
*/
function slashReservedSubdomain(
bytes _subdomain,
bytes32 _domainHash,
uint256 _rootPos,
bytes32[] _proof
)
external
{
require(reservedSubdomainsMerkleRoots.length > _rootPos, "Invalid Merkle Root");
require(
MerkleProof.verifyProof(
_proof,
reservedSubdomainsMerkleRoots[_rootPos],
keccak256(_subdomain)
),
"Invalid Proof."
);
slashSubdomain(_subdomain, _domainHash);
}
/**
* @notice slash account of invalid subdomain
* @param _subdomain raw value of offending subdomain
* @param _domainHash domain hash
* @param _offendingPos position of invalid character
*/
function slashInvalidSubdomain(
bytes _subdomain,
bytes32 _domainHash,
uint256 _offendingPos
)
external
{
require(_subdomain.length > _offendingPos, "Invalid position.");
byte b = _subdomain[_offendingPos];
require(!((b >= 48 && b <= 57) || (b >= 97 && b <= 122)), "Not invalid character.");
slashSubdomain(_subdomain, _domainHash);
}
function slashSubdomain(bytes _subdomain, bytes32 _domainHash) internal {
bytes32 userHash = keccak256(_subdomain);
bytes32 subdomainHash = keccak256(abi.encodePacked(_domainHash, userHash));
require(accounts[subdomainHash].creationTime > 0, "Username not registered.");
ens.setSubnodeOwner(_domainHash, userHash, address(this));
ens.setResolver(subdomainHash, address(0));
ens.setOwner(subdomainHash, address(0));
uint256 amountToTransfer = accounts[subdomainHash].tokenBalance;
delete accounts[subdomainHash];
if(amountToTransfer > 0){
require(token.transfer(msg.sender, amountToTransfer), "Error in transfer.");
}
emit SubdomainOwner(subdomainHash, address(0));
}
/**
* @notice Migrate account to new registry
* @param _userHash `msg.sender` owned subdomain hash
* @param _domainHash choosen contract owned domain hash
**/
function moveAccount(
bytes32 _userHash,
bytes32 _domainHash
)
external
{
bytes32 subdomainHash = keccak256(abi.encodePacked(_domainHash, _userHash));
require(msg.sender == accounts[subdomainHash].accountOwner, "Callable only by account owner.");
ENSSubdomainRegistry _newRegistry = ENSSubdomainRegistry(ens.owner(_domainHash));
Account memory account = accounts[subdomainHash];
delete accounts[subdomainHash];
//require(address(this) == _newRegistry.parentRegistry(), "Wrong update.");
token.approve(_newRegistry, account.tokenBalance);
_newRegistry.migrateAccount(
_userHash,
_domainHash,
account.tokenBalance,
account.creationTime,
account.accountOwner
);
}
/**
* @dev callabe only by parent registry to continue migration of domain
**/
function migrateDomain(
bytes32 _domain,
uint256 _price
)
external
onlyParentRegistry
{
require(ens.owner(_domain) == address(this), "ENS domain owner not transfered.");
assert(domains[_domain].state == NodeState.Free);
domains[_domain] = Domain(NodeState.Owned, _price);
}
/**
* @dev callable only by parent registry for continue user opt-in migration
* @param _userHash any subdomain hash coming from parent
* @param _domainHash choosen contract owned domain hash
* @param _tokenBalance amount being transferred
* @param _creationTime any value coming from parent
* @param _accountOwner accountOwner for opt-out/release at domain move
**/
function migrateAccount(
bytes32 _userHash,
bytes32 _domainHash,
uint256 _tokenBalance,
uint256 _creationTime,
address _accountOwner
)
external
onlyParentRegistry
{
bytes32 subdomainHash = keccak256(abi.encodePacked(_domainHash, _userHash));
accounts[subdomainHash] = Account(_tokenBalance, _creationTime, _accountOwner);
if (_tokenBalance > 0) {
require(
token.transferFrom(
parentRegistry,
address(this),
_tokenBalance
),
"Error moving funds from old registar."
);
}
}
/**
* @notice moves a domain to other Registry (will not move subdomains accounts)
* @param _newRegistry new registry hodling this domain
* @param _domain domain being moved
*/
function moveDomain(
ENSSubdomainRegistry _newRegistry,
bytes32 _domain
)
external
onlyController
{
require(domains[_domain].state == NodeState.Owned, "Wrong domain");
require(ens.owner(_domain) == address(this), "Domain not owned anymore.");
uint256 price = domains[_domain].price;
domains[_domain].state = NodeState.Moved;
ens.setOwner(_domain, _newRegistry);
_newRegistry.migrateDomain(_domain, price);
emit DomainMoved(_domain, _newRegistry);
}
/**
* @notice Controller include new domain available to register
* @param _domain domain owned by user registry being activated
* @param _price cost to register subnode from this node
*/
function setDomainPrice(
bytes32 _domain,
uint256 _price
)
external
onlyController
{
require(domains[_domain].state == NodeState.Free, "Domain state is not free");
require(ens.owner(_domain) == address(this), "Registry does not own domain");
domains[_domain] = Domain(NodeState.Owned, _price);
emit DomainPrice(_domain, _price);
}
/**
* @notice updates domain price
* @param _domain active domain being defined price
* @param _price new price
*/
function updateDomainPrice(
bytes32 _domain,
uint256 _price
)
external
onlyController
{
Domain storage domain = domains[_domain];
require(domain.state == NodeState.Owned, "Domain not owned");
domain.price = _price;
emit DomainPrice(_domain, _price);
}
/**
* @notice updates default public resolver for newly registred subdomains
* @param _resolver new default resolver
*/
function setResolver(
address _resolver
)
external
onlyController
{
resolver = PublicResolver(_resolver);
}
function getPrice(bytes32 _domainHash)
external
view
returns(uint256 subdomainPrice)
{
subdomainPrice = domains[_domainHash].price;
}
function getAccountBalance(bytes32 _subdomainHash)
external
view
returns(uint256 accountBalance)
{
accountBalance = accounts[_subdomainHash].tokenBalance;
}
function getAccountOwner(bytes32 _subdomainHash)
external
view
returns(address accountOwner)
{
accountOwner = accounts[_subdomainHash].accountOwner;
}
function getCreationTime(bytes32 _subdomainHash)
external
view
returns(uint256 creationTime)
{
creationTime = accounts[_subdomainHash].creationTime;
}
function getExpirationTime(bytes32 _subdomainHash)
external
view
returns(uint256 expirationTime)
{
expirationTime = accounts[_subdomainHash].creationTime + releaseDelay;
}
}