Test Contract Source

A copy of the Solidity source code file is included here as a reference when going through the examples in the Using document.

The source file is found at: <Python sys.prefix dir>/contracts/Test.sol

pragma solidity ^0.8;

// SPDX-FileCopyrightText: Copyright 2021 Stephen R. Newell
// SPDX-License-Identifier: MIT

/**
 *  @title Test Contract
 *
 *  @author Stephen Newell
 *
 *  @notice This is used to for testing simpleth methods. It is designed
 *  with very simple transactions, functions, and variables to support
 *  a wide variety of test cases.
 *
 *  @dev All changes must be made with the Python unit tests in mind. Be
 *  cautious that you do not break anything. See the `tests` directory for
 *  PyTest tests that use this contract.
 */
contract Test {

    /// @dev used to store enum
    enum Size {SMALL, MEDIUM, LARGE}
    Size public testEnum;

    /// @dev set by constructor. Has no other purpose.
    int public initNum;

    /// @dev used by the storeNums...() transactions
    uint[3] public nums = [uint(0), 1, 2];

    /// @dev used by sumNums() to hold the sum of nums[].
    uint public numsTotal;

    /// @dev address that constructed and deployed the contract.
    address public owner;

    /// @dev used to store an address
    address public testAddr;

    /// @dev used to store an array
    uint[3] public testArray;

    /// @dev used to store a boolean
    bool public testBool;

    /// @dev used to store a four-byte value
    bytes4 public testBytes4;

    /// @dev used to store a 32-byte value
    bytes32 public testBytes32;

    /// @dev used to store a byte array
    bytes public testBytes;

    /// @dev used to store a signed integer
    int public testInt;

    /// @dev used to store a string
    string public testStr;

    /// @dev used to store an unsigned integer
    uint public testUint;


    /**
     * @notice Emitted when new byte values are stored
     *
     * @param timestamp block time when nums were updated
     * @param testBytes4 a 4-byte value
     * @param testBytes32 a 32-byte value
     * @param testBytes a byte array value
     */
    event BytesStored(
        uint timestamp,
        bytes4 testBytes4,
        bytes32 testBytes32,
        bytes testBytes
    );

    /**
     * @notice Emitted when contract is destroyed
     *
     * @param timestamp block time when paid
     * @param amountGwei contract's ether balance sent to owner
     */
    event Destroyed(
        uint timestamp,
        uint amountGwei
    );

    /**
     * @notice Emitted when new num1 is stored
     *
     * @param timestamp block time when initNum was updated
     * @param divisor used to divide initNum
     * @param result resulting initNum
     */
    event InitNumDivided(
        uint timestamp,
        int divisor,
        int result
    );

    /**
     * @notice Emitted when num0 is stored
     *
     * @param timestamp block time when nums were updated
     * @param num0 stored in nums[0]
     */
    event Num0Stored(
        uint timestamp,
        uint num0
    );

    /**
     * @notice Emitted when new num1 is stored
     *
     * @param timestamp block time when nums were updated
     * @param num1 stored in nums[1]
     */
    event Num1Stored(
        uint timestamp,
        uint num1
    );

    /**
     * @notice Emitted when new num2 is stored
     *
     * @param timestamp block time when nums were updated
     * @param num2 stored in nums[2]
     */
    event Num2Stored(
        uint timestamp,
        uint num2
    );

    /**
     * @notice Emitted when nums[] are divided
     *
     * @param timestamp block time when nums divided
     * @param num0 value in nums[0] after dividing
     * @param num1 value in nums[1] after dividing
     * @param num2 value in nums[2] after dividing
     * @param divisor value used to divide nums[]
     */
    event NumsDivided(
        uint timestamp,
        uint num0,
        uint num1,
        uint num2,
        uint divisor
    );

    /**
     * @notice Emitted when a selected nums[] is stored
     *
     * @param timestamp block time when nums was updated
     * @param index into nums[]
     * @param num stored in nums[`index`]
     */
    event NumStored(
        uint timestamp,
        uint index,
        uint num
    );

    /**
     * @notice Emitted when new nums are stored
     *
     * @param timestamp block time when nums were updated
     * @param num0 stored in nums[0].  It is indexed for some test cases.
     * @param num1 stored in nums[1]
     * @param num2 stored in nums[2]
     */
    event NumsStored(
        uint timestamp,
        uint indexed num0,
        uint num1,
        uint num2
    );

    /**
     * @notice Emitted when nums were stored and then divided
     *
     * @param timestamp block time after nums[] divided
     */
    event NumsStoredAndDivided(uint timestamp);

    /**
     * @notice Emitted when new nums are stored along with
     * a value (in wei) sent as a payment.
     *
     * @param timestamp block time when nums were updated
     * @param num0 stored in nums[0]
     * @param num1 stored in nums[1]
     * @param num2 stored in nums[2]
     * @param paid amount of wei sent
     * @param balance amount of wei in contract's balance
     */
    event NumsStoredAndPaid(
        uint timestamp,
        uint num0,
        uint num1,
        uint num2,
        uint paid,
        uint balance
    );

    /**
     * @notice Emitted when nums were stored and then summed
     *
     * @param timestamp block time after total was stored
     */
    event NumsStoredAndSummed(uint timestamp);

    /**
     * @notice Emitted when nums[] total is stored
     *
     * @param timestamp block time when total is stored
     * @param num0 value in nums[0]
     * @param num1 value in nums[1]
     * @param num2 value in nums[2]
     * @param total sum of the three nums assigned to numsTotal
     */
    event NumsSummed(
        uint timestamp,
        uint num0,
        uint num1,
        uint num2,
        uint total
    );

    /**
     * @notice Emitted when owner is changed
     *
     * @param timestamp block time when owner was set
     * @param newOwner address of the new owner
     */
    event OwnerSet(
        uint timestamp,
        address newOwner
    );

    /**
     * @notice Emitted when contract address is sent ether
     *
     * @param timestamp block time when paid
     * @param sender address sending the ether
     * @param amountWei of ether received (in wei)
     */
    event Received(
        uint timestamp,
        address sender,
        uint amountWei
    );

    /**
     * @notice Emitted when the contract is deployed.
     *
     * @dev Parameters are arbitrary.
     *
     * @param timestamp block time, in epoch seconds, when deployed
     * @param sender becomes the address of owner
     * @param initNum value assigned with constructor()
     * @param Test address of this contract
     */
    event TestConstructed(
        uint timestamp,
        address indexed sender,
        int initNum,
        address Test
    );

    /**
     * @notice Emitted when nums[0] and nums[1] total is stored
     *
     * @param timestamp block time when total is stored
     * @param num0 value in nums[0]
     * @param num1 value in nums[1]
     * @param total sum of the first two nums assigned to numsTotal
     */
    event TwoNumsSummed(
        uint timestamp,
        uint num0,
        uint num1,
        uint total
    );

    /**
     * @notice Emitted when the four different types of variables
     * are stored
     *
     * @param timestamp block time when variables were updated
     * @param testBool value given to the boolean variable
     * @param testEnum value given to the enumerated variable
     * @param testUint value given to the unsigned integer variable
     * @param testInt value given to the signed integer variable
     * @param testAddr value given to the address variable
     * @param testStr value given to the string variable
     * @param testArray values given to the array
     */
    event TypesStored(
        uint timestamp,
        bool testBool,
        Size testEnum,
        uint testUint,
        int testInt,
        address testAddr,
        string testStr,
        uint[3] testArray
    );


    /**
     * @notice Guard function that requires the sender be the Club
     * owner.
     *
     * @dev Used for owner-only transactions.
     */
    modifier isOwner() {
        require(msg.sender == owner, "Must be owner");
        _;
    }


    /**
     * @notice Create a new Test contract on the blockchain.
     *
     * @dev msg.sender becomes contract owner. Emits
     * TestConstructed().
     *
     * @param _initNum value is stored in initNum variable
     */
    constructor(int _initNum) {
        owner = msg.sender;
        initNum = _initNum;
        emit TestConstructed(
            block.timestamp,
            msg.sender,
            initNum,
            address(this)
        );
    }

    /**
     * @notice Allows test of assert()
     *
     * @dev If _value <= 10, assert will fail and pass back a message.
     *
     * @param _value only used in assert() test. Greater than 10 passes
     * assert(). 10, or less, fails assert().
     */
    function assertGreaterThan10(int _value)
        public
        pure
    {
        assert(_value > 10);
    }

    /**
     * @notice Destroy the deployed test contract. Make it unusable.
     *
     * @dev This is irreversible. Once destroyed, a contract is still
     * on the blockchain and transactions can be sent to it, but they will
     * not have any effect. Any ether in the contract's balance is sent
     * to _to. After a contract is destroyed, its either is
     * inaccessible. Emits Destroyed event. Must be owner to use.
     *
     * @param _to address to receive contract's ether balance
     */
    function destroy(address payable _to)
        public
        isOwner
    {
        emit Destroyed(
            block.timestamp,
            address(this).balance
        );
        selfdestruct(_to);
    }

    /**
     * @notice Divides initNum by a divisor
     *
     * @dev Emits InitNumDivided(). Used to test for divide-by-zero
     * errors by using 0 for divisor and for non-integer results by using
     * 3, or other, for divisor
     *
     * @param _divisor divide initNum by this value
     */
    function divideInitNum(int _divisor)
        public
    {
        initNum = initNum / _divisor;
        emit InitNumDivided(
            block.timestamp,
            _divisor,
            initNum
        );
    }

    /**
     * @notice Divides values in nums[]. There is no test for
     * _divisor being zero. This is used to test a transaction
     * that fails.
     *
     * @dev Emits NumsDivided()
     */
    function divideNums(uint _divisor)
        public
    {
        nums[0] = nums[0] / _divisor;
        nums[1] = nums[1] / _divisor;
        nums[2] = nums[2] / _divisor;
        emit NumsDivided(
            block.timestamp,
            nums[0],
            nums[1],
            nums[2],
            _divisor
        );
    }

    /**
     * @notice Function to return the three test byte values
     *
     * @return testBytes4_ four-byte value in testBytes4
     * @return testBytes32_ 32-byte value in testBytes32
     * @return testBytes_ byte string in testBytes
     */
    function getBytes()
        public
        view
        returns(
            bytes4 testBytes4_,
            bytes32 testBytes32_,
            bytes memory testBytes_
        )
    {
        return (testBytes4, testBytes32, testBytes);
    }

    /**
     * @notice Function to return nums[index]
     *
     * @param index specifies the nums[] entry to return
     *
     * @return num value for nums[index]
     */
    function getNum(uint8 index) public view returns(uint num) {
        return nums[index];
    }

    /**
     * @notice Function to return nums[0]
     *
     * @return num the first element of nums[]
     */
    function getNum0() public view returns(uint num) {
        return nums[0];
    }

    /**
     * @notice Function to return an array
     *
     * @dev Shows how to return all values as a list
     *
     * @return nums all values in nums[]
     */
    function getNums() public view returns(uint[3] memory) {
        return nums;
    }

    /**
     * @notice Function to return multiple values
     *
     * @dev Shows how to return multiple values and types
     * set with storeTypes()
     *
     * @return testBool_ testBool value
     * @return testEnum_ testEnum value
     * @return testUint_ testUint value
     * @return testInt_ testInt value
     * @return testAddr_ testAddress value
     * @return testStr_ testStr value
     * @return testArray_ testArray value
     */
    function getTypes()
        public
        view
        returns(
            bool testBool_,
            Size testEnum_,
            uint testUint_,
            int testInt_,
            address testAddr_,
            string memory testStr_,
            uint[3] memory testArray_
        )
    {
        return (
            testBool,
            testEnum,
            testUint,
            testInt,
            testAddr,
            testStr,
            testArray
        );
    }

    /**
     * @notice Function with require() that fails
     *
     * @dev Require always passes badk a message.
     */
    function requireFailsFunction() public pure {
        uint256 test_value = 100;
        require(test_value == 1, "Function require failed");
    }

    /**
     * @notice Allows current owner to assign a new owner
     *
     * @dev Emits OwnerSet().
     *
     * @param _newOwner address of the account to be the new owner
     */
    function setOwner(address _newOwner)
        public
        isOwner
    {
        owner = _newOwner;
        emit OwnerSet(
            block.timestamp,
            _newOwner
        );
    }

    /**
     * @notice Stores various byte values
     *
     * @dev Emits BytesStored()
     *
     * @param _testBytes4 value to store in testBytes4
     * @param _testBytes32 value to store in testBytes32
     * @param _testBytes value to store in testBytes
     */
    function storeBytes(
            bytes4 _testBytes4,
            bytes32 _testBytes32,
            bytes memory _testBytes
        )
        public
    {
        testBytes4 = _testBytes4;
        testBytes32 = _testBytes32;
        testBytes = _testBytes;
        emit BytesStored(
            block.timestamp,
            testBytes4,
            testBytes32,
            testBytes
        );
    }

    /**
     * @notice Stores one of the nums[]
     *
     * @dev Emits NumStored(). Used to test for out of bounds
     * errors by giving bad value to `_index`.
     *
     * @param _index selects which nums[]
     * @param _num value to store in nums[`index`]
     */
    function storeNum(uint _index, uint _num)
        public
    {
        nums[_index] = _num;
        emit NumStored(
            block.timestamp,
            _index,
            nums[_index]
        );
    }

    /**
     * @notice Stores the three args in nums[]
     *
     * @dev Emits NumsStored()
     *
     * @param _num0 value to store in nums[0]
     * @param _num1 value to store in nums[1]
     * @param _num2 value to store in nums[2]
     */
    function storeNums(uint _num0, uint _num1, uint _num2)
        public
    {
        nums[0] = _num0;
        nums[1] = _num1;
        nums[2] = _num2;
        emit NumsStored(
            block.timestamp,
            nums[0],
            nums[1],
            nums[2]
        );
    }

    /**
     * @notice Stores the three args in nums[] and call
     * sumNums() to divide nums
     *
     * @dev There is no test to check for a _divisor of 0.
     * Zero is used in testing for this transaction to call
     * another transaction that fails.
     *
     * @param _num0 value to store in nums[0]
     * @param _num1 value to store in nums[1]
     * @param _num2 value to store in nums[2]
     * @param _divisor pass to divideNums() to divide
     * the three nums
     */
    function storeNumsAndDivide(
            uint _num0,
            uint _num1,
            uint _num2,
            uint _divisor
        )
        public
    {
        nums[0] = _num0;
        nums[1] = _num1;
        nums[2] = _num2;
        emit NumsStored(
            block.timestamp,
            nums[0],
            nums[1],
            nums[2]
        );
        divideNums(_divisor);
        emit NumsStoredAndDivided(block.timestamp);
    }

    /**
     * @notice Stores the three args in nums[] and accepts a payment.
     *
     * @dev Emits NumsStored()
     *
     * @param _num0 value to store in nums[0]
     * @param _num1 value to store in nums[1]
     * @param _num2 value to store in nums[2]
     */
    function storeNumsAndPay(uint _num0, uint _num1, uint _num2)
        public
        payable
    {
        nums[0] = _num0;
        nums[1] = _num1;
        nums[2] = _num2;
        emit NumsStoredAndPaid(
            block.timestamp,
            nums[0],
            nums[1],
            nums[2],
            msg.value,
            address(this).balance
        );
    }

    /**
     * @notice Stores the three args in nums[] and call
     * sumNums() to sum the nums
     *
     * @dev Emits NumsStored() and NumsStoredAndSummed()
     *
     * @param _num0 value to store in nums[0]
     * @param _num1 value to store in nums[1]
     * @param _num2 value to store in nums[2]
     */
    function storeNumsAndSum(uint _num0, uint _num1, uint _num2)
        public
    {
        nums[0] = _num0;
        nums[1] = _num1;
        nums[2] = _num2;
        emit NumsStored(
            block.timestamp,
            nums[0],
            nums[1],
            nums[2]
        );
        sumNums();
        emit NumsStoredAndSummed(block.timestamp);
    }

    /**
     * @notice Stores the three args in nums[] but does
     * not emit an event.
     *
     * @dev Same as NumsStored() but this transaction
     * does not emit NumsStored()
     *
     * @param _num0 value to store in nums[0]
     * @param _num1 value to store in nums[1]
     * @param _num2 value to store in nums[2]
     */
    function storeNumsWithNoEvent(
        uint _num0,
        uint _num1,
        uint _num2
    )
        public
    {
        nums[0] = _num0;
        nums[1] = _num1;
        nums[2] = _num2;
    }

    /**
     * @notice Stores the three args in nums[] and emits
     * three different events.
     *
     * @dev Same as NumsStored() but this transaction
     * emits Num0Stored(), Num1Stored(), Num2Stored()
     * instead of NumsStored().
     *
     * @param _num0 value to store in nums[0]
     * @param _num1 value to store in nums[1]
     * @param _num2 value to store in nums[2]
     */
    function storeNumsWithThreeEvents(
        uint _num0,
        uint _num1,
        uint _num2
    )
        public
    {
        nums[0] = _num0;
        nums[1] = _num1;
        nums[2] = _num2;
        emit Num0Stored(block.timestamp, nums[0]);
        emit Num1Stored(block.timestamp, nums[1]);
        emit Num2Stored(block.timestamp, nums[2]);
    }

    /**
     * @notice Stores a variety of data types into public state
     * variables
     *
     * @dev Emits TypesStored()
     *
     * @param _bool boolean to store in testBool
     * @param _enum enumerated Size to store in testEnum
     * @param _uint unsigned integer to store in testUint
     * @param _int signed integer to store into testUnt
     * @param _addr address to store into testAddr
     * @param _str string to store into testStr
     * @param _array array of three unsigned integers to store in testArray
     */
    function storeTypes(
        bool _bool,
        Size _enum,
        uint _uint,
        int _int,
        address _addr,
        string memory _str,
        uint[3] calldata _array
    )
        public
    {
        testBool = _bool;
        testEnum = _enum;
        testUint = _uint;
        testInt = _int;
        testAddr = _addr;
        testStr = _str;
        testArray = _array;
        emit TypesStored(
            block.timestamp,
            testBool,
            testEnum,
            testUint,
            testInt,
            testAddr,
            testStr,
            testArray
        );
    }

    /**
     * @notice Sums values in nums[] and stores in numsTotal
     *
     * @dev Emits NumsSummed()
     */
    function sumNums()
        public
    {
        numsTotal = nums[0] + nums[1] + nums[2];
        emit NumsSummed(
            block.timestamp,
            nums[0],
            nums[1],
            nums[2],
            numsTotal
        );
    }

    /**
     * @notice Sums values in nums[0] and nums[1] and stores in
     * numsTotal. Required to be owner to call
     *
     * @dev Emits TwoNumsSummed()
     */
    function sumTwoNums()
        public
    {
        require(msg.sender == owner, "must be owner to sum two nums");
        numsTotal = nums[0] + nums[1];
        emit TwoNumsSummed(
            block.timestamp,
            nums[0],
            nums[1],
            numsTotal
        );
    }

    /**
     * @notice Allows test of assert().
     *
     * @dev Always asserts. No event emitted.
     */
    function throwAssert()
        public
        pure
    {
        assert(false);
    }

    /**
     * @notice Allows test of revert() with no parameter for a message
     *
     * @dev Always reverts. No event emitted.
     */
    function throwRevert()
        public
        pure
    {
        revert();
    }

    /**
     * @notice Allows test of revert() with a description message
     *
     * @dev Always reverts. No event emitted.
     *
     * @param _message_str Passed back as the revert description.
     */
    function throwRevertWithMessage(string memory _message_str)
        public
        pure
    {
        revert(_message_str);
    }

    /**
     * @notice Fallback function to make contract payable
     *
     * @dev Adds value sent to contract balance
     */
    receive() external payable {
        emit Received(
            block.timestamp,
            msg.sender,
            msg.value
        );
    }
}