mirror of
				https://github.com/ethereum/solidity
				synced 2023-10-03 13:03:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			708 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			708 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
| 	This file is part of solidity.
 | |
| 
 | |
| 	solidity is free software: you can redistribute it and/or modify
 | |
| 	it under the terms of the GNU General Public License as published by
 | |
| 	the Free Software Foundation, either version 3 of the License, or
 | |
| 	(at your option) any later version.
 | |
| 
 | |
| 	solidity is distributed in the hope that it will be useful,
 | |
| 	but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| 	MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
| 	GNU General Public License for more details.
 | |
| 
 | |
| 	You should have received a copy of the GNU General Public License
 | |
| 	along with solidity.  If not, see <http://www.gnu.org/licenses/>.
 | |
| */
 | |
| // SPDX-License-Identifier: GPL-3.0
 | |
| /**
 | |
|  * @author Christian <c@ethdev.com>
 | |
|  * @date 2015
 | |
|  * Tests for a (comparatively) complex multisig wallet contract.
 | |
|  */
 | |
| 
 | |
| #include <libsolutil/LazyInit.h>
 | |
| 
 | |
| #include <string>
 | |
| #include <tuple>
 | |
| #include <optional>
 | |
| 
 | |
| #if defined(_MSC_VER)
 | |
| #pragma warning(push)
 | |
| #pragma warning(disable:4535) // calling _set_se_translator requires /EHa
 | |
| #endif
 | |
| #include <boost/test/unit_test.hpp>
 | |
| #if defined(_MSC_VER)
 | |
| #pragma warning(pop)
 | |
| #endif
 | |
| 
 | |
| #include <test/libsolidity/SolidityExecutionFramework.h>
 | |
| 
 | |
| using namespace std;
 | |
| using namespace solidity::test;
 | |
| using namespace solidity::util;
 | |
| 
 | |
| namespace solidity::frontend::test
 | |
