pragma solidity ^0.4.21;
/* ChiMarket is a contract that allows for buying and selling of CHI tokens for
used in the Aethia game (see https://aethia.co). The contract is designed to
discover the price of CHI (in ETH) through market forces (i.e. supply and
demand). The contract dynamically determines the price for each trade such that
the value of ETH and the value of CHI held by the contract are always the same.
This mechanism guarantees that you can always buy and sell any amount of CHI,
although the price might be outrageously high or low, depending on the amount
you are trying to buy/sell.
The contract provides to functions that can be used to query how much ETH
the contract is willing to pay for your CHI, or how much ETH you'll need to
buy CHI. You can call those functions without generating a transaction,
for example in the "Read Smart Contract" tab on Etherscan. This will give
you an estimate only, because the price might change by the time your TX confirms.
To avoid price surprises, this contract only supports limit buy and sells.
Limit buy and sell functions are used to convert CHI/ETH into ETH/CHI. limitSell
also takes a limit argument, which is the lowest amount of ETH you are willing
to accept for your trade. On the buy side, you should send more ETH than rrequired
to provide a cushin. You will always get the exact current price, the
limit argument is just to provide safety in case price changes dramatically.
If you send extra ETH, the excess will be returned to you in the same TX.
*/
// Interface to ERC721 functions used in this contract
interface ERC20token {
function balanceOf(address who) external view returns (uint256);
function transfer(address to, uint256 value) external returns (bool);
function allowance(address owner, address spender) external view returns (uint256);
function transferFrom(address from, address to, uint256 value) external returns (bool);
}
// Interface to ERC721 functions used in this contract
interface ERC721Token {
function transferFrom(address _from, address _to, uint256 _tokenId) external payable;
}
contract ChiMarket {
ERC20token ChiToken = ERC20token(0x71E1f8E809Dc8911FCAC95043bC94929a36505A5);
address owner;
uint256 market_halfspread;
modifier onlyOwner {
require(msg.sender == owner);
_;
}
function ChiMarket() public {
owner = msg.sender;
}
// Calculate the amount of ETH the contract pays out on a SELL
function calcSELLoffer(uint256 chi_amount) public view returns(uint256){
uint256 eth_balance = address(this).balance;
uint256 chi_balance = ChiToken.balanceOf(this);
uint256 eth_amount;
require(eth_balance > 0 && chi_balance > 0);
require(chi_balance + chi_amount >= chi_balance); // don't allow overflow
eth_amount = (chi_amount * eth_balance) / (chi_balance + chi_amount);
require(1000 * eth_amount >= eth_amount); // don't allow overflow
eth_amount = ((1000 - market_halfspread) * eth_amount) / 1000;
return eth_amount;
}
// Calculate the amount of ETH the contract requires on a BUY
// When this function is called from a payable function, the balance is updated
// already, so we need to subtract it through _offset_eth. Otherwise _offset_eth
// should be set to 0.
function calcBUYoffer(uint256 _chi_amount, uint256 _offset_eth) public view returns(uint256){
require(address(this).balance > _offset_eth); // no overflow
uint256 eth_balance = address(this).balance - _offset_eth;
uint256 chi_balance = ChiToken.balanceOf(this);
uint256 eth_amount;
require(eth_balance > 0 && chi_balance > 0);
require(chi_balance > _chi_amount); // must have enough CHI
require(chi_balance - _chi_amount <= chi_balance); // don't allow overflow
eth_amount = (_chi_amount * eth_balance) / (chi_balance - _chi_amount);
require(1000 * eth_amount >= eth_amount); // don't allow overflow
eth_amount = (1000 * eth_amount) / (1000 - market_halfspread);
return eth_amount;
}
// CHI buying function
// All of the ETH included in the TX is converted to CHI
// requires at least _min_chi_amount of CHI for that ETH, otherwise TX fails
function limitBuy(uint256 _chi_amount) public payable{
require(_chi_amount > 0);
uint256 eth_amount = calcBUYoffer(_chi_amount, msg.value);
require(eth_amount <= msg.value);
uint256 return_ETH_amount = msg.value - eth_amount;
require(return_ETH_amount < msg.value);
if(return_ETH_amount > 0){
msg.sender.transfer(return_ETH_amount); // return extra ETH
}
require(ChiToken.transfer(msg.sender, _chi_amount)); // send CHI tokens
}
// CHI selling function.
// sell _chi_amount of CHI
// require at least _min_eth_amount for that CHI, otherwise TX fails
// Make sure to set CHI allowance before calling this function
function limitSell(uint256 _chi_amount, uint256 _min_eth_amount) public {
require(ChiToken.allowance(msg.sender, this) >= _chi_amount);
uint256 eth_amount = calcSELLoffer(_chi_amount);
require(eth_amount >= _min_eth_amount);
require(eth_amount > 0);
require(ChiToken.transferFrom(msg.sender, this, _chi_amount));
msg.sender.transfer(eth_amount);
}
// Allows owner to move CHI (e.g. to an updated contract), also to rescue
// other ERC20 tokens sent by mistake.
function moveERC20Tokens(address _tokenContract, address _to, uint _val) public onlyOwner {
ERC20token token = ERC20token(_tokenContract);
require(token.transfer(_to, _val));
}
// Hopefully this doesn't get used, but it allows for gotchi rescue if someone sends
// their gotchi (or a cat) to the contract by mistake.
function moveERC721Tokens(address _tokenContract, address _to, uint256 _tid) public onlyOwner {
ERC721Token token = ERC721Token(_tokenContract);
token.transferFrom(this, _to, _tid);
}
// Allows the owner to move ether, for example to an updated contract
function moveEther(address _target, uint256 _amount) public onlyOwner {
require(_amount <= address(this).balance);
_target.transfer(_amount);
}
// Set the market spread (actually it's half of the spread).
function setSpread(uint256 _halfspread) public onlyOwner {
require(_halfspread <= 50);
market_halfspread = _halfspread;
}
// Allows for deposit of ETH and CHI at the same time (to avoid temporary imbalance
// in the market)
function depositBoth(uint256 _chi_amount) public payable onlyOwner {
require(ChiToken.allowance(msg.sender, this) >= _chi_amount);
require(ChiToken.transferFrom(msg.sender, this, _chi_amount));
}
// Allows for withdrawal of ETH and CHI at the same time (to avoid temporary imbalance
// in the market)
function withdrawBoth(uint256 _chi_amount, uint256 _eth_amount) public onlyOwner {
uint256 eth_balance = address(this).balance;
uint256 chi_balance = ChiToken.balanceOf(this);
require(_chi_amount <= chi_balance);
require(_eth_amount <= eth_balance);
msg.sender.transfer(_eth_amount);
require(ChiToken.transfer(msg.sender, _chi_amount));
}
// change the owner
function setOwner(address _owner) public onlyOwner {
owner = _owner;
}
// empty fallback payable to allow ETH deposits to the contract
function() public payable{
}
}
{
"compilationTarget": {
"ChiMarket.sol": "ChiMarket"
},
"evmVersion": "byzantium",
"libraries": {},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"constant":true,"inputs":[{"name":"_chi_amount","type":"uint256"},{"name":"_offset_eth","type":"uint256"}],"name":"calcBUYoffer","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_owner","type":"address"}],"name":"setOwner","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_target","type":"address"},{"name":"_amount","type":"uint256"}],"name":"moveEther","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_chi_amount","type":"uint256"},{"name":"_eth_amount","type":"uint256"}],"name":"withdrawBoth","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_chi_amount","type":"uint256"}],"name":"depositBoth","outputs":[],"payable":true,"stateMutability":"payable","type":"function"},{"constant":false,"inputs":[{"name":"_chi_amount","type":"uint256"},{"name":"_min_eth_amount","type":"uint256"}],"name":"limitSell","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"name":"chi_amount","type":"uint256"}],"name":"calcSELLoffer","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_tokenContract","type":"address"},{"name":"_to","type":"address"},{"name":"_val","type":"uint256"}],"name":"moveERC20Tokens","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_halfspread","type":"uint256"}],"name":"setSpread","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_chi_amount","type":"uint256"}],"name":"limitBuy","outputs":[],"payable":true,"stateMutability":"payable","type":"function"},{"constant":false,"inputs":[{"name":"_tokenContract","type":"address"},{"name":"_to","type":"address"},{"name":"_tid","type":"uint256"}],"name":"moveERC721Tokens","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"inputs":[],"payable":false,"stateMutability":"nonpayable","type":"constructor"},{"payable":true,"stateMutability":"payable","type":"fallback"}]