2017-07-12 13:46:33 +00:00
|
|
|
pragma solidity ^0.4.11;
|
|
|
|
import "../Oracles/Oracle.sol";
|
|
|
|
import "../Tokens/Token.sol";
|
|
|
|
import "../Utils/Math.sol";
|
|
|
|
|
|
|
|
|
|
|
|
/// @title Ultimate oracle contract - Allows to swap oracle result for ultimate oracle result
|
|
|
|
/// @author Stefan George - <stefan@gnosis.pm>
|
|
|
|
contract UltimateOracle is Oracle {
|
|
|
|
using Math for *;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Events
|
|
|
|
*/
|
|
|
|
event ForwardedOracleOutcomeAssignment(int outcome);
|
|
|
|
event OutcomeChallenge(address indexed sender, int outcome);
|
|
|
|
event OutcomeVote(address indexed sender, int outcome, uint amount);
|
|
|
|
event Withdrawal(address indexed sender, uint amount);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Storage
|
|
|
|
*/
|
|
|
|
Oracle public forwardedOracle;
|
|
|
|
Token public collateralToken;
|
|
|
|
uint8 public spreadMultiplier;
|
|
|
|
uint public challengePeriod;
|
|
|
|
uint public challengeAmount;
|
|
|
|
uint public frontRunnerPeriod;
|
|
|
|
|
|
|
|
int public forwardedOutcome;
|
|
|
|
uint public forwardedOutcomeSetTimestamp;
|
|
|
|
int public frontRunner;
|
|
|
|
uint public frontRunnerSetTimestamp;
|
|
|
|
|
|
|
|
uint public totalAmount;
|
|
|
|
mapping (int => uint) public totalOutcomeAmounts;
|
|
|
|
mapping (address => mapping (int => uint)) public outcomeAmounts;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Public functions
|
|
|
|
*/
|
|
|
|
/// @dev Constructor sets ultimate oracle properties
|
|
|
|
/// @param _forwardedOracle Oracle address
|
|
|
|
/// @param _collateralToken Collateral token address
|
|
|
|
/// @param _spreadMultiplier Defines the spread as a multiple of the money bet on other outcomes
|
|
|
|
/// @param _challengePeriod Time to challenge oracle outcome
|
|
|
|
/// @param _challengeAmount Amount to challenge the outcome
|
|
|
|
/// @param _frontRunnerPeriod Time to overbid the front-runner
|
2018-06-25 13:02:20 +00:00
|
|
|
constructor(
|
2017-07-12 13:46:33 +00:00
|
|
|
Oracle _forwardedOracle,
|
|
|
|
Token _collateralToken,
|
|
|
|
uint8 _spreadMultiplier,
|
|
|
|
uint _challengePeriod,
|
|
|
|
uint _challengeAmount,
|
|
|
|
uint _frontRunnerPeriod
|
|
|
|
)
|
|
|
|
public
|
|
|
|
{
|
|
|
|
// Validate inputs
|
2018-06-12 09:05:49 +00:00
|
|
|
require( address(_forwardedOracle) != address(0)
|
|
|
|
&& address(_collateralToken) != address(0)
|
2017-07-12 13:46:33 +00:00
|
|
|
&& _spreadMultiplier >= 2
|
|
|
|
&& _challengePeriod > 0
|
|
|
|
&& _challengeAmount > 0
|
|
|
|
&& _frontRunnerPeriod > 0);
|
|
|
|
forwardedOracle = _forwardedOracle;
|
|
|
|
collateralToken = _collateralToken;
|
|
|
|
spreadMultiplier = _spreadMultiplier;
|
|
|
|
challengePeriod = _challengePeriod;
|
|
|
|
challengeAmount = _challengeAmount;
|
|
|
|
frontRunnerPeriod = _frontRunnerPeriod;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// @dev Allows to set oracle outcome
|
|
|
|
function setForwardedOutcome()
|
|
|
|
public
|
|
|
|
{
|
|
|
|
// There was no challenge and the outcome was not set yet in the ultimate oracle but in the forwarded oracle
|
|
|
|
require( !isChallenged()
|
|
|
|
&& forwardedOutcomeSetTimestamp == 0
|
|
|
|
&& forwardedOracle.isOutcomeSet());
|
|
|
|
forwardedOutcome = forwardedOracle.getOutcome();
|
|
|
|
forwardedOutcomeSetTimestamp = now;
|
2018-06-27 08:35:38 +00:00
|
|
|
emit ForwardedOracleOutcomeAssignment(forwardedOutcome);
|
2017-07-12 13:46:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// @dev Allows to challenge the oracle outcome
|
|
|
|
/// @param _outcome Outcome to bid on
|
|
|
|
function challengeOutcome(int _outcome)
|
|
|
|
public
|
|
|
|
{
|
|
|
|
// There was no challenge yet or the challenge period expired
|
|
|
|
require( !isChallenged()
|
|
|
|
&& !isChallengePeriodOver()
|
2018-07-12 18:07:16 +00:00
|
|
|
&& collateralToken.transferFrom(msg.sender, address(this), challengeAmount));
|
2017-07-12 13:46:33 +00:00
|
|
|
outcomeAmounts[msg.sender][_outcome] = challengeAmount;
|
|
|
|
totalOutcomeAmounts[_outcome] = challengeAmount;
|
|
|
|
totalAmount = challengeAmount;
|
|
|
|
frontRunner = _outcome;
|
|
|
|
frontRunnerSetTimestamp = now;
|
2018-06-27 08:35:38 +00:00
|
|
|
emit OutcomeChallenge(msg.sender, _outcome);
|
2017-07-12 13:46:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// @dev Allows to challenge the oracle outcome
|
|
|
|
/// @param _outcome Outcome to bid on
|
|
|
|
/// @param amount Amount to bid
|
|
|
|
function voteForOutcome(int _outcome, uint amount)
|
|
|
|
public
|
|
|
|
{
|
|
|
|
uint maxAmount = (totalAmount - totalOutcomeAmounts[_outcome]).mul(spreadMultiplier);
|
|
|
|
if (amount > maxAmount)
|
|
|
|
amount = maxAmount;
|
|
|
|
// Outcome is challenged and front runner period is not over yet and tokens can be transferred
|
|
|
|
require( isChallenged()
|
|
|
|
&& !isFrontRunnerPeriodOver()
|
2018-07-12 18:07:16 +00:00
|
|
|
&& collateralToken.transferFrom(msg.sender, address(this), amount));
|
2017-07-12 13:46:33 +00:00
|
|
|
outcomeAmounts[msg.sender][_outcome] = outcomeAmounts[msg.sender][_outcome].add(amount);
|
|
|
|
totalOutcomeAmounts[_outcome] = totalOutcomeAmounts[_outcome].add(amount);
|
|
|
|
totalAmount = totalAmount.add(amount);
|
|
|
|
if (_outcome != frontRunner && totalOutcomeAmounts[_outcome] > totalOutcomeAmounts[frontRunner])
|
|
|
|
{
|
|
|
|
frontRunner = _outcome;
|
|
|
|
frontRunnerSetTimestamp = now;
|
|
|
|
}
|
2018-06-27 08:35:38 +00:00
|
|
|
emit OutcomeVote(msg.sender, _outcome, amount);
|
2017-07-12 13:46:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// @dev Withdraws winnings for user
|
|
|
|
/// @return Winnings
|
|
|
|
function withdraw()
|
|
|
|
public
|
|
|
|
returns (uint amount)
|
|
|
|
{
|
|
|
|
// Outcome was challenged and ultimate outcome decided
|
|
|
|
require(isFrontRunnerPeriodOver());
|
|
|
|
amount = totalAmount.mul(outcomeAmounts[msg.sender][frontRunner]) / totalOutcomeAmounts[frontRunner];
|
|
|
|
outcomeAmounts[msg.sender][frontRunner] = 0;
|
|
|
|
// Transfer earnings to contributor
|
|
|
|
require(collateralToken.transfer(msg.sender, amount));
|
2018-06-27 08:35:38 +00:00
|
|
|
emit Withdrawal(msg.sender, amount);
|
2017-07-12 13:46:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// @dev Checks if time to challenge the outcome is over
|
|
|
|
/// @return Is challenge period over?
|
|
|
|
function isChallengePeriodOver()
|
|
|
|
public
|
2018-07-02 09:14:28 +00:00
|
|
|
view
|
2017-07-12 13:46:33 +00:00
|
|
|
returns (bool)
|
|
|
|
{
|
|
|
|
return forwardedOutcomeSetTimestamp != 0 && now.sub(forwardedOutcomeSetTimestamp) > challengePeriod;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// @dev Checks if time to overbid the front runner is over
|
|
|
|
/// @return Is front runner period over?
|
|
|
|
function isFrontRunnerPeriodOver()
|
|
|
|
public
|
2018-07-02 09:14:28 +00:00
|
|
|
view
|
2017-07-12 13:46:33 +00:00
|
|
|
returns (bool)
|
|
|
|
{
|
|
|
|
return frontRunnerSetTimestamp != 0 && now.sub(frontRunnerSetTimestamp) > frontRunnerPeriod;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// @dev Checks if outcome was challenged
|
|
|
|
/// @return Is challenged?
|
|
|
|
function isChallenged()
|
|
|
|
public
|
2018-07-02 09:14:28 +00:00
|
|
|
view
|
2017-07-12 13:46:33 +00:00
|
|
|
returns (bool)
|
|
|
|
{
|
|
|
|
return frontRunnerSetTimestamp != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// @dev Returns if winning outcome is set
|
|
|
|
/// @return Is outcome set?
|
|
|
|
function isOutcomeSet()
|
|
|
|
public
|
2018-07-02 09:14:28 +00:00
|
|
|
view
|
2017-07-12 13:46:33 +00:00
|
|
|
returns (bool)
|
|
|
|
{
|
|
|
|
return isChallengePeriodOver() && !isChallenged()
|
|
|
|
|| isFrontRunnerPeriodOver();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// @dev Returns winning outcome
|
|
|
|
/// @return Outcome
|
|
|
|
function getOutcome()
|
|
|
|
public
|
2018-07-02 09:14:28 +00:00
|
|
|
view
|
2017-07-12 13:46:33 +00:00
|
|
|
returns (int)
|
|
|
|
{
|
|
|
|
if (isFrontRunnerPeriodOver())
|
|
|
|
return frontRunner;
|
|
|
|
return forwardedOutcome;
|
|
|
|
}
|
|
|
|
}
|