Problema relacionado con el gas en el momento de la migración del contrato

Error: No se pudo almacenar el código de contrato, verifique su cantidad de gas. (También traté de agregar gas en truffle.js pero no tuve éxito)

pragma solidity ^0.4.21;


contract Ownable {
  address public owner;
  event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
  constructor() public {
    owner = msg.sender;
  }
  modifier onlyOwner() {
    require(msg.sender == owner);
    _;
  }
  function transferOwnership(address newOwner) public onlyOwner {
    require(newOwner != address(0));
    emit OwnershipTransferred(owner, newOwner);
    owner = newOwner;
  }
}

// ######################################################################

library SafeMath {
  function mul(uint256 a, uint256 b) internal pure returns (uint256) {
    if (a == 0) {
      return 0;
    }
    uint256 c = a * b;
    assert(c / a == b);
    return c;
  }
  function div(uint256 a, uint256 b) internal pure returns (uint256) {
    uint256 c = a / b;
    return c;
  }
  function sub(uint256 a, uint256 b) internal pure returns (uint256) {
    assert(b <= a);
    return a - b;
  }
  function add(uint256 a, uint256 b) internal pure returns (uint256) {
    uint256 c = a + b;
    assert(c >= a);
    return c;
  }
}

// ######################################################################

contract ERC20Basic {
  function totalSupply() public view returns (uint256);
  function balanceOf(address who) public view returns (uint256);
  function transfer(address to, uint256 value) public returns (bool);
  event Transfer(address indexed from, address indexed to, uint256 value);
}

// ######################################################################

contract BasicToken is ERC20Basic {
  using SafeMath for uint256;

  mapping(address => uint256) balances;

  uint256 totalSupply_;

  function totalSupply() public view returns (uint256) {
    return totalSupply_;
  }

  function transfer(address _to, uint256 _value) public returns (bool) {
    require(_to != address(0));
    require(_value <= balances[msg.sender]);

    // SafeMath.sub will throw if there is not enough balance.
    balances[msg.sender] = balances[msg.sender].sub(_value);
    balances[_to] = balances[_to].add(_value);
    emit Transfer(msg.sender, _to, _value);
    return true;
  }

  function balanceOf(address _owner) public view returns (uint256 balance) {
    return balances[_owner];
  }

}

// ######################################################################

contract ERC20 is ERC20Basic {
  function allowance(address owner, address spender) public view returns (uint256);
  function transferFrom(address from, address to, uint256 value) public returns (bool);
  function approve(address spender, uint256 value) public returns (bool);
  event Approval(address indexed owner, address indexed spender, uint256 value);
}

// ######################################################################

contract StandardToken is ERC20, BasicToken {

  mapping (address => mapping (address => uint256)) internal allowed;

  function transferFrom(address _from, address _to, uint256 _value) public returns (bool) {
    require(_to != address(0));
    require(_value <= balances[_from]);
    require(_value <= allowed[_from][msg.sender]);

    balances[_from] = balances[_from].sub(_value);
    balances[_to] = balances[_to].add(_value);
    allowed[_from][msg.sender] = allowed[_from][msg.sender].sub(_value);
    emit Transfer(_from, _to, _value);
    return true;
  }

  function approve(address _spender, uint256 _value) public returns (bool) {
    allowed[msg.sender][_spender] = _value;
    emit Approval(msg.sender, _spender, _value);
    return true;
  }

  function allowance(address _owner, address _spender) public view returns (uint256) {
    return allowed[_owner][_spender];
  }

  function increaseApproval(address _spender, uint _addedValue) public returns (bool) {
    allowed[msg.sender][_spender] = allowed[msg.sender][_spender].add(_addedValue);
    emit Approval(msg.sender, _spender, allowed[msg.sender][_spender]);
    return true;
  }

  function decreaseApproval(address _spender, uint _subtractedValue) public returns (bool) {
    uint oldValue = allowed[msg.sender][_spender];
    if (_subtractedValue > oldValue) {
      allowed[msg.sender][_spender] = 0;
    } else {
      allowed[msg.sender][_spender] = oldValue.sub(_subtractedValue);
    }
    emit Approval(msg.sender, _spender, allowed[msg.sender][_spender]);
    return true;
  }

}

// ######################################################################

contract DeveloperToken is StandardToken {
  string public constant name = "DeveloperToken";
  string public constant symbol = "DEV";
  uint256 public constant decimals = 18;
}

