松露测试给出“错误:尝试运行调用合同功能的交易,但收件人地址___不是合同地址”

时间:2018-11-24 16:06:40

标签: javascript json ethereum solidity truffle

当我在终端上运行truffle test时,出现以下错误Error: Attempting to run transaction which calls a contract function, but recipient address 0x3ad2c00512808bd7fafa6dce844a583621f3df87 is not a contract address我不明白为什么得到此错误,因为如果我运行{{1},我的build文件夹是正常含义}终端中的地址与构建文件中的地址相同。当我运行松露迁移它工作。另外,每次我运行松露测试时,收件人地址总是会更改。我不知道该怎么办才能帮助您。

另一件事,只有在使用代码truffle migrate --reset时,才会发生此错误。管理员是msg.sender的=,这是ganache中的第一个帐户,所以我不知道出了什么问题。

我正在使用this视频。我已经完成了这个家伙的所有其他视频,一切都很好,直到现在15:11分钟他都进行了最终测试,对他来说,它可以工作,但是对我来说,它给了我上面的错误。

任何人,请帮助

这是我的测试(javascript)

selfdestruct(admin);

这是我的合同代码(牢固性)

var CinoCoin = artifacts.require("./CinoCoin.sol");
var CinoCoinSale = artifacts.require("./CinoCoinSale.sol");


contract('CinoCoinSale', function(accounts) {
  var tokenInstance;
  var tokenSaleInstance;
  var admin = accounts[0];
  var buyer = accounts[1];
  var tokenPrice = 1000000000000000; // in wei 0.01 ether
  var tokensAvailable = 750000;
  var numberOfTokens;

  it('initializes the contract with the correct values', function() {
    return CinoCoinSale.deployed().then(function(instance) {
      tokenSaleInstance = instance;
      return tokenSaleInstance.address
    }).then(function(address) {
      assert.notEqual(address, 0x0, 'has contract address');
      return tokenSaleInstance.tokenContract();
    }).then(function(address) {
      assert.notEqual(address, 0x0, 'has token contract address');
      return tokenSaleInstance.tokenPrice();
    }).then(function(price) {
      assert.equal(price, tokenPrice, 'token price is correct');
    });
  });

  it('facilitats token buying', function() {
    return CinoCoin.deployed().then(function(instance) {
      //Grab token instance first
      tokenInstance = instance;
      return CinoCoinSale.deployed();
    }).then(function(instance) {
      //Then grab token sale instance
      tokenSaleInstance = instance;
      //Provision 75% of all tokens to the token sale contract
      return tokenInstance.transfer(tokenSaleInstance.address, tokensAvailable, { from: admin})
    }).then(function(receipt) {
      numberOfTokens = 10;
      return tokenSaleInstance.buyTokens(numberOfTokens, { from: buyer, value: numberOfTokens * tokenPrice })
    }).then(function(receipt) {
      assert.equal(receipt.logs.length, 1, 'triggers one event');
      assert.equal(receipt.logs[0].event, 'Sell', 'should be the "Sell" event');
      assert.equal(receipt.logs[0].args._buyer, buyer, 'logs the account that purchased the tokens');
      assert.equal(receipt.logs[0].args._amount, numberOfTokens, 'logs the number of tokens purchased');
      return tokenSaleInstance.tokensSold();
    }).then(function(amount) {
      assert.equal(amount.toNumber(), numberOfTokens, 'increments the number of tokens sold');
      return tokenInstance.balanceOf(buyer);
    }).then(function(balance) {
      assert.equal(balance.toNumber(), numberOfTokens);
      return tokenInstance.balanceOf(tokenSaleInstance.address);
    }).then(function(balance) {
      assert.equal(balance.toNumber(), tokensAvailable - numberOfTokens);
      //Try to buy tokens different from the ether value
      return tokenSaleInstance.buyTokens(numberOfTokens, { from: buyer, value: 1 });
    }).then(assert.fail).catch(function(error) {
      assert(error.message.indexOf('revert') >= 0, 'msg.value must equal number of tokens in wei');
      return tokenSaleInstance.buyTokens(800000, { from: buyer, value: numberOfTokens * tokenPrice });
    }).then(assert.fail).catch(function(error) {
      assert(error.message.indexOf('revert') >= 0, 'connot purchase more tokens than available');
    });
  });

  it('ends token sale', function () {
    return CinoCoin.deployed().then(function(instance) {
      //Grab token instance first
      tokenInstance = instance;
      return CinoCoinSale.deployed();
    }).then(function(instance) {
      //Then grab token sale instance
      tokenSaleInstance = instance;
      //try to end sale from account other than admin
      return tokenSaleInstance.endSale({ from: buyer });
    }).then(assert.fail).catch(function(error) {
      assert(error.message.indexOf('revert' >= 0, 'must be admin to end sale'));
      //End sale as admin
      return tokenSaleInstance.endSale({ from: admin });
    }).then(function(receipt) {
      return tokenInstance.balanceOf(admin);
    }).then(function(balance) {
      assert.equal(balance.toNumber(), 999990, 'returns all unsold cino coins to admin');
      //Check that the token price was reset when selfFestruct was called
      return tokenSaleInstance.tokenPrice();
    }).then(function(price) {
      assert.equal(price.toNumber(), 0, 'token price was reset');
    });
  });
});