| {
 | |
| static char const* walletCode = R"DELIMITER(
 | |
| //sol Wallet
 | |
| // Multi-sig, daily-limited account proxy/wallet.
 | |
| // @authors:
 | |
| // Gav Wood <g@ethdev.com>
 | |
| // inheritable "property" contract that enables methods to be protected by requiring the acquiescence of either a
 | |
| // single, or, crucially, each of a number of, designated owners.
 | |
| // usage:
 | |
| // use modifiers onlyowner (just own owned) or onlymanyowners(hash), whereby the same hash must be provided by
 | |
| // some number (specified in constructor) of the set of owners (specified in the constructor, modifiable) before the
 | |
| // interior is executed.
 | |
| 
 | |
| pragma solidity >=0.4.0 <0.9.0;
 | |
| 
 | |
| contract multiowned {
 | |
| 
 | |
| 	// TYPES
 | |
| 
 | |
| 	// struct for the status of a pending operation.
 | |
| 	struct PendingState {
 | |
| 		uint yetNeeded;
 | |
| 		uint ownersDone;
 | |
| 		uint index;
 | |
| 	}
 | |
| 
 | |
| 	// EVENTS
 | |
| 
 | |
| 	// this contract only has five types of events: it can accept a confirmation, in which case
 | |
| 	// we record owner and operation (hash) alongside it.
 | |
| 	event Confirmation(address owner, bytes32 operation);
 | |
| 	event Revoke(address owner, bytes32 operation);
 | |
| 	// some others are in the case of an owner changing.
 | |
| 	event OwnerChanged(address oldOwner, address newOwner);
 | |
| 	event OwnerAdded(address newOwner);
 | |
| 	event OwnerRemoved(address oldOwner);
 | |
| 	// the last one is emitted if the required signatures change
 | |
| 	event RequirementChanged(uint newRequirement);
 | |
| 
 | |
| 	// MODIFIERS
 | |
| 
 | |
| 	// simple single-sig function modifier.
 | |
| 	modifier onlyowner {
 | |
| 		if (isOwner(msg.sender))
 | |
| 			_;
 | |
| 	}
 | |
| 	// multi-sig function modifier: the operation must have an intrinsic hash in order
 | |
| 	// that later attempts can be realised as the same underlying operation and
 | |
| 	// thus count as confirmations.
 | |
| 	modifier onlymanyowners(bytes32 _operation) {
 | |
| 		if (confirmAndCheck(_operation))
 | |
| 			_;
 | |
| 	}
 | |
| 
 | |
| 	// METHODS
 | |
| 
 | |
| 	// constructor is given number of sigs required to do protected "onlymanyowners" transactions
 | |
| 	// as well as the selection of addresses capable of confirming them.
 | |
| 	constructor(address[] memory _owners, uint _required) {
 | |
| 		m_numOwners = _owners.length + 1;
 | |
| 		m_owners[1] = uint160(msg.sender);
 | |
| 		m_ownerIndex[uint160(msg.sender)] = 1;
 | |
| 		for (uint i = 0; i < _owners.length; ++i)
 | |
| 		{
 | |
| 			m_owners[2 + i] = uint160(_owners[i]);
 | |
| 			m_ownerIndex[uint160(_owners[i])] = 2 + i;
 | |
| 		}
 | |
| 		m_required = _required;
 | |
| 	}
 | |
| 
 | |
| 	// Revokes a prior confirmation of the given operation
 | |
| 	function revoke(bytes32 _operation) external {
 | |
| 		uint ownerIndex = m_ownerIndex[uint160(msg.sender)];
 | |
| 		// make sure they're an owner
 | |
| 		if (ownerIndex == 0) return;
 | |
| 		uint ownerIndexBit = 2**ownerIndex;
 | |
| 		PendingState storage pending = m_pending[_operation];
 | |
| 		if (pending.ownersDone & ownerIndexBit > 0) {
 | |
| 			pending.yetNeeded++;
 | |
| 			pending.ownersDone -= ownerIndexBit;
 | |
| 			emit Revoke(msg.sender, _operation);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Replaces an owner `_from` with another `_to`.
 | |
| 	function changeOwner(address _from, address _to) onlymanyowners(keccak256(msg.data)) public virtual {
 | |
| 		if (isOwner(_to)) return;
 | |
| 		uint ownerIndex = m_ownerIndex[uint160(_from)];
 | |
| 		if (ownerIndex == 0) return;
 | |
| 
 | |
| 		clearPending();
 | |
| 		m_owners[ownerIndex] = uint160(_to);
 | |
| 		m_ownerIndex[uint160(_from)] = 0;
 | |
| 		m_ownerIndex[uint160(_to)] = ownerIndex;
 | |
| 		emit OwnerChanged(_from, _to);
 | |
| 	}
 | |
| 
 | |
| 	function addOwner(address _owner) onlymanyowners(keccak256(msg.data)) external {
 | |
| 		if (isOwner(_owner)) return;
 | |
| 
 | |
| 		clearPending();
 | |
| 		if (m_numOwners >= c_maxOwners)
 | |
| 			reorganizeOwners();
 | |
| 		if (m_numOwners >= c_maxOwners)
 | |
| 			return;
 | |
| 		m_numOwners++;
 | |
| 		m_owners[m_numOwners] = uint160(_owner);
 | |
| 		m_ownerIndex[uint160(_owner)] = m_numOwners;
 | |
| 		emit OwnerAdded(_owner);
 | |
| 	}
 | |
| 
 | |
| 	function removeOwner(address _owner) onlymanyowners(keccak256(msg.data)) external {
 | |
| 		uint ownerIndex = m_ownerIndex[uint160(_owner)];
 | |
| 		if (ownerIndex == 0) return;
 | |
| 		if (m_required > m_numOwners - 1) return;
 | |
| 
 | |
| 		m_owners[ownerIndex] = 0;
 | |
| 		m_ownerIndex[uint160(_owner)] = 0;
 | |
| 		clearPending();
 | |
| 		reorganizeOwners(); //make sure m_numOwner is equal to the number of owners and always points to the optimal free slot
 | |
| 		emit OwnerRemoved(_owner);
 | |
| 	}
 | |
| 
 | |
| 	function changeRequirement(uint _newRequired) onlymanyowners(keccak256(msg.data)) external {
 | |
| 		if (_newRequired > m_numOwners) return;
 | |
| 		m_required = _newRequired;
 | |
| 		clearPending();
 | |
| 		emit RequirementChanged(_newRequired);
 | |
| 	}
 | |
| 
 | |
| 	function isOwner(address _addr) public returns (bool) {
 | |
| 		return m_ownerIndex[uint160(_addr)] > 0;
 | |
| 	}
 | |
| 
 | |
| 	function hasConfirmed(bytes32 _operation, address _owner) public view returns (bool) {
 | |
| 		PendingState storage pending = m_pending[_operation];
 | |
| 		uint ownerIndex = m_ownerIndex[uint160(_owner)];
 | |
| 
 | |
| 		// make sure they're an owner
 | |
| 		if (ownerIndex == 0) return false;
 | |
| 
 | |
| 		// determine the bit to set for this owner.
 | |
| 		uint ownerIndexBit = 2**ownerIndex;
 | |
| 		if (pending.ownersDone & ownerIndexBit == 0) {
 | |
| 			return false;
 | |
| 		} else {
 | |
| 			return true;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// INTERNAL METHODS
 | |
| 
 | |
| 	function confirmAndCheck(bytes32 _operation) internal returns (bool) {
 | |
| 		// determine what index the present sender is:
 | |
| 		uint ownerIndex = m_ownerIndex[uint160(msg.sender)];
 | |
| 		// make sure they're an owner
 | |
| 		if (ownerIndex == 0) return false;
 | |
| 
 | |
| 		PendingState storage pending = m_pending[_operation];
 | |
| 		// if we're not yet working on this operation, switch over and reset the confirmation status.
 | |
| 		if (pending.yetNeeded == 0) {
 | |
| 			// reset count of confirmations needed.
 | |
| 			pending.yetNeeded = m_required;
 | |
| 			// reset which owners have confirmed (none) - set our bitmap to 0.
 | |
| 			pending.ownersDone = 0;
 | |
| 			m_pendingIndex.push(_operation);
 | |
| 			pending.index = m_pendingIndex.length - 1;
 | |
| 		}
 | |
| 		// determine the bit to set for this owner.
 | |
| 		uint ownerIndexBit = 2**ownerIndex;
 | |
| 		// make sure we (the message sender) haven't confirmed this operation previously.
 | |
| 		if (pending.ownersDone & ownerIndexBit == 0) {
 | |
| 			emit Confirmation(msg.sender, _operation);
 | |
| 			// ok - check if count is enough to go ahead.
 | |
| 			if (pending.yetNeeded <= 1) {
 | |
| 				// enough confirmations: reset and run interior.
 | |
| 				delete m_pendingIndex[m_pending[_operation].index];
 | |
| 				delete m_pending[_operation];
 | |
| 				return true;
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				// not enough: record that this owner in particular confirmed.
 | |
| 				pending.yetNeeded--;
 | |
| 				pending.ownersDone |= ownerIndexBit;
 | |
| 				return false;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	function reorganizeOwners() private returns (bool) {
 | |
| 		uint free = 1;
 | |
| 		while (free < m_numOwners)
 | |
| 		{
 | |
| 			while (free < m_numOwners && m_owners[free] != 0) free++;
 | |
| 			while (m_numOwners > 1 && m_owners[m_numOwners] == 0) m_numOwners--;
 | |
| 			if (free < m_numOwners && m_owners[m_numOwners] != 0 && m_owners[free] == 0)
 | |
| 			{
 | |
| 				m_owners[free] = m_owners[m_numOwners];
 | |
| 				m_ownerIndex[m_owners[free]] = free;
 | |
| 				m_owners[m_numOwners] = 0;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	function clearPending() internal virtual {
 | |
| 		uint length = m_pendingIndex.length;
 | |
| 		for (uint i = 0; i < length; ++i)
 | |
| 			if (m_pendingIndex[i] != 0)
 | |
| 				delete m_pending[m_pendingIndex[i]];
 | |
| 		delete m_pendingIndex;
 | |
| 	}
 | |
| 
 | |
| 	// FIELDS
 | |
| 
 | |
| 	// the number of owners that must confirm the same operation before it is run.
 | |
| 	uint public m_required;
 | |
| 	// pointer used to find a free slot in m_owners
 | |
| 	uint public m_numOwners;
 | |
| 
 | |
| 	// list of owners
 | |
| 	uint[256] m_owners;
 | |
| 	uint constant c_maxOwners = 250;
 | |
| 	// index on the list of owners to allow reverse lookup
 | |
| 	mapping(uint => uint) m_ownerIndex;
 | |
| 	// the ongoing operations.
 | |
| 	mapping(bytes32 => PendingState) m_pending;
 | |
| 	bytes32[] m_pendingIndex;
 | |
| }
 | |
| 
 | |
| // inheritable "property" contract that enables methods to be protected by placing a linear limit (specifiable)
 | |
| // on a particular resource per calendar day. is multiowned to allow the limit to be altered. resource that method
 | |
| // uses is specified in the modifier.
 | |
| abstract contract daylimit is multiowned {
 | |
| 
 | |
| 	// MODIFIERS
 | |
| 
 | |
| 	// simple modifier for daily limit.
 | |
| 	modifier limitedDaily(uint _value) {
 | |
| 		if (underLimit(_value))
 | |
| 			_;
 | |
| 	}
 | |
| 
 | |
| 	// METHODS
 | |
| 
 | |
| 	// constructor - stores initial daily limit and records the present day's index.
 | |
| 	constructor(uint _limit) {
 | |
| 		m_dailyLimit = _limit;
 | |
| 		m_lastDay = today();
 | |
| 	}
 | |
| 	// (re)sets the daily limit. needs many of the owners to confirm. doesn't alter the amount already spent today.
 | |
| 	function setDailyLimit(uint _newLimit) onlymanyowners(keccak256(msg.data)) external {
 | |
| 		m_dailyLimit = _newLimit;
 | |
| 	}
 | |
| 	// (re)sets the daily limit. needs many of the owners to confirm. doesn't alter the amount already spent today.
 | |
| 	function resetSpentToday() onlymanyowners(keccak256(msg.data)) external {
 | |
| 		m_spentToday = 0;
 | |
| 	}
 | |
| 
 | |
| 	// INTERNAL METHODS
 | |
| 
 | |
| 	// checks to see if there is at least `_value` left from the daily limit today. if there is, subtracts it and
 | |
| 	// returns true. otherwise just returns false.
 | |
| 	function underLimit(uint _value) internal onlyowner returns (bool) {
 | |
| 		// reset the spend limit if we're on a different day to last time.
 | |
| 		if (today() > m_lastDay) {
 | |
| 			m_spentToday = 0;
 | |
| 			m_lastDay = today();
 | |
| 		}
 | |
| 		// check to see if there's enough left - if so, subtract and return true.
 | |
| 		if (m_spentToday + _value >= m_spentToday && m_spentToday + _value <= m_dailyLimit) {
 | |
| 			m_spentToday += _value;
 | |
| 			return true;
 | |
| 		}
 | |
| 		return false;
 | |
| 	}
 | |
| 	// determines today's index.
 | |
| 	function today() private view returns (uint) { return block.timestamp / 1 days; }
 | |
| 
 | |
| 	// FIELDS
 | |
| 
 | |
| 	uint public m_dailyLimit;
 | |
| 	uint m_spentToday;
 | |
| 	uint m_lastDay;
 | |
| }
 | |
| 
 | |
| // interface contract for multisig proxy contracts; see below for docs.
 | |
| abstract contract multisig {
 | |
| 
 | |
| 	// EVENTS
 | |
| 
 | |
| 	// logged events:
 | |
| 	// Funds has arrived into the wallet (record how much).
 | |
| 	event Deposit(address from, uint value);
 | |
| 	// Single transaction going out of the wallet (record who signed for it, how much, and to whom it's going).
 | |
| 	event SingleTransact(address owner, uint value, address to, bytes data);
 | |
| 	// Multi-sig transaction going out of the wallet (record who signed for it last, the operation hash, how much, and to whom it's going).
 | |
| 	event MultiTransact(address owner, bytes32 operation, uint value, address to, bytes data);
 | |
| 	// Confirmation still needed for a transaction.
 | |
| 	event ConfirmationNeeded(bytes32 operation, address initiator, uint value, address to, bytes data);
 | |
| 
 | |
| 	// FUNCTIONS
 | |
| 
 | |
| 	// TODO: document
 | |
| 	function changeOwner(address _from, address _to) public virtual;
 | |
| 	function execute(address _to, uint _value, bytes calldata _data) external virtual returns (bytes32);
 | |
| 	function confirm(bytes32 _h) public virtual returns (bool);
 | |
| }
 | |
| 
 | |
| // usage:
 | |
| // bytes32 h = Wallet(w).from(oneOwner).transact(to, value, data);
 | |
| // Wallet(w).from(anotherOwner).confirm(h);
 | |
| contract Wallet is multisig, multiowned, daylimit {
 | |
| 
 | |
| 	// TYPES
 | |
| 
 | |
| 	// Transaction structure to remember details of transaction lest it need be saved for a later call.
 | |
| 	struct Transaction {
 | |
| 		address to;
 | |
| 		uint value;
 | |
| 		bytes data;
 | |
| 	}
 | |
| 
 | |
| 	// METHODS
 | |
| 
 | |
| 	// constructor - just pass on the owner array to the multiowned and
 | |
| 	// the limit to daylimit
 | |
| 	constructor(address[] memory _owners, uint _required, uint _daylimit) payable
 | |
| 			multiowned(_owners, _required) daylimit(_daylimit) {
 | |
| 	}
 | |
| 
 | |
| 	function changeOwner(address _from, address _to) public override(multiowned, multisig) {
 | |
| 		multiowned.changeOwner(_from, _to);
 | |
| 	}
 | |
| 	// destroys the contract sending everything to `_to`.
 | |
| 	function shutdown(address payable _to) onlymanyowners(keccak256(msg.data)) external {
 | |
| 		selfdestruct(_to);
 | |
| 	}
 | |
| 
 | |
| 	// gets called for plain ether transfers
 | |
| 	receive() external payable {
 | |
| 		// did we actually receive value?
 | |
| 		if (msg.value > 0)
 | |
| 			emit Deposit(msg.sender, msg.value);
 | |
| 	}
 | |
| 
 | |
| 	// Outside-visible transact entry point. Executes transaction immediately if below daily spend limit.
 | |
| 	// If not, goes into multisig process. We provide a hash on return to allow the sender to provide
 | |
| 	// shortcuts for the other confirmations (allowing them to avoid replicating the _to, _value
 | |
| 	// and _data arguments). They still get the option of using them if they want, anyways.
 | |
| 	function execute(address _to, uint _value, bytes calldata _data) external override onlyowner returns (bytes32 _r) {
 | |
| 		// first, take the opportunity to check that we're under the daily limit.
 | |
| 		if (underLimit(_value)) {
 | |
| 			emit SingleTransact(msg.sender, _value, _to, _data);
 | |
| 			// yes - just execute the call.
 | |
| 			_to.call{value: _value}(_data);
 | |
| 			return 0;
 | |
| 		}
 | |
| 		// determine our operation hash.
 | |
| 		_r = keccak256(abi.encodePacked(msg.data, block.number));
 | |
| 		if (!confirm(_r) && m_txs[_r].to == 0x0000000000000000000000000000000000000000) {
 | |
| 			m_txs[_r].to = _to;
 | |
| 			m_txs[_r].value = _value;
 | |
| 			m_txs[_r].data = _data;
 | |
| 			emit ConfirmationNeeded(_r, msg.sender, _value, _to, _data);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// confirm a transaction through just the hash. we use the previous transactions map, m_txs, in order
 | |
| 	// to determine the body of the transaction from the hash provided.
 | |
| 	function confirm(bytes32 _h) onlymanyowners(_h) public override returns (bool) {
 | |
| 		if (m_txs[_h].to != 0x0000000000000000000000000000000000000000) {
 | |
| 			m_txs[_h].to.call{value: m_txs[_h].value}(m_txs[_h].data);
 | |
| 			emit MultiTransact(msg.sender, _h, m_txs[_h].value, m_txs[_h].to, m_txs[_h].data);
 | |
| 			delete m_txs[_h];
 | |
| 			return true;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// INTERNAL METHODS
 | |
| 
 | |
| 	function clearPending() internal override {
 | |
| 		uint length = m_pendingIndex.length;
 | |
| 		for (uint i = 0; i < length; ++i)
 | |
| 			delete m_txs[m_pendingIndex[i]];
 | |
| 		super.clearPending();
 | |
| 	}
 | |
| 
 | |
| 	// FIELDS
 | |
| 
 | |
| 	// pending transactions we have at present.
 | |
| 	mapping (bytes32 => Transaction) m_txs;
 | |
| }
 | |
| )DELIMITER";
 | |
| 
 | |
| static LazyInit<bytes> s_compiledWallet;
 | |
| 
 | |
| class WalletTestFramework: public SolidityExecutionFramework
 | |
| {
 | |
| protected:
 | |
| 	void deployWallet(
 | |
| 		u256 const& _value = 0,
 | |
| 		vector<h160> const& _owners = vector<h160>{},
 | |
| 		u256 _required = 1,
 | |
| 		u256 _dailyLimit = 0
 | |
| 	)
 | |
| 	{
 | |
| 		bytes const& compiled = s_compiledWallet.init([&]{
 | |
| 			return compileContract(walletCode, "Wallet");
 | |
| 		});
 | |
| 
 | |
| 		bytes args = encodeArgs(u256(0x60), _required, _dailyLimit, u256(_owners.size()), _owners);
 | |
| 		sendMessage(compiled + args, true, _value);
 | |
| 		BOOST_REQUIRE(m_transactionSuccessful);
 | |
| 		BOOST_REQUIRE(!m_output.empty());
 | |
| 	}
 | |
| };
 | |
| 
 | |
| /// This is a test suite that tests optimised code!
 | |
| BOOST_FIXTURE_TEST_SUITE(SolidityWallet, WalletTestFramework)
 | |
| 
 | |
| BOOST_AUTO_TEST_CASE(creation)
 | |
| {
 | |
| 	deployWallet(200);
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", m_sender) == encodeArgs(true));
 | |
| 	bool v2 = !solidity::test::CommonOptions::get().useABIEncoderV1;
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", h256(~0)) == (v2 ? encodeArgs() : encodeArgs(false)));
 | |
| }
 | |
| 
 | |
| BOOST_AUTO_TEST_CASE(add_owners)
 | |
| {
 | |
| 	deployWallet(200);
 | |
| 	h160 originalOwner = m_sender;
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", account(1)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", account(1)) == encodeArgs(true));
 | |
| 	// now let the new owner add someone
 | |
| 	sendEther(account(1), 10 * ether);
 | |
| 	m_sender = account(1);
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", h160(0x13)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", h160(0x13)) == encodeArgs(true));
 | |
| 	// and check that a non-owner cannot add a new owner
 | |
| 	m_sender = account(0);
 | |
| 	sendEther(account(2), 10 * ether);
 | |
| 	m_sender = account(2);
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", h160(0x20)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", h160(0x20)) == encodeArgs(false));
 | |
| 	// finally check that all the owners are there
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", originalOwner) == encodeArgs(true));
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", account(1)) == encodeArgs(true));
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", h160(0x13)) == encodeArgs(true));
 | |
| }
 | |
| 
 | |
| BOOST_AUTO_TEST_CASE(change_owners)
 | |
| {
 | |
| 	deployWallet(200);
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", h160(0x12)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", h160(0x12)) == encodeArgs(true));
 | |
| 	BOOST_REQUIRE(callContractFunction("changeOwner(address,address)", h160(0x12), h160(0x13)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", h160(0x12)) == encodeArgs(false));
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", h160(0x13)) == encodeArgs(true));
 | |
| }
 | |
| 
 | |
| BOOST_AUTO_TEST_CASE(remove_owner)
 | |
| {
 | |
| 	deployWallet(200);
 | |
| 	// add 10 owners
 | |
| 	for (unsigned i = 0; i < 10; ++i)
 | |
| 	{
 | |
| 		BOOST_REQUIRE(callContractFunction("addOwner(address)", h160(0x12 + i)) == encodeArgs());
 | |
| 		BOOST_REQUIRE(callContractFunction("isOwner(address)", h160(0x12 + i)) == encodeArgs(true));
 | |
| 	}
 | |
| 	// check they are there again
 | |
| 	for (unsigned i = 0; i < 10; ++i)
 | |
| 		BOOST_REQUIRE(callContractFunction("isOwner(address)", h160(0x12 + i)) == encodeArgs(true));
 | |
| 	// remove the odd owners
 | |
| 	for (unsigned i = 0; i < 10; ++i)
 | |
| 		if (i % 2 == 1)
 | |
| 			BOOST_REQUIRE(callContractFunction("removeOwner(address)", h160(0x12 + i)) == encodeArgs());
 | |
| 	// check the result
 | |
| 	for (unsigned i = 0; i < 10; ++i)
 | |
| 		BOOST_REQUIRE(callContractFunction("isOwner(address)", h160(0x12 + i)) == encodeArgs(i % 2 == 0));
 | |
| 	// add them again
 | |
| 	for (unsigned i = 0; i < 10; ++i)
 | |
| 		if (i % 2 == 1)
 | |
| 			BOOST_REQUIRE(callContractFunction("addOwner(address)", h160(0x12 + i)) == encodeArgs());
 | |
| 	// check everyone is there
 | |
| 	for (unsigned i = 0; i < 10; ++i)
 | |
| 		BOOST_REQUIRE(callContractFunction("isOwner(address)", h160(0x12 + i)) == encodeArgs(true));
 | |
| }
 | |
| 
 | |
| BOOST_AUTO_TEST_CASE(initial_owners)
 | |
| {
 | |
| 	vector<h160> owners{
 | |
| 		h160("0x42c56279432962a17176998a4747d1b4d6ed4367"),
 | |
| 		h160("0xd4d4669f5ba9f4c27d38ef02a358c339b5560c47"),
 | |
| 		h160("0xe6716f9544a56c530d868e4bfbacb172315bdead"),
 | |
| 		h160("0x775e18be7a50a0abb8a4e82b1bd697d79f31fe04"),
 | |
| 		h160("0xf4dd5c3794f1fd0cdc0327a83aa472609c806e99"),
 | |
| 		h160("0x4c9113886af165b2de069d6e99430647e94a9fff"),
 | |
| 		h160("0x3fb1cd2cd96c6d5c0b5eb3322d807b34482481d4")
 | |
| 	};
 | |
| 	deployWallet(0, owners, 4, 2);
 | |
| 	BOOST_CHECK(callContractFunction("m_numOwners()") == encodeArgs(u256(8)));
 | |
| 	BOOST_CHECK(callContractFunction("isOwner(address)", m_sender) == encodeArgs(true));
 | |
| 	for (h160 const& owner: owners)
 | |
| 	{
 | |
| 		BOOST_CHECK(callContractFunction("isOwner(address)", owner) == encodeArgs(true));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| BOOST_AUTO_TEST_CASE(multisig_value_transfer)
 | |
| {
 | |
| 	deployWallet(200);
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", account(1)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", account(2)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", account(3)) == encodeArgs());
 | |
| 	// 4 owners, set required to 3
 | |
| 	BOOST_REQUIRE(callContractFunction("changeRequirement(uint256)", u256(3)) == encodeArgs());
 | |
| 	h160 destination = h160("0x5c6d6026d3fb35cd7175fd0054ae8df50d8f8b41");
 | |
| 	BOOST_CHECK_EQUAL(balanceAt(destination), 0);
 | |
| 	m_sender = account(0);
 | |
| 	sendEther(account(1), 10 * ether);
 | |
| 	m_sender = account(1);
 | |
| 	auto ophash = callContractFunction("execute(address,uint256,bytes)", destination, 100, 0x60, 0x00);
 | |
| 	BOOST_CHECK_EQUAL(balanceAt(destination), 0);
 | |
| 	m_sender = account(0);
 | |
| 	sendEther(account(2), 10 * ether);
 | |
| 	m_sender = account(2);
 | |
| 	callContractFunction("confirm(bytes32)", ophash);
 | |
| 	BOOST_CHECK_EQUAL(balanceAt(destination), 0);
 | |
| 	m_sender = account(0);
 | |
| 	sendEther(account(3), 10 * ether);
 | |
| 	m_sender = account(3);
 | |
| 	callContractFunction("confirm(bytes32)", ophash);
 | |
| 	// now it should go through
 | |
| 	BOOST_CHECK_EQUAL(balanceAt(destination), 100);
 | |
| }
 | |
| 
 | |
| BOOST_AUTO_TEST_CASE(revoke_addOwner)
 | |
| {
 | |
| 	deployWallet();
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", account(1)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", account(2)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", account(3)) == encodeArgs());
 | |
| 	// 4 owners, set required to 3
 | |
| 	BOOST_REQUIRE(callContractFunction("changeRequirement(uint256)", u256(3)) == encodeArgs());
 | |
| 	// add a new owner
 | |
| 	h160 deployer = m_sender;
 | |
| 	h256 opHash = util::keccak256(FixedHash<4>(util::keccak256("addOwner(address)")).asBytes() + h256(0x33).asBytes());
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", h160(0x33)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", h160(0x33)) == encodeArgs(false));
 | |
| 	m_sender = account(0);
 | |
| 	sendEther(account(1), 10 * ether);
 | |
| 	m_sender = account(1);
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", h160(0x33)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", h160(0x33)) == encodeArgs(false));
 | |
| 	// revoke one confirmation
 | |
| 	m_sender = deployer;
 | |
| 	BOOST_REQUIRE(callContractFunction("revoke(bytes32)", opHash) == encodeArgs());
 | |
| 	m_sender = account(0);
 | |
| 	sendEther(account(2), 10 * ether);
 | |
| 	m_sender = account(2);
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", h160(0x33)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", h160(0x33)) == encodeArgs(false));
 | |
| 	m_sender = account(0);
 | |
| 	sendEther(account(3), 10 * ether);
 | |
| 	m_sender = account(3);
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", h160(0x33)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("isOwner(address)", h160(0x33)) == encodeArgs(true));
 | |
| }
 | |
| 
 | |
| BOOST_AUTO_TEST_CASE(revoke_transaction)
 | |
| {
 | |
| 	deployWallet(200);
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", account(1)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", account(2)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", account(3)) == encodeArgs());
 | |
| 	// 4 owners, set required to 3
 | |
| 	BOOST_REQUIRE(callContractFunction("changeRequirement(uint256)", u256(3)) == encodeArgs());
 | |
| 	// create a transaction
 | |
| 	h160 deployer = m_sender;
 | |
| 	h160 destination = h160("0x5c6d6026d3fb35cd7175fd0054ae8df50d8f8b41");
 | |
| 	BOOST_CHECK_EQUAL(balanceAt(destination), 0);
 | |
| 	m_sender = account(0);
 | |
| 	sendEther(account(1), 10 * ether);
 | |
| 	m_sender = account(1);
 | |
| 	auto opHash = callContractFunction("execute(address,uint256,bytes)", destination, 100, 0x60, 0x00);
 | |
| 	BOOST_CHECK_EQUAL(balanceAt(destination), 0);
 | |
| 	m_sender = account(0);
 | |
| 	sendEther(account(2), 10 * ether);
 | |
| 	m_sender = account(2);
 | |
| 	callContractFunction("confirm(bytes32)", opHash);
 | |
| 	BOOST_CHECK_EQUAL(balanceAt(destination), 0);
 | |
| 	m_sender = account(0);
 | |
| 	sendEther(account(1), 10 * ether);
 | |
| 	m_sender = account(1);
 | |
| 	BOOST_REQUIRE(callContractFunction("revoke(bytes32)", opHash) == encodeArgs());
 | |
| 	m_sender = deployer;
 | |
| 	callContractFunction("confirm(bytes32)", opHash);
 | |
| 	BOOST_CHECK_EQUAL(balanceAt(destination), 0);
 | |
| 	m_sender = account(0);
 | |
| 	sendEther(account(3), 10 * ether);
 | |
| 	m_sender = account(3);
 | |
| 	callContractFunction("confirm(bytes32)", opHash);
 | |
| 	// now it should go through
 | |
| 	BOOST_CHECK_EQUAL(balanceAt(destination), 100);
 | |
| }
 | |
| 
 | |
| BOOST_AUTO_TEST_CASE(daylimit)
 | |
| {
 | |
| 	deployWallet(200);
 | |
| 	BOOST_REQUIRE(callContractFunction("m_dailyLimit()") == encodeArgs(u256(0)));
 | |
| 	BOOST_REQUIRE(callContractFunction("setDailyLimit(uint256)", u256(100)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("m_dailyLimit()") == encodeArgs(u256(100)));
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", account(1)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", account(2)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("addOwner(address)", account(3)) == encodeArgs());
 | |
| 	// 4 owners, set required to 3
 | |
| 	BOOST_REQUIRE(callContractFunction("changeRequirement(uint256)", u256(3)) == encodeArgs());
 | |
| 
 | |
| 	// try to send tx over daylimit
 | |
| 	h160 destination = h160("0x5c6d6026d3fb35cd7175fd0054ae8df50d8f8b41");
 | |
| 	BOOST_CHECK_EQUAL(balanceAt(destination), 0);
 | |
| 	sendEther(account(1), 10 * ether);
 | |
| 	m_sender = account(1);
 | |
| 	BOOST_REQUIRE(
 | |
| 		callContractFunction("execute(address,uint256,bytes)", destination, 150, 0x60, 0x00) !=
 | |
| 		encodeArgs(u256(0))
 | |
| 	);
 | |
| 	BOOST_CHECK_EQUAL(balanceAt(destination), 0);
 | |
| 	// try to send tx under daylimit by stranger
 | |
| 	m_sender = account(0);
 | |
| 	sendEther(account(4), 10 * ether);
 | |
| 	m_sender = account(4);
 | |
| 	BOOST_REQUIRE(
 | |
| 		callContractFunction("execute(address,uint256,bytes)", destination, 90, 0x60, 0x00) ==
 | |
| 		encodeArgs(u256(0))
 | |
| 	);
 | |
| 	BOOST_CHECK_EQUAL(balanceAt(destination), 0);
 | |
| 	// now send below limit by owner
 | |
| 	m_sender = account(0);
 | |
| 	sendEther(account(1), 10 * ether);
 | |
| 	BOOST_REQUIRE(
 | |
| 		callContractFunction("execute(address,uint256,bytes)", destination, 90, 0x60, 0x00) ==
 | |
| 		encodeArgs(u256(0))
 | |
| 	);
 | |
| 	BOOST_CHECK_EQUAL(balanceAt(destination), 90);
 | |
| }
 | |
| 
 | |
| BOOST_AUTO_TEST_CASE(daylimit_constructor)
 | |
| {
 | |
| 	deployWallet(200, {}, 1, 20);
 | |
| 	BOOST_REQUIRE(callContractFunction("m_dailyLimit()") == encodeArgs(u256(20)));
 | |
| 	BOOST_REQUIRE(callContractFunction("setDailyLimit(uint256)", u256(30)) == encodeArgs());
 | |
| 	BOOST_REQUIRE(callContractFunction("m_dailyLimit()") == encodeArgs(u256(30)));
 | |
| }
 | |
| 
 | |
| //@todo test data calls
 | |
| 
 | |
| BOOST_AUTO_TEST_SUITE_END()
 | |
| 
 | |
| } // end namespaces
 |