// ######################################################################

contract DeveloperCrowdsale is Ownable {
  using SafeMath for uint256;

  // ============= Token Distribution ================
  uint256 public maxTokens = 100000000 *(10 ** 18);
  uint256 public tokensForEcosystem = 80000000 *(10 ** 18);
  uint256 public tokensForBounty = 4000000 *(10 ** 18);
  uint256 public totalTokensForSale = 15000000 *(10 ** 18);
  uint256 public tokenForAirdrop = 1000000 *(10 ** 18);

  ERC20 public token;

  address public wallet;

  uint256 public rate;

  uint256 public weiRaised;

  uint256 public cap;
  uint256 public goal;

  // start and end timestamps where investments are allowed (both inclusive)
  uint256 public startTime;
  uint256 public endTime;

  bool public mintingFinished = false;

  mapping(address => uint256) balances;

  event Mint(address indexed to, uint256 amount);
  event MintFinished();
  event TokenPurchase(address indexed purchaser, address indexed beneficiary, uint256 value, uint256 amount);

  constructor() public {
    rate = 1000;
    wallet = msg.sender;
    token = new DeveloperToken();
    startTime = 1527186660000;  // new Date('05/25/2018 00:01').getTime();
    endTime = 1529951340000; //  new Date('06/25/2018 23:59').getTime();
    cap = 42500 *(10 ** 18);
    goal = 7500 *(10 ** 18);
  }

  function () external payable {
    uint256 tokensThatWillBeMintedAfterPurchase = msg.value.mul(rate);
    require(token.totalSupply() + tokensThatWillBeMintedAfterPurchase < totalTokensForSale);
    buyTokens(msg.sender);
  }

  // ====================== Crowdsale Price Management =================
  function setCrowdsalePrice() public onlyOwner {
      if (weiRaised <= 2500000 *(10 ** 18)) {
        setCurrentRate(1000);
      } else if (weiRaised <= 5000000 *(10 ** 18)) {
        setCurrentRate(500);
      } else if (weiRaised <= 10000000 *(10 ** 18)) {
        setCurrentRate(333);
      } else if (weiRaised <= 15000000 *(10 ** 18)) {
        setCurrentRate(250);
      }
  }

  // Change the current rate
  function setCurrentRate(uint256 _rate) private {
      rate = _rate;
  }

  function buyTokens(address _beneficiary) public payable {
    require(validPurchase());

    uint256 weiAmount = msg.value;

    // calculate token amount to be created
    uint256 tokens = _getTokenAmount(weiAmount);

    // update state
    weiRaised = weiRaised.add(weiAmount);

    _processPurchase(_beneficiary, tokens);
    emit TokenPurchase(msg.sender, _beneficiary, weiAmount, tokens);

    _forwardFunds();
  }

  function _deliverTokens(address _beneficiary, uint256 _tokenAmount) internal {
    token.transfer(_beneficiary, _tokenAmount);

  }

  function _processPurchase(address _beneficiary, uint256 _tokenAmount) internal {
    _deliverTokens(_beneficiary, _tokenAmount);
  }

  function _getTokenAmount(uint256 _weiAmount) internal view returns (uint256) {
    return _weiAmount.mul(rate);
  }

  function _forwardFunds() internal {
    token.totalSupply() + msg.value.mul(rate);
    wallet.transfer(msg.value);
  }

  // Finish: Mint Extra Tokens as needed before finalizing the Crowdsale.
  function finish(address _airdrop, address _ecosystemFund, address _bountyFund) public onlyOwner {
    require(!mintingFinished);
    require(hasEnded());
    uint256 alreadyMinted = token.totalSupply();
    require(alreadyMinted < maxTokens);

    uint256 unsoldTokens = totalTokensForSale - alreadyMinted;
    if (unsoldTokens > 0) {
      tokensForEcosystem = tokensForEcosystem + unsoldTokens;
    }

    token.totalSupply().add(tokensForEcosystem);
    token.totalSupply().add(tokensForBounty);
    token.totalSupply().add(tokenForAirdrop);

    balances[_airdrop].add(tokenForAirdrop);
    balances[_ecosystemFund].add(tokensForEcosystem);
    balances[_bountyFund].add(tokensForBounty);
    mintingFinished = true ;
  }

  // @return true if the transaction can buy tokens
  function validPurchase() internal view returns (bool) {
    bool withinCap = weiRaised.add(msg.value) <= cap;
    bool withinPeriod = now >= startTime && now <= endTime;
    bool nonZeroPurchase = msg.value != 0;
    return withinPeriod && nonZeroPurchase && withinCap;
  }

  // @return true if crowdsale event has ended
  function hasEnded() public view returns (bool) {
    bool capReached = weiRaised >= cap;
    return now > endTime || capReached;
  }

}

