relayer-registry/contracts/RelayerRegistry.sol

138 lines
4.4 KiB
Solidity
Raw Normal View History

2021-08-11 17:04:24 +02:00
//SPDX-License-Identifier: MIT
pragma solidity ^0.7.0;
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/math/SafeMath.sol";
2021-08-13 16:06:29 +02:00
import "@openzeppelin/contracts-upgradeable/proxy/Initializable.sol";
2021-08-11 17:04:24 +02:00
interface IENS {
function owner(bytes32 node) external view returns (address);
}
2021-08-20 11:30:44 +02:00
contract RelayerRegistry is Initializable {
2021-08-11 17:04:24 +02:00
using SafeMath for uint256;
IERC20 public constant TORN = IERC20(0x77777FeDdddFfC19Ff86DB637967013e6C6A116C);
IENS public constant ENS = IENS(0x00000000000C2E074eC69A0dFb2997BA6C7d2e1e);
2021-08-20 11:30:44 +02:00
address public constant GOV = 0x5efda50f22d34F262c29268506C5Fa42cB56A1Ce;
2021-08-13 16:06:29 +02:00
address public withdrawalProxy;
2021-08-11 17:04:24 +02:00
mapping(address => Relayer) public relayers;
2021-08-20 11:30:44 +02:00
uint256 public txFee; // TODO should rely on baseFee for a tx
uint256 public minStake;
2021-08-20 11:30:44 +02:00
uint256 public govFeeSplitPercent;
uint256 public totalTornBurned;
event Register(bytes32 ensHash, address relayer);
event Stake(address indexed relayer, uint256 stake);
event Kick(address indexed relayer, bool confiscation);
2021-08-23 11:59:28 +02:00
event Transaction(address indexed relayer, uint256 fee);
event NewMinStake(uint256 stake);
event NewTxFee(uint256 fee);
2021-08-13 19:42:57 +02:00
event NewWithdrawalProxy(address proxy);
2021-08-20 11:30:44 +02:00
event NewGovFeeSplitPercent(uint256 govFeeSplitPercent);
2021-08-23 11:59:28 +02:00
modifier onlyWithdrawalProxy() {
require(msg.sender == withdrawalProxy, "only withdrawal proxy");
_;
}
2021-08-11 17:04:24 +02:00
2021-08-23 11:59:28 +02:00
modifier onlyGovernance() {
2021-08-20 11:30:44 +02:00
require(msg.sender == GOV, "only governance");
_;
}
2021-08-11 17:04:24 +02:00
struct Relayer {
uint256 balance;
bytes32 ensHash;
mapping(address => bool) addresses;
2021-08-11 17:04:24 +02:00
}
2021-08-23 11:59:28 +02:00
/// @dev since this is an implementation contract used by upgradability proxy
constructor() {
initialize(address(0), 0, 0, 0);
}
function initialize(
address _withdrawalProxy,
uint256 _txFee,
uint256 _minStake,
uint256 _govFeeSplitPercent
) public initializer {
withdrawalProxy = _withdrawalProxy;
txFee = _txFee;
minStake = _minStake;
2021-08-20 11:30:44 +02:00
govFeeSplitPercent = _govFeeSplitPercent;
2021-08-11 17:04:24 +02:00
}
2021-08-23 11:59:28 +02:00
function register(uint256 _stake, bytes32 _ensHash) external {
require(msg.sender == ENS.owner(_ensHash), "only owner of the ENS name");
Relayer storage relayer = relayers[msg.sender];
require(relayer.ensHash == bytes32(0) && _stake >= minStake);
_updateStake(_stake, relayer);
relayer.addresses[msg.sender] = true;
relayer.ensHash = _ensHash;
emit Register(_ensHash, msg.sender);
2021-08-11 17:04:24 +02:00
}
2021-08-23 11:59:28 +02:00
function stake(uint256 _stake) external {
2021-08-11 17:04:24 +02:00
Relayer storage relayer = relayers[msg.sender];
require(relayer.ensHash != bytes32(0), "Only registered relayers can top up balance");
_updateStake(_stake, relayer);
}
2021-08-23 11:59:28 +02:00
function transaction(address _sender, address _feeReceiver) external onlyWithdrawalProxy {
Relayer storage relayer = relayers[_feeReceiver];
2021-08-23 11:59:28 +02:00
if (relayer.ensHash != bytes32(0)) {
require(relayer.addresses[_sender], "only registered relayer can send");
uint256 fee = relayer.balance > txFee ? txFee : relayer.balance;
relayer.balance -= fee;
totalTornBurned += fee;
emit Transaction(_feeReceiver, fee);
}
2021-08-11 17:04:24 +02:00
}
2021-08-20 11:30:44 +02:00
function distributeFees() external {
2021-08-23 11:59:28 +02:00
uint256 _totalTornBurned = totalTornBurned;
uint256 govAmount = totalTornBurned.mul(govFeeSplitPercent).div(100);
2021-08-20 11:30:44 +02:00
TORN.transfer(GOV, govAmount);
TORN.transfer(address(0), _totalTornBurned.sub(govAmount));
}
function setTxFee(uint256 _txFee) external onlyGovernance {
2021-08-11 17:04:24 +02:00
txFee = _txFee;
emit NewTxFee(_txFee);
2021-08-11 17:04:24 +02:00
}
2021-08-20 11:30:44 +02:00
function setMinStake(uint256 _stake) external onlyGovernance {
minStake = _stake;
emit NewMinStake(_stake);
}
2021-08-20 11:30:44 +02:00
function setWithdrawalProxy(address _proxy) external onlyGovernance {
2021-08-13 19:42:57 +02:00
withdrawalProxy = _proxy;
emit NewWithdrawalProxy(_proxy);
}
2021-08-20 11:30:44 +02:00
function setGovFeeSplitPercent(uint256 _govFeeSplitPercent) external onlyGovernance {
require(_govFeeSplitPercent <= 100, "Percent should an integer number from 0 to 100");
govFeeSplitPercent = _govFeeSplitPercent;
emit NewGovFeeSplitPercent(_govFeeSplitPercent);
}
function kick(address _relayer, bool confiscateStake) external onlyGovernance {
2021-08-11 17:04:24 +02:00
Relayer storage relayer = relayers[_relayer];
2021-08-23 11:59:28 +02:00
TORN.transfer(confiscateStake ? GOV : _relayer, relayer.balance);
2021-08-11 17:04:24 +02:00
relayer.balance = 0;
emit Kick(_relayer, confiscateStake);
}
2021-08-23 11:59:28 +02:00
function _updateStake(uint256 _stake, Relayer storage relayer) private {
TORN.transferFrom(msg.sender, address(this), _stake);
relayer.balance = relayer.balance.add(_stake);
emit Stake(msg.sender, _stake);
2021-08-11 17:04:24 +02:00
}
}