mirror of
				https://github.com/ethereum/solidity
				synced 2023-10-03 13:03:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			380 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			380 lines
		
	
	
		
			12 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/>.
 | |
| 
 | |
| 	The Implementation originally from https://msdn.microsoft.com/en-us/library/windows/desktop/aa365592(v=vs.85).aspx
 | |
| */
 | |
| /// @file RPCSession.cpp
 | |
| /// Low-level IPC communication between the test framework and the Ethereum node.
 | |
| 
 | |
| #include <test/RPCSession.h>
 | |
| 
 | |
| #include <test/Options.h>
 | |
| 
 | |
| #include <liblangutil/EVMVersion.h>
 | |
| 
 | |
| #include <libdevcore/CommonData.h>
 | |
| 
 | |
| #include <libdevcore/JSON.h>
 | |
| 
 | |
| #include <string>
 | |
| #include <stdio.h>
 | |
| #include <thread>
 | |
| #include <chrono>
 | |
| 
 | |
| using namespace std;
 | |
| using namespace dev;
 | |
| 
 | |
| IPCSocket::IPCSocket(string const& _path): m_path(_path)
 | |
| {
 | |
| #if defined(_WIN32)
 | |
| 	m_socket = CreateFile(
 | |
| 		m_path.c_str(),   // pipe name
 | |
| 		GENERIC_READ |  // read and write access
 | |
| 		GENERIC_WRITE,
 | |
| 		0,              // no sharing
 | |
| 		NULL,           // default security attribute
 | |
| 		OPEN_EXISTING,  // opens existing pipe
 | |
| 		0,              // default attributes
 | |
| 		NULL);          // no template file
 | |
| 
 | |
| 	if (m_socket == INVALID_HANDLE_VALUE)
 | |
| 		BOOST_FAIL("Error creating IPC socket object!");
 | |
| 
 | |
| #else
 | |
| 	if (_path.length() >= sizeof(sockaddr_un::sun_path))
 | |
| 		BOOST_FAIL("Error opening IPC: socket path is too long!");
 | |
| 
 | |
| 	struct sockaddr_un saun;
 | |
| 	memset(&saun, 0, sizeof(sockaddr_un));
 | |
| 	saun.sun_family = AF_UNIX;
 | |
| 	strcpy(saun.sun_path, _path.c_str());
 | |
| 
 | |
| // http://idletechnology.blogspot.ca/2011/12/unix-domain-sockets-on-osx.html
 | |
| //
 | |
| // SUN_LEN() might be optimal, but it seemingly affects the portability,
 | |
| // with at least Android missing this macro.  Just using the sizeof() for
 | |
| // structure seemingly works, and would only have the side-effect of
 | |
| // sending larger-than-required packets over the socket.  Given that this
 | |
| // code is only used for unit-tests, that approach seems simpler.
 | |
| #if defined(__APPLE__)
 | |
| 	saun.sun_len = sizeof(struct sockaddr_un);
 | |
| #endif //  defined(__APPLE__)
 | |
| 
 | |
| 	if ((m_socket = socket(AF_UNIX, SOCK_STREAM, 0)) < 0)
 | |
| 		BOOST_FAIL("Error creating IPC socket object");
 | |
| 
 | |
| 	if (connect(m_socket, reinterpret_cast<struct sockaddr const*>(&saun), sizeof(struct sockaddr_un)) < 0)
 | |
| 	{
 | |
| 		close(m_socket);
 | |
| 		BOOST_FAIL("Error connecting to IPC socket: " << _path);
 | |
| 	}
 | |
| #endif
 | |
| }
 | |
| 
 | |
| string IPCSocket::sendRequest(string const& _req)
 | |
