/*
Copyright 2019 The Hydro Protocol Foundation
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
pragma solidity 0.5.8;
pragma experimental ABIEncoderV2;
library Decimal {
using SafeMath for uint256;
uint256 constant BASE = 10**18;
function one()
internal
pure
returns (uint256)
{
return BASE;
}
function onePlus(
uint256 d
)
internal
pure
returns (uint256)
{
return d.add(BASE);
}
function mulFloor(
uint256 target,
uint256 d
)
internal
pure
returns (uint256)
{
return target.mul(d) / BASE;
}
function mulCeil(
uint256 target,
uint256 d
)
internal
pure
returns (uint256)
{
return target.mul(d).divCeil(BASE);
}
function divFloor(
uint256 target,
uint256 d
)
internal
pure
returns (uint256)
{
return target.mul(BASE).div(d);
}
function divCeil(
uint256 target,
uint256 d
)
internal
pure
returns (uint256)
{
return target.mul(BASE).divCeil(d);
}
}
contract Ownable {
address private _owner;
event OwnershipTransferred(
address indexed previousOwner,
address indexed newOwner
);
/** @dev The Ownable constructor sets the original `owner` of the contract to the sender account. */
constructor()
internal
{
_owner = msg.sender;
emit OwnershipTransferred(address(0), _owner);
}
/** @return the address of the owner. */
function owner()
public
view
returns(address)
{
return _owner;
}
/** @dev Throws if called by any account other than the owner. */
modifier onlyOwner() {
require(isOwner(), "NOT_OWNER");
_;
}
/** @return true if `msg.sender` is the owner of the contract. */
function isOwner()
public
view
returns(bool)
{
return msg.sender == _owner;
}
/** @dev Allows the current owner to relinquish control of the contract.
* @notice Renouncing to ownership will leave the contract without an owner.
* It will not be possible to call the functions with the `onlyOwner`
* modifier anymore.
*/
function renounceOwnership()
public
onlyOwner
{
emit OwnershipTransferred(_owner, address(0));
_owner = address(0);
}
/** @dev Allows the current owner to transfer control of the contract to a newOwner.
* @param newOwner The address to transfer ownership to.
*/
function transferOwnership(
address newOwner
)
public
onlyOwner
{
require(newOwner != address(0), "INVALID_OWNER");
emit OwnershipTransferred(_owner, newOwner);
_owner = newOwner;
}
}
library SafeMath {
// Multiplies two numbers, reverts on overflow.
function mul(
uint256 a,
uint256 b
)
internal
pure
returns (uint256)
{
if (a == 0) {
return 0;
}
uint256 c = a * b;
require(c / a == b, "MUL_ERROR");
return c;
}
// Integer division of two numbers truncating the quotient, reverts on division by zero.
function div(
uint256 a,
uint256 b
)
internal
pure
returns (uint256)
{
require(b > 0, "DIVIDING_ERROR");
return a / b;
}
function divCeil(
uint256 a,
uint256 b
)
internal
pure
returns (uint256)
{
uint256 quotient = div(a, b);
uint256 remainder = a - quotient * b;
if (remainder > 0) {
return quotient + 1;
} else {
return quotient;
}
}
// Subtracts two numbers, reverts on overflow (i.e. if subtrahend is greater than minuend).
function sub(
uint256 a,
uint256 b
)
internal
pure
returns (uint256)
{
require(b <= a, "SUB_ERROR");
return a - b;
}
function sub(
int256 a,
uint256 b
)
internal
pure
returns (int256)
{
require(b <= 2**255-1, "INT256_SUB_ERROR");
int256 c = a - int256(b);
require(c <= a, "INT256_SUB_ERROR");
return c;
}
// Adds two numbers, reverts on overflow.
function add(
uint256 a,
uint256 b
)
internal
pure
returns (uint256)
{
uint256 c = a + b;
require(c >= a, "ADD_ERROR");
return c;
}
function add(
int256 a,
uint256 b
)
internal
pure
returns (int256)
{
require(b <= 2**255 - 1, "INT256_ADD_ERROR");
int256 c = a + int256(b);
require(c >= a, "INT256_ADD_ERROR");
return c;
}
// Divides two numbers and returns the remainder (unsigned integer modulo), reverts when dividing by zero.
function mod(
uint256 a,
uint256 b
)
internal
pure
returns (uint256)
{
require(b != 0, "MOD_ERROR");
return a % b;
}
/**
* Check the amount of precision lost by calculating multiple * (numerator / denominator). To
* do this, we check the remainder and make sure it's proportionally less than 0.1%. So we have:
*
* ((numerator * multiple) % denominator) 1
* -------------------------------------- < ----
* numerator * multiple 1000
*
* To avoid further division, we can move the denominators to the other sides and we get:
*
* ((numerator * multiple) % denominator) * 1000 < numerator * multiple
*
* Since we want to return true if there IS a rounding error, we simply flip the sign and our
* final equation becomes:
*
* ((numerator * multiple) % denominator) * 1000 >= numerator * multiple
*
* @param numerator The numerator of the proportion
* @param denominator The denominator of the proportion
* @param multiple The amount we want a proportion of
* @return Boolean indicating if there is a rounding error when calculating the proportion
*/
function isRoundingError(
uint256 numerator,
uint256 denominator,
uint256 multiple
)
internal
pure
returns (bool)
{
// numerator.mul(multiple).mod(denominator).mul(1000) >= numerator.mul(multiple)
return mul(mod(mul(numerator, multiple), denominator), 1000) >= mul(numerator, multiple);
}
/**
* Takes an amount (multiple) and calculates a proportion of it given a numerator/denominator
* pair of values. The final value will be rounded down to the nearest integer value.
*
* This function will revert the transaction if rounding the final value down would lose more
* than 0.1% precision.
*
* @param numerator The numerator of the proportion
* @param denominator The denominator of the proportion
* @param multiple The amount we want a proportion of
* @return The final proportion of multiple rounded down
*/
function getPartialAmountFloor(
uint256 numerator,
uint256 denominator,
uint256 multiple
)
internal
pure
returns (uint256)
{
require(!isRoundingError(numerator, denominator, multiple), "ROUNDING_ERROR");
// numerator.mul(multiple).div(denominator)
return div(mul(numerator, multiple), denominator);
}
/**
* Returns the smaller integer of the two passed in.
*
* @param a Unsigned integer
* @param b Unsigned integer
* @return The smaller of the two integers
*/
function min(
uint256 a,
uint256 b
)
internal
pure
returns (uint256)
{
return a < b ? a : b;
}
}
contract FeedPriceOracle is Ownable {
using SafeMath for uint256;
address[] public assets;
uint256 public price;
uint256 public lastBlockNumber;
uint256 public validBlockNumber;
uint256 public maxChangeRate;
uint256 public minPrice;
uint256 public maxPrice;
event PriceFeed(
uint256 price,
uint256 blockNumber
);
constructor (
address[] memory _assets,
uint256 _validBlockNumber,
uint256 _maxChangeRate,
uint256 _minPrice,
uint256 _maxPrice
)
public
{
assets = _assets;
setParams(
_validBlockNumber,
_maxChangeRate,
_minPrice,
_maxPrice
);
}
function setParams(
uint256 _validBlockNumber,
uint256 _maxChangeRate,
uint256 _minPrice,
uint256 _maxPrice
)
public
onlyOwner
{
require(_minPrice <= _maxPrice, "MIN_PRICE_MUST_LESS_OR_EQUAL_THAN_MAX_PRICE");
validBlockNumber = _validBlockNumber;
maxChangeRate = _maxChangeRate;
minPrice = _minPrice;
maxPrice = _maxPrice;
}
function feed(
uint256 newPrice
)
external
onlyOwner
{
require(newPrice > 0, "PRICE_MUST_GREATER_THAN_0");
require(lastBlockNumber < block.number, "BLOCKNUMBER_WRONG");
require(newPrice <= maxPrice, "PRICE_EXCEED_MAX_LIMIT");
require(newPrice >= minPrice, "PRICE_EXCEED_MIN_LIMIT");
if (price > 0) {
uint256 changeRate = Decimal.divFloor(newPrice, price);
if (changeRate > Decimal.one()) {
changeRate = changeRate.sub(Decimal.one());
} else {
changeRate = Decimal.one().sub(changeRate);
}
require(changeRate <= maxChangeRate, "PRICE_CHANGE_RATE_EXCEED");
}
price = newPrice;
lastBlockNumber = block.number;
emit PriceFeed(price, lastBlockNumber);
}
function isValidAsset(
address asset
)
private
view
returns (bool)
{
for (uint256 i = 0; i < assets.length; i++ ) {
if (assets[i] == asset) {
return true;
}
}
return false;
}
function getPrice(
address _asset
)
external
view
returns (uint256)
{
require(isValidAsset(_asset), "ASSET_NOT_MATCH");
require(block.number.sub(lastBlockNumber) <= validBlockNumber, "PRICE_EXPIRED");
return price;
}
}
{
"compilationTarget": {
"FeedPriceOracle.sol": "FeedPriceOracle"
},
"evmVersion": "petersburg",
"libraries": {},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"constant":true,"inputs":[],"name":"lastBlockNumber","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"name":"_asset","type":"address"}],"name":"getPrice","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[],"name":"renounceOwnership","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_validBlockNumber","type":"uint256"},{"name":"_maxChangeRate","type":"uint256"},{"name":"_minPrice","type":"uint256"},{"name":"_maxPrice","type":"uint256"}],"name":"setParams","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"maxChangeRate","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"owner","outputs":[{"name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"isOwner","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"price","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"name":"","type":"uint256"}],"name":"assets","outputs":[{"name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"maxPrice","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"minPrice","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"newPrice","type":"uint256"}],"name":"feed","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"validBlockNumber","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"inputs":[{"name":"_assets","type":"address[]"},{"name":"_validBlockNumber","type":"uint256"},{"name":"_maxChangeRate","type":"uint256"},{"name":"_minPrice","type":"uint256"},{"name":"_maxPrice","type":"uint256"}],"payable":false,"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"name":"price","type":"uint256"},{"indexed":false,"name":"blockNumber","type":"uint256"}],"name":"PriceFeed","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"previousOwner","type":"address"},{"indexed":true,"name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"}]