Este es mi archivo de migración:

var SafeMath = artifacts.require("./SafeMath.sol");
var DeveloperCrowdsale =  artifacts.require("./DeveloperCrowdsale.sol");


module.exports = function(deployer) {
  deployer.deploy(SafeMath);
  deployer.link(SafeMath, DeveloperCrowdsale);
  deployer.deploy(DeveloperCrowdsale);
};

};

Este es mi trufa.js:

module.exports = {
    networks: {
         development: {
              host: "localhost",
              port: 8545,
              network_id: "*" // Match any network id
            }
       }
};
Si está utilizando ganache-cli (vio su problema en Gitter), intente aumentar el límite de gas para ver si realmente es un problema de gas. ejecute ganache-cli con -l 9999999999la opción, por ejemplo.
¿Por qué estáis lanzando contratos individuales para contratos de los que queréis heredar? ¿Por qué no simplemente importarlos?
@KakiMasterOfTime, lo hice pero tuve el mismo problema.
@ReyHaynes, en el flujo de stackover no puedo copiar y pegar varios archivos, por eso hice un archivo individual. Espero que entiendas.
La versión @chiragmaliwal Web3 0.x que usa truffle usa el problema del gas como un error genérico, técnicamente puede significar cualquier cosa. mi consejo es que intente estar seguro de las variables que pasa a los constructores de su contrato e intente aislar la implementación del contrato que genera el error de origen del error. Si puede hacer un repositorio en github de su proyecto de trufas y enviar el enlace, lo veré yo mismo.
@KakiMasterOfTime, consulte el repositorio de github: github.com/cmaliwal/DeveloperToken
@chiragmaliwal Solo necesita implementar DeveloperCrowdsale dentro de su secuencia de comandos de migración.deployer.deploy(DeveloperCrowdsale);
@ZackMcGinnis, ¿Qué pasa con las matemáticas seguras y otras cosas?
@chiragmaliwal ya que los importa, se incluirán en la implementación de DeveloperCrowdsale (junto con cualquier otro archivo de solidez del que herede)
¡Oooh! Genial... ¡Me salvas el día! Gracias @ZackMcGinnis. También estaba pensando lo mismo, pero no estaba seguro acerca de este enfoque.
@ZackMcGinnis, según su sugerencia, hice lo mismo y el contrato se implementó correctamente. Pero no veo ninguna información relacionada con el token ERC20 en etherscan. Por favor, eche un vistazo: rinkeby.etherscan.io/tx/…
@ZackMcGinnis, cuando agrego esta dirección en metsmask (agregar tokens) no obtengo ningún dato relacionado con mi token, obtengo 0 TKN
Mmm interesante. Dentro de su DeveloperCrowdsalecontrato, tratando de cambiar ERC20 public tokena DeveloperToken public token.
@ZackMcGinnis, no tuvo ningún éxito. Todavía tengo algunos problemas. Por favor, eche un vistazo: ropsten.etherscan.io/tx/…
intente registrar en un evento la dirección del "nuevo DeveloperToken" que está creando.

Respuestas (2)

No necesita implementar todos los contratos, en su caso solo la biblioteca SafeMath y el contrato DeveloperCrowdsale.

module.exports = function(deployer) {
  deployer.deploy(SafeMath);
  deployer.link(SafeMath, DeveloperCrowdsale);
  deployer.deploy(DeveloperCrowdsale);
};
Bueno ! Entendí tu punto. contract implementado con éxito, pero cuando intenté agregar tokens en metamask, no obtengo los detalles del token, por lo que parece que algo está mal en la implementación. por favor, eche un vistazo: ethereum.stackexchange.com/questions/47208/…
No menciones, estoy muy cerca de mi respuesta. Aceptaré tu respuesta.

editar: Aprendido en las versiones más nuevas, debería llamarse constructor. en tu caso esta mal escrito

Esto es solo un error tipográfico y actualicé el código.
esta no es la razón del error.