| {
 | |
| #if defined(_WIN32)
 | |
| 	// Write to the pipe.
 | |
| 	DWORD cbWritten;
 | |
| 	BOOL fSuccess = WriteFile(
 | |
| 		m_socket,               // pipe handle
 | |
| 		_req.c_str(),           // message
 | |
| 		_req.size(),            // message length
 | |
| 		&cbWritten,             // bytes written
 | |
| 		NULL);                  // not overlapped
 | |
| 
 | |
| 	if (!fSuccess || (_req.size() != cbWritten))
 | |
| 		BOOST_FAIL("WriteFile to pipe failed");
 | |
| 
 | |
| 	// Read from the pipe.
 | |
| 	DWORD cbRead;
 | |
| 	fSuccess = ReadFile(
 | |
| 		m_socket,          // pipe handle
 | |
| 		m_readBuf,         // buffer to receive reply
 | |
| 		sizeof(m_readBuf), // size of buffer
 | |
| 		&cbRead,           // number of bytes read
 | |
| 		NULL);             // not overlapped
 | |
| 
 | |
| 	if (!fSuccess)
 | |
| 		BOOST_FAIL("ReadFile from pipe failed");
 | |
| 
 | |
| 	return string(m_readBuf, m_readBuf + cbRead);
 | |
| #else
 | |
| 	if (send(m_socket, _req.c_str(), _req.length(), 0) != (ssize_t)_req.length())
 | |
| 		BOOST_FAIL("Writing on IPC failed.");
 | |
| 
 | |
| 	auto start = chrono::steady_clock::now();
 | |
| 	ssize_t ret;
 | |
| 	do
 | |
| 	{
 | |
| 		ret = recv(m_socket, m_readBuf, sizeof(m_readBuf), 0);
 | |
| 		// Also consider closed socket an error.
 | |
| 		if (ret < 0)
 | |
| 			BOOST_FAIL("Reading on IPC failed.");
 | |
| 	}
 | |
| 	while (
 | |
| 		ret == 0 &&
 | |
| 		chrono::duration_cast<chrono::milliseconds>(chrono::steady_clock::now() - start).count() < m_readTimeOutMS
 | |
| 	);
 | |
| 
 | |
| 	if (ret == 0)
 | |
| 		BOOST_FAIL("Timeout reading on IPC.");
 | |
| 
 | |
| 	return string(m_readBuf, m_readBuf + ret);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| RPCSession& RPCSession::instance(const string& _path)
 | |
| {
 | |
| 	static RPCSession session(_path);
 | |
| 	BOOST_REQUIRE_EQUAL(session.m_ipcSocket.path(), _path);
 | |
| 	return session;
 | |
| }
 | |
| 
 | |
| string RPCSession::eth_getCode(string const& _address, string const& _blockNumber)
 | |
| {
 | |
| 	return rpcCall("eth_getCode", { quote(_address), quote(_blockNumber) }).asString();
 | |
| }
 | |
| 
 | |
| Json::Value RPCSession::eth_getBlockByNumber(string const& _blockNumber, bool _fullObjects)
 | |
| {
 | |
| 	// NOTE: to_string() converts bool to 0 or 1
 | |
| 	return rpcCall("eth_getBlockByNumber", { quote(_blockNumber), _fullObjects ? "true" : "false" });
 | |
| }
 | |
| 
 | |
| RPCSession::TransactionReceipt RPCSession::eth_getTransactionReceipt(string const& _transactionHash)
 | |
| {
 | |
| 	TransactionReceipt receipt;
 | |
| 	Json::Value const result = rpcCall("eth_getTransactionReceipt", { quote(_transactionHash) });
 | |
| 	BOOST_REQUIRE(!result.isNull());
 | |
| 	receipt.gasUsed = result["gasUsed"].asString();
 | |
| 	receipt.contractAddress = result["contractAddress"].asString();
 | |
| 	receipt.blockNumber = result["blockNumber"].asString();
 | |
| 	if (m_receiptHasStatusField)
 | |
| 	{
 | |
| 		BOOST_REQUIRE(!result["status"].isNull());
 | |
| 		receipt.status = result["status"].asString();
 | |
| 	}
 | |
| 	for (auto const& log: result["logs"])
 | |
| 	{
 | |
| 		LogEntry entry;
 | |
| 		entry.address = log["address"].asString();
 | |
| 		entry.data = log["data"].asString();
 | |
| 		for (auto const& topic: log["topics"])
 | |
| 			entry.topics.push_back(topic.asString());
 | |
| 		receipt.logEntries.push_back(entry);
 | |
| 	}
 | |
| 	return receipt;
 | |
| }
 | |
| 
 | |
| string RPCSession::eth_sendTransaction(TransactionData const& _td)
 | |
| {
 | |
| 	return rpcCall("eth_sendTransaction", { _td.toJson() }).asString();
 | |
| }
 | |
| 
 | |
| string RPCSession::eth_call(TransactionData const& _td, string const& _blockNumber)
 | |
| {
 | |
| 	return rpcCall("eth_call", { _td.toJson(), quote(_blockNumber) }).asString();
 | |
| }
 | |
| 
 | |
| string RPCSession::eth_sendTransaction(string const& _transaction)
 | |
| {
 | |
| 	return rpcCall("eth_sendTransaction", { _transaction }).asString();
 | |
| }
 | |
| 
 | |
| string RPCSession::eth_getBalance(string const& _address, string const& _blockNumber)
 | |
| {
 | |
| 	string address = (_address.length() == 20) ? "0x" + _address : _address;
 | |
| 	return rpcCall("eth_getBalance", { quote(address), quote(_blockNumber) }).asString();
 | |
| }
 | |
| 
 | |
| string RPCSession::eth_getStorageRoot(string const& _address, string const& _blockNumber)
 | |
| {
 | |
| 	string address = (_address.length() == 20) ? "0x" + _address : _address;
 | |
| 	return rpcCall("eth_getStorageRoot", { quote(address), quote(_blockNumber) }).asString();
 | |
| }
 | |
| 
 | |
| string RPCSession::eth_gasPrice()
 | |
| {
 | |
| 	return rpcCall("eth_gasPrice").asString();
 | |
| }
 | |
| 
 | |
| void RPCSession::personal_unlockAccount(string const& _address, string const& _password, int _duration)
 | |
| {
 | |
| 	BOOST_REQUIRE_MESSAGE(
 | |
| 		rpcCall("personal_unlockAccount", { quote(_address), quote(_password), to_string(_duration) }),
 | |
| 		"Error unlocking account " + _address
 | |
| 	);
 | |
| }
 | |
| 
 | |
| string RPCSession::personal_newAccount(string const& _password)
 | |
| {
 | |
| 	string addr = rpcCall("personal_newAccount", { quote(_password) }).asString();
 | |
| 	BOOST_TEST_MESSAGE("Created account " + addr);
 | |
| 	return addr;
 | |
| }
 | |
| 
 | |
| void RPCSession::test_setChainParams(vector<string> const& _accounts)
 | |
| {
 | |
| 	string forks;
 | |
| 	if (test::Options::get().evmVersion() >= solidity::EVMVersion::tangerineWhistle())
 | |
| 		forks += "\"EIP150ForkBlock\": \"0x00\",\n";
 | |
| 	if (test::Options::get().evmVersion() >= solidity::EVMVersion::spuriousDragon())
 | |
| 		forks += "\"EIP158ForkBlock\": \"0x00\",\n";
 | |
| 	if (test::Options::get().evmVersion() >= solidity::EVMVersion::byzantium())
 | |
| 	{
 | |
| 		forks += "\"byzantiumForkBlock\": \"0x00\",\n";
 | |
| 		m_receiptHasStatusField = true;
 | |
| 	}
 | |
| 	if (test::Options::get().evmVersion() >= solidity::EVMVersion::constantinople())
 | |
| 		forks += "\"constantinopleForkBlock\": \"0x00\",\n";
 | |
| 	static string const c_configString = R"(
 | |
| 	{
 | |
| 		"sealEngine": "NoProof",
 | |
| 		"params": {
 | |
| 			"accountStartNonce": "0x00",
 | |
| 			"maximumExtraDataSize": "0x1000000",
 | |
| 			"blockReward": "0x",
 | |
| 			"allowFutureBlocks": true,
 | |
| 			)" + forks + R"(
 | |
| 			"homesteadForkBlock": "0x00"
 | |
| 		},
 | |
| 		"genesis": {
 | |
| 			"author": "0000000000000010000000000000000000000000",
 | |
| 			"timestamp": "0x00",
 | |
| 			"parentHash": "0x0000000000000000000000000000000000000000000000000000000000000000",
 | |
| 			"extraData": "0x",
 | |
| 			"gasLimit": "0x1000000000000",
 | |
| 			"mixHash": "0x0000000000000000000000000000000000000000000000000000000000000000",
 | |
| 			"nonce": "0x0000000000000042",
 | |
| 			"difficulty": "131072"
 | |
|         },
 | |
| 		"accounts": {
 | |
| 			"0000000000000000000000000000000000000001": { "wei": "1", "precompiled": { "name": "ecrecover", "linear": { "base": 3000, "word": 0 } } },
 | |
| 			"0000000000000000000000000000000000000002": { "wei": "1", "precompiled": { "name": "sha256", "linear": { "base": 60, "word": 12 } } },
 | |
| 			"0000000000000000000000000000000000000003": { "wei": "1", "precompiled": { "name": "ripemd160", "linear": { "base": 600, "word": 120 } } },
 | |
| 			"0000000000000000000000000000000000000004": { "wei": "1", "precompiled": { "name": "identity", "linear": { "base": 15, "word": 3 } } },
 | |
| 			"0000000000000000000000000000000000000005": { "wei": "1", "precompiled": { "name": "modexp" } },
 | |
| 			"0000000000000000000000000000000000000006": { "wei": "1", "precompiled": { "name": "alt_bn128_G1_add", "linear": { "base": 500, "word": 0 } } },
 | |
| 			"0000000000000000000000000000000000000007": { "wei": "1", "precompiled": { "name": "alt_bn128_G1_mul", "linear": { "base": 40000, "word": 0 } } },
 | |
| 			"0000000000000000000000000000000000000008": { "wei": "1", "precompiled": { "name": "alt_bn128_pairing_product" } }
 | |
| 		}
 | |
| 	}
 | |
| 	)";
 | |
| 
 | |
| 	Json::Value config;
 | |
| 	BOOST_REQUIRE(jsonParseStrict(c_configString, config));
 | |
| 	for (auto const& account: _accounts)
 | |
| 		config["accounts"][account]["wei"] = "0x100000000000000000000000000000000000000000";
 | |
| 	test_setChainParams(jsonCompactPrint(config));
 | |
| }
 | |
| 
 | |
| void RPCSession::test_setChainParams(string const& _config)
 | |
| {
 | |
| 	BOOST_REQUIRE(rpcCall("test_setChainParams", { _config }) == true);
 | |
| }
 | |
| 
 | |
| void RPCSession::test_rewindToBlock(size_t _blockNr)
 | |
| {
 | |
| 	BOOST_REQUIRE(rpcCall("test_rewindToBlock", { to_string(_blockNr) }) == true);
 | |
| }
 | |
| 
 | |
| void RPCSession::test_mineBlocks(int _number)
 | |
| {
 | |
| 	u256 startBlock = fromBigEndian<u256>(fromHex(rpcCall("eth_blockNumber").asString()));
 | |
| 	BOOST_REQUIRE(rpcCall("test_mineBlocks", { to_string(_number) }, true) == true);
 | |
| 	BOOST_REQUIRE(fromBigEndian<u256>(fromHex(rpcCall("eth_blockNumber").asString())) == startBlock + _number);
 | |
| }
 | |
| 
 | |
| void RPCSession::test_modifyTimestamp(size_t _timestamp)
 | |
| {
 | |
| 	BOOST_REQUIRE(rpcCall("test_modifyTimestamp", { to_string(_timestamp) }) == true);
 | |
| }
 | |
| 
 | |
| Json::Value RPCSession::rpcCall(string const& _methodName, vector<string> const& _args, bool _canFail)
 | |
| {
 | |
| 	string request = "{\"jsonrpc\":\"2.0\",\"method\":\"" + _methodName + "\",\"params\":[";
 | |
| 	for (size_t i = 0; i < _args.size(); ++i)
 | |
| 	{
 | |
| 		request += _args[i];
 | |
| 		if (i + 1 != _args.size())
 | |
| 			request += ", ";
 | |
| 	}
 | |
| 
 | |
| 	request += "],\"id\":" + to_string(m_rpcSequence) + "}";
 | |
| 	++m_rpcSequence;
 | |
| 
 | |
| 	BOOST_TEST_MESSAGE("Request: " + request);
 | |
| 	string reply = m_ipcSocket.sendRequest(request);
 | |
| 	BOOST_TEST_MESSAGE("Reply: " + reply);
 | |
| 
 | |
| 	Json::Value result;
 | |
| 	string errorMsg;
 | |
| 	if (!jsonParseStrict(reply, result, &errorMsg))
 | |
| 		BOOST_FAIL("Failed to parse JSON-RPC response: " + errorMsg);
 | |
| 
 | |
| 	if (!result.isMember("id") || !result["id"].isUInt())
 | |
| 		BOOST_FAIL("Badly formatted JSON-RPC response (missing or non-integer \"id\")");
 | |
| 	if (result["id"].asUInt() != (m_rpcSequence - 1))
 | |
| 		BOOST_FAIL(
 | |
| 			"Response identifier mismatch. "
 | |
| 			"Expected " + to_string(m_rpcSequence - 1) + " but got " + to_string(result["id"].asUInt()) + "."
 | |
| 		);
 | |
| 
 | |
| 	if (result.isMember("error"))
 | |
| 	{
 | |
| 		if (_canFail)
 | |
| 			return Json::Value();
 | |
| 
 | |
| 		BOOST_FAIL("Error on JSON-RPC call: " + result["error"]["message"].asString());
 | |
| 	}
 | |
| 	return result["result"];
 | |
| }
 | |
