mirror of
				https://github.com/ethereum/solidity
				synced 2023-10-03 13:03:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			540 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			540 lines
		
	
	
		
			13 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
 | |
| #include <tools/solidityUpgrade/SourceUpgrade.h>
 | |
| 
 | |
| #include <liblangutil/Exceptions.h>
 | |
| #include <liblangutil/SourceReferenceFormatter.h>
 | |
| 
 | |
| #include <libsolidity/ast/AST.h>
 | |
| 
 | |
| #include <boost/filesystem.hpp>
 | |
| #include <boost/filesystem/operations.hpp>
 | |
| #include <boost/algorithm/string.hpp>
 | |
| 
 | |
| #ifdef _WIN32 // windows
 | |
| 	#include <io.h>
 | |
| 	#define isatty _isatty
 | |
| 	#define fileno _fileno
 | |
| #else // unix
 | |
| 	#include <unistd.h>
 | |
| #endif
 | |
| 
 | |
| 
 | |
| namespace po = boost::program_options;
 | |
| namespace fs = boost::filesystem;
 | |
| 
 | |
| using namespace solidity;
 | |
| using namespace solidity::langutil;
 | |
| using namespace solidity::tools;
 | |
| using namespace solidity::util;
 | |
| using namespace solidity::frontend;
 | |
| using namespace std;
 | |
| 
 | |
| 
 | |
| static string const g_argHelp = "help";
 | |
| static string const g_argVersion = "version";
 | |
| static string const g_argInputFile = "input-file";
 | |
| static string const g_argModules = "modules";
 | |
| static string const g_argDryRun = "dry-run";
 | |
| static string const g_argUnsafe = "unsafe";
 | |
| static string const g_argVerbose = "verbose";
 | |
| static string const g_argIgnoreMissingFiles = "ignore-missing";
 | |
| static string const g_argAllowPaths = "allow-paths";
 | |
| 
 | |
| namespace
 | |
| {
 | |
| 
 | |
| ostream& out()
 | |
| {
 | |
| 	return cout;
 | |
| }
 | |
| 
 | |
| AnsiColorized log()
 | |
| {
 | |
| 	return AnsiColorized(cout, true, {});
 | |
| }
 | |
| 
 | |
| AnsiColorized success()
 | |
| {
 | |
| 	return AnsiColorized(cout, true, {formatting::CYAN});
 | |
| }
 | |
| 
 | |
| AnsiColorized warning()
 | |
| {
 | |
| 	return AnsiColorized(cout, true, {formatting::YELLOW});
 | |
| }
 | |
| 
 | |
| AnsiColorized error()
 | |
| {
 | |
| 	return AnsiColorized(cout, true, {formatting::MAGENTA});
 | |
| }
 | |
| 
 | |
| void logVersion()
 | |
| {
 | |
| 	/// TODO Replace by variable that can be set during build.
 | |
| 	out() << "0.1.0" << endl;
 | |
| }
 | |
| 
 | |
| void logProgress()
 | |
| {
 | |
| 	out() << ".";
 | |
| 	out().flush();
 | |
| }
 | |
| 
 | |
| }
 | |
| 
 | |
| bool SourceUpgrade::parseArguments(int _argc, char** _argv)
 | |
| {
 | |
| 	po::options_description desc(R"(solidity-upgrade, the Solidity upgrade assistant.
 | |
| 
 | |
| The solidity-upgrade tool can help upgrade smart contracts to breaking language features.
 | |
| 
 | |
| It does not support all breaking changes for each version, but will hopefully assist
 | |
| upgrading your contracts to the desired Solidity version.
 | |
| 
 | |
| solidity-upgrade is distributed in the hope that it will be useful,
 | |
| but WITHOUT ANY WARRANTY. Please be careful when running upgrades on
 | |
| your contracts.
 | |
| 
 | |
| Usage: solidity-upgrade [options] contract.sol
 | |
| 
 | |
| Allowed options)",
 | |
| 		po::options_description::m_default_line_length,
 | |
| 		po::options_description::m_default_line_length - 23
 | |
| 	);
 | |
| 	desc.add_options()
 | |
