/* 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 . */ /** * @date 2017 * Unit tests for the metadata output. */ #include #include #include #include #include #include #include #include using namespace std; namespace solidity::frontend::test { namespace { map requireParsedCBORMetadata(bytes const& _bytecode) { bytes cborMetadata = solidity::test::onlyMetadata(_bytecode); BOOST_REQUIRE(!cborMetadata.empty()); std::optional> tmp = solidity::test::parseCBORMetadata(cborMetadata); BOOST_REQUIRE(tmp); return *tmp; } optional compileAndCheckLicenseMetadata(string const& _contractName, char const* _sourceCode) { CompilerStack compilerStack; compilerStack.setSources({{"A.sol", std::string(_sourceCode)}}); BOOST_REQUIRE_MESSAGE(compilerStack.compile(), "Compiling contract failed"); std::string const& serialisedMetadata = compilerStack.metadata(_contractName); BOOST_CHECK(solidity::test::isValidMetadata(serialisedMetadata)); Json::Value metadata; BOOST_REQUIRE(util::jsonParseStrict(serialisedMetadata, metadata)); BOOST_CHECK_EQUAL(metadata["sources"].size(), 1); BOOST_REQUIRE(metadata["sources"].isMember("A.sol")); if (metadata["sources"]["A.sol"].isMember("license")) { BOOST_REQUIRE(metadata["sources"]["A.sol"]["license"].isString()); return metadata["sources"]["A.sol"]["license"].asString(); } else return nullopt; } } 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) public {} } )"; for (auto release: std::set{true, VersionIsRelease}) for (auto metadataHash: set{ CompilerStack::MetadataHash::IPFS, CompilerStack::MetadataHash::Bzzr1, CompilerStack::MetadataHash::None }) { CompilerStack compilerStack; compilerStack.overwriteReleaseFlag(release); compilerStack.setSources({{"", std::string(sourceCode)}}); compilerStack.setEVMVersion(solidity::test::CommonOptions::get().evmVersion()); compilerStack.setOptimiserSettings(solidity::test::CommonOptions::get().optimize); compilerStack.setMetadataHash(metadataHash); BOOST_REQUIRE_MESSAGE(compilerStack.compile(), "Compiling contract failed"); bytes const& bytecode = compilerStack.runtimeObject("test").bytecode; std::string const& metadata = compilerStack.metadata("test"); BOOST_CHECK(solidity::test::isValidMetadata(metadata)); auto const cborMetadata = requireParsedCBORMetadata(bytecode); if (metadataHash == CompilerStack::MetadataHash::None) BOOST_CHECK(cborMetadata.size() == 1); else { bytes hash; string hashMethod; if (metadataHash == CompilerStack::MetadataHash::IPFS) { hash = util::ipfsHash(metadata); BOOST_REQUIRE(hash.size() == 34); hashMethod = "ipfs"; } else { hash = util::bzzr1Hash(metadata).asBytes(); BOOST_REQUIRE(hash.size() == 32); hashMethod = "bzzr1"; } BOOST_CHECK(cborMetadata.size() == 2); BOOST_CHECK(cborMetadata.count(hashMethod) == 1); BOOST_CHECK(cborMetadata.at(hashMethod) == util::toHex(hash)); } BOOST_CHECK(cborMetadata.count("solc") == 1); if (release) BOOST_CHECK(cborMetadata.at("solc") == util::toHex(VersionCompactBytes)); else BOOST_CHECK(cborMetadata.at("solc") == VersionStringStrict); } } 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) public {} } )"; for (auto release: set{true, VersionIsRelease}) for (auto metadataHash: set{ CompilerStack::MetadataHash::IPFS, CompilerStack::MetadataHash::Bzzr1, CompilerStack::MetadataHash::None }) { CompilerStack compilerStack; compilerStack.overwriteReleaseFlag(release); compilerStack.setSources({{"", std::string(sourceCode)}}); compilerStack.setEVMVersion(solidity::test::CommonOptions::get().evmVersion()); compilerStack.setOptimiserSettings(solidity::test::CommonOptions::get().optimize); compilerStack.setMetadataHash(metadataHash); BOOST_REQUIRE_MESSAGE(compilerStack.compile(), "Compiling contract failed"); bytes const& bytecode = compilerStack.runtimeObject("test").bytecode; std::string const& metadata = compilerStack.metadata("test"); BOOST_CHECK(solidity::test::isValidMetadata(metadata)); auto const cborMetadata = requireParsedCBORMetadata(bytecode); if (metadataHash == CompilerStack::MetadataHash::None) BOOST_CHECK(cborMetadata.size() == 2); else { bytes hash; string hashMethod; if (metadataHash == CompilerStack::MetadataHash::IPFS) { hash = util::ipfsHash(metadata); BOOST_REQUIRE(hash.size() == 34); hashMethod = "ipfs"; } else { hash = util::bzzr1Hash(metadata).asBytes(); BOOST_REQUIRE(hash.size() == 32); hashMethod = "bzzr1"; } BOOST_CHECK(cborMetadata.size() == 3); BOOST_CHECK(cborMetadata.count(hashMethod) == 1); BOOST_CHECK(cborMetadata.at(hashMethod) == util::toHex(hash)); } BOOST_CHECK(cborMetadata.count("solc") == 1); if (release) BOOST_CHECK(cborMetadata.at("solc") == util::toHex(VersionCompactBytes)); else BOOST_CHECK(cborMetadata.at("solc") == VersionStringStrict); BOOST_CHECK(cborMetadata.count("experimental") == 1); BOOST_CHECK(cborMetadata.at("experimental") == "true"); } } BOOST_AUTO_TEST_CASE(metadata_relevant_sources) { CompilerStack compilerStack; char const* sourceCodeA = R"( pragma solidity >=0.0; contract A { function g(function(uint) external returns (uint) x) public {} } )"; char const* sourceCodeB = R"( pragma solidity >=0.0; contract B { function g(function(uint) external returns (uint) x) public {} } )"; compilerStack.setSources({ {"A", std::string(sourceCodeA)}, {"B", std::string(sourceCodeB)}, }); compilerStack.setEVMVersion(solidity::test::CommonOptions::get().evmVersion()); compilerStack.setOptimiserSettings(solidity::test::CommonOptions::get().optimize); BOOST_REQUIRE_MESSAGE(compilerStack.compile(), "Compiling contract failed"); std::string const& serialisedMetadata = compilerStack.metadata("A"); BOOST_CHECK(solidity::test::isValidMetadata(serialisedMetadata)); Json::Value metadata; BOOST_REQUIRE(util::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* sourceCodeA = R"( pragma solidity >=0.0; contract A { function g(function(uint) external returns (uint) x) public virtual {} } )"; char const* sourceCodeB = R"( pragma solidity >=0.0; import "./A"; contract B is A { function g(function(uint) external returns (uint) x) public virtual override {} } )"; char const* sourceCodeC = R"( pragma solidity >=0.0; import "./B"; contract C is B { function g(function(uint) external returns (uint) x) public override {} } )"; compilerStack.setSources({ {"A", std::string(sourceCodeA)}, {"B", std::string(sourceCodeB)}, {"C", std::string(sourceCodeC)} }); compilerStack.setEVMVersion(solidity::test::CommonOptions::get().evmVersion()); compilerStack.setOptimiserSettings(solidity::test::CommonOptions::get().optimize); BOOST_REQUIRE_MESSAGE(compilerStack.compile(), "Compiling contract failed"); std::string const& serialisedMetadata = compilerStack.metadata("C"); BOOST_CHECK(solidity::test::isValidMetadata(serialisedMetadata)); Json::Value metadata; BOOST_REQUIRE(util::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_CASE(metadata_useLiteralContent) { // Check that the metadata contains "useLiteralContent" char const* sourceCode = R"( pragma solidity >=0.0; contract test { } )"; auto check = [](char const* _src, bool _literal) { CompilerStack compilerStack; compilerStack.setSources({{"", std::string(_src)}}); compilerStack.setEVMVersion(solidity::test::CommonOptions::get().evmVersion()); compilerStack.setOptimiserSettings(solidity::test::CommonOptions::get().optimize); compilerStack.useMetadataLiteralSources(_literal); BOOST_REQUIRE_MESSAGE(compilerStack.compile(), "Compiling contract failed"); string metadata_str = compilerStack.metadata("test"); Json::Value metadata; util::jsonParseStrict(metadata_str, metadata); BOOST_CHECK(solidity::test::isValidMetadata(metadata_str)); BOOST_CHECK(metadata.isMember("settings")); BOOST_CHECK(metadata["settings"].isMember("metadata")); BOOST_CHECK(metadata["settings"]["metadata"].isMember("bytecodeHash")); if (_literal) { BOOST_CHECK(metadata["settings"]["metadata"].isMember("useLiteralContent")); BOOST_CHECK(metadata["settings"]["metadata"]["useLiteralContent"].asBool()); } }; check(sourceCode, true); check(sourceCode, false); } BOOST_AUTO_TEST_CASE(metadata_viair) { char const* sourceCode = R"( pragma solidity >=0.0; contract test { } )"; auto check = [](char const* _src, bool _viair) { CompilerStack compilerStack; compilerStack.setSources({{"", std::string(_src)}}); compilerStack.setEVMVersion(solidity::test::CommonOptions::get().evmVersion()); compilerStack.setOptimiserSettings(solidity::test::CommonOptions::get().optimize); compilerStack.setViaIR(_viair); BOOST_REQUIRE_MESSAGE(compilerStack.compile(), "Compiling contract failed"); string metadata_str = compilerStack.metadata("test"); Json::Value metadata; util::jsonParseStrict(metadata_str, metadata); BOOST_CHECK(solidity::test::isValidMetadata(metadata_str)); BOOST_CHECK(metadata.isMember("settings")); if (_viair) { BOOST_CHECK(metadata["settings"].isMember("viaIR")); BOOST_CHECK(metadata["settings"]["viaIR"].asBool()); } }; check(sourceCode, true); check(sourceCode, false); } BOOST_AUTO_TEST_CASE(metadata_revert_strings) { CompilerStack compilerStack; char const* sourceCodeA = R"( pragma solidity >=0.0; contract A { } )"; compilerStack.setSources({{"A", std::string(sourceCodeA)}}); compilerStack.setRevertStringBehaviour(RevertStrings::Strip); BOOST_REQUIRE_MESSAGE(compilerStack.compile(), "Compiling contract failed"); std::string const& serialisedMetadata = compilerStack.metadata("A"); BOOST_CHECK(solidity::test::isValidMetadata(serialisedMetadata)); Json::Value metadata; BOOST_REQUIRE(util::jsonParseStrict(serialisedMetadata, metadata)); BOOST_CHECK_EQUAL(metadata["settings"]["debug"]["revertStrings"], "strip"); } BOOST_AUTO_TEST_CASE(metadata_license_missing) { char const* sourceCode = R"( pragma solidity >=0.0; contract C { } )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == nullopt); } BOOST_AUTO_TEST_CASE(metadata_license_gpl3) { // Can't use a raw string here due to the stylechecker. char const* sourceCode = "// NOTE: we also add trailing whitespace after the license, to see it is trimmed.\n" "// SPDX-License-Identifier: GPL-3.0 \n" "pragma solidity >=0.0;\n" "contract C {\n" "}\n"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == "GPL-3.0"); } BOOST_AUTO_TEST_CASE(metadata_license_whitespace_before_spdx) { char const* sourceCode = R"( // SPDX-License-Identifier: GPL-3.0 contract C {} )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == "GPL-3.0"); } BOOST_AUTO_TEST_CASE(metadata_license_whitespace_after_colon) { char const* sourceCode = R"( // SPDX-License-Identifier: GPL-3.0 contract C {} )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == "GPL-3.0"); } BOOST_AUTO_TEST_CASE(metadata_license_gpl3_or_apache2) { char const* sourceCode = R"( // SPDX-License-Identifier: GPL-3.0 OR Apache-2.0 pragma solidity >=0.0; contract C { } )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == "GPL-3.0 OR Apache-2.0"); } BOOST_AUTO_TEST_CASE(metadata_license_ignored_unicode) { char const* sourceCode = R"( // SPDX-License-Identifier: ⡉⡊⡋⡌⡍⡎⡏⡐⡑⡒ pragma solidity >=0.0; contract C { } )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == nullopt); } BOOST_AUTO_TEST_CASE(metadata_license_ignored_stray_unicode) { char const* sourceCode = R"( // SPDX-License-Identifier: GPL-3.0 ⡉⡊⡋⡌⡍⡎⡏⡐⡑⡒ pragma solidity >=0.0; contract C { } )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == "GPL-3.0"); } BOOST_AUTO_TEST_CASE(metadata_license_bidi_marks) { char const* sourceCode = "// \xE2\x80\xAE""0.3-LPG :reifitnedI-esneciL-XDPS\xE2\x80\xAC\n" "// NOTE: The text above is reversed using Unicode directional marks. In raw form it would look like this:\n" "// 0.3-LPG :reifitnedI-esneciL-XDPS\n" "contract C {}\n"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == nullopt); } BOOST_AUTO_TEST_CASE(metadata_license_bottom) { char const* sourceCode = R"( contract C {} // SPDX-License-Identifier: GPL-3.0 )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == "GPL-3.0"); } BOOST_AUTO_TEST_CASE(metadata_cr_endings) { char const* sourceCode = "// SPDX-License-Identifier: GPL-3.0\r" "contract C {}\r"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == "GPL-3.0"); } BOOST_AUTO_TEST_CASE(metadata_crlf_endings) { char const* sourceCode = "// SPDX-License-Identifier: GPL-3.0\r\n" "contract C {}\r\n"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == "GPL-3.0"); } BOOST_AUTO_TEST_CASE(metadata_license_in_string) { char const* sourceCode = R"( contract C { bytes license = "// SPDX-License-Identifier: GPL-3.0"; } )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == nullopt); } BOOST_AUTO_TEST_CASE(metadata_license_in_contract) { char const* sourceCode = R"( contract C { // SPDX-License-Identifier: GPL-3.0 } )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == nullopt); } BOOST_AUTO_TEST_CASE(metadata_license_missing_colon) { char const* sourceCode = R"( // SPDX-License-Identifier GPL-3.0 contract C {} )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == nullopt); } BOOST_AUTO_TEST_CASE(metadata_license_multiline) { char const* sourceCode = R"( /* SPDX-License-Identifier: GPL-3.0 */ contract C {} )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == "GPL-3.0"); } BOOST_AUTO_TEST_CASE(metadata_license_natspec) { char const* sourceCode = R"( /// SPDX-License-Identifier: GPL-3.0 contract C {} )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == "GPL-3.0"); } BOOST_AUTO_TEST_CASE(metadata_license_natspec_multiline) { char const* sourceCode = R"( /** SPDX-License-Identifier: GPL-3.0 */ contract C {} )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == "GPL-3.0"); } BOOST_AUTO_TEST_CASE(metadata_license_no_whitespace) { char const* sourceCode = R"( //SPDX-License-Identifier:GPL-3.0 contract C {} )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == "GPL-3.0"); } BOOST_AUTO_TEST_CASE(metadata_license_no_whitespace_multiline) { char const* sourceCode = R"( /*SPDX-License-Identifier:GPL-3.0*/ contract C {} )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == "GPL-3.0"); } BOOST_AUTO_TEST_CASE(metadata_license_nonempty_line) { char const* sourceCode = R"( pragma solidity >= 0.0; // SPDX-License-Identifier: GPL-3.0 contract C {} )"; BOOST_CHECK(compileAndCheckLicenseMetadata("C", sourceCode) == "GPL-3.0"); } BOOST_AUTO_TEST_SUITE_END() }