mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
1376 lines
47 KiB
C++
1376 lines
47 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
|
|
|
|
/// Unit tests for solc/CommandLineInterface.h
|
|
|
|
#include <solc/CommandLineInterface.h>
|
|
#include <solc/Exceptions.h>
|
|
|
|
#include <test/solc/Common.h>
|
|
|
|
#include <test/Common.h>
|
|
#include <test/libsolidity/util/SoltestErrors.h>
|
|
#include <liblangutil/SemVerHandler.h>
|
|
#include <test/FilesystemUtils.h>
|
|
|
|
#include <libsolutil/JSON.h>
|
|
#include <libsolutil/TemporaryDirectory.h>
|
|
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
#include <range/v3/view/transform.hpp>
|
|
|
|
#include <map>
|
|
#include <ostream>
|
|
#include <set>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
using namespace std;
|
|
using namespace solidity::frontend;
|
|
using namespace solidity::test;
|
|
using namespace solidity::util;
|
|
using namespace solidity::langutil;
|
|
|
|
using PathSet = set<boost::filesystem::path>;
|
|
|
|
#define TEST_CASE_NAME (boost::unit_test::framework::current_test_case().p_name)
|
|
|
|
namespace
|
|
{
|
|
|
|
ostream& operator<<(ostream& _out, vector<ImportRemapper::Remapping> const& _remappings)
|
|
{
|
|
static auto remappingToString = [](auto const& _remapping)
|
|
{
|
|
return _remapping.context + ":" + _remapping.prefix + "=" + _remapping.target;
|
|
};
|
|
|
|
_out << "[" << joinHumanReadable(_remappings | ranges::views::transform(remappingToString)) << "]";
|
|
return _out;
|
|
}
|
|
|
|
ostream& operator<<(ostream& _out, map<string, string> const& _map)
|
|
{
|
|
_out << "{" << endl;
|
|
for (auto const& [key, value]: _map)
|
|
_out << "" << key << ": " << value << "," << endl;
|
|
_out << "}";
|
|
|
|
return _out;
|
|
}
|
|
|
|
ostream& operator<<(ostream& _out, PathSet const& _paths)
|
|
{
|
|
static auto pathString = [](auto const& _path) { return _path.string(); };
|
|
|
|
_out << "{" << joinHumanReadable(_paths | ranges::views::transform(pathString)) << "}";
|
|
return _out;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
namespace boost::test_tools::tt_detail
|
|
{
|
|
|
|
// Boost won't find the << operator unless we put it in the std namespace which is illegal.
|
|
// The recommended solution is to overload print_log_value<> struct and make it use our operator.
|
|
|
|
template<>
|
|
struct print_log_value<vector<ImportRemapper::Remapping>>
|
|
{
|
|
void operator()(std::ostream& _out, vector<ImportRemapper::Remapping> const& _value) { ::operator<<(_out, _value); }
|
|
};
|
|
|
|
template<>
|
|
struct print_log_value<map<string, string>>
|
|
{
|
|
void operator()(std::ostream& _out, map<string, string> const& _value) { ::operator<<(_out, _value); }
|
|
};
|
|
|
|
template<>
|
|
struct print_log_value<PathSet>
|
|
{
|
|
void operator()(std::ostream& _out, PathSet const& _value) { ::operator<<(_out, _value); }
|
|
};
|
|
|
|
} // namespace boost::test_tools::tt_detail
|
|
|
|
namespace solidity::frontend::test
|
|
{
|
|
|
|
BOOST_AUTO_TEST_SUITE(CommandLineInterfaceTest)
|
|
|
|
BOOST_AUTO_TEST_CASE(help)
|
|
{
|
|
OptionsReaderAndMessages result = runCLI({"solc", "--help"}, "");
|
|
|
|
BOOST_TEST(result.success);
|
|
BOOST_TEST(boost::starts_with(result.stdoutContent, "solc, the Solidity commandline compiler."));
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.options.input.mode == InputMode::Help);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(license)
|
|
{
|
|
OptionsReaderAndMessages result = runCLI({"solc", "--license"}, "");
|
|
|
|
BOOST_TEST(result.success);
|
|
BOOST_TEST(boost::starts_with(result.stdoutContent, "Most of the code is licensed under GPLv3"));
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.options.input.mode == InputMode::License);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(version)
|
|
{
|
|
OptionsReaderAndMessages result = runCLI({"solc", "--version"}, "");
|
|
|
|
BOOST_TEST(result.success);
|
|
BOOST_TEST(boost::ends_with(result.stdoutContent, "Version: " + solidity::frontend::VersionString + "\n"));
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.options.input.mode == InputMode::Version);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(multiple_input_modes)
|
|
{
|
|
array<string, 10> inputModeOptions = {
|
|
"--help",
|
|
"--license",
|
|
"--version",
|
|
"--standard-json",
|
|
"--link",
|
|
"--assemble",
|
|
"--strict-assembly",
|
|
"--yul",
|
|
"--import-ast",
|
|
"--import-asm-json",
|
|
};
|
|
string expectedMessage =
|
|
"The following options are mutually exclusive: "
|
|
"--help, --license, --version, --standard-json, --link, --assemble, --strict-assembly, --yul, --import-ast, --lsp, --import-asm-json. "
|
|
"Select at most one.";
|
|
|
|
for (string const& mode1: inputModeOptions)
|
|
for (string const& mode2: inputModeOptions)
|
|
if (mode1 != mode2)
|
|
BOOST_CHECK_EXCEPTION(
|
|
parseCommandLineAndReadInputFiles({"solc", mode1, mode2}),
|
|
CommandLineValidationError,
|
|
[&](auto const& _exception) { BOOST_TEST(_exception.what() == expectedMessage); return true; }
|
|
);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_input)
|
|
{
|
|
TemporaryDirectory tempDir1(TEST_CASE_NAME);
|
|
TemporaryDirectory tempDir2(TEST_CASE_NAME);
|
|
createFilesWithParentDirs({tempDir1.path() / "input1.sol"});
|
|
createFilesWithParentDirs({tempDir2.path() / "input2.sol"});
|
|
|
|
boost::filesystem::path expectedRootPath = FileReader::normalizeCLIRootPathForVFS(tempDir1);
|
|
boost::filesystem::path expectedDir1 = expectedRootPath / tempDir1.path().relative_path();
|
|
boost::filesystem::path expectedDir2 = expectedRootPath / tempDir2.path().relative_path();
|
|
soltestAssert(expectedDir1.is_absolute() || expectedDir1.root_path() == "/", "");
|
|
soltestAssert(expectedDir2.is_absolute() || expectedDir2.root_path() == "/", "");
|
|
|
|
vector<ImportRemapper::Remapping> expectedRemappings = {
|
|
{"", "a", "b/c/d"},
|
|
{"a", "b", "c/d/e/"},
|
|
};
|
|
map<string, string> expectedSources = {
|
|
{"<stdin>", ""},
|
|
{(expectedDir1 / "input1.sol").generic_string(), ""},
|
|
{(expectedDir2 / "input2.sol").generic_string(), ""},
|
|
};
|
|
PathSet expectedAllowedPaths = {
|
|
boost::filesystem::canonical(tempDir1),
|
|
boost::filesystem::canonical(tempDir2),
|
|
"b/c",
|
|
"c/d/e",
|
|
};
|
|
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles({
|
|
"solc",
|
|
"a=b/c/d",
|
|
(tempDir1.path() / "input1.sol").string(),
|
|
(tempDir2.path() / "input2.sol").string(),
|
|
"a:b=c/d/e/",
|
|
"-",
|
|
});
|
|
|
|
BOOST_TEST(result.success);
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.options.input.mode == InputMode::Compiler);
|
|
BOOST_TEST(result.options.input.addStdin);
|
|
BOOST_CHECK_EQUAL(result.options.input.remappings, expectedRemappings);
|
|
BOOST_CHECK_EQUAL(result.reader.sourceUnits(), expectedSources);
|
|
BOOST_CHECK_EQUAL(result.reader.allowedDirectories(), expectedAllowedPaths);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_ignore_missing_some_files_exist)
|
|
{
|
|
TemporaryDirectory tempDir1(TEST_CASE_NAME);
|
|
TemporaryDirectory tempDir2(TEST_CASE_NAME);
|
|
createFilesWithParentDirs({tempDir1.path() / "input1.sol"});
|
|
|
|
boost::filesystem::path expectedRootPath = FileReader::normalizeCLIRootPathForVFS(tempDir1);
|
|
boost::filesystem::path expectedDir1 = expectedRootPath / tempDir1.path().relative_path();
|
|
soltestAssert(expectedDir1.is_absolute() || expectedDir1.root_path() == "/", "");
|
|
|
|
// NOTE: Allowed paths should not be added for skipped files.
|
|
map<string, string> expectedSources = {{(expectedDir1 / "input1.sol").generic_string(), ""}};
|
|
PathSet expectedAllowedPaths = {boost::filesystem::canonical(tempDir1)};
|
|
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles({
|
|
"solc",
|
|
(tempDir1.path() / "input1.sol").string(),
|
|
(tempDir2.path() / "input2.sol").string(),
|
|
"--ignore-missing",
|
|
});
|
|
BOOST_TEST(result.success);
|
|
BOOST_TEST(result.stderrContent == "\"" + (tempDir2.path() / "input2.sol").string() + "\" is not found. Skipping.\n");
|
|
BOOST_TEST(result.options.input.mode == InputMode::Compiler);
|
|
BOOST_TEST(!result.options.input.addStdin);
|
|
BOOST_CHECK_EQUAL(result.reader.sourceUnits(), expectedSources);
|
|
BOOST_CHECK_EQUAL(result.reader.allowedDirectories(), expectedAllowedPaths);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_ignore_missing_no_files_exist)
|
|
{
|
|
TemporaryDirectory tempDir(TEST_CASE_NAME);
|
|
|
|
string expectedMessage =
|
|
"\"" + (tempDir.path() / "input1.sol").string() + "\" is not found. Skipping.\n"
|
|
"\"" + (tempDir.path() / "input2.sol").string() + "\" is not found. Skipping.\n"
|
|
"All specified input files either do not exist or are not regular files.\n";
|
|
|
|
OptionsReaderAndMessages result = runCLI({
|
|
"solc",
|
|
(tempDir.path() / "input1.sol").string(),
|
|
(tempDir.path() / "input2.sol").string(),
|
|
"--ignore-missing",
|
|
});
|
|
BOOST_TEST(!result.success);
|
|
BOOST_TEST(result.stderrContent == expectedMessage);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_not_a_file)
|
|
{
|
|
TemporaryDirectory tempDir(TEST_CASE_NAME);
|
|
|
|
string expectedMessage = "\"" + tempDir.path().string() + "\" is not a valid file.";
|
|
|
|
BOOST_CHECK_EXCEPTION(
|
|
parseCommandLineAndReadInputFiles({"solc", tempDir.path().string()}),
|
|
CommandLineValidationError,
|
|
[&](auto const& _exception) { BOOST_TEST(_exception.what() == expectedMessage); return true; }
|
|
);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(standard_json_base_path)
|
|
{
|
|
TemporaryDirectory tempDir(TEST_CASE_NAME);
|
|
TemporaryWorkingDirectory tempWorkDir(tempDir.path().root_path());
|
|
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles({
|
|
"solc",
|
|
"--standard-json",
|
|
"--base-path=" + tempDir.path().string(),
|
|
});
|
|
BOOST_TEST(result.success);
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.options.input.mode == InputMode::StandardJson);
|
|
BOOST_TEST(result.options.input.addStdin);
|
|
BOOST_TEST(result.options.input.paths.empty());
|
|
BOOST_TEST(result.reader.sourceUnits().empty());
|
|
BOOST_TEST(result.reader.allowedDirectories().empty());
|
|
BOOST_TEST(result.reader.basePath() == "/" / tempDir.path().relative_path());
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(standard_json_no_input_file)
|
|
{
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles({"solc", "--standard-json"});
|
|
BOOST_TEST(result.success);
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.options.input.mode == InputMode::StandardJson);
|
|
BOOST_TEST(result.options.input.addStdin);
|
|
BOOST_TEST(result.options.input.paths.empty());
|
|
BOOST_TEST(result.reader.sourceUnits().empty());
|
|
BOOST_TEST(result.reader.allowedDirectories().empty());
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(standard_json_dash)
|
|
{
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles({"solc", "--standard-json", "-"});
|
|
BOOST_TEST(result.success);
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.options.input.mode == InputMode::StandardJson);
|
|
BOOST_TEST(result.options.input.addStdin);
|
|
BOOST_TEST(result.reader.sourceUnits().empty());
|
|
BOOST_TEST(result.reader.allowedDirectories().empty());
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(standard_json_one_input_file)
|
|
{
|
|
TemporaryDirectory tempDir(TEST_CASE_NAME);
|
|
createFilesWithParentDirs({tempDir.path() / "input.json"});
|
|
|
|
vector<string> commandLine = {"solc", "--standard-json", (tempDir.path() / "input.json").string()};
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles(commandLine);
|
|
BOOST_TEST(result.success);
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.options.input.mode == InputMode::StandardJson);
|
|
BOOST_TEST(!result.options.input.addStdin);
|
|
BOOST_TEST(result.options.input.paths == PathSet{tempDir.path() / "input.json"});
|
|
BOOST_TEST(result.reader.allowedDirectories().empty());
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(standard_json_two_input_files)
|
|
{
|
|
string expectedMessage =
|
|
"Too many input files for --standard-json.\n"
|
|
"Please either specify a single file name or provide its content on standard input.";
|
|
|
|
BOOST_CHECK_EXCEPTION(
|
|
parseCommandLineAndReadInputFiles({"solc", "--standard-json", "input1.json", "input2.json"}),
|
|
CommandLineValidationError,
|
|
[&](auto const& _exception) { BOOST_TEST(_exception.what() == expectedMessage); return true; }
|
|
);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(standard_json_one_input_file_and_stdin)
|
|
{
|
|
string expectedMessage =
|
|
"Too many input files for --standard-json.\n"
|
|
"Please either specify a single file name or provide its content on standard input.";
|
|
|
|
BOOST_CHECK_EXCEPTION(
|
|
parseCommandLineAndReadInputFiles({"solc", "--standard-json", "input1.json", "-"}),
|
|
CommandLineValidationError,
|
|
[&](auto const& _exception) { BOOST_TEST(_exception.what() == expectedMessage); return true; }
|
|
);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(standard_json_ignore_missing)
|
|
{
|
|
TemporaryDirectory tempDir(TEST_CASE_NAME);
|
|
|
|
// This option is pretty much useless Standard JSON mode.
|
|
string expectedMessage =
|
|
"All specified input files either do not exist or are not regular files.";
|
|
|
|
BOOST_CHECK_EXCEPTION(
|
|
parseCommandLineAndReadInputFiles({
|
|
"solc",
|
|
"--standard-json",
|
|
(tempDir.path() / "input.json").string(),
|
|
"--ignore-missing",
|
|
}),
|
|
CommandLineValidationError,
|
|
[&](auto const& _exception) { BOOST_TEST(_exception.what() == expectedMessage); return true; }
|
|
);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(standard_json_remapping)
|
|
{
|
|
string expectedMessage =
|
|
"Import remappings are not accepted on the command line in Standard JSON mode.\n"
|
|
"Please put them under 'settings.remappings' in the JSON input.";
|
|
|
|
BOOST_CHECK_EXCEPTION(
|
|
parseCommandLineAndReadInputFiles({"solc", "--standard-json", "a=b"}),
|
|
CommandLineValidationError,
|
|
[&](auto const& _exception) { BOOST_TEST(_exception.what() == expectedMessage); return true; }
|
|
);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_paths_to_source_unit_names_no_base_path)
|
|
{
|
|
TemporaryDirectory tempDirCurrent(TEST_CASE_NAME);
|
|
TemporaryDirectory tempDirOther(TEST_CASE_NAME);
|
|
TemporaryWorkingDirectory tempWorkDir(tempDirCurrent);
|
|
soltestAssert(tempDirCurrent.path().is_absolute(), "");
|
|
soltestAssert(tempDirOther.path().is_absolute(), "");
|
|
|
|
// NOTE: On macOS the path usually contains symlinks which prevents base path from being stripped.
|
|
// Use canonical() to resolve symnlinks and get consistent results on all platforms.
|
|
boost::filesystem::path currentDirNoSymlinks = boost::filesystem::canonical(tempDirCurrent);
|
|
boost::filesystem::path otherDirNoSymlinks = boost::filesystem::canonical(tempDirOther);
|
|
|
|
boost::filesystem::path expectedOtherDir = "/" / otherDirNoSymlinks.relative_path();
|
|
soltestAssert(expectedOtherDir.is_absolute() || expectedOtherDir.root_path() == "/", "");
|
|
|
|
vector<string> commandLine = {
|
|
"solc",
|
|
"contract1.sol", // Relative path
|
|
"c/d/contract2.sol", // Relative path with subdirectories
|
|
currentDirNoSymlinks.string() + "/contract3.sol", // Absolute path inside working dir
|
|
otherDirNoSymlinks.string() + "/contract4.sol", // Absolute path outside of working dir
|
|
};
|
|
|
|
CommandLineOptions expectedOptions;
|
|
expectedOptions.input.paths = {
|
|
"contract1.sol",
|
|
"c/d/contract2.sol",
|
|
currentDirNoSymlinks / "contract3.sol",
|
|
otherDirNoSymlinks / "contract4.sol",
|
|
};
|
|
expectedOptions.modelChecker.initialize = true;
|
|
|
|
map<string, string> expectedSources = {
|
|
{"contract1.sol", ""},
|
|
{"c/d/contract2.sol", ""},
|
|
{"contract3.sol", ""},
|
|
{expectedOtherDir.generic_string() + "/contract4.sol", ""},
|
|
};
|
|
|
|
FileReader::FileSystemPathSet expectedAllowedDirectories = {
|
|
currentDirNoSymlinks / "c/d",
|
|
currentDirNoSymlinks,
|
|
otherDirNoSymlinks,
|
|
};
|
|
|
|
createFilesWithParentDirs(expectedOptions.input.paths);
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles(commandLine);
|
|
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.stdoutContent == "");
|
|
BOOST_REQUIRE(result.success);
|
|
BOOST_TEST(result.options == expectedOptions);
|
|
BOOST_TEST(result.reader.sourceUnits() == expectedSources);
|
|
BOOST_TEST(result.reader.allowedDirectories() == expectedAllowedDirectories);
|
|
BOOST_TEST(result.reader.basePath() == "");
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_paths_to_source_unit_names_base_path_same_as_work_dir)
|
|
{
|
|
TemporaryDirectory tempDirCurrent(TEST_CASE_NAME);
|
|
TemporaryDirectory tempDirOther(TEST_CASE_NAME);
|
|
TemporaryWorkingDirectory tempWorkDir(tempDirCurrent);
|
|
soltestAssert(tempDirCurrent.path().is_absolute(), "");
|
|
soltestAssert(tempDirOther.path().is_absolute(), "");
|
|
|
|
// NOTE: On macOS the path usually contains symlinks which prevents base path from being stripped.
|
|
// Use canonical() to resolve symnlinks and get consistent results on all platforms.
|
|
boost::filesystem::path currentDirNoSymlinks = boost::filesystem::canonical(tempDirCurrent);
|
|
boost::filesystem::path otherDirNoSymlinks = boost::filesystem::canonical(tempDirOther);
|
|
|
|
boost::filesystem::path expectedWorkDir = "/" / boost::filesystem::current_path().relative_path();
|
|
boost::filesystem::path expectedOtherDir = "/" / otherDirNoSymlinks.relative_path();
|
|
soltestAssert(expectedWorkDir.is_absolute() || expectedWorkDir.root_path() == "/", "");
|
|
soltestAssert(expectedOtherDir.is_absolute() || expectedOtherDir.root_path() == "/", "");
|
|
|
|
vector<string> commandLine = {
|
|
"solc",
|
|
"--base-path=" + currentDirNoSymlinks.string(),
|
|
"contract1.sol", // Relative path
|
|
"c/d/contract2.sol", // Relative path with subdirectories
|
|
currentDirNoSymlinks.string() + "/contract3.sol", // Absolute path inside working dir
|
|
otherDirNoSymlinks.string() + "/contract4.sol", // Absolute path outside of working dir
|
|
};
|
|
|
|
CommandLineOptions expectedOptions;
|
|
expectedOptions.input.paths = {
|
|
"contract1.sol",
|
|
"c/d/contract2.sol",
|
|
currentDirNoSymlinks / "contract3.sol",
|
|
otherDirNoSymlinks / "contract4.sol",
|
|
};
|
|
expectedOptions.input.basePath = currentDirNoSymlinks;
|
|
expectedOptions.modelChecker.initialize = true;
|
|
|
|
map<string, string> expectedSources = {
|
|
{"contract1.sol", ""},
|
|
{"c/d/contract2.sol", ""},
|
|
{"contract3.sol", ""},
|
|
{expectedOtherDir.generic_string() + "/contract4.sol", ""},
|
|
};
|
|
|
|
FileReader::FileSystemPathSet expectedAllowedDirectories = {
|
|
currentDirNoSymlinks / "c/d",
|
|
currentDirNoSymlinks,
|
|
otherDirNoSymlinks,
|
|
};
|
|
|
|
createFilesWithParentDirs(expectedOptions.input.paths);
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles(commandLine);
|
|
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.stdoutContent == "");
|
|
BOOST_REQUIRE(result.success);
|
|
BOOST_TEST(result.options == expectedOptions);
|
|
BOOST_TEST(result.reader.sourceUnits() == expectedSources);
|
|
BOOST_TEST(result.reader.allowedDirectories() == expectedAllowedDirectories);
|
|
BOOST_TEST(result.reader.basePath() == expectedWorkDir);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_paths_to_source_unit_names_base_path_different_from_work_dir)
|
|
{
|
|
TemporaryDirectory tempDirCurrent(TEST_CASE_NAME);
|
|
TemporaryDirectory tempDirOther(TEST_CASE_NAME);
|
|
TemporaryDirectory tempDirBase(TEST_CASE_NAME);
|
|
TemporaryWorkingDirectory tempWorkDir(tempDirCurrent);
|
|
soltestAssert(tempDirCurrent.path().is_absolute(), "");
|
|
soltestAssert(tempDirOther.path().is_absolute(), "");
|
|
soltestAssert(tempDirBase.path().is_absolute(), "");
|
|
|
|
// NOTE: On macOS the path usually contains symlinks which prevents base path from being stripped.
|
|
// Use canonical() to resolve symnlinks and get consistent results on all platforms.
|
|
boost::filesystem::path currentDirNoSymlinks = boost::filesystem::canonical(tempDirCurrent);
|
|
boost::filesystem::path otherDirNoSymlinks = boost::filesystem::canonical(tempDirOther);
|
|
boost::filesystem::path baseDirNoSymlinks = boost::filesystem::canonical(tempDirBase);
|
|
|
|
boost::filesystem::path expectedWorkDir = "/" / boost::filesystem::current_path().relative_path();
|
|
boost::filesystem::path expectedCurrentDir = "/" / currentDirNoSymlinks.relative_path();
|
|
boost::filesystem::path expectedOtherDir = "/" / otherDirNoSymlinks.relative_path();
|
|
boost::filesystem::path expectedBaseDir = "/" / baseDirNoSymlinks.relative_path();
|
|
soltestAssert(expectedWorkDir.is_absolute() || expectedWorkDir.root_path() == "/", "");
|
|
soltestAssert(expectedCurrentDir.is_absolute() || expectedCurrentDir.root_path() == "/", "");
|
|
soltestAssert(expectedOtherDir.is_absolute() || expectedOtherDir.root_path() == "/", "");
|
|
soltestAssert(expectedBaseDir.is_absolute() || expectedBaseDir.root_path() == "/", "");
|
|
|
|
vector<string> commandLine = {
|
|
"solc",
|
|
"--base-path=" + baseDirNoSymlinks.string(),
|
|
"contract1.sol", // Relative path
|
|
"c/d/contract2.sol", // Relative path with subdirectories
|
|
currentDirNoSymlinks.string() + "/contract3.sol", // Absolute path inside working dir
|
|
otherDirNoSymlinks.string() + "/contract4.sol", // Absolute path outside of working dir
|
|
baseDirNoSymlinks.string() + "/contract5.sol", // Absolute path inside base path
|
|
};
|
|
|
|
CommandLineOptions expectedOptions;
|
|
expectedOptions.input.paths = {
|
|
"contract1.sol",
|
|
"c/d/contract2.sol",
|
|
currentDirNoSymlinks / "contract3.sol",
|
|
otherDirNoSymlinks / "contract4.sol",
|
|
baseDirNoSymlinks / "contract5.sol",
|
|
};
|
|
expectedOptions.input.basePath = baseDirNoSymlinks;
|
|
expectedOptions.modelChecker.initialize = true;
|
|
|
|
map<string, string> expectedSources = {
|
|
{expectedWorkDir.generic_string() + "/contract1.sol", ""},
|
|
{expectedWorkDir.generic_string() + "/c/d/contract2.sol", ""},
|
|
{expectedCurrentDir.generic_string() + "/contract3.sol", ""},
|
|
{expectedOtherDir.generic_string() + "/contract4.sol", ""},
|
|
{"contract5.sol", ""},
|
|
};
|
|
|
|
FileReader::FileSystemPathSet expectedAllowedDirectories = {
|
|
currentDirNoSymlinks / "c/d",
|
|
currentDirNoSymlinks,
|
|
otherDirNoSymlinks,
|
|
baseDirNoSymlinks,
|
|
};
|
|
|
|
createFilesWithParentDirs(expectedOptions.input.paths);
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles(commandLine);
|
|
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.stdoutContent == "");
|
|
BOOST_REQUIRE(result.success);
|
|
BOOST_TEST(result.options == expectedOptions);
|
|
BOOST_TEST(result.reader.sourceUnits() == expectedSources);
|
|
BOOST_TEST(result.reader.allowedDirectories() == expectedAllowedDirectories);
|
|
BOOST_TEST(result.reader.basePath() == expectedBaseDir);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_paths_to_source_unit_names_relative_base_path)
|
|
{
|
|
TemporaryDirectory tempDirCurrent(TEST_CASE_NAME);
|
|
TemporaryDirectory tempDirOther(TEST_CASE_NAME);
|
|
TemporaryWorkingDirectory tempWorkDir(tempDirCurrent);
|
|
soltestAssert(tempDirCurrent.path().is_absolute(), "");
|
|
soltestAssert(tempDirOther.path().is_absolute(), "");
|
|
|
|
// NOTE: On macOS the path usually contains symlinks which prevents base path from being stripped.
|
|
// Use canonical() to resolve symnlinks and get consistent results on all platforms.
|
|
boost::filesystem::path currentDirNoSymlinks = boost::filesystem::canonical(tempDirCurrent);
|
|
boost::filesystem::path otherDirNoSymlinks = boost::filesystem::canonical(tempDirOther);
|
|
|
|
boost::filesystem::path expectedWorkDir = "/" / boost::filesystem::current_path().relative_path();
|
|
boost::filesystem::path expectedOtherDir = "/" / otherDirNoSymlinks.relative_path();
|
|
soltestAssert(expectedWorkDir.is_absolute() || expectedWorkDir.root_path() == "/", "");
|
|
soltestAssert(expectedOtherDir.is_absolute() || expectedOtherDir.root_path() == "/", "");
|
|
|
|
vector<string> commandLine = {
|
|
"solc",
|
|
"--base-path=base",
|
|
"contract1.sol", // Relative path outside of base path
|
|
"base/contract2.sol", // Relative path inside base path
|
|
currentDirNoSymlinks.string() + "/contract3.sol", // Absolute path inside working dir
|
|
currentDirNoSymlinks.string() + "/base/contract4.sol", // Absolute path inside base path
|
|
otherDirNoSymlinks.string() + "/contract5.sol", // Absolute path outside of working dir
|
|
otherDirNoSymlinks.string() + "/base/contract6.sol", // Absolute path outside of working dir
|
|
};
|
|
|
|
CommandLineOptions expectedOptions;
|
|
expectedOptions.input.paths = {
|
|
"contract1.sol",
|
|
"base/contract2.sol",
|
|
currentDirNoSymlinks / "contract3.sol",
|
|
currentDirNoSymlinks / "base/contract4.sol",
|
|
otherDirNoSymlinks / "contract5.sol",
|
|
otherDirNoSymlinks / "base/contract6.sol",
|
|
};
|
|
expectedOptions.input.basePath = "base";
|
|
expectedOptions.modelChecker.initialize = true;
|
|
|
|
map<string, string> expectedSources = {
|
|
{expectedWorkDir.generic_string() + "/contract1.sol", ""},
|
|
{"contract2.sol", ""},
|
|
{expectedWorkDir.generic_string() + "/contract3.sol", ""},
|
|
{"contract4.sol", ""},
|
|
{expectedOtherDir.generic_string() + "/contract5.sol", ""},
|
|
{expectedOtherDir.generic_string() + "/base/contract6.sol", ""},
|
|
};
|
|
|
|
FileReader::FileSystemPathSet expectedAllowedDirectories = {
|
|
currentDirNoSymlinks / "base",
|
|
currentDirNoSymlinks,
|
|
otherDirNoSymlinks,
|
|
otherDirNoSymlinks / "base",
|
|
};
|
|
|
|
createFilesWithParentDirs(expectedOptions.input.paths);
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles(commandLine);
|
|
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.stdoutContent == "");
|
|
BOOST_REQUIRE(result.success);
|
|
BOOST_TEST(result.options == expectedOptions);
|
|
BOOST_TEST(result.reader.sourceUnits() == expectedSources);
|
|
BOOST_TEST(result.reader.allowedDirectories() == expectedAllowedDirectories);
|
|
BOOST_TEST(result.reader.basePath() == expectedWorkDir / "base");
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_paths_to_source_unit_names_normalization_and_weird_names)
|
|
{
|
|
TemporaryDirectory tempDir({"x/y/z"}, TEST_CASE_NAME);
|
|
TemporaryWorkingDirectory tempWorkDir(tempDir.path() / "x/y/z");
|
|
soltestAssert(tempDir.path().is_absolute(), "");
|
|
|
|
string uncPath = "//" + tempDir.path().relative_path().generic_string();
|
|
soltestAssert(FileReader::isUNCPath(uncPath), "");
|
|
|
|
boost::filesystem::path tempDirNoSymlinks = boost::filesystem::canonical(tempDir);
|
|
|
|
boost::filesystem::path expectedWorkDir = "/" / boost::filesystem::current_path().relative_path();
|
|
soltestAssert(expectedWorkDir.is_absolute() || expectedWorkDir.root_path() == "/", "");
|
|
|
|
vector<string> commandLine = {
|
|
"solc",
|
|
|
|
#if !defined(_WIN32)
|
|
// URLs. We interpret them as local paths.
|
|
// Note that : is not allowed in file names on Windows.
|
|
"file://c/d/contract1.sol",
|
|
"file:///c/d/contract2.sol",
|
|
"https://example.com/contract3.sol",
|
|
#endif
|
|
|
|
// Redundant slashes
|
|
"a/b//contract4.sol",
|
|
"a/b///contract5.sol",
|
|
"a/b////contract6.sol",
|
|
|
|
// Dot segments
|
|
"./a/b/contract7.sol",
|
|
"././a/b/contract8.sol",
|
|
"a/./b/contract9.sol",
|
|
"a/././b/contract10.sol",
|
|
|
|
// Dot dot segments
|
|
"../a/b/contract11.sol",
|
|
"../../a/b/contract12.sol",
|
|
"a/../b/contract13.sol",
|
|
"a/b/../../contract14.sol",
|
|
tempDirNoSymlinks.string() + "/x/y/z/a/../b/contract15.sol",
|
|
tempDirNoSymlinks.string() + "/x/y/z/a/b/../../contract16.sol",
|
|
|
|
// Dot dot segments going beyond filesystem root
|
|
"/../" + tempDir.path().relative_path().generic_string() + "/contract17.sol",
|
|
"/../../" + tempDir.path().relative_path().generic_string() + "/contract18.sol",
|
|
|
|
#if !defined(_WIN32)
|
|
// Name conflict with source unit name of stdin.
|
|
// Note that < and > are not allowed in file names on Windows.
|
|
"<stdin>",
|
|
|
|
// UNC paths on UNIX just resolve into normal paths. On Windows this would be an network
|
|
// share (and an error unless the share actually exists so I can't test it here).
|
|
uncPath + "/contract19.sol",
|
|
|
|
// Windows paths on non-Windows systems.
|
|
// Note that on Windows we tested them already just by using absolute paths.
|
|
"a\\b\\contract20.sol",
|
|
"C:\\a\\b\\contract21.sol",
|
|
#endif
|
|
};
|
|
|
|
CommandLineOptions expectedOptions;
|
|
expectedOptions.input.paths = {
|
|
#if !defined(_WIN32)
|
|
"file://c/d/contract1.sol",
|
|
"file:///c/d/contract2.sol",
|
|
"https://example.com/contract3.sol",
|
|
#endif
|
|
|
|
"a/b//contract4.sol",
|
|
"a/b///contract5.sol",
|
|
"a/b////contract6.sol",
|
|
|
|
"./a/b/contract7.sol",
|
|
"././a/b/contract8.sol",
|
|
"a/./b/contract9.sol",
|
|
"a/././b/contract10.sol",
|
|
|
|
"../a/b/contract11.sol",
|
|
"../../a/b/contract12.sol",
|
|
"a/../b/contract13.sol",
|
|
"a/b/../../contract14.sol",
|
|
tempDirNoSymlinks.string() + "/x/y/z/a/../b/contract15.sol",
|
|
tempDirNoSymlinks.string() + "/x/y/z/a/b/../../contract16.sol",
|
|
|
|
"/../" + tempDir.path().relative_path().string() + "/contract17.sol",
|
|
"/../../" + tempDir.path().relative_path().string() + "/contract18.sol",
|
|
|
|
#if !defined(_WIN32)
|
|
"<stdin>",
|
|
|
|
uncPath + "/contract19.sol",
|
|
|
|
"a\\b\\contract20.sol",
|
|
"C:\\a\\b\\contract21.sol",
|
|
#endif
|
|
};
|
|
expectedOptions.modelChecker.initialize = true;
|
|
|
|
map<string, string> expectedSources = {
|
|
#if !defined(_WIN32)
|
|
{"file:/c/d/contract1.sol", ""},
|
|
{"file:/c/d/contract2.sol", ""},
|
|
{"https:/example.com/contract3.sol", ""},
|
|
#endif
|
|
|
|
{"a/b/contract4.sol", ""},
|
|
{"a/b/contract5.sol", ""},
|
|
{"a/b/contract6.sol", ""},
|
|
|
|
{"a/b/contract7.sol", ""},
|
|
{"a/b/contract8.sol", ""},
|
|
{"a/b/contract9.sol", ""},
|
|
{"a/b/contract10.sol", ""},
|
|
|
|
{expectedWorkDir.parent_path().generic_string() + "/a/b/contract11.sol", ""},
|
|
{expectedWorkDir.parent_path().parent_path().generic_string() + "/a/b/contract12.sol", ""},
|
|
{"b/contract13.sol", ""},
|
|
{"contract14.sol", ""},
|
|
{"b/contract15.sol", ""},
|
|
{"contract16.sol", ""},
|
|
|
|
{"/" + tempDir.path().relative_path().generic_string() + "/contract17.sol", ""},
|
|
{"/" + tempDir.path().relative_path().generic_string() + "/contract18.sol", ""},
|
|
|
|
#if !defined(_WIN32)
|
|
{"<stdin>", ""},
|
|
|
|
{uncPath + "/contract19.sol", ""},
|
|
|
|
{"a\\b\\contract20.sol", ""},
|
|
{"C:\\a\\b\\contract21.sol", ""},
|
|
#endif
|
|
};
|
|
|
|
FileReader::FileSystemPathSet expectedAllowedDirectories = {
|
|
#if !defined(_WIN32)
|
|
tempDirNoSymlinks / "x/y/z/file:/c/d",
|
|
tempDirNoSymlinks / "x/y/z/https:/example.com",
|
|
#endif
|
|
tempDirNoSymlinks / "x/y/z/a/b",
|
|
tempDirNoSymlinks / "x/y/z",
|
|
tempDirNoSymlinks / "x/y/z/b",
|
|
tempDirNoSymlinks / "x/y/a/b",
|
|
tempDirNoSymlinks / "x/a/b",
|
|
tempDirNoSymlinks,
|
|
#if !defined(_WIN32)
|
|
boost::filesystem::canonical(uncPath),
|
|
#endif
|
|
};
|
|
|
|
createFilesWithParentDirs(expectedOptions.input.paths);
|
|
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles(commandLine);
|
|
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.stdoutContent == "");
|
|
BOOST_REQUIRE(result.success);
|
|
BOOST_TEST(result.options == expectedOptions);
|
|
BOOST_TEST(result.reader.sourceUnits() == expectedSources);
|
|
BOOST_TEST(result.reader.allowedDirectories() == expectedAllowedDirectories);
|
|
BOOST_TEST(result.reader.basePath() == expectedOptions.input.basePath);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_paths_to_source_unit_names_symlinks)
|
|
{
|
|
TemporaryDirectory tempDir({"r/"}, TEST_CASE_NAME);
|
|
createFilesWithParentDirs({tempDir.path() / "x/y/z/contract.sol"});
|
|
TemporaryWorkingDirectory tempWorkDir(tempDir.path() / "r");
|
|
|
|
if (
|
|
!createSymlinkIfSupportedByFilesystem("../x/y", tempDir.path() / "r/sym", true) ||
|
|
!createSymlinkIfSupportedByFilesystem("contract.sol", tempDir.path() / "x/y/z/contract_symlink.sol", false)
|
|
)
|
|
return;
|
|
|
|
boost::filesystem::path expectedWorkDir = "/" / boost::filesystem::current_path().relative_path();
|
|
soltestAssert(expectedWorkDir.is_absolute() || expectedWorkDir.root_path() == "/", "");
|
|
|
|
vector<string> commandLine = {
|
|
"solc",
|
|
|
|
"--base-path=../r/sym/z/",
|
|
"sym/z/contract.sol", // File accessed directly + same dir symlink as base path
|
|
"../x/y/z/contract.sol", // File accessed directly + different dir symlink than base path
|
|
"sym/z/contract_symlink.sol", // File accessed via symlink + same dir symlink as base path
|
|
"../x/y/z/contract_symlink.sol", // File accessed via symlink + different dir symlink than base path
|
|
};
|
|
|
|
CommandLineOptions expectedOptions;
|
|
expectedOptions.input.paths = {
|
|
"sym/z/contract.sol",
|
|
"../x/y/z/contract.sol",
|
|
"sym/z/contract_symlink.sol",
|
|
"../x/y/z/contract_symlink.sol",
|
|
};
|
|
expectedOptions.input.basePath = "../r/sym/z/";
|
|
expectedOptions.modelChecker.initialize = true;
|
|
|
|
map<string, string> expectedSources = {
|
|
{"contract.sol", ""},
|
|
{(expectedWorkDir.parent_path() / "x/y/z/contract.sol").generic_string(), ""},
|
|
{"contract_symlink.sol", ""},
|
|
{(expectedWorkDir.parent_path() / "x/y/z/contract_symlink.sol").generic_string(), ""},
|
|
};
|
|
|
|
FileReader::FileSystemPathSet expectedAllowedDirectories = {
|
|
boost::filesystem::canonical(tempDir) / "x/y/z",
|
|
};
|
|
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles(commandLine);
|
|
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.stdoutContent == "");
|
|
BOOST_REQUIRE(result.success);
|
|
BOOST_TEST(result.options == expectedOptions);
|
|
BOOST_TEST(result.reader.sourceUnits() == expectedSources);
|
|
BOOST_TEST(result.reader.allowedDirectories() == expectedAllowedDirectories);
|
|
BOOST_TEST(result.reader.basePath() == expectedWorkDir / "sym/z/");
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_paths_to_source_unit_names_base_path_and_stdin)
|
|
{
|
|
TemporaryDirectory tempDir(TEST_CASE_NAME);
|
|
TemporaryWorkingDirectory tempWorkDir(tempDir);
|
|
boost::filesystem::create_directories(tempDir.path() / "base");
|
|
|
|
boost::filesystem::path expectedWorkDir = "/" / boost::filesystem::current_path().relative_path();
|
|
|
|
vector<string> commandLine = {"solc", "--base-path=base", "-"};
|
|
|
|
CommandLineOptions expectedOptions;
|
|
expectedOptions.input.addStdin = true;
|
|
expectedOptions.input.basePath = "base";
|
|
expectedOptions.modelChecker.initialize = true;
|
|
|
|
map<string, string> expectedSources = {
|
|
{"<stdin>", ""},
|
|
};
|
|
FileReader::FileSystemPathSet expectedAllowedDirectories = {};
|
|
|
|
createFilesWithParentDirs(expectedOptions.input.paths);
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles(commandLine);
|
|
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_TEST(result.stdoutContent == "");
|
|
BOOST_REQUIRE(result.success);
|
|
BOOST_TEST(result.options == expectedOptions);
|
|
BOOST_TEST(result.reader.sourceUnits() == expectedSources);
|
|
BOOST_TEST(result.reader.allowedDirectories() == expectedAllowedDirectories);
|
|
BOOST_TEST(result.reader.basePath() == expectedWorkDir / "base");
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_include_paths)
|
|
{
|
|
TemporaryDirectory tempDir({"base/", "include/", "lib/nested/"}, TEST_CASE_NAME);
|
|
TemporaryWorkingDirectory tempWorkDir(tempDir);
|
|
|
|
string const preamble =
|
|
"// SPDX-License-Identifier: GPL-3.0\n"
|
|
"pragma solidity >=0.0;\n";
|
|
string const mainContractSource = preamble +
|
|
"import \"contract.sol\";\n"
|
|
"import \"contract_via_callback.sol\";\n"
|
|
"import \"include.sol\";\n"
|
|
"import \"include_via_callback.sol\";\n"
|
|
"import \"nested.sol\";\n"
|
|
"import \"nested_via_callback.sol\";\n"
|
|
"import \"lib.sol\";\n"
|
|
"import \"lib_via_callback.sol\";\n";
|
|
|
|
createFilesWithParentDirs(
|
|
{
|
|
tempDir.path() / "base/contract.sol",
|
|
tempDir.path() / "base/contract_via_callback.sol",
|
|
tempDir.path() / "include/include.sol",
|
|
tempDir.path() / "include/include_via_callback.sol",
|
|
tempDir.path() / "lib/nested/nested.sol",
|
|
tempDir.path() / "lib/nested/nested_via_callback.sol",
|
|
tempDir.path() / "lib/lib.sol",
|
|
tempDir.path() / "lib/lib_via_callback.sol",
|
|
},
|
|
preamble
|
|
);
|
|
createFilesWithParentDirs({tempDir.path() / "base/main.sol"}, mainContractSource);
|
|
|
|
boost::filesystem::path canonicalWorkDir = boost::filesystem::canonical(tempDir);
|
|
boost::filesystem::path expectedWorkDir = "/" / canonicalWorkDir.relative_path();
|
|
|
|
vector<string> commandLine = {
|
|
"solc",
|
|
"--no-color",
|
|
"--base-path=base/",
|
|
"--include-path=include/",
|
|
"--include-path=lib/nested",
|
|
"--include-path=lib/",
|
|
"base/main.sol",
|
|
"base/contract.sol",
|
|
"include/include.sol",
|
|
"lib/nested/nested.sol",
|
|
"lib/lib.sol",
|
|
};
|
|
|
|
CommandLineOptions expectedOptions;
|
|
expectedOptions.input.paths = {
|
|
"base/main.sol",
|
|
"base/contract.sol",
|
|
"include/include.sol",
|
|
"lib/nested/nested.sol",
|
|
"lib/lib.sol",
|
|
};
|
|
expectedOptions.input.basePath = "base/";
|
|
expectedOptions.input.includePaths = {
|
|
"include/",
|
|
"lib/nested",
|
|
"lib/",
|
|
};
|
|
expectedOptions.formatting.coloredOutput = false;
|
|
expectedOptions.modelChecker.initialize = true;
|
|
|
|
map<string, string> expectedSources = {
|
|
{"main.sol", mainContractSource},
|
|
{"contract.sol", preamble},
|
|
{"contract_via_callback.sol", preamble},
|
|
{"include.sol", preamble},
|
|
{"include_via_callback.sol", preamble},
|
|
{"nested.sol", preamble},
|
|
{"nested_via_callback.sol", preamble},
|
|
{"lib.sol", preamble},
|
|
{"lib_via_callback.sol", preamble},
|
|
};
|
|
|
|
vector<boost::filesystem::path> expectedIncludePaths = {
|
|
expectedWorkDir / "include/",
|
|
expectedWorkDir / "lib/nested",
|
|
expectedWorkDir / "lib/",
|
|
};
|
|
|
|
FileReader::FileSystemPathSet expectedAllowedDirectories = {
|
|
canonicalWorkDir / "base",
|
|
canonicalWorkDir / "include",
|
|
canonicalWorkDir / "lib/nested",
|
|
canonicalWorkDir / "lib",
|
|
};
|
|
|
|
string const expectedStdoutContent = "Compiler run successful. No contracts to compile.\n";
|
|
OptionsReaderAndMessages result = runCLI(commandLine, "");
|
|
|
|
BOOST_TEST(result.stderrContent == "");
|
|
if (SemVerVersion{string(VersionString)}.isPrerelease())
|
|
BOOST_TEST(result.stdoutContent == "");
|
|
else
|
|
BOOST_TEST(result.stdoutContent == expectedStdoutContent);
|
|
BOOST_REQUIRE(result.success);
|
|
BOOST_TEST(result.options == expectedOptions);
|
|
BOOST_TEST(result.reader.sourceUnits() == expectedSources);
|
|
BOOST_TEST(result.reader.includePaths() == expectedIncludePaths);
|
|
BOOST_TEST(result.reader.allowedDirectories() == expectedAllowedDirectories);
|
|
BOOST_TEST(result.reader.basePath() == expectedWorkDir / "base/");
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_no_contracts_to_compile)
|
|
{
|
|
string const contractSource = R"(
|
|
// SPDX-License-Identifier: GPL-3.0
|
|
pragma solidity >=0.0;
|
|
enum Status { test }
|
|
)";
|
|
|
|
string const expectedStdoutContent = "Compiler run successful. No contracts to compile.\n";
|
|
OptionsReaderAndMessages result = runCLI({"solc", "-"}, contractSource);
|
|
|
|
if (SemVerVersion{string(VersionString)}.isPrerelease())
|
|
BOOST_TEST(result.stdoutContent == "");
|
|
else
|
|
BOOST_TEST(result.stdoutContent == expectedStdoutContent);
|
|
BOOST_REQUIRE(result.success);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_no_output)
|
|
{
|
|
string const contractSource = R"(
|
|
// SPDX-License-Identifier: GPL-3.0
|
|
pragma solidity >=0.0;
|
|
abstract contract A {
|
|
function B() public virtual returns(uint);
|
|
})";
|
|
|
|
string const expectedStdoutContent = "Compiler run successful. No output generated.\n";
|
|
OptionsReaderAndMessages result = runCLI({"solc", "-"}, contractSource);
|
|
|
|
if (SemVerVersion{string(VersionString)}.isPrerelease())
|
|
BOOST_TEST(result.stdoutContent == "");
|
|
else
|
|
BOOST_TEST(result.stdoutContent == expectedStdoutContent);
|
|
BOOST_REQUIRE(result.success);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(standard_json_include_paths)
|
|
{
|
|
TemporaryDirectory tempDir({"base/", "include/", "lib/nested/"}, TEST_CASE_NAME);
|
|
TemporaryWorkingDirectory tempWorkDir(tempDir);
|
|
|
|
string const preamble =
|
|
"// SPDX-License-Identifier: GPL-3.0\n"
|
|
"pragma solidity >=0.0;\n";
|
|
string const mainContractSource = preamble +
|
|
"import 'contract_via_callback.sol';\n"
|
|
"import 'include_via_callback.sol';\n"
|
|
"import 'nested_via_callback.sol';\n"
|
|
"import 'lib_via_callback.sol';\n";
|
|
|
|
string const standardJsonInput = R"(
|
|
{
|
|
"language": "Solidity",
|
|
"sources": {
|
|
"main.sol": {"content": ")" + mainContractSource + R"("}
|
|
}
|
|
}
|
|
)";
|
|
|
|
createFilesWithParentDirs(
|
|
{
|
|
tempDir.path() / "base/contract_via_callback.sol",
|
|
tempDir.path() / "include/include_via_callback.sol",
|
|
tempDir.path() / "lib/nested/nested_via_callback.sol",
|
|
tempDir.path() / "lib/lib_via_callback.sol",
|
|
},
|
|
preamble
|
|
);
|
|
|
|
boost::filesystem::path expectedWorkDir = "/" / boost::filesystem::canonical(tempDir).relative_path();
|
|
|
|
vector<string> commandLine = {
|
|
"solc",
|
|
"--base-path=base/",
|
|
"--include-path=include/",
|
|
"--include-path=lib/nested",
|
|
"--include-path=lib/",
|
|
"--standard-json",
|
|
};
|
|
|
|
CommandLineOptions expectedOptions;
|
|
expectedOptions.input.mode = InputMode::StandardJson;
|
|
expectedOptions.input.paths = {};
|
|
expectedOptions.input.addStdin = true;
|
|
expectedOptions.input.basePath = "base/";
|
|
expectedOptions.input.includePaths = {
|
|
"include/",
|
|
"lib/nested",
|
|
"lib/",
|
|
};
|
|
expectedOptions.modelChecker.initialize = false;
|
|
|
|
// NOTE: Source code from Standard JSON does not end up in FileReader. This is not a problem
|
|
// because FileReader is only used once to initialize the compiler stack and after that
|
|
// its sources are irrelevant (even though the callback still stores everything it loads).
|
|
map<string, string> expectedSources = {
|
|
{"contract_via_callback.sol", preamble},
|
|
{"include_via_callback.sol", preamble},
|
|
{"nested_via_callback.sol", preamble},
|
|
{"lib_via_callback.sol", preamble},
|
|
};
|
|
|
|
vector<boost::filesystem::path> expectedIncludePaths = {
|
|
expectedWorkDir / "include/",
|
|
expectedWorkDir / "lib/nested",
|
|
expectedWorkDir / "lib/",
|
|
};
|
|
|
|
FileReader::FileSystemPathSet expectedAllowedDirectories = {};
|
|
|
|
OptionsReaderAndMessages result = runCLI(commandLine, standardJsonInput);
|
|
|
|
Json::Value parsedStdout;
|
|
string jsonParsingErrors;
|
|
BOOST_TEST(util::jsonParseStrict(result.stdoutContent, parsedStdout, &jsonParsingErrors));
|
|
BOOST_TEST(jsonParsingErrors == "");
|
|
for (Json::Value const& errorDict: parsedStdout["errors"])
|
|
// The error list might contain pre-release compiler warning
|
|
BOOST_TEST(errorDict["severity"] != "error");
|
|
BOOST_TEST(
|
|
(parsedStdout["sources"].getMemberNames() | ranges::to<set>) ==
|
|
(expectedSources | ranges::views::keys | ranges::to<set>) + set<string>{"main.sol"}
|
|
);
|
|
|
|
BOOST_REQUIRE(result.success);
|
|
BOOST_TEST(result.options == expectedOptions);
|
|
BOOST_TEST(result.reader.sourceUnits() == expectedSources);
|
|
BOOST_TEST(result.reader.includePaths() == expectedIncludePaths);
|
|
BOOST_TEST(result.reader.allowedDirectories() == expectedAllowedDirectories);
|
|
BOOST_TEST(result.reader.basePath() == expectedWorkDir / "base/");
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_include_paths_empty_path)
|
|
{
|
|
TemporaryDirectory tempDir({"base/", "include/"}, TEST_CASE_NAME);
|
|
TemporaryWorkingDirectory tempWorkDir(tempDir);
|
|
createFilesWithParentDirs({tempDir.path() / "base/main.sol"});
|
|
|
|
string expectedMessage = "Empty values are not allowed in --include-path.";
|
|
|
|
BOOST_CHECK_EXCEPTION(
|
|
parseCommandLineAndReadInputFiles({
|
|
"solc",
|
|
"--base-path=base/",
|
|
"--include-path", "include/",
|
|
"--include-path", "",
|
|
"base/main.sol",
|
|
}),
|
|
CommandLineValidationError,
|
|
[&](auto const& _exception) { BOOST_TEST(_exception.what() == expectedMessage); return true; }
|
|
);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_include_paths_without_base_path)
|
|
{
|
|
TemporaryDirectory tempDir(TEST_CASE_NAME);
|
|
TemporaryWorkingDirectory tempWorkDir(tempDir);
|
|
createFilesWithParentDirs({tempDir.path() / "contract.sol"});
|
|
|
|
string expectedMessage = "--include-path option requires a non-empty base path.";
|
|
|
|
BOOST_CHECK_EXCEPTION(
|
|
parseCommandLineAndReadInputFiles({"solc", "--include-path", "include/", "contract.sol"}),
|
|
CommandLineValidationError,
|
|
[&](auto const& _exception) { BOOST_TEST(_exception.what() == expectedMessage); return true; }
|
|
);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_include_paths_should_detect_source_unit_name_collisions)
|
|
{
|
|
TemporaryDirectory tempDir({"dir1/", "dir2/", "dir3/"}, TEST_CASE_NAME);
|
|
TemporaryWorkingDirectory tempWorkDir(tempDir);
|
|
createFilesWithParentDirs({
|
|
"dir1/contract1.sol",
|
|
"dir1/contract2.sol",
|
|
"dir2/contract1.sol",
|
|
"dir2/contract2.sol",
|
|
});
|
|
|
|
boost::filesystem::path expectedWorkDir = "/" / boost::filesystem::canonical(tempDir).relative_path();
|
|
|
|
string expectedMessage =
|
|
"Source unit name collision detected. "
|
|
"The specified values of base path and/or include paths would result in multiple "
|
|
"input files being assigned the same source unit name:\n"
|
|
"contract1.sol matches: "
|
|
"\"" + (expectedWorkDir / "dir1/contract1.sol").generic_string() + "\", "
|
|
"\"" + (expectedWorkDir / "dir2/contract1.sol").generic_string() + "\"\n"
|
|
"contract2.sol matches: "
|
|
"\"" + (expectedWorkDir / "dir1/contract2.sol").generic_string() + "\", "
|
|
"\"" + (expectedWorkDir / "dir2/contract2.sol").generic_string() + "\"\n";
|
|
|
|
{
|
|
// import "contract1.sol" and import "contract2.sol" would be ambiguous:
|
|
BOOST_CHECK_EXCEPTION(
|
|
parseCommandLineAndReadInputFiles({
|
|
"solc",
|
|
"--base-path=dir1/",
|
|
"--include-path=dir2/",
|
|
"dir1/contract1.sol",
|
|
"dir2/contract1.sol",
|
|
"dir1/contract2.sol",
|
|
"dir2/contract2.sol",
|
|
}),
|
|
CommandLineValidationError,
|
|
[&](auto const& _exception) { BOOST_TEST(_exception.what() == expectedMessage); return true; }
|
|
);
|
|
}
|
|
|
|
{
|
|
// import "contract1.sol" and import "contract2.sol" would be ambiguous:
|
|
BOOST_CHECK_EXCEPTION(
|
|
parseCommandLineAndReadInputFiles({
|
|
"solc",
|
|
"--base-path=dir3/",
|
|
"--include-path=dir1/",
|
|
"--include-path=dir2/",
|
|
"dir1/contract1.sol",
|
|
"dir2/contract1.sol",
|
|
"dir1/contract2.sol",
|
|
"dir2/contract2.sol",
|
|
}),
|
|
CommandLineValidationError,
|
|
[&](auto const& _exception) { BOOST_TEST(_exception.what() == expectedMessage); return true; }
|
|
);
|
|
}
|
|
|
|
{
|
|
// No conflict if files with the same name exist but only one is given to the compiler.
|
|
vector<string> commandLine = {
|
|
"solc",
|
|
"--base-path=dir3/",
|
|
"--include-path=dir1/",
|
|
"--include-path=dir2/",
|
|
"dir1/contract1.sol",
|
|
"dir1/contract2.sol",
|
|
};
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles(commandLine);
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_REQUIRE(result.success);
|
|
}
|
|
|
|
{
|
|
// The same file specified multiple times is not a conflict.
|
|
vector<string> commandLine = {
|
|
"solc",
|
|
"--base-path=dir3/",
|
|
"--include-path=dir1/",
|
|
"--include-path=dir2/",
|
|
"dir1/contract1.sol",
|
|
"dir1/contract1.sol",
|
|
"./dir1/contract1.sol",
|
|
};
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles(commandLine);
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_REQUIRE(result.success);
|
|
}
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_include_paths_should_allow_duplicate_paths)
|
|
{
|
|
TemporaryDirectory tempDir({"dir1/", "dir2/"}, TEST_CASE_NAME);
|
|
TemporaryWorkingDirectory tempWorkDir(tempDir);
|
|
createFilesWithParentDirs({"dir1/contract.sol"});
|
|
|
|
boost::filesystem::path expectedWorkDir = "/" / boost::filesystem::canonical(tempDir).relative_path();
|
|
boost::filesystem::path expectedTempDir = "/" / tempDir.path().relative_path();
|
|
|
|
vector<string> commandLine = {
|
|
"solc",
|
|
"--base-path=dir1/",
|
|
"--include-path", "dir1",
|
|
"--include-path", "dir1",
|
|
"--include-path", "dir1/",
|
|
"--include-path", "dir1/",
|
|
"--include-path", "./dir1/",
|
|
"--include-path", "dir2/../dir1/",
|
|
"--include-path", (tempDir.path() / "dir1/").string(),
|
|
"--include-path", (expectedWorkDir / "dir1/").string(),
|
|
"--include-path", "dir1/",
|
|
"dir1/contract.sol",
|
|
};
|
|
|
|
// Duplicates do not affect the result but are not removed from the include path list.
|
|
vector<boost::filesystem::path> expectedIncludePaths = {
|
|
expectedWorkDir / "dir1",
|
|
expectedWorkDir / "dir1",
|
|
expectedWorkDir / "dir1/",
|
|
expectedWorkDir / "dir1/",
|
|
expectedWorkDir / "dir1/",
|
|
expectedWorkDir / "dir1/",
|
|
// NOTE: On macOS expectedTempDir usually contains a symlink and therefore for us it's
|
|
// different from expectedWorkDir.
|
|
expectedTempDir / "dir1/",
|
|
expectedWorkDir / "dir1/",
|
|
expectedWorkDir / "dir1/",
|
|
};
|
|
|
|
OptionsReaderAndMessages result = parseCommandLineAndReadInputFiles(commandLine);
|
|
BOOST_TEST(result.stderrContent == "");
|
|
BOOST_REQUIRE(result.success);
|
|
BOOST_TEST(result.reader.includePaths() == expectedIncludePaths);
|
|
BOOST_TEST(result.reader.basePath() == expectedWorkDir / "dir1/");
|
|
}
|
|
|
|
BOOST_AUTO_TEST_CASE(cli_include_paths_ambiguous_import)
|
|
{
|
|
TemporaryDirectory tempDir({"base/", "include/"}, TEST_CASE_NAME);
|
|
TemporaryWorkingDirectory tempWorkDir(tempDir);
|
|
|
|
string const preamble =
|
|
"// SPDX-License-Identifier: GPL-3.0\n"
|
|
"pragma solidity >=0.0;\n";
|
|
string const mainContractSource = preamble +
|
|
// Ambiguous: both base/contract.sol and include/contract.sol match the import.
|
|
"import \"contract.sol\";";
|
|
|
|
createFilesWithParentDirs({"base/contract.sol", "include/contract.sol"}, preamble);
|
|
|
|
boost::filesystem::path expectedWorkDir = "/" / boost::filesystem::canonical(tempDir).relative_path();
|
|
|
|
vector<string> commandLine = {
|
|
"solc",
|
|
"--no-color",
|
|
"--base-path=base/",
|
|
"--include-path=include/",
|
|
"-",
|
|
};
|
|
|
|
string expectedMessage =
|
|
"Error: Source \"contract.sol\" not found: Ambiguous import. "
|
|
"Multiple matching files found inside base path and/or include paths: \"" +
|
|
(expectedWorkDir / "base/contract.sol").generic_string() + "\", \"" +
|
|
(expectedWorkDir / "include/contract.sol").generic_string() + "\".\n"
|
|
" --> <stdin>:3:1:\n"
|
|
" |\n"
|
|
"3 | import \"contract.sol\";\n"
|
|
" | ^^^^^^^^^^^^^^^^^^^^^^\n\n";
|
|
|
|
OptionsReaderAndMessages result = runCLI(commandLine, mainContractSource);
|
|
BOOST_TEST(result.stderrContent == expectedMessage);
|
|
BOOST_REQUIRE(!result.success);
|
|
}
|
|
|
|
BOOST_AUTO_TEST_SUITE_END()
|
|
|
|
} // namespace solidity::frontend::test
|