它说当我注释掉pragma solidity ^0.4.23; import "./CinoCoin.sol"; contract CinoCoinSale { address admin; CinoCoin public tokenContract; uint256 public tokenPrice; uint256 public tokensSold; event Sell(address _buyer, uint256 _amount); function CinoCoinSale(CinoCoin _tokenContract, uint256 _tokenPrice) public { //Assign an admin / an external account connected to the blockchain that has certain priviliges admin = msg.sender; //Assign token contract tokenContract = _tokenContract; //Token Price how much the token will cost tokenPrice = _tokenPrice; } //multiply function for function multiply(uint x, uint y) internal pure returns (uint z) { require(y == 0 || (z = x * y) / y == x); } //Buy tokens function buyTokens(uint256 _numberOfTokens) public payable { //Require that the value is equal to tokens require(msg.value == multiply(_numberOfTokens, tokenPrice)); //Require that there are enough tokens in the contrat require(tokenContract.balanceOf(this) >= _numberOfTokens); //Require the transfer is successful require(tokenContract.transfer(msg.sender, _numberOfTokens)); //Keep track of number of tokens sold tokensSold += _numberOfTokens; //Trigger a sell event Sell(msg.sender, _numberOfTokens); } //ending token CinoCoinSale function endSale()public { //Only an admin can end the end the sale require(msg.sender == admin); //Transfer the amount of token in the sale back to the admin require(tokenContract.transfer(admin, tokenContract.balanceOf(this))); //Destroy contract selfdestruct(admin); } } 时我的测试通过了,所以这似乎是一个探索

感谢您的帮助

1 个答案:

答案 0 :(得分:0)

很高兴您已修复错误。在查看您的代码时,我注意到由于大的Promise链,很难遵循测试文件。我将文件转换为async / await,将来可能会更易于维护和调试。

我还注意到您正在尝试通过手动检查日志并捕获还原异常来断言发出的事件和还原。实际上,我写了一个库来简化此过程,因此我也为此添加了代码。

可以使用npm安装该库:

npm install truffle-assertions

此后,新的测试代码应该可以使用。我希望这对您有所帮助,并祝您Dapp开发顺利。

const CinoCoin = artifacts.require("CinoCoin");
const CinoCoinSale = artifacts.require("CinoCoinSale");
const truffleAssert = require("truffle-assertions");

contract('CinoCoinSale', function(accounts) {
  let tokenInstance;
  let tokenSaleInstance;
  let admin = accounts[0];
  let buyer = accounts[1];
  let tokenPrice = 1000000000000000; // in wei 0.01 ether
  let tokensAvailable = 750000;

  it('initializes the contract with the correct values', async function() {
    tokenInstance = await CinoCoin.deployed();
    tokenSaleInstance = await CinoCoinSale.deployed();

    assert.notEqual(tokenSaleInstance.address, 0x0, 'has contract address');
    assert.notEqual(await tokenSaleInstance.tokenContract(), 0x0, 'has token contract address');
    assert.equal(await tokenSaleInstance.tokenPrice(), tokenPrice, 'token price is correct');
    assert.equal(await tokenSaleInstance.admin(), admin, 'admin is correct');
  });

  it('facilitates token buying', async function() {
    tokenInstance = await CinoCoin.deployed();
    tokenSaleInstance = await CinoCoinSale.deployed();
    await tokenInstance.transfer(tokenSaleInstance.address, tokensAvailable, { from: admin });

    const numberOfTokens = 10;
    const receipt = await tokenSaleInstance.buyTokens(numberOfTokens, { from: buyer, value: numberOfTokens * tokenPrice });

    truffleAssert.eventEmitted(receipt, 'Sell', (ev) => {
      return ev._buyer === buyer && ev._amount.toNumber() === numberOfTokens;
    });

    const tokensSold = await tokenSaleInstance.tokensSold();
    assert.equal(tokensSold.toNumber(), numberOfTokens, 'increments the number of tokens sold');

    const buyerBalance = await tokenInstance.balanceOf(tokenSaleInstance.address);
    assert.equal(buyerBalance.toNumber(), numberOfTokens);

    const tokenSaleBalance = await tokenInstance.balanceOf(tokenSaleInstance.address);
    assert.equal(tokenSaleBalance.toNumber(), tokensAvailable - numberOfTokens);

    truffleAssert.reverts(
      tokenSaleInstance.buyTokens(numberOfTokens, { from: buyer, value: 1 }),
      null,
      'msg.value must equal number of tokens in wei'
    );

    truffleAssert.reverts(
      tokenSaleInstance.buyTokens(800000, { from: buyer, value: numberOfTokens * tokenPrice }),
      null,
      'connot purchase more tokens than available'
    );
  });

  it('ends token sale', async function () {
    tokenInstance = await CinoCoin.deployed();
    tokenSaleInstance = await CinoCoinSale.deployed();

    truffleAssert.reverts(tokenSaleInstance.endSale({ from: buyer }), null, 'must be admin to end sale');

    await tokenSaleInstance.endSale({ from: admin });

    const adminBalance = await tokenInstance.balanceOf(admin);
    assert.equal(adminBalance.toNumber(), 999990, 'returns all unsold cino coins to admin');

    const tokenPrice = await tokenSaleInstance.tokenPrice();
    assert.equal(tokenPrice.toNumber(), 0, 'token price was reset');
  });
});