簡體   English   中英

測試我的 Solidity 智能合約時遇到問題(帶松露)

[英]Having a Problem Testing my Solidity Smart Contract (w/ Truffle)

所以我正在構建一個基本的 NF 令牌。 創建了一個基本的鑄造功能和映射。 我用松露測試應用程序,嘗試了可靠性測試和 JS 測試。 出現以下錯誤。

Using network 'development'.

Compiling ./contracts/NFCertificate.sol...
Compiling ./node_modules/openzeppelin-solidity/contracts/AddressUtils.sol...
Compiling ./node_modules/openzeppelin-solidity/contracts/introspection/ERC165.sol...
Compiling ./node_modules/openzeppelin-solidity/contracts/introspection/SupportsInterfaceWithLookup.sol...
Compiling ./node_modules/openzeppelin-solidity/contracts/math/SafeMath.sol...
Compiling ./node_modules/openzeppelin-solidity/contracts/ownership/Ownable.sol...
Compiling ./node_modules/openzeppelin-solidity/contracts/token/ERC721/ERC721Basic.sol...
Compiling ./node_modules/openzeppelin-solidity/contracts/token/ERC721/ERC721BasicToken.sol...
Compiling ./node_modules/openzeppelin-solidity/contracts/token/ERC721/ERC721Receiver.sol...
Compiling ./test/TestCertificate.sol...
Compiling truffle/Assert.sol...
Compiling truffle/DeployedAddresses.sol...

Compilation warnings encountered:

/Users/aditya/Desktop/Work & Hobbies/Ideas/Blockchain/Blockchain Development/Ethereum:dApp/CertificateContract/contracts/NFCertificate.sol:26:35: Warning: This function only accepts a single "bytes" argument. Please use "abi.encodePacked(...)" or a similar function to encode the data.
        uint256 tokenId = uint256(keccak256(certificateNum, msg.sender, title, message));
                                  ^---------------------------------------------------^



  TestCertificate
    1) testNumber
    > No events were emitted

  Contract: NFCertificate
    2) It should return the same number
    > No events were emitted


  0 passing (1s)
  2 failing

  1) TestCertificate
       testNumber:
     Error: VM Exception while processing transaction: revert
      at Object.InvalidResponse (/usr/local/lib/node_modules/truffle/build/webpack:/~/web3/lib/web3/errors.js:38:1)
      at /usr/local/lib/node_modules/truffle/build/webpack:/~/web3/lib/web3/requestmanager.js:86:1
      at /usr/local/lib/node_modules/truffle/build/webpack:/packages/truffle-provider/wrapper.js:134:1
      at XMLHttpRequest.request.onreadystatechange (/usr/local/lib/node_modules/truffle/build/webpack:/~/web3/lib/web3/httpprovider.js:128:1)
      at XMLHttpRequestEventTarget.dispatchEvent (/usr/local/lib/node_modules/truffle/build/webpack:/~/xhr2/lib/xhr2.js:64:1)
      at XMLHttpRequest._setReadyState (/usr/local/lib/node_modules/truffle/build/webpack:/~/xhr2/lib/xhr2.js:354:1)
      at XMLHttpRequest._onHttpResponseEnd (/usr/local/lib/node_modules/truffle/build/webpack:/~/xhr2/lib/xhr2.js:509:1)
      at IncomingMessage.<anonymous> (/usr/local/lib/node_modules/truffle/build/webpack:/~/xhr2/lib/xhr2.js:469:1)
      at endReadableNT (_stream_readable.js:1081:12)
      at process._tickCallback (internal/process/next_tick.js:63:19)

  2) Contract: NFCertificate
       It should return the same number:
     TypeError: instance.returnNumbers is not a function
      at Context.<anonymous> (test/TestCertificate.js:9:31)
      at /usr/local/lib/node_modules/truffle/build/webpack:/packages/truffle-core/lib/testing/testrunner.js:135:1
      at /usr/local/lib/node_modules/truffle/build/webpack:/~/web3/lib/web3/property.js:119:1
      at /usr/local/lib/node_modules/truffle/build/webpack:/~/web3/lib/web3/requestmanager.js:89:1
      at /usr/local/lib/node_modules/truffle/build/webpack:/packages/truffle-provider/wrapper.js:134:1
      at XMLHttpRequest.request.onreadystatechange (/usr/local/lib/node_modules/truffle/build/webpack:/~/web3/lib/web3/httpprovider.js:128:1)
      at XMLHttpRequestEventTarget.dispatchEvent (/usr/local/lib/node_modules/truffle/build/webpack:/~/xhr2/lib/xhr2.js:64:1)
      at XMLHttpRequest._setReadyState (/usr/local/lib/node_modules/truffle/build/webpack:/~/xhr2/lib/xhr2.js:354:1)
      at XMLHttpRequest._onHttpResponseEnd (/usr/local/lib/node_modules/truffle/build/webpack:/~/xhr2/lib/xhr2.js:509:1)
      at IncomingMessage.<anonymous> (/usr/local/lib/node_modules/truffle/build/webpack:/~/xhr2/lib/xhr2.js:469:1)
      at endReadableNT (_stream_readable.js:1081:12)
      at process._tickCallback (internal/process/next_tick.js:63:19)