| 		(g_argHelp.c_str(), "Show help message and exit.")
 | |
| 		(g_argVersion.c_str(), "Show version and exit.")
 | |
| 		(
 | |
| 			g_argAllowPaths.c_str(),
 | |
| 			po::value<string>()->value_name("path(s)"),
 | |
| 			"Allow a given path for imports. A list of paths can be supplied by separating them "
 | |
| 			"with a comma."
 | |
| 		)
 | |
| 		(g_argIgnoreMissingFiles.c_str(), "Ignore missing files.")
 | |
| 		(
 | |
| 			g_argModules.c_str(),
 | |
| 			po::value<string>()->value_name("module(s)"),
 | |
| 			"Only activate a specific upgrade module. A list of "
 | |
| 			"modules can be supplied by separating them with a comma."
 | |
| 		)
 | |
| 		(g_argDryRun.c_str(), "Apply changes in-memory only and don't write to input file.")
 | |
| 		(g_argVerbose.c_str(), "Print logs, errors and changes. Shortens output of upgrade patches.")
 | |
| 		(g_argUnsafe.c_str(), "Accept *unsafe* changes.");
 | |
| 
 | |
| 
 | |
| 	po::options_description allOptions = desc;
 | |
| 	allOptions.add_options()("input-file", po::value<vector<string>>(), "input file");
 | |
| 
 | |
| 	po::positional_options_description filesPositions;
 | |
| 	filesPositions.add("input-file", -1);
 | |
| 
 | |
| 	// parse the compiler arguments
 | |
| 	try
 | |
| 	{
 | |
| 		po::command_line_parser cmdLineParser(_argc, _argv);
 | |
| 		cmdLineParser.style(
 | |
| 			po::command_line_style::default_style & (~po::command_line_style::allow_guessing)
 | |
| 		);
 | |
| 		cmdLineParser.options(allOptions).positional(filesPositions);
 | |
| 		po::store(cmdLineParser.run(), m_args);
 | |
| 	}
 | |
| 	catch (po::error const& _exception)
 | |
