文件 1 的 10:BerezkaDaoManager.sol
pragma solidity ^0.8.4;
import "@openzeppelin/contracts/access/Ownable.sol";
contract BerezkaDaoManager is Ownable {
struct Dao {
address agent;
address tokens;
}
mapping(address => Dao) public daoConfig;
function _agentAddress(
address _token
) public view returns (address) {
address agentAddress = daoConfig[_token].agent;
require(agentAddress != address(0), "NO_DAO_FOR_TOKEN");
return agentAddress;
}
function addDao(
address _token,
address _tokens,
address _agent
) public onlyOwner {
daoConfig[_token] = Dao(_agent, _tokens);
}
function deleteDao(address _token) public onlyOwner {
delete daoConfig[_token];
}
}
文件 2 的 10:BerezkaOracleClient.sol
pragma solidity ^0.8.4;
import "@openzeppelin/contracts/access/Ownable.sol";
contract BerezkaOracleClient is Ownable {
address public oracleAddress = 0xAb66dE3DF08318922bb4cE15553E4C2dCf9187A1;
uint256 public signatureValidityDuractionSec = 3600;
modifier withValidOracleData(
address _token,
uint256 _optimisticPrice,
uint256 _optimisticPriceTimestamp,
bytes memory _signature
) {
require(_optimisticPrice > 0, "ZERO_OPTIMISTIC_PRICE");
require(
isValidSignatureDate(_optimisticPriceTimestamp),
"EXPIRED_PRICE_DATA"
);
require(
isValidSignature(
_optimisticPrice,
_optimisticPriceTimestamp,
_token,
_signature
),
"INVALID_SIGNATURE"
);
_;
}
function isValidSignatureDate(uint256 _optimisticPriceTimestamp)
public
view
returns (bool)
{
return computeSignatureDateDelta(_optimisticPriceTimestamp) <= signatureValidityDuractionSec;
}
function computeSignatureDateDelta(uint256 _optimisticPriceTimestamp)
public
view
returns (uint256)
{
uint256 timeDelta = 0;
if (_optimisticPriceTimestamp >= block.timestamp) {
timeDelta = _optimisticPriceTimestamp - block.timestamp;
} else {
timeDelta = block.timestamp - _optimisticPriceTimestamp;
}
return timeDelta;
}
function isValidSignature(
uint256 _price,
uint256 _timestamp,
address _token,
bytes memory _signature
) public view returns (bool) {
return recover(_price, _timestamp, _token, _signature) == oracleAddress;
}
function recover(
uint256 _price,
uint256 _timestamp,
address _token,
bytes memory _signature
) public pure returns (address) {
bytes32 dataHash = keccak256(
abi.encodePacked(_price, _timestamp, _token)
);
bytes32 signedMessageHash = keccak256(
abi.encodePacked("\x19Ethereum Signed Message:\n32", dataHash)
);
(bytes32 r, bytes32 s, uint8 v) = splitSignature(_signature);
address signer = ecrecover(signedMessageHash, v, r, s);
return signer;
}
function splitSignature(bytes memory sig)
public
pure
returns (
bytes32 r,
bytes32 s,
uint8 v
)
{
require(sig.length == 65, "invalid signature length");
assembly {
r := mload(add(sig, 32))
s := mload(add(sig, 64))
v := byte(0, mload(add(sig, 96)))
}
}
function setSignatureValidityDurationSec(
uint256 _signatureValidityDuractionSec
) public onlyOwner {
require(_signatureValidityDuractionSec > 0);
signatureValidityDuractionSec = _signatureValidityDuractionSec;
}
function setOracleAddress(address _oracleAddres) public onlyOwner {
oracleAddress = _oracleAddres;
}
}
文件 3 的 10:BerezkaStableCoinManager.sol
pragma solidity ^0.8.4;
import "@openzeppelin/contracts/access/Ownable.sol";
import "@openzeppelin/contracts/token/ERC20/extensions/IERC20Metadata.sol";
contract BerezkaStableCoinManager is Ownable {
mapping(address => bool) public whitelist;
modifier isWhitelisted(
address _targetToken
) {
require(whitelist[_targetToken], "INVALID_TOKEN_TO_DEPOSIT");
_;
}
function computeExchange(
uint256 _amount,
uint256 _price,
address _targetToken
) public view returns (uint256) {
IERC20Metadata targetToken = IERC20Metadata(_targetToken);
uint256 result = _amount * _price / 10 ** (24 - targetToken.decimals());
require(result > 0, "INVALID_TOKEN_AMOUNT");
return result;
}
function addWhitelistTokens(address[] memory _whitelisted)
public
onlyOwner
{
for (uint256 i = 0; i < _whitelisted.length; i++) {
whitelist[_whitelisted[i]] = true;
}
}
function removeWhitelistTokens(address[] memory _whitelisted)
public
onlyOwner
{
for (uint256 i = 0; i < _whitelisted.length; i++) {
whitelist[_whitelisted[i]] = false;
}
}
}
文件 4 的 10:Context.sol
pragma solidity ^0.8.0;
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}
文件 5 的 10:IAgent.sol
pragma solidity ^0.8.4;
interface IAgent {
function transfer(
address _token,
address _to,
uint256 _value
) external;
}
文件 6 的 10:IERC20.sol
pragma solidity ^0.8.0;
interface IERC20 {
function totalSupply() external view returns (uint256);
function balanceOf(address account) external view returns (uint256);
function transfer(address recipient, uint256 amount) external returns (bool);
function allowance(address owner, address spender) external view returns (uint256);
function approve(address spender, uint256 amount) external returns (bool);
function transferFrom(
address sender,
address recipient,
uint256 amount
) external returns (bool);
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
}
文件 7 的 10:IERC20Metadata.sol
pragma solidity ^0.8.0;
import "../IERC20.sol";
interface IERC20Metadata is IERC20 {
function name() external view returns (string memory);
function symbol() external view returns (string memory);
function decimals() external view returns (uint8);
}
文件 8 的 10:ITokens.sol
pragma solidity ^0.8.4;
interface ITokens {
function burn(address _holder, uint256 _amount) external;
function mint(address _holder, uint256 _amount) external;
}
文件 9 的 10:Ownable.sol
pragma solidity ^0.8.0;
import "../utils/Context.sol";
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor() {
_setOwner(_msgSender());
}
function owner() public view virtual returns (address) {
return _owner;
}
modifier onlyOwner() {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
_;
}
function renounceOwnership() public virtual onlyOwner {
_setOwner(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_setOwner(newOwner);
}
function _setOwner(address newOwner) private {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
文件 10 的 10:Withdraw.sol
pragma solidity ^0.8.4;
import "./api/IAgent.sol";
import "./api/ITokens.sol";
import "./common/BerezkaOracleClient.sol";
import "./common/BerezkaDaoManager.sol";
import "./common/BerezkaStableCoinManager.sol";
contract BerezkaWithdraw is
BerezkaOracleClient,
BerezkaDaoManager,
BerezkaStableCoinManager
{
event WithdrawSuccessEvent(
address indexed daoToken,
uint256 daoTokenAmount,
address indexed stableToken,
uint256 stableTokenAmount,
address indexed sender,
uint256 price,
uint256 timestamp
);
function withdraw(
uint256 _amount,
address _token,
address _targetToken,
uint256 _optimisticPrice,
uint256 _optimisticPriceTimestamp,
bytes memory _signature
)
public
withValidOracleData(
_token,
_optimisticPrice,
_optimisticPriceTimestamp,
_signature
)
isWhitelisted(_targetToken)
{
require(_amount > 0, "ZERO_TOKEN_AMOUNT");
_checkUserBalance(_amount, _token, msg.sender);
uint256 optimisticAmount = computeExchange(
_amount,
_optimisticPrice,
_targetToken
);
_doWithdraw(
_amount,
_token,
_targetToken,
msg.sender,
optimisticAmount
);
emit WithdrawSuccessEvent(
_token,
_amount,
_targetToken,
optimisticAmount,
msg.sender,
_optimisticPrice,
_optimisticPriceTimestamp
);
}
function _doWithdraw(
uint256 _amount,
address _token,
address _targetToken,
address _user,
uint256 _optimisticAmount
) internal {
address agentAddress = _agentAddress(_token);
IERC20 targetToken = IERC20(_targetToken);
require(
targetToken.balanceOf(agentAddress) >= _optimisticAmount,
"INSUFFICIENT_FUNDS_ON_AGENT"
);
IAgent agent = IAgent(agentAddress);
agent.transfer(_targetToken, _user, _optimisticAmount);
ITokens tokens = ITokens(daoConfig[_token].tokens);
tokens.burn(_user, _amount);
}
function _checkUserBalance(
uint256 _amount,
address _token,
address _user
) internal view {
IERC20 token = IERC20(_token);
require(
token.balanceOf(_user) >= _amount,
"NOT_ENOUGH_TOKENS_TO_BURN_ON_BALANCE"
);
}
}
{
"compilationTarget": {
"contracts/Withdraw.sol": "BerezkaWithdraw"
},
"evmVersion": "istanbul",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 1000
},
"remappings": []
}
[{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"daoToken","type":"address"},{"indexed":false,"internalType":"uint256","name":"daoTokenAmount","type":"uint256"},{"indexed":true,"internalType":"address","name":"stableToken","type":"address"},{"indexed":false,"internalType":"uint256","name":"stableTokenAmount","type":"uint256"},{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":false,"internalType":"uint256","name":"price","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"timestamp","type":"uint256"}],"name":"WithdrawSuccessEvent","type":"event"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"_agentAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_tokens","type":"address"},{"internalType":"address","name":"_agent","type":"address"}],"name":"addDao","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"_whitelisted","type":"address[]"}],"name":"addWhitelistTokens","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"uint256","name":"_price","type":"uint256"},{"internalType":"address","name":"_targetToken","type":"address"}],"name":"computeExchange","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_optimisticPriceTimestamp","type":"uint256"}],"name":"computeSignatureDateDelta","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"daoConfig","outputs":[{"internalType":"address","name":"agent","type":"address"},{"internalType":"address","name":"tokens","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"deleteDao","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_price","type":"uint256"},{"internalType":"uint256","name":"_timestamp","type":"uint256"},{"internalType":"address","name":"_token","type":"address"},{"internalType":"bytes","name":"_signature","type":"bytes"}],"name":"isValidSignature","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_optimisticPriceTimestamp","type":"uint256"}],"name":"isValidSignatureDate","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"oracleAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_price","type":"uint256"},{"internalType":"uint256","name":"_timestamp","type":"uint256"},{"internalType":"address","name":"_token","type":"address"},{"internalType":"bytes","name":"_signature","type":"bytes"}],"name":"recover","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address[]","name":"_whitelisted","type":"address[]"}],"name":"removeWhitelistTokens","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_oracleAddres","type":"address"}],"name":"setOracleAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_signatureValidityDuractionSec","type":"uint256"}],"name":"setSignatureValidityDurationSec","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"signatureValidityDuractionSec","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes","name":"sig","type":"bytes"}],"name":"splitSignature","outputs":[{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"},{"internalType":"uint8","name":"v","type":"uint8"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"whitelist","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_targetToken","type":"address"},{"internalType":"uint256","name":"_optimisticPrice","type":"uint256"},{"internalType":"uint256","name":"_optimisticPriceTimestamp","type":"uint256"},{"internalType":"bytes","name":"_signature","type":"bytes"}],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]