- pragma solidity ^0.5.0;
- /**
- * @dev Interface of the ERC20 standard as defined in the EIP. Does not include
- * the optional functions; to access them see {ERC20Detailed}.
- */
- interface IERC20 {
- /**
- * @dev Returns the amount of tokens in existence.
- */
- function totalSupply() external view returns (uint256);
- /**
- * @dev Returns the amount of tokens owned by `account`.
- */
- function balanceOf(address account) external view returns (uint256);
- /**
- * @dev Moves `amount` tokens from the caller's account to `recipient`.
- *
- * Returns a boolean value indicating whether the operation succeeded.
- *
- * Emits a {Transfer} event.
- */
- function transfer(address recipient, uint256 amount) external returns (bool);
- /**
- * @dev Returns the remaining number of tokens that `spender` will be
- * allowed to spend on behalf of `owner` through {transferFrom}. This is
- * zero by default.
- *
- * This value changes when {approve} or {transferFrom} are called.
- */
- function allowance(address owner, address spender) external view returns (uint256);
- /**
- * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
- *
- * Returns a boolean value indicating whether the operation succeeded.
- *
- * IMPORTANT: Beware that changing an allowance with this method brings the risk
- * that someone may use both the old and the new allowance by unfortunate
- * transaction ordering. One possible solution to mitigate this race
- * condition is to first reduce the spender's allowance to 0 and set the
- * desired value afterwards:
- * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
- *
- * Emits an {Approval} event.
- */
- function approve(address spender, uint256 amount) external returns (bool);
- /**
- * @dev Moves `amount` tokens from `sender` to `recipient` using the
- * allowance mechanism. `amount` is then deducted from the caller's
- * allowance.
- *
- * Returns a boolean value indicating whether the operation succeeded.
- *
- * Emits a {Transfer} event.
- */
- function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
- /**
- * @dev Emitted when `value` tokens are moved from one account (`from`) to
- * another (`to`).
- *
- * Note that `value` may be zero.
- */
- event Transfer(address indexed from, address indexed to, uint256 value);
- /**
- * @dev Emitted when the allowance of a `spender` for an `owner` is set by
- * a call to {approve}. `value` is the new allowance.
- */
- event Approval(address indexed owner, address indexed spender, uint256 value);
- }
- pragma solidity ^0.5.0;
- /**
- * @dev Contract module that helps prevent reentrant calls to a function.
- *
- * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
- * available, which can be applied to functions to make sure there are no nested
- * (reentrant) calls to them.
- *
- * Note that because there is a single `nonReentrant` guard, functions marked as
- * `nonReentrant` may not call one another. This can be worked around by making
- * those functions `private`, and then adding `external` `nonReentrant` entry
- * points to them.
- *
- * _Since v2.5.0:_ this module is now much more gas efficient, given net gas
- * metering changes introduced in the Istanbul hardfork.
- */
- contract ReentrancyGuard {
- bool private _notEntered;
- constructor () internal {
- // Storing an initial non-zero value makes deployment a bit more
- // expensive, but in exchange the refund on every call to nonReentrant
- // will be lower in amount. Since refunds are capped to a percetange of
- // the total transaction's gas, it is best to keep them low in cases
- // like this one, to increase the likelihood of the full refund coming
- // into effect.
- _notEntered = true;
- }
- /**
- * @dev Prevents a contract from calling itself, directly or indirectly.
- * Calling a `nonReentrant` function from another `nonReentrant`
- * function is not supported. It is possible to prevent this from happening
- * by making the `nonReentrant` function external, and make it call a
- * `private` function that does the actual work.
- */
- modifier nonReentrant() {
- // On the first call to nonReentrant, _notEntered will be true
- require(_notEntered, "ReentrancyGuard: reentrant call");
- // Any calls to nonReentrant after this point will fail
- _notEntered = false;
- _;
- // By storing the original value once again, a refund is triggered (see
- // https://eips.ethereum.org/EIPS/eip-2200)
- _notEntered = true;
- }
- }
- pragma solidity ^0.5.0;
- /*
- * @dev Provides information about the current execution context, including the
- * sender of the transaction and its data. While these are generally available
- * via msg.sender and msg.data, they should not be accessed in such a direct
- * manner, since when dealing with GSN meta-transactions the account sending and
- * paying for execution may not be the actual sender (as far as an application
- * is concerned).
- *
- * This contract is only required for intermediate, library-like contracts.
- */
- contract Context {
- // Empty internal constructor, to prevent people from mistakenly deploying
- // an instance of this contract, which should be used via inheritance.
- constructor () internal { }
- // solhint-disable-previous-line no-empty-blocks
- function _msgSender() internal view returns (address payable) {
- return msg.sender;
- }
- function _msgData() internal view returns (bytes memory) {
- this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
- return msg.data;
- }
- }
- pragma solidity ^0.5.0;
- /**
- * @dev Wrappers over Solidity's arithmetic operations with added overflow
- * checks.
- *
- * Arithmetic operations in Solidity wrap on overflow. This can easily result
- * in bugs, because programmers usually assume that an overflow raises an
- * error, which is the standard behavior in high level programming languages.
- * `SafeMath` restores this intuition by reverting the transaction when an
- * operation overflows.
- *
- * Using this library instead of the unchecked operations eliminates an entire
- * class of bugs, so it's recommended to use it always.
- */
- library SafeMath {
- /**
- * @dev Returns the addition of two unsigned integers, reverting on
- * overflow.
- *
- * Counterpart to Solidity's `+` operator.
- *
- * Requirements:
- * - Addition cannot overflow.
- */
- function add(uint256 a, uint256 b) internal pure returns (uint256) {
- uint256 c = a + b;
- require(c >= a, "SafeMath: addition overflow");
- return c;
- }
- /**
- * @dev Returns the subtraction of two unsigned integers, reverting on
- * overflow (when the result is negative).
- *
- * Counterpart to Solidity's `-` operator.
- *
- * Requirements:
- * - Subtraction cannot overflow.
- */
- function sub(uint256 a, uint256 b) internal pure returns (uint256) {
- return sub(a, b, "SafeMath: subtraction overflow");
- }
- /**
- * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
- * overflow (when the result is negative).
- *
- * Counterpart to Solidity's `-` operator.
- *
- * Requirements:
- * - Subtraction cannot overflow.
- *
- * _Available since v2.4.0._
- */
- function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
- require(b <= a, errorMessage);
- uint256 c = a - b;
- return c;
- }
- /**
- * @dev Returns the multiplication of two unsigned integers, reverting on
- * overflow.
- *
- * Counterpart to Solidity's `*` operator.
- *
- * Requirements:
- * - Multiplication cannot overflow.
- */
- function mul(uint256 a, uint256 b) internal pure returns (uint256) {
- // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
- // benefit is lost if 'b' is also tested.
- // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
- if (a == 0) {
- return 0;
- }
- uint256 c = a * b;
- require(c / a == b, "SafeMath: multiplication overflow");
- return c;
- }
- /**
- * @dev Returns the integer division of two unsigned integers. Reverts on
- * division by zero. The result is rounded towards zero.
- *
- * Counterpart to Solidity's `/` operator. Note: this function uses a
- * `revert` opcode (which leaves remaining gas untouched) while Solidity
- * uses an invalid opcode to revert (consuming all remaining gas).
- *
- * Requirements:
- * - The divisor cannot be zero.
- */
- function div(uint256 a, uint256 b) internal pure returns (uint256) {
- return div(a, b, "SafeMath: division by zero");
- }
- /**
- * @dev Returns the integer division of two unsigned integers. Reverts with custom message on
- * division by zero. The result is rounded towards zero.
- *
- * Counterpart to Solidity's `/` operator. Note: this function uses a
- * `revert` opcode (which leaves remaining gas untouched) while Solidity
- * uses an invalid opcode to revert (consuming all remaining gas).
- *
- * Requirements:
- * - The divisor cannot be zero.
- *
- * _Available since v2.4.0._
- */
- function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
- // Solidity only automatically asserts when dividing by 0
- require(b > 0, errorMessage);
- uint256 c = a / b;
- // assert(a == b * c + a % b); // There is no case in which this doesn't hold
- return c;
- }
- /**
- * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
- * Reverts when dividing by zero.
- *
- * Counterpart to Solidity's `%` operator. This function uses a `revert`
- * opcode (which leaves remaining gas untouched) while Solidity uses an
- * invalid opcode to revert (consuming all remaining gas).
- *
- * Requirements:
- * - The divisor cannot be zero.
- */
- function mod(uint256 a, uint256 b) internal pure returns (uint256) {
- return mod(a, b, "SafeMath: modulo by zero");
- }
- /**
- * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
- * Reverts with custom message when dividing by zero.
- *
- * Counterpart to Solidity's `%` operator. This function uses a `revert`
- * opcode (which leaves remaining gas untouched) while Solidity uses an
- * invalid opcode to revert (consuming all remaining gas).
- *
- * Requirements:
- * - The divisor cannot be zero.
- *
- * _Available since v2.4.0._
- */
- function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
- require(b != 0, errorMessage);
- return a % b;
- }
- }
- pragma solidity ^0.5.0;
- /**
- * @title SafeERC20
- * @dev Wrappers around ERC20 operations that throw on failure (when the token
- * contract returns false). Tokens that return no value (and instead revert or
- * throw on failure) are also supported, non-reverting calls are assumed to be
- * successful.
- * To use this library you can add a `using SafeERC20 for ERC20;` statement to your contract,
- * which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
- */
- library SafeERC20 {
- using SafeMath for uint256;
- using Address for address;
- function safeTransfer(IERC20 token, address to, uint256 value) internal {
- callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
- }
- function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal {
- callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
- }
- function safeApprove(IERC20 token, address spender, uint256 value) internal {
- // safeApprove should only be called when setting an initial allowance,
- // or when resetting it to zero. To increase and decrease it, use
- // 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
- // solhint-disable-next-line max-line-length
- require((value == 0) || (token.allowance(address(this), spender) == 0),
- "SafeERC20: approve from non-zero to non-zero allowance"
- );
- callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
- }
- function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal {
- uint256 newAllowance = token.allowance(address(this), spender).add(value);
- callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
- }
- function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal {
- uint256 newAllowance = token.allowance(address(this), spender).sub(value, "SafeERC20: decreased allowance below zero");
- callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
- }
- /**
- * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
- * on the return value: the return value is optional (but if data is returned, it must not be false).
- * @param token The token targeted by the call.
- * @param data The call data (encoded using abi.encode or one of its variants).
- */
- function callOptionalReturn(IERC20 token, bytes memory data) private {
- // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
- // we're implementing it ourselves.
- // A Solidity high level call has three parts:
- // 1. The target address is checked to verify it contains contract code
- // 2. The call itself is made, and success asserted
- // 3. The return value is decoded, which in turn checks the size of the returned data.
- // solhint-disable-next-line max-line-length
- require(address(token).isContract(), "SafeERC20: call to non-contract");
- // solhint-disable-next-line avoid-low-level-calls
- (bool success, bytes memory returndata) = address(token).call(data);
- require(success, "SafeERC20: low-level call failed");
- if (returndata.length > 0) { // Return data is optional
- // solhint-disable-next-line max-line-length
- require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
- }
- }
- }
- pragma solidity ^0.5.0;
- /**
- * @dev Standard math utilities missing in the Solidity language.
- */
- library Math {
- /**
- * @dev Returns the largest of two numbers.
- */
- function max(uint256 a, uint256 b) internal pure returns (uint256) {
- return a >= b ? a : b;
- }
- /**
- * @dev Returns the smallest of two numbers.
- */
- function min(uint256 a, uint256 b) internal pure returns (uint256) {
- return a < b ? a : b;
- }
- /**
- * @dev Returns the average of two numbers. The result is rounded towards
- * zero.
- */
- function average(uint256 a, uint256 b) internal pure returns (uint256) {
- // (a + b) / 2 can overflow, so we distribute
- return (a / 2) + (b / 2) + ((a % 2 + b % 2) / 2);
- }
- }
- pragma solidity ^0.5.0;
- /**
- * @title Crowdsale
- * @dev Crowdsale is a base contract for managing a token crowdsale,
- * allowing investors to purchase tokens with ether. This contract implements
- * such functionality in its most fundamental form and can be extended to provide additional
- * functionality and/or custom behavior.
- * The external interface represents the basic interface for purchasing tokens, and conforms
- * the base architecture for crowdsales. It is *not* intended to be modified / overridden.
- * The internal interface conforms the extensible and modifiable surface of crowdsales. Override
- * the methods to add functionality. Consider using 'super' where appropriate to concatenate
- * behavior.
- */
- contract Crowdsale is Context, ReentrancyGuard {
- using SafeMath for uint256;
- using SafeERC20 for IERC20;
- // The token being sold
- IERC20 private _token;
- // Address where funds are collected
- address payable private _wallet;
- // How many token units a buyer gets per wei.
- // The rate is the conversion between wei and the smallest and indivisible token unit.
- // So, if you are using a rate of 1 with a ERC20Detailed token with 3 decimals called TOK
- // 1 wei will give you 1 unit, or 0.001 TOK.
- uint256 private _rate;
- // Amount of wei raised
- uint256 private _weiRaised;
- /**
- * Event for token purchase logging
- * @param purchaser who paid for the tokens
- * @param beneficiary who got the tokens
- * @param value weis paid for purchase
- * @param amount amount of tokens purchased
- */
- event TokensPurchased(address indexed purchaser, address indexed beneficiary, uint256 value, uint256 amount);
- /**
- * @param rate Number of token units a buyer gets per wei
- * @dev The rate is the conversion between wei and the smallest and indivisible
- * token unit. So, if you are using a rate of 1 with a ERC20Detailed token
- * with 3 decimals called TOK, 1 wei will give you 1 unit, or 0.001 TOK.
- * @param wallet Address where collected funds will be forwarded to
- * @param token Address of the token being sold
- */
- constructor (uint256 rate, address payable wallet, IERC20 token) public {
- require(rate > 0, "Crowdsale: rate is 0");
- require(wallet != address(0), "Crowdsale: wallet is the zero address");
- require(address(token) != address(0), "Crowdsale: token is the zero address");
- _rate = rate;
- _wallet = wallet;
- _token = token;
- }
- /**
- * @dev fallback function ***DO NOT OVERRIDE***
- * Note that other contracts will transfer funds with a base gas stipend
- * of 2300, which is not enough to call buyTokens. Consider calling
- * buyTokens directly when purchasing tokens from a contract.
- */
- function () external payable {
- buyTokens(_msgSender());
- }
- /**
- * @return the token being sold.
- */
- function token() public view returns (IERC20) {
- return _token;
- }
- /**
- * @return the address where funds are collected.
- */
- function wallet() public view returns (address payable) {
- return _wallet;
- }
- /**
- * @return the number of token units a buyer gets per wei.
- */
- function rate() public view returns (uint256) {
- return _rate;
- }
- /**
- * @return the amount of wei raised.
- */
- function weiRaised() public view returns (uint256) {
- return _weiRaised;
- }
- /**
- * @dev low level token purchase ***DO NOT OVERRIDE***
- * This function has a non-reentrancy guard, so it shouldn't be called by
- * another `nonReentrant` function.
- * @param beneficiary Recipient of the token purchase
- */
- function buyTokens(address beneficiary) public nonReentrant payable {
- uint256 weiAmount = msg.value;
- _preValidatePurchase(beneficiary, weiAmount);
- // calculate token amount to be created
- uint256 tokens = _getTokenAmount(weiAmount);
- // update state
- _weiRaised = _weiRaised.add(weiAmount);
- _processPurchase(beneficiary, tokens);
- emit TokensPurchased(_msgSender(), beneficiary, weiAmount, tokens);
- _updatePurchasingState(beneficiary, weiAmount);
- _forwardFunds();
- _postValidatePurchase(beneficiary, weiAmount);
- }
- /**
- * @dev Validation of an incoming purchase. Use require statements to revert state when conditions are not met.
- * Use `super` in contracts that inherit from Crowdsale to extend their validations.
- * Example from CappedCrowdsale.sol's _preValidatePurchase method:
- * super._preValidatePurchase(beneficiary, weiAmount);
- * require(weiRaised().add(weiAmount) <= cap);
- * @param beneficiary Address performing the token purchase
- * @param weiAmount Value in wei involved in the purchase
- */
- function _preValidatePurchase(address beneficiary, uint256 weiAmount) internal view {
- require(beneficiary != address(0), "Crowdsale: beneficiary is the zero address");
- require(weiAmount != 0, "Crowdsale: weiAmount is 0");
- this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
- }
- /**
- * @dev Validation of an executed purchase. Observe state and use revert statements to undo rollback when valid
- * conditions are not met.
- * @param beneficiary Address performing the token purchase
- * @param weiAmount Value in wei involved in the purchase
- */
- function _postValidatePurchase(address beneficiary, uint256 weiAmount) internal view {
- // solhint-disable-previous-line no-empty-blocks
- }
- /**
- * @dev Source of tokens. Override this method to modify the way in which the crowdsale ultimately gets and sends
- * its tokens.
- * @param beneficiary Address performing the token purchase
- * @param tokenAmount Number of tokens to be emitted
- */
- function _deliverTokens(address beneficiary, uint256 tokenAmount) internal {
- _token.safeTransfer(beneficiary, tokenAmount);
- }
- /**
- * @dev Executed when a purchase has been validated and is ready to be executed. Doesn't necessarily emit/send
- * tokens.
- * @param beneficiary Address receiving the tokens
- * @param tokenAmount Number of tokens to be purchased
- */
- function _processPurchase(address beneficiary, uint256 tokenAmount) internal {
- _deliverTokens(beneficiary, tokenAmount);
- }
- /**
- * @dev Override for extensions that require an internal state to check for validity (current user contributions,
- * etc.)
- * @param beneficiary Address receiving the tokens
- * @param weiAmount Value in wei involved in the purchase
- */
- function _updatePurchasingState(address beneficiary, uint256 weiAmount) internal {
- // solhint-disable-previous-line no-empty-blocks
- }
- /**
- * @dev Override to extend the way in which ether is converted to tokens.
- * @param weiAmount Value in wei to be converted into tokens
- * @return Number of tokens that can be purchased with the specified _weiAmount
- */
- function _getTokenAmount(uint256 weiAmount) internal view returns (uint256) {
- return weiAmount.mul(_rate);
- }
- /**
- * @dev Determines how ETH is stored/forwarded on purchases.
- */
- function _forwardFunds() internal {
- _wallet.transfer(msg.value);
- }
- }
- pragma solidity ^0.5.0;
- /**
- * @title AllowanceCrowdsale
- * @dev Extension of Crowdsale where tokens are held by a wallet, which approves an allowance to the crowdsale.
- */
- contract AllowanceCrowdsale is Crowdsale {
- using SafeMath for uint256;
- using SafeERC20 for IERC20;
- address private _tokenWallet;
- /**
- * @dev Constructor, takes token wallet address.
- * @param tokenWallet Address holding the tokens, which has approved allowance to the crowdsale.
- */
- constructor (address tokenWallet) public {
- require(tokenWallet != address(0), "AllowanceCrowdsale: token wallet is the zero address");
- _tokenWallet = tokenWallet;
- }
- /**
- * @return the address of the wallet that will hold the tokens.
- */
- function tokenWallet() public view returns (address) {
- return _tokenWallet;
- }
- /**
- * @dev Checks the amount of tokens left in the allowance.
- * @return Amount of tokens left in the allowance
- */
- function remainingTokens() public view returns (uint256) {
- return Math.min(token().balanceOf(_tokenWallet), token().allowance(_tokenWallet, address(this)));
- }
- /**
- * @dev Overrides parent behavior by transferring tokens from wallet.
- * @param beneficiary Token purchaser
- * @param tokenAmount Amount of tokens purchased
- */
- function _deliverTokens(address beneficiary, uint256 tokenAmount) internal {
- token().safeTransferFrom(_tokenWallet, beneficiary, tokenAmount);
- }
- }
- pragma solidity ^0.5.0;
- /**
- * @title TimedCrowdsale
- * @dev Crowdsale accepting contributions only within a time frame.
- */
- contract TimedCrowdsale is Crowdsale {
- using SafeMath for uint256;
- uint256 private _openingTime;
- uint256 private _closingTime;
- /**
- * Event for crowdsale extending
- * @param newClosingTime new closing time
- * @param prevClosingTime old closing time
- */
- event TimedCrowdsaleExtended(uint256 prevClosingTime, uint256 newClosingTime);
- /**
- * @dev Reverts if not in crowdsale time range.
- */
- modifier onlyWhileOpen {
- require(isOpen(), "TimedCrowdsale: not open");
- _;
- }
- /**
- * @dev Constructor, takes crowdsale opening and closing times.
- * @param openingTime Crowdsale opening time
- * @param closingTime Crowdsale closing time
- */
- constructor (uint256 openingTime, uint256 closingTime) public {
- // solhint-disable-next-line not-rely-on-time
- require(openingTime >= block.timestamp, "TimedCrowdsale: opening time is before current time");
- // solhint-disable-next-line max-line-length
- require(closingTime > openingTime, "TimedCrowdsale: opening time is not before closing time");
- _openingTime = openingTime;
- _closingTime = closingTime;
- }
- /**
- * @return the crowdsale opening time.
- */
- function openingTime() public view returns (uint256) {
- return _openingTime;
- }
- /**
- * @return the crowdsale closing time.
- */
- function closingTime() public view returns (uint256) {
- return _closingTime;
- }
- /**
- * @return true if the crowdsale is open, false otherwise.
- */
- function isOpen() public view returns (bool) {
- // solhint-disable-next-line not-rely-on-time
- return block.timestamp >= _openingTime && block.timestamp <= _closingTime;
- }
- /**
- * @dev Checks whether the period in which the crowdsale is open has already elapsed.
- * @return Whether crowdsale period has elapsed
- */
- function hasClosed() public view returns (bool) {
- // solhint-disable-next-line not-rely-on-time
- return block.timestamp > _closingTime;
- }
- /**
- * @dev Extend parent behavior requiring to be within contributing period.
- * @param beneficiary Token purchaser
- * @param weiAmount Amount of wei contributed
- */
- function _preValidatePurchase(address beneficiary, uint256 weiAmount) internal onlyWhileOpen view {
- super._preValidatePurchase(beneficiary, weiAmount);
- }
- /**
- * @dev Extend crowdsale.
- * @param newClosingTime Crowdsale closing time
- */
- function _extendTime(uint256 newClosingTime) internal {
- require(!hasClosed(), "TimedCrowdsale: already closed");
- // solhint-disable-next-line max-line-length
- require(newClosingTime > _closingTime, "TimedCrowdsale: new closing time is before current closing time");
- emit TimedCrowdsaleExtended(_closingTime, newClosingTime);
- _closingTime = newClosingTime;
- }
- }
- pragma solidity ^0.5.0;
- /**
- * @title IncreasingPriceCrowdsale
- * @dev Extension of Crowdsale contract that increases the price of tokens linearly in time.
- * Note that what should be provided to the constructor is the initial and final _rates_, that is,
- * the amount of tokens per wei contributed. Thus, the initial rate must be greater than the final rate.
- */
- contract IncreasingPriceCrowdsale is TimedCrowdsale {
- using SafeMath for uint256;
- uint256 private _initialRate;
- uint256 private _finalRate;
- /**
- * @dev Constructor, takes initial and final rates of tokens received per wei contributed.
- * @param initialRate Number of tokens a buyer gets per wei at the start of the crowdsale
- * @param finalRate Number of tokens a buyer gets per wei at the end of the crowdsale
- */
- constructor (uint256 initialRate, uint256 finalRate) public {
- require(finalRate > 0, "IncreasingPriceCrowdsale: final rate is 0");
- // solhint-disable-next-line max-line-length
- require(initialRate > finalRate, "IncreasingPriceCrowdsale: initial rate is not greater than final rate");
- _initialRate = initialRate;
- _finalRate = finalRate;
- }
- /**
- * The base rate function is overridden to revert, since this crowdsale doesn't use it, and
- * all calls to it are a mistake.
- */
- function rate() public view returns (uint256) {
- revert("IncreasingPriceCrowdsale: rate() called");
- }
- /**
- * @return the initial rate of the crowdsale.
- */
- function initialRate() public view returns (uint256) {
- return _initialRate;
- }
- /**
- * @return the final rate of the crowdsale.
- */
- function finalRate() public view returns (uint256) {
- return _finalRate;
- }
- /**
- * @dev Returns the rate of tokens per wei at the present time.
- * Note that, as price _increases_ with time, the rate _decreases_.
- * @return The number of tokens a buyer gets per wei at a given time
- */
- function getCurrentRate() public view returns (uint256) {
- if (!isOpen()) {
- return 0;
- }
- // solhint-disable-next-line not-rely-on-time
- uint256 elapsedTime = block.timestamp.sub(openingTime());
- uint256 timeRange = closingTime().sub(openingTime());
- uint256 rateRange = _initialRate.sub(_finalRate);
- return _initialRate.sub(elapsedTime.mul(rateRange).div(timeRange));
- }
- /**
- * @dev Overrides parent method taking into account variable rate.
- * @param weiAmount The value in wei to be converted into tokens
- * @return The number of tokens _weiAmount wei will buy at present time
- */
- function _getTokenAmount(uint256 weiAmount) internal view returns (uint256) {
- uint256 currentRate = getCurrentRate();
- return currentRate.mul(weiAmount);
- }
- }
- contract TestToken is ERC20 {
- using SafeMath for uint256;
- string public symbol = "TEST";
- string public name = "TEST Token";
- uint8 public decimals = 18;
- uint256 public _totalSupply = 140000000 * 10**uint256(decimals);
- }
- contract TestCrowdsale is Crowdsale, AllowanceCrowdsale, TimedCrowdsale
- {
- constructor(
- uint256 _openingTime,
- uint256 _closingTime,
- uint256 _rate,
- address _wallet,
- address _tokenWallet,
- address _token
- )
- public
- AllowanceCrowdsale(_tokenWallet)
- TimedCrowdsale(_openingTime, _closingTime)
- {
- }
- }
TestCrowdsale.sol