dodoMysteryBox finish
This commit is contained in:
@@ -20,4 +20,6 @@
|
|||||||
|
|
||||||
- contracts/external/ERC1155/
|
- contracts/external/ERC1155/
|
||||||
|
|
||||||
- contracts/lib/RandomGenerator.sol
|
- contracts/lib/RandomGenerator.sol
|
||||||
|
|
||||||
|
- contracts/SmartRoute/proxies/DODOMysteryBoxProxy.sol
|
||||||
@@ -10,9 +10,10 @@ import {SafeERC20} from "../lib/SafeERC20.sol";
|
|||||||
import {SafeMath} from "../lib/SafeMath.sol";
|
import {SafeMath} from "../lib/SafeMath.sol";
|
||||||
import {IRandomGenerator} from "../lib/RandomGenerator.sol";
|
import {IRandomGenerator} from "../lib/RandomGenerator.sol";
|
||||||
import {InitializableOwnable} from "../lib/InitializableOwnable.sol";
|
import {InitializableOwnable} from "../lib/InitializableOwnable.sol";
|
||||||
|
import {ReentrancyGuard} from "../lib/ReentrancyGuard.sol";
|
||||||
import {ERC1155} from "../external/ERC1155/ERC1155.sol";
|
import {ERC1155} from "../external/ERC1155/ERC1155.sol";
|
||||||
|
|
||||||
contract DODOMysteryBox is ERC1155, InitializableOwnable {
|
contract DODOMysteryBox is ERC1155, InitializableOwnable, ReentrancyGuard {
|
||||||
using SafeMath for uint256;
|
using SafeMath for uint256;
|
||||||
using SafeERC20 for IERC20;
|
using SafeERC20 for IERC20;
|
||||||
|
|
||||||
@@ -27,13 +28,15 @@ contract DODOMysteryBox is ERC1155, InitializableOwnable {
|
|||||||
uint256[][] public _PRIZE_SET_; // Interval index => tokenIds
|
uint256[][] public _PRIZE_SET_; // Interval index => tokenIds
|
||||||
mapping(uint256 => bool) _TOKEN_ID_FLAG_;
|
mapping(uint256 => bool) _TOKEN_ID_FLAG_;
|
||||||
|
|
||||||
uint256 constant totalInterval = 1000;
|
|
||||||
|
|
||||||
// ============ Event =============
|
// ============ Event =============
|
||||||
event ChangeRandomGenerator(address randomGenerator);
|
event ChangeRandomGenerator(address randomGenerator);
|
||||||
event ChangeTicketUnit(uint256 newTicketUnit);
|
event ChangeTicketUnit(uint256 newTicketUnit);
|
||||||
event RetriveTicket(address to, uint256 amount);
|
event RetriveTicket(address to, uint256 amount);
|
||||||
event BurnTicket(uint256 amount);
|
event BurnTicket(uint256 amount);
|
||||||
|
event RedeemPrize(address to, uint256 ticketInput, uint256 ticketNum);
|
||||||
|
event SetProbInterval();
|
||||||
|
event SetPrizeSet();
|
||||||
|
event SetPrizeSetByIndex(uint256 index);
|
||||||
|
|
||||||
function init(
|
function init(
|
||||||
address owner,
|
address owner,
|
||||||
@@ -44,8 +47,10 @@ contract DODOMysteryBox is ERC1155, InitializableOwnable {
|
|||||||
uint256[] memory probIntervals,
|
uint256[] memory probIntervals,
|
||||||
uint256[][] memory prizeSet
|
uint256[][] memory prizeSet
|
||||||
) public {
|
) public {
|
||||||
require(probIntervals.length == prizeSet.length, "DODOMysteryBox:PARAM_NOT_MATCH");
|
require(
|
||||||
|
probIntervals.length == prizeSet.length && probIntervals.length > 0,
|
||||||
|
"DODOMysteryBox: PARAM_NOT_INVALID"
|
||||||
|
);
|
||||||
initOwner(owner);
|
initOwner(owner);
|
||||||
_setURI(baseUri);
|
_setURI(baseUri);
|
||||||
|
|
||||||
@@ -57,7 +62,7 @@ contract DODOMysteryBox is ERC1155, InitializableOwnable {
|
|||||||
_setPrizeSet(prizeSet);
|
_setPrizeSet(prizeSet);
|
||||||
}
|
}
|
||||||
|
|
||||||
function redeemPrize(address to) external {
|
function redeemPrize(address to) preventReentrant external {
|
||||||
uint256 ticketBalance = IERC20(_TICKET_).balanceOf(address(this));
|
uint256 ticketBalance = IERC20(_TICKET_).balanceOf(address(this));
|
||||||
uint256 ticketInput = ticketBalance.sub(_TICKET_RESERVE_);
|
uint256 ticketInput = ticketBalance.sub(_TICKET_RESERVE_);
|
||||||
uint256 ticketNum = ticketInput.div(_TICKET_UNIT_);
|
uint256 ticketNum = ticketInput.div(_TICKET_UNIT_);
|
||||||
@@ -65,13 +70,28 @@ contract DODOMysteryBox is ERC1155, InitializableOwnable {
|
|||||||
for (uint256 i = 0; i < ticketNum; i++) {
|
for (uint256 i = 0; i < ticketNum; i++) {
|
||||||
_redeemSinglePrize(to);
|
_redeemSinglePrize(to);
|
||||||
}
|
}
|
||||||
_TICKET_RESERVE_ = _TICKET_RESERVE_.add(ticketBalance);
|
_TICKET_RESERVE_ = _TICKET_RESERVE_.add(ticketInput);
|
||||||
|
emit RedeemPrize(to, ticketInput, ticketNum);
|
||||||
|
}
|
||||||
|
|
||||||
|
// =============== View ================
|
||||||
|
function getRarityByTokenId(uint256 tokenId) external view returns (uint256) {
|
||||||
|
require(_TOKEN_ID_FLAG_[tokenId], "DODOMysteryBox: TOKEN_ID_NOT_FOUND");
|
||||||
|
for (uint256 i = 0; i < _PRIZE_SET_.length; i++) {
|
||||||
|
uint256[] memory curPrizes = _PRIZE_SET_[i];
|
||||||
|
for (uint256 j = 0; j < curPrizes.length; j++) {
|
||||||
|
if(tokenId == curPrizes[j]) {
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ============ Internal ============
|
// ============ Internal ============
|
||||||
|
|
||||||
function _redeemSinglePrize(address to) internal {
|
function _redeemSinglePrize(address to) internal {
|
||||||
uint256 random = IRandomGenerator(_RANDOM_GENERATOR_).random() % totalInterval;
|
uint256 range = _PROB_INTERVAL_[_PROB_INTERVAL_.length - 1];
|
||||||
|
uint256 random = IRandomGenerator(_RANDOM_GENERATOR_).random(gasleft()) % range;
|
||||||
uint256 i;
|
uint256 i;
|
||||||
for (i = 0; i < _PROB_INTERVAL_.length; i++) {
|
for (i = 0; i < _PROB_INTERVAL_.length; i++) {
|
||||||
if (random <= _PROB_INTERVAL_[i]) {
|
if (random <= _PROB_INTERVAL_[i]) {
|
||||||
@@ -84,34 +104,58 @@ contract DODOMysteryBox is ERC1155, InitializableOwnable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function _setProbInterval(uint256[] memory probIntervals) internal {
|
function _setProbInterval(uint256[] memory probIntervals) internal {
|
||||||
uint256 sum;
|
for (uint256 i = 1; i < probIntervals.length; i++) {
|
||||||
for (uint256 i = 0; i < probIntervals.length; i++) {
|
require(probIntervals[i] > probIntervals[i - 1], "DODOMysteryBox: INTERVAL_INVALID");
|
||||||
require(probIntervals[i] > 0, "DODOMysteryBox: INTERVAL_INVALID");
|
|
||||||
sum += probIntervals[i];
|
|
||||||
_PROB_INTERVAL_.push(probIntervals[i]);
|
|
||||||
}
|
}
|
||||||
require(sum == totalInterval, "DODOMysteryBox: TOTAL_INTERVAL_INVALID");
|
_PROB_INTERVAL_ = probIntervals;
|
||||||
|
emit SetProbInterval();
|
||||||
}
|
}
|
||||||
|
|
||||||
function _setPrizeSet(uint256[][] memory prizeSet) internal {
|
function _setPrizeSet(uint256[][] memory prizeSet) internal {
|
||||||
for (uint256 i = 0; i < prizeSet.length; i++) {
|
for (uint256 i = 0; i < prizeSet.length; i++) {
|
||||||
uint256[] memory curPrizes = prizeSet[i];
|
uint256[] memory curPrizes = prizeSet[i];
|
||||||
require(curPrizes.length > 0, "DODOMysteryBox: PRIZES_INVALID");
|
require(curPrizes.length > 0, "DODOMysteryBox: PRIZES_INVALID");
|
||||||
_PRIZE_SET_.push();
|
|
||||||
for (uint256 j = 0; j < curPrizes.length; j++) {
|
for (uint256 j = 0; j < curPrizes.length; j++) {
|
||||||
uint256 curTokenId = prizeSet[i][j];
|
uint256 curTokenId = prizeSet[i][j];
|
||||||
if(_TOKEN_ID_FLAG_[curTokenId]){
|
if (_TOKEN_ID_FLAG_[curTokenId]) {
|
||||||
require(false, "DODOMysteryBox: TOKEN_ID_INVALID");
|
require(false, "DODOMysteryBox: TOKEN_ID_INVALID");
|
||||||
}else {
|
} else {
|
||||||
_PRIZE_SET_[i].push(curTokenId);
|
|
||||||
_TOKEN_ID_FLAG_[curTokenId] = true;
|
_TOKEN_ID_FLAG_[curTokenId] = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
_PRIZE_SET_ = prizeSet;
|
||||||
|
emit SetPrizeSet();
|
||||||
}
|
}
|
||||||
|
|
||||||
// ================= Owner ===================
|
// ================= Owner ===================
|
||||||
|
|
||||||
|
function setProbInterval(uint256[] memory probIntervals) external onlyOwner {
|
||||||
|
require(probIntervals.length > 0, "DODOMysteryBox: PARAM_NOT_INVALID");
|
||||||
|
_setProbInterval(probIntervals);
|
||||||
|
}
|
||||||
|
|
||||||
|
function setPrzieSet(uint256[][] memory prizeSet) external onlyOwner {
|
||||||
|
require(prizeSet.length == _PROB_INTERVAL_.length, "DODOMysteryBox: PARAM_NOT_INVALID");
|
||||||
|
_setPrizeSet(prizeSet);
|
||||||
|
}
|
||||||
|
|
||||||
|
function setPrizeSetByIndex(uint256 index, uint256[] memory prizes) external onlyOwner {
|
||||||
|
require(
|
||||||
|
prizes.length > 0 && index < _PRIZE_SET_.length,
|
||||||
|
"DODOMysteryBox: PARAM_NOT_INVALID"
|
||||||
|
);
|
||||||
|
for (uint256 i = 0; i < prizes.length; i++) {
|
||||||
|
if (_TOKEN_ID_FLAG_[prizes[i]]) {
|
||||||
|
require(false, "DODOMysteryBox: TOKEN_ID_INVALID");
|
||||||
|
} else {
|
||||||
|
_TOKEN_ID_FLAG_[prizes[i]] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_PRIZE_SET_[index] = prizes;
|
||||||
|
emit SetPrizeSetByIndex(index);
|
||||||
|
}
|
||||||
|
|
||||||
function updateRandomGenerator(address newRandomGenerator) external onlyOwner {
|
function updateRandomGenerator(address newRandomGenerator) external onlyOwner {
|
||||||
require(newRandomGenerator != address(0));
|
require(newRandomGenerator != address(0));
|
||||||
_RANDOM_GENERATOR_ = newRandomGenerator;
|
_RANDOM_GENERATOR_ = newRandomGenerator;
|
||||||
|
|||||||
78
contracts/SmartRoute/proxies/DODOMysteryBoxProxy.sol
Normal file
78
contracts/SmartRoute/proxies/DODOMysteryBoxProxy.sol
Normal file
@@ -0,0 +1,78 @@
|
|||||||
|
/*
|
||||||
|
|
||||||
|
Copyright 2020 DODO ZOO.
|
||||||
|
SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
pragma solidity 0.6.9;
|
||||||
|
|
||||||
|
import {IDODOApproveProxy} from "../DODOApproveProxy.sol";
|
||||||
|
import {IERC20} from "../../intf/IERC20.sol";
|
||||||
|
import {IWETH} from "../../intf/IWETH.sol";
|
||||||
|
import {SafeMath} from "../../lib/SafeMath.sol";
|
||||||
|
import {SafeERC20} from "../../lib/SafeERC20.sol";
|
||||||
|
import {ReentrancyGuard} from "../../lib/ReentrancyGuard.sol";
|
||||||
|
|
||||||
|
interface IDODOMysteryBox {
|
||||||
|
function _TICKET_() external view returns (address);
|
||||||
|
function redeemPrize(address to) external;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @title DODO MysteryBoxProxy
|
||||||
|
* @author DODO Breeder
|
||||||
|
*
|
||||||
|
* @notice Entrance of MysteryBox in DODO platform
|
||||||
|
*/
|
||||||
|
contract DODOMysteryBoxProxy is ReentrancyGuard {
|
||||||
|
using SafeMath for uint256;
|
||||||
|
|
||||||
|
// ============ Storage ============
|
||||||
|
|
||||||
|
address constant _ETH_ADDRESS_ = 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE;
|
||||||
|
address public immutable _WETH_;
|
||||||
|
address public immutable _DODO_APPROVE_PROXY_;
|
||||||
|
|
||||||
|
// ============ Events ============
|
||||||
|
event RedeemPrize(address indexed account, address indexed mysteryBox, uint256 ticketAmount);
|
||||||
|
|
||||||
|
fallback() external payable {}
|
||||||
|
|
||||||
|
receive() external payable {}
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
address payable weth,
|
||||||
|
address dodoApproveProxy
|
||||||
|
) public {
|
||||||
|
_WETH_ = weth;
|
||||||
|
_DODO_APPROVE_PROXY_ = dodoApproveProxy;
|
||||||
|
}
|
||||||
|
|
||||||
|
function redeemPrize(
|
||||||
|
address dodoMysteryBox,
|
||||||
|
uint256 ticketAmount,
|
||||||
|
uint8 flag // 0 - ERC20, 1 - quoteInETH
|
||||||
|
) external payable preventReentrant {
|
||||||
|
_deposit(msg.sender, dodoMysteryBox, IDODOMysteryBox(dodoMysteryBox)._TICKET_(), ticketAmount, flag == 1);
|
||||||
|
IDODOMysteryBox(dodoMysteryBox).redeemPrize(msg.sender);
|
||||||
|
emit RedeemPrize(msg.sender, dodoMysteryBox, ticketAmount);
|
||||||
|
}
|
||||||
|
|
||||||
|
function _deposit(
|
||||||
|
address from,
|
||||||
|
address to,
|
||||||
|
address token,
|
||||||
|
uint256 amount,
|
||||||
|
bool isETH
|
||||||
|
) internal {
|
||||||
|
if (isETH) {
|
||||||
|
if (amount > 0) {
|
||||||
|
IWETH(_WETH_).deposit{value: amount}();
|
||||||
|
if (to != address(this)) SafeERC20.safeTransfer(IERC20(_WETH_), to, amount);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
IDODOApproveProxy(_DODO_APPROVE_PROXY_).claimTokens(token, from, to, amount);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -8,7 +8,7 @@
|
|||||||
pragma solidity 0.6.9;
|
pragma solidity 0.6.9;
|
||||||
|
|
||||||
interface IRandomGenerator {
|
interface IRandomGenerator {
|
||||||
function random() external view returns (uint256);
|
function random(uint256 seed) external view returns (uint256);
|
||||||
}
|
}
|
||||||
|
|
||||||
interface IDODOMidPrice {
|
interface IDODOMidPrice {
|
||||||
@@ -16,7 +16,7 @@ interface IDODOMidPrice {
|
|||||||
}
|
}
|
||||||
|
|
||||||
contract RandomGenerator {
|
contract RandomGenerator {
|
||||||
address[] internal pools;
|
address[] public pools;
|
||||||
|
|
||||||
constructor(address[] memory _pools) public {
|
constructor(address[] memory _pools) public {
|
||||||
for (uint256 i = 0; i < pools.length; i++) {
|
for (uint256 i = 0; i < pools.length; i++) {
|
||||||
@@ -24,11 +24,11 @@ contract RandomGenerator {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function random() external view returns (uint256) {
|
function random(uint256 seed) external view returns (uint256) {
|
||||||
uint256 priceSum;
|
uint256 priceSum;
|
||||||
for (uint256 i = 0; i < pools.length; i++) {
|
for (uint256 i = 0; i < pools.length; i++) {
|
||||||
priceSum += IDODOMidPrice(pools[i]).getMidPrice();
|
priceSum += IDODOMidPrice(pools[i]).getMidPrice();
|
||||||
}
|
}
|
||||||
return uint256(keccak256(abi.encodePacked(blockhash(block.number-1), priceSum)));
|
return uint256(keccak256(abi.encodePacked(blockhash(block.number-1), priceSum, seed)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -131,3 +131,8 @@ Init DODONFTProxyAddress Tx: 0x1ea94a3bedab5e35fec95071738066762a63253a68bc535ac
|
|||||||
DODOApproveProxy unlockAddProxy tx: 0xb8472dd2df9baeeff89d982299cbfc2d90e18c20baff97577fba959541f2514f
|
DODOApproveProxy unlockAddProxy tx: 0xb8472dd2df9baeeff89d982299cbfc2d90e18c20baff97577fba959541f2514f
|
||||||
DODOApproveProxy addDODOProxy tx: 0x8aa82620b6735ddfad0d55d3e9a752c3056e7d83664077e0240c0215a2e0017d
|
DODOApproveProxy addDODOProxy tx: 0x8aa82620b6735ddfad0d55d3e9a752c3056e7d83664077e0240c0215a2e0017d
|
||||||
Add AdminList on DODONFTRegistry Tx: 0x20f41302401122d950049e66458f307257743c0d395e06af607215b0f2532643
|
Add AdminList on DODONFTRegistry Tx: 0x20f41302401122d950049e66458f307257743c0d395e06af607215b0f2532643
|
||||||
|
====================================================
|
||||||
|
network type: development
|
||||||
|
Deploy time: 2021/4/12 下午9:22:30
|
||||||
|
Deploy type: NFT
|
||||||
|
multiSigAddress: undefined
|
||||||
|
|||||||
@@ -138,11 +138,11 @@ describe("DODONFT", () => {
|
|||||||
var erc721Instance = contracts.getContractWithAddress(contracts.ERC721, erc721Address);
|
var erc721Instance = contracts.getContractWithAddress(contracts.ERC721, erc721Address);
|
||||||
await erc721Instance.methods.safeTransferFrom(author, vaultAddress, 0).send(ctx.sendParam(author));
|
await erc721Instance.methods.safeTransferFrom(author, vaultAddress, 0).send(ctx.sendParam(author));
|
||||||
|
|
||||||
// var quoteToken = "0x156595bAF85D5C29E91d959889B022d952190A64";
|
var quoteToken = "0x156595bAF85D5C29E91d959889B022d952190A64";
|
||||||
// var vaultPreOwner = "0xaac153c1344cA14497A5dd22b1F70C28793625aa";
|
var vaultPreOwner = "0x7e83d9d94837eE82F0cc18a691da6f42F03F1d86";
|
||||||
// var stakeToken = "0x854b0f89BAa9101e49Bfb357A38071C9db5d0DFa";
|
// var stakeToken = "0x854b0f89BAa9101e49Bfb357A38071C9db5d0DFa";
|
||||||
var quoteToken = ctx.USDT.options.address;
|
// var quoteToken = ctx.USDT.options.address;
|
||||||
var vaultPreOwner = author;
|
// var vaultPreOwner = author;
|
||||||
var stakeToken = "0x0000000000000000000000000000000000000000";
|
var stakeToken = "0x0000000000000000000000000000000000000000";
|
||||||
|
|
||||||
var dvmParams = [
|
var dvmParams = [
|
||||||
@@ -168,18 +168,18 @@ describe("DODONFT", () => {
|
|||||||
).encodeABI();
|
).encodeABI();
|
||||||
console.log("data:", callData);
|
console.log("data:", callData);
|
||||||
|
|
||||||
await logGas(await nftVaultInstance.methods.createFragment(
|
// await logGas(await nftVaultInstance.methods.createFragment(
|
||||||
ctx.NFTProxy.options.address,
|
// ctx.NFTProxy.options.address,
|
||||||
callData
|
// callData
|
||||||
), ctx.sendParam(author), "createFragment");
|
// ), ctx.sendParam(author), "createFragment");
|
||||||
|
|
||||||
let [fragAddress, , dvmAddress] = await ctx.getRegistry(ctx, vaultAddress);
|
// let [fragAddress, , dvmAddress] = await ctx.getRegistry(ctx, vaultAddress);
|
||||||
|
|
||||||
var dvmInstance = contracts.getContractWithAddress(contracts.DVM_NAME, dvmAddress);
|
// var dvmInstance = contracts.getContractWithAddress(contracts.DVM_NAME, dvmAddress);
|
||||||
var midPrice = await dvmInstance.methods.getMidPrice().call();
|
// var midPrice = await dvmInstance.methods.getMidPrice().call();
|
||||||
assert(midPrice, mweiStr("1"));
|
// assert(midPrice, mweiStr("1"));
|
||||||
let newVaultOwner = await nftVaultInstance.methods._OWNER_().call();
|
// let newVaultOwner = await nftVaultInstance.methods._OWNER_().call();
|
||||||
assert(fragAddress, newVaultOwner);
|
// assert(fragAddress, newVaultOwner);
|
||||||
});
|
});
|
||||||
|
|
||||||
it.only("stakeToFeeDistributor", async () => {
|
it.only("stakeToFeeDistributor", async () => {
|
||||||
|
|||||||
Reference in New Issue
Block a user