| 	{
 | |
| 		error() << _exception.what() << endl;
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	if (m_args.count(g_argHelp) || (isatty(fileno(stdin)) && _argc == 1))
 | |
| 	{
 | |
| 		out() << endl;
 | |
| 		log() << desc;
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	if (m_args.count(g_argVersion))
 | |
| 	{
 | |
| 		logVersion();
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	if (m_args.count(g_argModules))
 | |
| 	{
 | |
| 		vector<string> moduleArgs;
 | |
| 		auto modules = boost::split(
 | |
| 			moduleArgs, m_args[g_argModules].as<string>(), boost::is_any_of(",")
 | |
| 		);
 | |
| 
 | |
| 		/// All modules are activated by default. Clear them before activating single ones.
 | |
| 		m_suite.deactivateModules();
 | |
| 
 | |
| 		for (string const& module: modules)
 | |
| 		{
 | |
| 			if (module == "constructor")
 | |
| 				m_suite.activateModule(Module::ConstructorKeyword);
 | |
| 			else if (module == "visibility")
 | |
| 				m_suite.activateModule(Module::VisibilitySpecifier);
 | |
| 			else if (module == "abstract")
 | |
| 				m_suite.activateModule(Module::AbstractContract);
 | |
| 			else if (module == "override")
 | |
| 				m_suite.activateModule(Module::OverridingFunction);
 | |
| 			else if (module == "virtual")
 | |
| 				m_suite.activateModule(Module::VirtualFunction);
 | |
| 			else if (module == "dotsyntax")
 | |
| 				m_suite.activateModule(Module::DotSyntax);
 | |
| 			else if (module == "now")
 | |
| 				m_suite.activateModule(Module::NowKeyword);
 | |
| 			else if (module == "constructor-visibility")
 | |
| 				m_suite.activateModule(Module::ConstrutorVisibility);
 | |
| 			else
 | |
| 			{
 | |
| 				error() << "Unknown upgrade module \"" + module + "\"" << endl;
 | |
| 				return false;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/// TODO Share with solc commandline interface.
 | |
| 	if (m_args.count(g_argAllowPaths))
 | |
| 	{
 | |
| 		vector<string> paths;
 | |
| 		auto allowedPaths = boost::split(
 | |
| 			paths, m_args[g_argAllowPaths].as<string>(), boost::is_any_of(",")
 | |
| 		);
 | |
| 		for (string const& path: allowedPaths)
 | |
| 		{
 | |
| 			auto filesystem_path = boost::filesystem::path(path);
 | |
| 			/// If the given path had a trailing slash, the Boost filesystem
 | |
| 			/// path will have it's last component set to '.'. This breaks
 | |
| 			/// path comparison in later parts of the code, so we need to strip
 | |
| 			/// it.
 | |
| 			if (filesystem_path.filename() == ".")
 | |
| 				filesystem_path.remove_filename();
 | |
| 			m_allowedDirectories.push_back(filesystem_path);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void SourceUpgrade::printPrologue()
 | |
| {
 | |
| 	out() << endl;
 | |
| 	out() << endl;
 | |
| 
 | |
| 	log() <<
 | |
| 		"solidity-upgrade does not support all breaking changes for each version." <<
 | |
| 		endl <<
 | |
| 		"Please run `solidity-upgrade --help` and get a list of implemented upgrades." <<
 | |
| 		endl <<
 | |
| 		endl;
 | |
| 
 | |
| 	log() <<
 | |
| 		"Running analysis (and upgrade) on given source files." <<
 | |
| 		endl;
 | |
| }
 | |
| 
 | |
| bool SourceUpgrade::processInput()
 | |
| {
 | |
| 	if (!readInputFiles())
 | |
| 		return false;
 | |
| 
 | |
| 	resetCompiler(fileReader());
 | |
| 
 | |
| 	tryCompile();
 | |
| 	runUpgrade();
 | |
| 
 | |
| 	printStatistics();
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void SourceUpgrade::tryCompile() const
 | |
| {
 | |
| 	bool verbose = m_args.count(g_argVerbose);
 | |
| 
 | |
| 	if (verbose)
 | |
| 		log() << "Running compilation phases." << endl << endl;
 | |
| 	else
 | |
| 		logProgress();
 | |
| 
 | |
| 	try
 | |
| 	{
 | |
| 		if (m_compiler->parse())
 | |
| 		{
 | |
| 			if (m_compiler->analyze())
 | |
| 				m_compiler->compile();
 | |
| 			else
 | |
| 				if (verbose)
 | |
| 				{
 | |
| 					error() <<
 | |
| 						"Compilation errors that solidity-upgrade may resolve occurred." <<
 | |
| 						endl <<
 | |
| 						endl;
 | |
| 
 | |
| 					printErrors();
 | |
| 				}
 | |
| 		}
 | |
| 		else
 | |
| 			if (verbose)
 | |
| 			{
 | |
| 				error() <<
 | |
| 					"Compilation errors that solidity-upgrade cannot resolve occurred." <<
 | |
| 					endl <<
 | |
| 					endl;
 | |
| 
 | |
| 				printErrors();
 | |
| 			}
 | |
| 	}
 | |
| 	catch (Exception const& _exception)
 | |
| 	{
 | |
| 		error() << "Exception during compilation: " << boost::diagnostic_information(_exception) << endl;
 | |
| 	}
 | |
| 	catch (std::exception const& _exception)
 | |
| 	{
 | |
| 		error() << "Exception during compilation: " << boost::diagnostic_information(_exception) << endl;
 | |
| 	}
 | |
| 	catch (...)
 | |
| 	{
 | |
| 		error() << "Unknown exception during compilation: " << boost::current_exception_diagnostic_information() << endl;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void SourceUpgrade::runUpgrade()
 | |
| {
 | |
| 	bool recompile = true;
 | |
| 
 | |
| 	while (recompile && !m_compiler->errors().empty())
 | |
| 	{
 | |
| 		for (auto& sourceCode: m_sourceCodes)
 | |
| 		{
 | |
| 			recompile = analyzeAndUpgrade(sourceCode);
 | |
| 			if (recompile)
 | |
| 				break;
 | |
| 		}
 | |
| 
 | |
| 		if (recompile)
 | |
| 		{
 | |
| 			m_suite.reset();
 | |
| 			resetCompiler();
 | |
| 			tryCompile();
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool SourceUpgrade::analyzeAndUpgrade(pair<string, string> const& _sourceCode)
 | |
| {
 | |
| 	bool applyUnsafe = m_args.count(g_argUnsafe);
 | |
| 	bool verbose = m_args.count(g_argVerbose);
 | |
| 
 | |
| 	if (verbose)
 | |
| 		log() << "Analyzing and upgrading " << _sourceCode.first << "." << endl;
 | |
| 
 | |
| 	if (m_compiler->state() >= CompilerStack::State::AnalysisPerformed)
 | |
| 		m_suite.analyze(*m_compiler, m_compiler->ast(_sourceCode.first));
 | |
| 
 | |
| 	if (!m_suite.changes().empty())
 | |
| 	{
 | |
| 		auto& change = m_suite.changes().front();
 | |
| 
 | |
| 		if (verbose)
 | |
| 			change.log(*m_compiler, true);
 | |
| 
 | |
| 		if (change.level() == UpgradeChange::Level::Safe)
 | |
| 		{
 | |
| 			applyChange(_sourceCode, change);
 | |
| 			return true;
 | |
| 		}
 | |
| 		else if (change.level() == UpgradeChange::Level::Unsafe)
 | |
| 		{
 | |
| 			if (applyUnsafe)
 | |
| 			{
 | |
| 				applyChange(_sourceCode, change);
 | |
| 				return true;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return false;
 | |
| }
 | |
| 
 | |
| void SourceUpgrade::applyChange(
 | |
| 	pair<string, string> const& _sourceCode,
 | |
| 	UpgradeChange& _change
 | |
| )
 | |
| {
 | |
| 	bool dryRun = m_args.count(g_argDryRun);
 | |
| 	bool verbose = m_args.count(g_argVerbose);
 | |
| 
 | |
| 	if (verbose)
 | |
| 	{
 | |
| 		log() << "Applying change to " << _sourceCode.first << endl << endl;
 | |
| 		log() << _change.patch();
 | |
| 	}
 | |
| 
 | |
| 	m_sourceCodes[_sourceCode.first] = _change.apply(_sourceCode.second);
 | |
| 
 | |
| 	if (!dryRun)
 | |
| 		writeInputFile(_sourceCode.first, m_sourceCodes[_sourceCode.first]);
 | |
| }
 | |
| 
 | |
| void SourceUpgrade::printErrors() const
 | |
| {
 | |
| 	langutil::SourceReferenceFormatter formatter{cout, *m_compiler, true, false};
 | |
| 
 | |
| 	for (auto const& error: m_compiler->errors())
 | |
| 		if (error->type() != langutil::Error::Type::Warning)
 | |
| 			formatter.printErrorInformation(*error);
 | |
| }
 | |
| 
 | |
| void SourceUpgrade::printStatistics() const
 | |
| {
 | |
| 	out() << endl;
 | |
| 	out() << endl;
 | |
| 	out() << "After upgrade:" << endl;
 | |
| 	out() << endl;
 | |
| 	error() << "Found " << m_compiler->errors().size() << " errors." << endl;
 | |
| 	success() << "Found " << m_suite.changes().size() << " upgrades." << endl;
 | |
| }
 | |
| 
 | |
| bool SourceUpgrade::readInputFiles()
 | |
| {
 | |
| 	bool ignoreMissing = m_args.count(g_argIgnoreMissingFiles);
 | |
| 
 | |
| 	/// TODO Share with solc commandline interface.
 | |
| 	if (m_args.count(g_argInputFile))
 | |
| 		for (string path: m_args[g_argInputFile].as<vector<string>>())
 | |
| 		{
 | |
| 			boost::filesystem::path infile = path;
 | |
| 			if (!boost::filesystem::exists(infile))
 | |
| 			{
 | |
| 				if (!ignoreMissing)
 | |
| 				{
 | |
| 					error() << infile << " is not found." << endl;
 | |
| 					return false;
 | |
| 				}
 | |
| 				else
 | |
| 					error() << infile << " is not found. Skipping." << endl;
 | |
| 
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			if (!boost::filesystem::is_regular_file(infile))
 | |
| 			{
 | |
| 				if (!ignoreMissing)
 | |
| 				{
 | |
| 					error() << infile << " is not a valid file." << endl;
 | |
| 					return false;
 | |
| 				}
 | |
| 				else
 | |
| 					error() << infile << " is not a valid file. Skipping." << endl;
 | |
| 
 | |
| 				continue;
 | |
| 			}
 | |
| 
 | |
| 			m_sourceCodes[infile.generic_string()] = readFileAsString(infile);
 | |
| 		}
 | |
| 
 | |
| 	if (m_sourceCodes.size() == 0)
 | |
| 	{
 | |
| 		warning() << "No input files given." << endl;
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool SourceUpgrade::writeInputFile(string const& _path, string const& _source)
 | |
| {
 | |
| 	bool verbose = m_args.count(g_argVerbose);
 | |
| 
 | |
| 	if (verbose)
 | |
| 	{
 | |
| 		out() << endl;
 | |
| 		log() << "Writing to input file " << _path << "." << endl;
 | |
| 	}
 | |
| 
 | |
| 	ofstream file(_path, ios::trunc);
 | |
| 	file << _source;
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| ReadCallback::Callback SourceUpgrade::fileReader()
 | |
| {
 | |
| 	/// TODO Share with solc commandline interface.
 | |
| 	ReadCallback::Callback fileReader = [this](string const&, string const& _path)
 | |
| 	{
 | |
| 		try
 | |
| 		{
 | |
| 			boost::filesystem::path path = _path;
 | |
| 			boost::filesystem::path canonicalPath = boost::filesystem::weakly_canonical(path);
 | |
| 			bool isAllowed = false;
 | |
| 			for (auto const& allowedDir: m_allowedDirectories)
 | |
| 			{
 | |
| 				// If dir is a prefix of boostPath, we are fine.
 | |
| 				if (
 | |
| 					std::distance(allowedDir.begin(), allowedDir.end()) <= std::distance(canonicalPath.begin(), canonicalPath.end()) &&
 | |
| 					std::equal(allowedDir.begin(), allowedDir.end(), canonicalPath.begin())
 | |
| 				)
 | |
| 				{
 | |
| 					isAllowed = true;
 | |
| 					break;
 | |
| 				}
 | |
| 			}
 | |
| 			if (!isAllowed)
 | |
| 				return ReadCallback::Result{false, "File outside of allowed directories."};
 | |
| 
 | |
| 			if (!boost::filesystem::exists(canonicalPath))
 | |
| 				return ReadCallback::Result{false, "File not found."};
 | |
| 
 | |
| 			if (!boost::filesystem::is_regular_file(canonicalPath))
 | |
| 				return ReadCallback::Result{false, "Not a valid file."};
 | |
| 
 | |
| 			string contents = readFileAsString(canonicalPath);
 | |
| 			m_sourceCodes[path.generic_string()] = contents;
 | |
| 			return ReadCallback::Result{true, contents};
 | |
| 		}
 | |
| 		catch (Exception const& _exception)
 | |
| 		{
 | |
| 			return ReadCallback::Result{false, "Exception in read callback: " + boost::diagnostic_information(_exception)};
 | |
| 		}
 | |
| 		catch (...)
 | |
| 		{
 | |
| 			return ReadCallback::Result{false, "Unknown exception in read callback: " + boost::current_exception_diagnostic_information()};
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	return fileReader;
 | |
| }
 | |
| 
 | |
| void SourceUpgrade::resetCompiler()
 | |
| {
 | |
| 	m_compiler->reset();
 | |
| 	m_compiler->setSources(m_sourceCodes);
 | |
| 	m_compiler->setParserErrorRecovery(true);
 | |
| }
 | |
| 
 | |
| void SourceUpgrade::resetCompiler(ReadCallback::Callback const& _callback)
 | |
| {
 | |
| 	m_compiler = std::make_unique<CompilerStack>(_callback);
 | |
| 	m_compiler->setSources(m_sourceCodes);
 | |
| 	m_compiler->setParserErrorRecovery(true);
 | |
| }
 |