Skip to content

Instantly share code, notes, and snippets.

@pipermerriam
Created March 30, 2016 15:10
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save pipermerriam/f4ab133f591c2eb871ee3756c7aa4404 to your computer and use it in GitHub Desktop.
Save pipermerriam/f4ab133f591c2eb871ee3756c7aa4404 to your computer and use it in GitHub Desktop.
/// @title DigixDAO Contract Interfaces
contract ConfigInterface {
address public owner;
mapping (address => bool) admins;
mapping (bytes32 => address) addressMap;
mapping (bytes32 => bool) boolMap;
mapping (bytes32 => bytes32) bytesMap;
mapping (bytes32 => uint256) uintMap;
/// @notice setConfigAddress sets configuration `_key` to `_val`
/// @param _key The key name of the configuration.
/// @param _val The value of the configuration.
/// @return Whether the configuration setting was successful or not.
function setConfigAddress(bytes32 _key, address _val) returns (bool success);
/// @notice setConfigBool sets configuration `_key` to `_val`
/// @param _key The key name of the configuration.
/// @param _val The value of the configuration.
/// @return Whether the configuration setting was successful or not.
function setConfigBool(bytes32 _key, bool _val) returns (bool success);
/// @notice setConfigBytes sets configuration `_key` to `_val`
/// @param _key The key name of the configuration.
/// @param _val The value of the configuration.
/// @return Whether the configuration setting was successful or not.
function setConfigBytes(bytes32 _key, bytes32 _val) returns (bool success);
/// @notice setConfigUint `_key` to `_val`
/// @param _key The key name of the configuration.
/// @param _val The value of the configuration.
/// @return Whether the configuration setting was successful or not.
function setConfigUint(bytes32 _key, uint256 _val) returns (bool success);
/// @notice getConfigAddress gets configuration `_key`'s value
/// @param _key The key name of the configuration.
/// @return The configuration value
function getConfigAddress(bytes32 _key) returns (address val);
/// @notice getConfigBool gets configuration `_key`'s value
/// @param _key The key name of the configuration.
/// @return The configuration value
function getConfigBool(bytes32 _key) returns (bool val);
/// @notice getConfigBytes gets configuration `_key`'s value
/// @param _key The key name of the configuration.
/// @return The configuration value
function getConfigBytes(bytes32 _key) returns (bytes32 val);
/// @notice getConfigUint gets configuration `_key`'s value
/// @param _key The key name of the configuration.
/// @return The configuration value
function getConfigUint(bytes32 _key) returns (uint256 val);
/// @notice addAdmin sets `_admin` as configuration admin
/// @return Whether the configuration setting was successful or not.
function addAdmin(address _admin) returns (bool success);
/// @notice removeAdmin removes `_admin`'s rights
/// @param _admin The key name of the configuration.
/// @return Whether the configuration setting was successful or not.
function removeAdmin(address _admin) returns (bool success);
}
contract TokenInterface {
struct User {
bool locked;
uint256 balance;
uint256 badges;
mapping (address => uint256) allowed;
}
mapping (address => User) users;
mapping (address => uint256) balances;
mapping (address => mapping (address => uint256)) allowed;
mapping (address => bool) seller;
address config;
address owner;
address dao;
bool locked;
/// @return total amount of tokens
uint256 public totalSupply;
uint256 public totalBadges;
/// @param _owner The address from which the balance will be retrieved
/// @return The balance
function balanceOf(address _owner) constant returns (uint256 balance);
/// @param _owner The address from which the badge count will be retrieved
/// @return The badges count
function badgesOf(address _owner) constant returns (uint256 badge);
/// @notice send `_value` tokens to `_to` from `msg.sender`
/// @param _to The address of the recipient
/// @param _value The amount of tokens to be transfered
/// @return Whether the transfer was successful or not
function transfer(address _to, uint256 _value) returns (bool success);
/// @notice send `_value` badges to `_to` from `msg.sender`
/// @param _to The address of the recipient
/// @param _value The amount of tokens to be transfered
/// @return Whether the transfer was successful or not
function sendBadge(address _to, uint256 _value) returns (bool success);
/// @notice send `_value` tokens to `_to` from `_from` on the condition it is approved by `_from`
/// @param _from The address of the sender
/// @param _to The address of the recipient
/// @param _value The amount of tokens to be transfered
/// @return Whether the transfer was successful or not
function transferFrom(address _from, address _to, uint256 _value) returns (bool success);
/// @notice `msg.sender` approves `_spender` to spend `_value` tokens on its behalf
/// @param _spender The address of the account able to transfer the tokens
/// @param _value The amount of tokens to be approved for transfer
/// @return Whether the approval was successful or not
function approve(address _spender, uint256 _value) returns (bool success);
/// @param _owner The address of the account owning tokens
/// @param _spender The address of the account able to transfer the tokens
/// @return Amount of remaining tokens of _owner that _spender is allowed to spend
function allowance(address _owner, address _spender) constant returns (uint256 remaining);
/// @notice mint `_amount` of tokens to `_owner`
/// @param _owner The address of the account receiving the tokens
/// @param _amount The amount of tokens to mint
/// @return Whether or not minting was successful
function mint(address _owner, uint256 _amount) returns (bool success);
/// @notice mintBadge Mint `_amount` badges to `_owner`
/// @param _owner The address of the account receiving the tokens
/// @param _amount The amount of tokens to mint
/// @return Whether or not minting was successful
function mintBadge(address _owner, uint256 _amount) returns (bool success);
function registerDao(address _dao) returns (bool success);
function registerSeller(address _tokensales) returns (bool success);
event Transfer(address indexed _from, address indexed _to, uint256 _value);
event SendBadge(address indexed _from, address indexed _to, uint256 _amount);
event Approval(address indexed _owner, address indexed _spender, uint256 _value);
}
contract TokenSalesInterface {
struct SaleProxy {
address payout;
bool isProxy;
}
struct SaleStatus {
bool founderClaim;
uint256 releasedTokens;
uint256 releasedBadges;
uint256 claimers;
}
struct Info {
uint256 totalWei;
uint256 totalCents;
uint256 realCents;
uint256 amount;
}
struct SaleConfig {
uint256 startDate;
uint256 periodTwo;
uint256 periodThree;
uint256 endDate;
uint256 goal;
uint256 cap;
uint256 badgeCost;
uint256 founderAmount;
address founderWallet;
}
struct Buyer {
uint256 centsTotal;
uint256 weiTotal;
bool claimed;
}
Info saleInfo;
SaleConfig saleConfig;
SaleStatus saleStatus;
address config;
address owner;
bool locked;
uint256 public ethToCents;
mapping (address => Buyer) buyers;
mapping (address => SaleProxy) proxies;
/// @notice Calculates the parts per billion 1⁄1,000,000,000 of `_a` to `_b`
/// @param _a The antecedent
/// @param _c The consequent
/// @return Part per billion value
function ppb(uint256 _a, uint256 _c) public constant returns (uint256 b);
/// @notice Calculates the share from `_total` based on `_contrib`
/// @param _contrib The contributed amount in USD
/// @param _total The total amount raised in USD
/// @return Total number of shares
function calcShare(uint256 _contrib, uint256 _total) public constant returns (uint256 share);
/// @notice Calculates the current USD cents value of `_wei`
/// @param _wei the amount of wei
/// @return The USD cents value
function weiToCents(uint256 _wei) public constant returns (uint256 centsvalue);
function proxyPurchase(address _user) returns (bool success);
/// @notice Send msg.value purchase for _user.
/// @param _user The account to be credited
/// @return Success if purchase was accepted
function purchase(address _user, uint256 _amount) private returns (bool success);
/// @notice Get crowdsale information for `_user`
/// @param _user The account to be queried
/// @return `centstotal` the total amount of USD cents contributed
/// @return `weitotal` the total amount in wei contributed
/// @return `share` the current token shares earned
/// @return `badges` the number of proposer badges earned
/// @return `claimed` is true if the tokens and badges have been claimed
function userInfo(address _user) public constant returns (uint256 centstotal, uint256 weitotal, uint256 share, uint badges, bool claimed);
/// @notice Get the crowdsale information from msg.sender (see userInfo)
function myInfo() public constant returns (uint256 centstotal, uint256 weitotal, uint256 share, uint badges, bool claimed);
/// @notice get the total amount of wei raised for the crowdsale
/// @return The amount of wei raised
function totalWei() public constant returns (uint);
/// @notice get the total USD value in cents raised for the crowdsale
/// @return the amount USD cents
function totalCents() public constant returns (uint);
/// @notice get the current crowdsale information
/// @return `startsale` The unix timestamp for the start of the crowdsale and the first period modifier
/// @return `two` The unix timestamp for the start of the second period modifier
/// @return `three` The unix timestamp for the start of the third period modifier
/// @return `endsale` The unix timestamp of the end of crowdsale
/// @return `totalwei` The total amount of wei raised
/// @return `totalcents` The total number of USD cents raised
/// @return `amount` The amount of DGD tokens available for the crowdsale
/// @return `goal` The USD value goal for the crowdsale
/// @return `famount` Founders endowment
/// @return `faddress` Founder wallet address
/*function getSaleInfo() public constant returns (uint256 startsale, uint256 two, uint256 three, uint256 endsale, uint256 totalwei, uint256 totalcents, uint256 amount, uint256 goal, uint256 famount, address faddress);*/
function claimFor(address _user) returns (bool success);
/// @notice Allows msg.sender to claim the DGD tokens and badges if the goal is reached or refunds the ETH contributed if goal is not reached at the end of the crowdsale
function claim() returns (bool success);
function claimFounders() returns (bool success);
/// @notice See if the crowdsale goal has been reached
function goalReached() public constant returns (bool reached);
/// @notice Get the current sale period
/// @return `saleperiod` 0 = Outside of the crowdsale period, 1 = First reward period, 2 = Second reward period, 3 = Final crowdsale period.
function getPeriod() public constant returns (uint saleperiod);
/// @notice Get the date for the start of the crowdsale
/// @return `date` The unix timestamp for the start
function startDate() public constant returns (uint date);
/// @notice Get the date for the second reward period of the crowdsale
/// @return `date` The unix timestamp for the second period
function periodTwo() public constant returns (uint date);
/// @notice Get the date for the final period of the crowdsale
/// @return `date` The unix timestamp for the final period
function periodThree() public constant returns (uint date);
/// @notice Get the date for the end of the crowdsale
/// @return `date` The unix timestamp for the end of the crowdsale
function endDate() public constant returns (uint date);
/// @notice Check if crowdsale has ended
/// @return `ended` If the crowdsale has ended
function isEnded() public constant returns (bool ended);
/// @notice Send raised funds from the crowdsale to the DAO
/// @return `success` if the send succeeded
function sendFunds() public returns (bool success);
//function regProxy(address _payment, address _payout) returns (bool success);
function regProxy(address _payout) returns (bool success);
function getProxy(address _payout) public returns (address proxy);
function getPayout(address _proxy) public returns (address payout, bool isproxy);
function unlock() public returns (bool success);
function getSaleStatus() public constant returns (bool fclaim, uint256 reltokens, uint256 relbadges, uint256 claimers);
function getSaleInfo() public constant returns (uint256 weiamount, uint256 cents, uint256 realcents, uint256 amount);
function getSaleConfig() public constant returns (uint256 start, uint256 two, uint256 three, uint256 end, uint256 goal, uint256 cap, uint256 badgecost, uint256 famount, address fwallet);
event Purchase(uint256 indexed _exchange, uint256 indexed _rate, uint256 indexed _cents);
event Claim(address indexed _user, uint256 indexed _amount, uint256 indexed _badges);
}
contract ProxyPayment {
address payout;
address tokenSales;
address owner;
function ProxyPayment(address _payout, address _tokenSales) {
payout = _payout;
tokenSales = _tokenSales;
owner = _payout;
}
function () {
if (!TokenSalesInterface(tokenSales).proxyPurchase.value(msg.value).gas(106000)(payout)) throw;
}
}
contract TokenSales is TokenSalesInterface {
modifier ifOwner() {
if (msg.sender != owner) throw;
_
}
modifier ifOOrigin() {
if (tx.origin != owner) throw;
_
}
mapping (address => address) proxyPayouts;
uint256 public WEI_PER_ETH = 1000000000000000000;
uint256 public BILLION = 1000000000;
uint256 public CENTS = 100;
function TokenSales(address _config) {
owner = msg.sender;
config = _config;
saleStatus.founderClaim = false;
saleStatus.releasedTokens = 0;
saleStatus.releasedBadges = 0;
saleStatus.claimers = 0;
saleConfig.startDate = ConfigInterface(_config).getConfigUint("sale1:period1");
saleConfig.periodTwo = ConfigInterface(_config).getConfigUint("sale1:period2");
saleConfig.periodThree = ConfigInterface(_config).getConfigUint("sale1:period3");
saleConfig.endDate = ConfigInterface(_config).getConfigUint("sale1:end");
saleConfig.founderAmount = ConfigInterface(_config).getConfigUint("sale1:famount") * BILLION;
saleConfig.founderWallet = ConfigInterface(_config).getConfigAddress("sale1:fwallet");
saleConfig.goal = ConfigInterface(_config).getConfigUint("sale1:goal") * CENTS;
saleConfig.cap = ConfigInterface(_config).getConfigUint("sale1:cap") * CENTS;
saleConfig.badgeCost = ConfigInterface(_config).getConfigUint("sale1:badgecost") * CENTS;
saleInfo.amount = ConfigInterface(_config).getConfigUint("sale1:amount") * BILLION;
saleInfo.totalWei = 0;
saleInfo.totalCents = 0;
saleInfo.realCents;
saleStatus.founderClaim = false;
locked = true;
}
function () {
if (getPeriod() == 0) throw;
uint256 _amount = msg.value;
address _sender;
if (proxies[msg.sender].isProxy == true) {
_sender = proxies[msg.sender].payout;
} else {
_sender = msg.sender;
}
if (!purchase(_sender, _amount)) throw;
}
function proxyPurchase(address _user) returns (bool success) {
return purchase(_user, msg.value);
}
function purchase(address _user, uint256 _amount) private returns (bool success) {
uint256 _cents = weiToCents(_amount);
if ((saleInfo.realCents + _cents) > saleConfig.cap) return false;
uint256 _wei = _amount;
uint256 _modifier;
uint _period = getPeriod();
if ((_period == 0) || (_cents == 0)) {
return false;
} else {
if (_period == 3) _modifier = 100;
if (_period == 2) _modifier = 115;
if (_period == 1) _modifier = 130;
uint256 _creditwei = _amount;
uint256 _creditcents = (weiToCents(_creditwei) * _modifier * 10000) / 1000000 ;
buyers[_user].centsTotal += _creditcents;
buyers[_user].weiTotal += _creditwei;
saleInfo.totalCents += _creditcents;
saleInfo.realCents += _cents;
saleInfo.totalWei += _creditwei;
Purchase(ethToCents, _modifier, _creditcents);
return true;
}
}
function ppb(uint256 _a, uint256 _c) public constant returns (uint256 b) {
b = (BILLION * _a + _c / 2) / _c;
return b;
}
function calcShare(uint256 _contrib, uint256 _total) public constant returns (uint256 share) {
uint256 _ppb = ppb(_contrib, _total);
share = ((_ppb * saleInfo.amount) / BILLION);
return share;
}
function weiToCents(uint256 _wei) public constant returns (uint256 centsvalue) {
centsvalue = ((_wei * 100000 / WEI_PER_ETH) * ethToCents) / 100000;
return centsvalue;
}
function setEthToCents(uint256 _eth) ifOwner returns (bool success) {
ethToCents = _eth;
success = true;
return success;
}
function getSaleStatus() public constant returns (bool fclaim, uint256 reltokens, uint256 relbadges, uint256 claimers) {
return (saleStatus.founderClaim, saleStatus.releasedTokens, saleStatus.releasedBadges, saleStatus.claimers);
}
function getSaleInfo() public constant returns (uint256 weiamount, uint256 cents, uint256 realcents, uint256 amount) {
return (saleInfo.totalWei, saleInfo.totalCents, saleInfo.realCents, saleInfo.amount);
}
function getSaleConfig() public constant returns (uint256 start, uint256 two, uint256 three, uint256 end, uint256 goal, uint256 cap, uint256 badgecost, uint256 famount, address fwallet) {
return (saleConfig.startDate, saleConfig.periodTwo, saleConfig.periodThree, saleConfig.endDate, saleConfig.goal, saleConfig.cap, saleConfig.badgeCost, saleConfig.founderAmount, saleConfig.founderWallet);
}
function goalReached() public constant returns (bool reached) {
reached = (saleInfo.totalCents >= saleConfig.goal);
return reached;
}
function claim() returns (bool success) {
return claimFor(msg.sender);
}
function claimFor(address _user) returns (bool success) {
if ( (now < saleConfig.endDate) || (buyers[_user].claimed == true) ) {
return true;
}
if (!goalReached()) {
if (!address(_user).send(buyers[_user].weiTotal)) throw;
buyers[_user].claimed = true;
return true;
}
if (goalReached()) {
address _tokenc = ConfigInterface(config).getConfigAddress("ledger");
uint256 _tokens = calcShare(buyers[_user].centsTotal, saleInfo.totalCents);
uint256 _badges = buyers[_user].centsTotal / saleConfig.badgeCost;
if ((TokenInterface(_tokenc).mint(msg.sender, _tokens)) && (TokenInterface(_tokenc).mintBadge(_user, _badges))) {
saleStatus.releasedTokens += _tokens;
saleStatus.releasedBadges += _badges;
saleStatus.claimers += 1;
buyers[_user].claimed = true;
Claim(_user, _tokens, _badges);
return true;
} else {
return false;
}
}
}
function claimFounders() returns (bool success) {
if (saleStatus.founderClaim == true) return false;
if (now < saleConfig.endDate) return false;
if (!goalReached()) return false;
address _tokenc = ConfigInterface(config).getConfigAddress("ledger");
uint256 _tokens = saleConfig.founderAmount;
uint256 _badges = 4;
address _faddr = saleConfig.founderWallet;
if ((TokenInterface(_tokenc).mint(_faddr, _tokens)) && (TokenInterface(_tokenc).mintBadge(_faddr, _badges))) {
saleStatus.founderClaim = true;
saleStatus.releasedTokens += _tokens;
saleStatus.releasedBadges += _badges;
saleStatus.claimers += 1;
Claim(_faddr, _tokens, _badges);
return true;
} else {
return false;
}
}
function getPeriod() public constant returns (uint saleperiod) {
if ((now > saleConfig.endDate) || (now < saleConfig.startDate)) {
saleperiod = 0;
return saleperiod;
}
if (now >= saleConfig.periodThree) {
saleperiod = 3;
return saleperiod;
}
if (now >= saleConfig.periodTwo) {
saleperiod = 2;
return saleperiod;
}
if (now < saleConfig.periodTwo) {
saleperiod = 1;
return saleperiod;
}
}
function userInfo(address _user) public constant returns (uint256 centstotal, uint256 weitotal, uint256 share, uint badges, bool claimed) {
share = calcShare(buyers[_user].centsTotal, saleInfo.totalCents);
badges = buyers[_user].centsTotal / saleConfig.badgeCost;
return (buyers[_user].centsTotal, buyers[_user].weiTotal, share, badges, buyers[_user].claimed);
}
function myInfo() public constant returns (uint256 centstotal, uint256 weitotal, uint256 share, uint badges, bool claimed) {
return userInfo(msg.sender);
}
function totalWei() public constant returns (uint) {
return saleInfo.totalWei;
}
function totalCents() public constant returns (uint) {
return saleInfo.totalCents;
}
function startDate() public constant returns (uint date) {
return saleConfig.startDate;
}
function periodTwo() public constant returns (uint date) {
return saleConfig.periodTwo;
}
function periodThree() public constant returns (uint date) {
return saleConfig.periodThree;
}
function endDate() public constant returns (uint date) {
return saleConfig.endDate;
}
function isEnded() public constant returns (bool ended) {
return (now >= endDate());
}
function sendFunds() public returns (bool success) {
if (locked) return false;
if (!goalReached()) return false;
if (!isEnded()) return false;
address _dao = ConfigInterface(config).getConfigAddress("sale1:dao");
if (_dao == 0x0000000000000000000000000000000000000000) return false;
return _dao.send(totalWei());
}
function regProxy(address _payout) ifOOrigin returns (bool success) {
address _proxy = new ProxyPayment(_payout, address(this));
proxies[_proxy].payout = _payout;
proxies[_proxy].isProxy = true;
proxyPayouts[_payout] = _proxy;
return true;
}
function getProxy(address _payout) public returns (address proxy) {
return proxyPayouts[_payout];
}
function getPayout(address _proxy) public returns (address payout, bool isproxy) {
return (proxies[_proxy].payout, proxies[_proxy].isProxy);
}
function unlock() ifOwner public returns (bool success) {
locked = false;
return true;
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment