mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
bd48f181b5
Introduce formatExceptionInformation
330 lines
11 KiB
C++
330 lines
11 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/>.
|
|
*/
|
|
/**
|
|
* @author Christian <c@ethdev.com>
|
|
* @date 2014
|
|
* JSON interface for the solidity compiler to be used from Javascript.
|
|
*/
|
|
|
|
#include <string>
|
|
#include <functional>
|
|
#include <iostream>
|
|
#include <json/json.h>
|
|
#include <libdevcore/Common.h>
|
|
#include <libdevcore/CommonData.h>
|
|
#include <libdevcore/CommonIO.h>
|
|
#include <libdevcore/JSON.h>
|
|
#include <libevmasm/Instruction.h>
|
|
#include <libevmasm/GasMeter.h>
|
|
#include <libsolidity/parsing/Scanner.h>
|
|
#include <libsolidity/parsing/Parser.h>
|
|
#include <libsolidity/ast/ASTPrinter.h>
|
|
#include <libsolidity/analysis/NameAndTypeResolver.h>
|
|
#include <libsolidity/interface/Exceptions.h>
|
|
#include <libsolidity/interface/CompilerStack.h>
|
|
#include <libsolidity/interface/SourceReferenceFormatter.h>
|
|
#include <libsolidity/ast/ASTJsonConverter.h>
|
|
#include <libsolidity/interface/Version.h>
|
|
|
|
using namespace std;
|
|
using namespace dev;
|
|
using namespace solidity;
|
|
|
|
extern "C" {
|
|
/// Callback used to retrieve additional source files. "Returns" two pointers that should be
|
|
/// heap-allocated and are free'd by the caller.
|
|
typedef void (*CStyleReadFileCallback)(char const* _path, char** o_contents, char** o_error);
|
|
}
|
|
|
|
Json::Value functionHashes(ContractDefinition const& _contract)
|
|
{
|
|
Json::Value functionHashes(Json::objectValue);
|
|
for (auto const& it: _contract.interfaceFunctions())
|
|
functionHashes[it.second->externalSignature()] = toHex(it.first.ref());
|
|
return functionHashes;
|
|
}
|
|
|
|
Json::Value gasToJson(GasEstimator::GasConsumption const& _gas)
|
|
{
|
|
if (_gas.isInfinite || _gas.value > std::numeric_limits<Json::LargestUInt>::max())
|
|
return Json::Value(Json::nullValue);
|
|
else
|
|
return Json::Value(Json::LargestUInt(_gas.value));
|
|
}
|
|
|
|
Json::Value estimateGas(CompilerStack const& _compiler, string const& _contract)
|
|
{
|
|
Json::Value gasEstimates(Json::objectValue);
|
|
using Gas = GasEstimator::GasConsumption;
|
|
if (!_compiler.assemblyItems(_contract) && !_compiler.runtimeAssemblyItems(_contract))
|
|
return gasEstimates;
|
|
if (eth::AssemblyItems const* items = _compiler.assemblyItems(_contract))
|
|
{
|
|
Gas gas = GasEstimator::functionalEstimation(*items);
|
|
u256 bytecodeSize(_compiler.runtimeObject(_contract).bytecode.size());
|
|
Json::Value creationGas(Json::arrayValue);
|
|
creationGas[0] = gasToJson(gas);
|
|
creationGas[1] = gasToJson(bytecodeSize * eth::GasCosts::createDataGas);
|
|
gasEstimates["creation"] = creationGas;
|
|
}
|
|
if (eth::AssemblyItems const* items = _compiler.runtimeAssemblyItems(_contract))
|
|
{
|
|
ContractDefinition const& contract = _compiler.contractDefinition(_contract);
|
|
Json::Value externalFunctions(Json::objectValue);
|
|
for (auto it: contract.interfaceFunctions())
|
|
{
|
|
string sig = it.second->externalSignature();
|
|
externalFunctions[sig] = gasToJson(GasEstimator::functionalEstimation(*items, sig));
|
|
}
|
|
if (contract.fallbackFunction())
|
|
externalFunctions[""] = gasToJson(GasEstimator::functionalEstimation(*items, "INVALID"));
|
|
gasEstimates["external"] = externalFunctions;
|
|
Json::Value internalFunctions(Json::objectValue);
|
|
for (auto const& it: contract.definedFunctions())
|
|
{
|
|
if (it->isPartOfExternalInterface() || it->isConstructor())
|
|
continue;
|
|
size_t entry = _compiler.functionEntryPoint(_contract, *it);
|
|
GasEstimator::GasConsumption gas = GasEstimator::GasConsumption::infinite();
|
|
if (entry > 0)
|
|
gas = GasEstimator::functionalEstimation(*items, entry, *it);
|
|
FunctionType type(*it);
|
|
string sig = it->name() + "(";
|
|
auto paramTypes = type.parameterTypes();
|
|
for (auto it = paramTypes.begin(); it != paramTypes.end(); ++it)
|
|
sig += (*it)->toString() + (it + 1 == paramTypes.end() ? "" : ",");
|
|
sig += ")";
|
|
internalFunctions[sig] = gasToJson(gas);
|
|
}
|
|
gasEstimates["internal"] = internalFunctions;
|
|
}
|
|
return gasEstimates;
|
|
}
|
|
|
|
string compile(StringMap const& _sources, bool _optimize, CStyleReadFileCallback _readCallback)
|
|
{
|
|
Json::Value output(Json::objectValue);
|
|
Json::Value errors(Json::arrayValue);
|
|
ReadFile::Callback readCallback;
|
|
if (_readCallback)
|
|
{
|
|
readCallback = [=](string const& _path)
|
|
{
|
|
char* contents_c = nullptr;
|
|
char* error_c = nullptr;
|
|
_readCallback(_path.c_str(), &contents_c, &error_c);
|
|
ReadFile::Result result;
|
|
result.success = true;
|
|
if (!contents_c && !error_c)
|
|
{
|
|
result.success = false;
|
|
result.contentsOrErrorMessage = "File not found.";
|
|
}
|
|
if (contents_c)
|
|
{
|
|
result.success = true;
|
|
result.contentsOrErrorMessage = string(contents_c);
|
|
free(contents_c);
|
|
}
|
|
if (error_c)
|
|
{
|
|
result.success = false;
|
|
result.contentsOrErrorMessage = string(error_c);
|
|
free(error_c);
|
|
}
|
|
return result;
|
|
};
|
|
}
|
|
CompilerStack compiler(readCallback);
|
|
auto scannerFromSourceName = [&](string const& _sourceName) -> solidity::Scanner const& { return compiler.scanner(_sourceName); };
|
|
bool success = false;
|
|
try
|
|
{
|
|
compiler.addSources(_sources);
|
|
bool succ = compiler.compile(_optimize);
|
|
for (auto const& error: compiler.errors())
|
|
{
|
|
auto err = dynamic_pointer_cast<Error const>(error);
|
|
errors.append(SourceReferenceFormatter::formatExceptionInformation(
|
|
*error,
|
|
(err->type() == Error::Type::Warning) ? "Warning" : "Error",
|
|
scannerFromSourceName
|
|
));
|
|
}
|
|
success = succ; // keep success false on exception
|
|
}
|
|
catch (Error const& error)
|
|
{
|
|
errors.append(SourceReferenceFormatter::formatExceptionInformation(error, error.typeName(), scannerFromSourceName));
|
|
}
|
|
catch (CompilerError const& exception)
|
|
{
|
|
errors.append(SourceReferenceFormatter::formatExceptionInformation(exception, "Compiler error (" + exception.lineInfo() + ")", scannerFromSourceName));
|
|
}
|
|
catch (InternalCompilerError const& exception)
|
|
{
|
|
errors.append(SourceReferenceFormatter::formatExceptionInformation(exception, "Internal compiler error (" + exception.lineInfo() + ")", scannerFromSourceName));
|
|
}
|
|
catch (UnimplementedFeatureError const& exception)
|
|
{
|
|
errors.append(SourceReferenceFormatter::formatExceptionInformation(exception, "Unimplemented feature (" + exception.lineInfo() + ")", scannerFromSourceName));
|
|
}
|
|
catch (Exception const& exception)
|
|
{
|
|
errors.append("Exception during compilation: " + boost::diagnostic_information(exception));
|
|
}
|
|
catch (...)
|
|
{
|
|
errors.append("Unknown exception during compilation.");
|
|
}
|
|
|
|
if (errors.size() > 0)
|
|
output["errors"] = errors;
|
|
|
|
if (success)
|
|
{
|
|
try
|
|
{
|
|
output["contracts"] = Json::Value(Json::objectValue);
|
|
for (string const& contractName: compiler.contractNames())
|
|
{
|
|
Json::Value contractData(Json::objectValue);
|
|
contractData["interface"] = dev::jsonCompactPrint(compiler.interface(contractName));
|
|
contractData["bytecode"] = compiler.object(contractName).toHex();
|
|
contractData["runtimeBytecode"] = compiler.runtimeObject(contractName).toHex();
|
|
contractData["opcodes"] = solidity::disassemble(compiler.object(contractName).bytecode);
|
|
contractData["metadata"] = compiler.onChainMetadata(contractName);
|
|
contractData["functionHashes"] = functionHashes(compiler.contractDefinition(contractName));
|
|
contractData["gasEstimates"] = estimateGas(compiler, contractName);
|
|
auto sourceMap = compiler.sourceMapping(contractName);
|
|
contractData["srcmap"] = sourceMap ? *sourceMap : "";
|
|
auto runtimeSourceMap = compiler.runtimeSourceMapping(contractName);
|
|
contractData["srcmapRuntime"] = runtimeSourceMap ? *runtimeSourceMap : "";
|
|
ostringstream unused;
|
|
contractData["assembly"] = compiler.streamAssembly(unused, contractName, _sources, true);
|
|
output["contracts"][contractName] = contractData;
|
|
}
|
|
}
|
|
catch (...)
|
|
{
|
|
output["errors"].append("Unknown exception while generating contract data output.");
|
|
}
|
|
|
|
try
|
|
{
|
|
// Do not taint the internal error list
|
|
ErrorList formalErrors;
|
|
if (compiler.prepareFormalAnalysis(&formalErrors))
|
|
output["formal"]["why3"] = compiler.formalTranslation();
|
|
if (!formalErrors.empty())
|
|
{
|
|
Json::Value errors(Json::arrayValue);
|
|
for (auto const& error: formalErrors)
|
|
errors.append(SourceReferenceFormatter::formatExceptionInformation(
|
|
*error,
|
|
(error->type() == Error::Type::Warning) ? "Warning" : "Error",
|
|
scannerFromSourceName
|
|
));
|
|
output["formal"]["errors"] = errors;
|
|
}
|
|
}
|
|
catch (...)
|
|
{
|
|
output["errors"].append("Unknown exception while generating formal method output.");
|
|
}
|
|
|
|
try
|
|
{
|
|
// Indices into this array are used to abbreviate source names in source locations.
|
|
output["sourceList"] = Json::Value(Json::arrayValue);
|
|
for (auto const& source: compiler.sourceNames())
|
|
output["sourceList"].append(source);
|
|
output["sources"] = Json::Value(Json::objectValue);
|
|
for (auto const& source: compiler.sourceNames())
|
|
output["sources"][source]["AST"] = ASTJsonConverter(compiler.ast(source), compiler.sourceIndices()).json();
|
|
}
|
|
catch (...)
|
|
{
|
|
output["errors"].append("Unknown exception while generating source name output.");
|
|
}
|
|
}
|
|
|
|
try
|
|
{
|
|
return dev::jsonCompactPrint(output);
|
|
}
|
|
catch (...)
|
|
{
|
|
return "{\"errors\":[\"Unknown error while generating JSON.\"]}";
|
|
}
|
|
}
|
|
|
|
string compileMulti(string const& _input, bool _optimize, CStyleReadFileCallback _readCallback = nullptr)
|
|
{
|
|
Json::Reader reader;
|
|
Json::Value input;
|
|
if (!reader.parse(_input, input, false))
|
|
{
|
|
Json::Value errors(Json::arrayValue);
|
|
errors.append("Error parsing input JSON: " + reader.getFormattedErrorMessages());
|
|
Json::Value output(Json::objectValue);
|
|
output["errors"] = errors;
|
|
return dev::jsonCompactPrint(output);
|
|
}
|
|
else
|
|
{
|
|
StringMap sources;
|
|
Json::Value jsonSources = input["sources"];
|
|
if (jsonSources.isObject())
|
|
for (auto const& sourceName: jsonSources.getMemberNames())
|
|
sources[sourceName] = jsonSources[sourceName].asString();
|
|
return compile(sources, _optimize, _readCallback);
|
|
}
|
|
}
|
|
|
|
string compileSingle(string const& _input, bool _optimize)
|
|
{
|
|
StringMap sources;
|
|
sources[""] = _input;
|
|
return compile(sources, _optimize, nullptr);
|
|
}
|
|
|
|
static string s_outputBuffer;
|
|
|
|
extern "C"
|
|
{
|
|
extern char const* version()
|
|
{
|
|
return VersionString.c_str();
|
|
}
|
|
extern char const* compileJSON(char const* _input, bool _optimize)
|
|
{
|
|
s_outputBuffer = compileSingle(_input, _optimize);
|
|
return s_outputBuffer.c_str();
|
|
}
|
|
extern char const* compileJSONMulti(char const* _input, bool _optimize)
|
|
{
|
|
s_outputBuffer = compileMulti(_input, _optimize);
|
|
return s_outputBuffer.c_str();
|
|
}
|
|
extern char const* compileJSONCallback(char const* _input, bool _optimize, CStyleReadFileCallback _readCallback)
|
|
{
|
|
s_outputBuffer = compileMulti(_input, _optimize, _readCallback);
|
|
return s_outputBuffer.c_str();
|
|
}
|
|
}
|