測試腳本在合約中運行一個簡單的內部函數,該函數返回 int = 1000,並將其與測試中聲明的預期變量(讓 expected = 1000)進行比較。 這是JS測試腳本

  import assertRevert from "zeppelin- 
  solidity/test/helpers/assertRevert";

  const NFCertificate = artifacts.require("NFCertificate");

  contract("NFCertificate", () => {
      it("It should return the same number", function() {
          let instance = NFCertificate.deployed();
          let expected = 1000;
          assert.equal(instance.returnNumber(), expected);
      });
  });

這是 Solidity 測試腳本:

pragma solidity ^0.4.20;

import "truffle/Assert.sol";
import "truffle/DeployedAddresses.sol";
import "../contracts/NFCertificate.sol";

contract TestCertificate {

  function testNumber() public {
    NFCertificate cert = NFCertificate(DeployedAddresses.NFCertificate());
    uint expected = 1000;
    Assert.equal(cert.returnNumber(), expected, "Numbers should be equal");
  }
}

我還嘗試測試鑄造的令牌 ID,以及基於我聲明的映射的令牌所有者,我遇到了同樣的問題。 在用 javascript 編寫測試合約時,它無法識別原始 NFT 合約中的功能。 在可靠地編寫測試合約時,它幾乎總是說“錯誤:處理事務時出現 VM 異常:還原”,沒有別的。

最后,這是我試圖測試的合同。 感謝任何和所有幫助,我對編碼和以太坊非常陌生,所以我可能犯了很多錯誤。

pragma solidity ^0.4.24;

import '../node_modules/openzeppelin-solidity/contracts/token/ERC721/ERC721BasicToken.sol';
import '../node_modules/openzeppelin-solidity/contracts/ownership/Ownable.sol';

contract NFCertificate is ERC721BasicToken, Ownable {
    struct Certificate {
        uint certNum;
        uint256 tokenId;
        bytes32 title;
        bytes32 message;
        address owner;
    }

    mapping (uint256 => address) tokenToOwner;
    mapping (address => uint256) ownerToToken;
    mapping (uint256 => string) tokenIdToName;

    event returnNumbers(uint number);

    Certificate[] public certificates;

    function createCert(bytes32 title, bytes32 message) public returns (bytes32){
        uint certificateNum = certificates.length - 1; 
        uint256 tokenId = uint256(keccak256(certificateNum, msg.sender, title, message));
        certificates.push(Certificate(certificateNum++, tokenId, title, message, msg.sender));
        tokenToOwner[tokenId] = msg.sender;
        ownerToToken[msg.sender] = tokenId;
        _mint(msg.sender, tokenId);
    }

    function returnNumber() public returns(uint) {
        uint number = 1000;
        returnNumbers(number);
        return number;
    }

    function whatTokensDoYouOwn(address owner) public view returns(uint256) {
        return ownerToToken[owner]; 
    }


}

NFCertificate.deployed()返回一個 promise, instance.returnNumber() 所以JS應該是:

contract("NFCertificate", () => {
  it("It should return the same number", async function() {
      let instance = await NFCertificate.deployed();
      let expected = 1000;
      assert.equal(await instance.returnNumber(), expected);
  });
});

由於returnNumbers是一個事件,它應該使用emit關鍵字emit 以大寫開頭的事件樣式也很好,否則它可能看起來像一個函數。 所以event ReturnNumbers(uint number);

function returnNumber() public returns(uint) {
    uint number = 1000;
    emit ReturnNumbers(number);
    return number;
}

添加 async/await 行應該可以修復您的 JavaScript 測試。 如果您稍后想斷言您的ReturnNumbers事件已正確發出,我建議使用我的truffle-assertions庫,其中包含斷言事件已發出或尚未發出的函數。 它還包括以直接方式斷言恢復和其他故障的功能。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM