mirror of
				https://github.com/ethereum/solidity
				synced 2023-10-03 13:03:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			170 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			170 lines
		
	
	
		
			5.8 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/>.
 | 
						|
 */
 | 
						|
/**
 | 
						|
 * @date 2017
 | 
						|
 * Unit tests for the metadata output.
 | 
						|
 */
 | 
						|
 | 
						|
#include <test/Metadata.h>
 | 
						|
#include <test/Options.h>
 | 
						|
#include <libsolidity/interface/CompilerStack.h>
 | 
						|
#include <libdevcore/SwarmHash.h>
 | 
						|
#include <libdevcore/JSON.h>
 | 
						|
 | 
						|
namespace dev
 | 
						|
{
 | 
						|
namespace solidity
 | 
						|
{
 | 
						|
namespace test
 | 
						|
{
 | 
						|
 | 
						|
BOOST_AUTO_TEST_SUITE(Metadata)
 | 
						|
 | 
						|
BOOST_AUTO_TEST_CASE(metadata_stamp)
 | 
						|
{
 | 
						|
	// Check that the metadata stamp is at the end of the runtime bytecode.
 | 
						|
	char const* sourceCode = R"(
 | 
						|
		pragma solidity >=0.0;
 | 
						|
		pragma experimental __testOnlyAnalysis;
 | 
						|
		contract test {
 | 
						|
			function g(function(uint) external returns (uint) x) {}
 | 
						|
		}
 | 
						|
	)";
 | 
						|
	CompilerStack compilerStack;
 | 
						|
	compilerStack.addSource("", std::string(sourceCode));
 | 
						|
	compilerStack.setEVMVersion(dev::test::Options::get().evmVersion());
 | 
						|
	compilerStack.setOptimiserSettings(dev::test::Options::get().optimize);
 | 
						|
	BOOST_REQUIRE_MESSAGE(compilerStack.compile(), "Compiling contract failed");
 | 
						|
	bytes const& bytecode = compilerStack.runtimeObject("test").bytecode;
 | 
						|
	std::string const& metadata = compilerStack.metadata("test");
 | 
						|
	BOOST_CHECK(dev::test::isValidMetadata(metadata));
 | 
						|
	bytes hash = dev::swarmHash(metadata).asBytes();
 | 
						|
	BOOST_REQUIRE(hash.size() == 32);
 | 
						|
	BOOST_REQUIRE(bytecode.size() >= 2);
 | 
						|
	size_t metadataCBORSize = (size_t(bytecode.end()[-2]) << 8) + size_t(bytecode.end()[-1]);
 | 
						|
	BOOST_REQUIRE(metadataCBORSize < bytecode.size() - 2);
 | 
						|
	bytes expectation = bytes{0xa1, 0x65, 'b', 'z', 'z', 'r', '0', 0x58, 0x20} + hash;
 | 
						|
	BOOST_CHECK(std::equal(expectation.begin(), expectation.end(), bytecode.end() - metadataCBORSize - 2));
 | 
						|
}
 | 
						|
 | 
						|
BOOST_AUTO_TEST_CASE(metadata_stamp_experimental)
 | 
						|
{
 | 
						|
	// Check that the metadata stamp is at the end of the runtime bytecode.
 | 
						|
	char const* sourceCode = R"(
 | 
						|
		pragma solidity >=0.0;
 | 
						|
		pragma experimental __test;
 | 
						|
		contract test {
 | 
						|
			function g(function(uint) external returns (uint) x) {}
 | 
						|
		}
 | 
						|
	)";
 | 
						|
	CompilerStack compilerStack;
 | 
						|
	compilerStack.addSource("", std::string(sourceCode));
 | 
						|
	compilerStack.setEVMVersion(dev::test::Options::get().evmVersion());
 | 
						|
	compilerStack.setOptimiserSettings(dev::test::Options::get().optimize);
 | 
						|
	BOOST_REQUIRE_MESSAGE(compilerStack.compile(), "Compiling contract failed");
 | 
						|
	bytes const& bytecode = compilerStack.runtimeObject("test").bytecode;
 | 
						|
	std::string const& metadata = compilerStack.metadata("test");
 | 
						|
	BOOST_CHECK(dev::test::isValidMetadata(metadata));
 | 
						|
	bytes hash = dev::swarmHash(metadata).asBytes();
 | 
						|
	BOOST_REQUIRE(hash.size() == 32);
 | 
						|
	BOOST_REQUIRE(bytecode.size() >= 2);
 | 
						|
	size_t metadataCBORSize = (size_t(bytecode.end()[-2]) << 8) + size_t(bytecode.end()[-1]);
 | 
						|
	BOOST_REQUIRE(metadataCBORSize < bytecode.size() - 2);
 | 
						|
	bytes expectation =
 | 
						|
		bytes{0xa2, 0x65, 'b', 'z', 'z', 'r', '0', 0x58, 0x20} +
 | 
						|
		hash +
 | 
						|
		bytes{0x6c, 'e', 'x', 'p', 'e', 'r', 'i', 'm', 'e', 'n', 't', 'a', 'l', 0xf5};
 | 
						|
	BOOST_CHECK(std::equal(expectation.begin(), expectation.end(), bytecode.end() - metadataCBORSize - 2));
 | 
						|
}
 | 
						|
 | 
						|
BOOST_AUTO_TEST_CASE(metadata_relevant_sources)
 | 
						|
{
 | 
						|
	CompilerStack compilerStack;
 | 
						|
	char const* sourceCode = R"(
 | 
						|
		pragma solidity >=0.0;
 | 
						|
		contract A {
 | 
						|
			function g(function(uint) external returns (uint) x) {}
 | 
						|
		}
 | 
						|
	)";
 | 
						|
	compilerStack.addSource("A", std::string(sourceCode));
 | 
						|
	sourceCode = R"(
 | 
						|
		pragma solidity >=0.0;
 | 
						|
		contract B {
 | 
						|
			function g(function(uint) external returns (uint) x) {}
 | 
						|
		}
 | 
						|
	)";
 | 
						|
	compilerStack.addSource("B", std::string(sourceCode));
 | 
						|
	compilerStack.setEVMVersion(dev::test::Options::get().evmVersion());
 | 
						|
	compilerStack.setOptimiserSettings(dev::test::Options::get().optimize);
 | 
						|
	BOOST_REQUIRE_MESSAGE(compilerStack.compile(), "Compiling contract failed");
 | 
						|
 | 
						|
	std::string const& serialisedMetadata = compilerStack.metadata("A");
 | 
						|
	BOOST_CHECK(dev::test::isValidMetadata(serialisedMetadata));
 | 
						|
	Json::Value metadata;
 | 
						|
	BOOST_REQUIRE(jsonParseStrict(serialisedMetadata, metadata));
 | 
						|
 | 
						|
	BOOST_CHECK_EQUAL(metadata["sources"].size(), 1);
 | 
						|
	BOOST_CHECK(metadata["sources"].isMember("A"));
 | 
						|
}
 | 
						|
 | 
						|
BOOST_AUTO_TEST_CASE(metadata_relevant_sources_imports)
 | 
						|
{
 | 
						|
	CompilerStack compilerStack;
 | 
						|
	char const* sourceCode = R"(
 | 
						|
		pragma solidity >=0.0;
 | 
						|
		contract A {
 | 
						|
			function g(function(uint) external returns (uint) x) {}
 | 
						|
		}
 | 
						|
	)";
 | 
						|
	compilerStack.addSource("A", std::string(sourceCode));
 | 
						|
	sourceCode = R"(
 | 
						|
		pragma solidity >=0.0;
 | 
						|
		import "./A";
 | 
						|
		contract B is A {
 | 
						|
			function g(function(uint) external returns (uint) x) {}
 | 
						|
		}
 | 
						|
	)";
 | 
						|
	compilerStack.addSource("B", std::string(sourceCode));
 | 
						|
	sourceCode = R"(
 | 
						|
		pragma solidity >=0.0;
 | 
						|
		import "./B";
 | 
						|
		contract C is B {
 | 
						|
			function g(function(uint) external returns (uint) x) {}
 | 
						|
		}
 | 
						|
	)";
 | 
						|
	compilerStack.addSource("C", std::string(sourceCode));
 | 
						|
	compilerStack.setEVMVersion(dev::test::Options::get().evmVersion());
 | 
						|
	compilerStack.setOptimiserSettings(dev::test::Options::get().optimize);
 | 
						|
	BOOST_REQUIRE_MESSAGE(compilerStack.compile(), "Compiling contract failed");
 | 
						|
 | 
						|
	std::string const& serialisedMetadata = compilerStack.metadata("C");
 | 
						|
	BOOST_CHECK(dev::test::isValidMetadata(serialisedMetadata));
 | 
						|
	Json::Value metadata;
 | 
						|
	BOOST_REQUIRE(jsonParseStrict(serialisedMetadata, metadata));
 | 
						|
 | 
						|
	BOOST_CHECK_EQUAL(metadata["sources"].size(), 3);
 | 
						|
	BOOST_CHECK(metadata["sources"].isMember("A"));
 | 
						|
	BOOST_CHECK(metadata["sources"].isMember("B"));
 | 
						|
	BOOST_CHECK(metadata["sources"].isMember("C"));
 | 
						|
}
 | 
						|
 | 
						|
BOOST_AUTO_TEST_SUITE_END()
 | 
						|
 | 
						|
}
 | 
						|
}
 | 
						|
}
 |