| 
 | |
| string const& RPCSession::accountCreate()
 | |
| {
 | |
| 	m_accounts.push_back(personal_newAccount(""));
 | |
| 	personal_unlockAccount(m_accounts.back(), "", 100000);
 | |
| 	return m_accounts.back();
 | |
| }
 | |
| 
 | |
| string const& RPCSession::accountCreateIfNotExists(size_t _id)
 | |
| {
 | |
| 	while ((_id + 1) > m_accounts.size())
 | |
| 		accountCreate();
 | |
| 	return m_accounts[_id];
 | |
| }
 | |
| 
 | |
| RPCSession::RPCSession(const string& _path):
 | |
| 	m_ipcSocket(_path)
 | |
| {
 | |
| 	accountCreate();
 | |
| 	// This will pre-fund the accounts create prior.
 | |
| 	test_setChainParams(m_accounts);
 | |
| }
 | |
| 
 | |
| string RPCSession::TransactionData::toJson() const
 | |
| {
 | |
| 	Json::Value json;
 | |
| 	json["from"] = (from.length() == 20) ? "0x" + from : from;
 | |
| 	json["to"] = (to.length() == 20 || to == "") ? "0x" + to :  to;
 | |
| 	json["gas"] = gas;
 | |
| 	json["gasprice"] = gasPrice;
 | |
| 	json["value"] = value;
 | |
| 	json["data"] = data;
 | |
| 	return jsonCompactPrint(json);
 | |
| }
 |