solidity/solc/jsonCompiler.cpp

314 lines
9.8 KiB
C++
Raw Normal View History

2015-01-28 07:50:53 +00:00
/*
This file is part of solidity.
2015-01-28 07:50:53 +00:00
solidity is free software: you can redistribute it and/or modify
2015-01-28 07:50:53 +00:00
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,
2015-01-28 07:50:53 +00:00
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/>.
2015-01-28 07:50:53 +00:00
*/
/**
* @author Christian <c@ethdev.com>
* @date 2014
* JSON interface for the solidity compiler to be used from Javascript.
*/
#include <string>
2016-02-22 01:13:41 +00:00
#include <functional>
2015-01-28 07:50:53 +00:00
#include <iostream>
#include <json/json.h>
#include <libdevcore/Common.h>
#include <libdevcore/CommonData.h>
#include <libdevcore/CommonIO.h>
2016-11-15 17:33:28 +00:00
#include <libdevcore/JSON.h>
2016-04-01 20:11:01 +00:00
#include <libevmasm/Instruction.h>
#include <libevmasm/GasMeter.h>
2015-10-20 22:21:52 +00:00
#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>
2015-01-28 07:50:53 +00:00
using namespace std;
using namespace dev;
using namespace solidity;
2016-01-26 15:25:36 +00:00
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);
}
2015-05-15 15:00:08 +00:00
Json::Value functionHashes(ContractDefinition const& _contract)
{
Json::Value functionHashes(Json::objectValue);
2015-08-31 16:44:29 +00:00
for (auto const& it: _contract.interfaceFunctions())
2015-05-15 15:00:08 +00:00
functionHashes[it.second->externalSignature()] = toHex(it.first.ref());
return functionHashes;
}
/// Translates a gas value as a string to a JSON number or null
Json::Value gasToJson(Json::Value const& _value)
{
if (_value.isObject())
{
Json::Value ret = Json::objectValue;
for (auto const& sig: _value.getMemberNames())
ret[sig] = gasToJson(_value[sig]);
return ret;
}
if (_value == "infinite")
return Json::Value(Json::nullValue);
u256 value(_value.asString());
if (value > std::numeric_limits<Json::LargestUInt>::max())
return Json::Value(Json::nullValue);
else
return Json::Value(Json::LargestUInt(value));
}
2015-06-01 11:25:02 +00:00
Json::Value estimateGas(CompilerStack const& _compiler, string const& _contract)
{
2017-04-10 13:52:05 +00:00
Json::Value estimates = _compiler.gasEstimates(_contract);
Json::Value output(Json::objectValue);
if (estimates["creation"].isObject())
2015-06-01 11:25:02 +00:00
{
2017-04-10 13:52:05 +00:00
Json::Value creation(Json::arrayValue);
creation[0] = gasToJson(estimates["creation"]["executionCost"]);
creation[1] = gasToJson(estimates["creation"]["codeDepositCost"]);
2017-04-10 13:52:05 +00:00
output["creation"] = creation;
2015-06-01 11:25:02 +00:00
}
else
output["creation"] = Json::objectValue;
output["external"] = gasToJson(estimates.get("external", Json::objectValue));
output["internal"] = gasToJson(estimates.get("internal", Json::objectValue));
2017-04-10 13:52:05 +00:00
return output;
2015-06-01 11:25:02 +00:00
}
2016-01-26 15:25:36 +00:00
string compile(StringMap const& _sources, bool _optimize, CStyleReadFileCallback _readCallback)
2015-01-28 07:50:53 +00:00
{
Json::Value output(Json::objectValue);
2015-09-21 17:43:56 +00:00
Json::Value errors(Json::arrayValue);
2017-04-10 11:48:41 +00:00
ReadFile::Callback readCallback;
2016-01-26 15:25:36 +00:00
if (_readCallback)
{
readCallback = [=](string const& _path)
2016-01-26 15:25:36 +00:00
{
char* contents_c = nullptr;
char* error_c = nullptr;
_readCallback(_path.c_str(), &contents_c, &error_c);
2017-04-10 11:48:41 +00:00
ReadFile::Result result;
result.success = true;
2016-01-26 15:25:36 +00:00
if (!contents_c && !error_c)
{
result.success = false;
2017-03-16 23:59:36 +00:00
result.contentsOrErrorMessage = "File not found.";
}
2016-01-26 15:25:36 +00:00
if (contents_c)
{
result.success = true;
2017-03-16 23:59:36 +00:00
result.contentsOrErrorMessage = string(contents_c);
2016-01-26 15:25:36 +00:00
free(contents_c);
}
if (error_c)
{
result.success = false;
2017-03-16 23:59:36 +00:00
result.contentsOrErrorMessage = string(error_c);
2016-01-26 15:25:36 +00:00
free(error_c);
}
return result;
2016-01-26 15:25:36 +00:00
};
}
2016-08-15 14:58:01 +00:00
CompilerStack compiler(readCallback);
2016-02-22 01:13:41 +00:00
auto scannerFromSourceName = [&](string const& _sourceName) -> solidity::Scanner const& { return compiler.scanner(_sourceName); };
2015-10-02 20:54:30 +00:00
bool success = false;
2015-01-28 07:50:53 +00:00
try
{
2015-10-15 15:12:22 +00:00
compiler.addSources(_sources);
bool succ = compiler.compile(_optimize);
2015-10-02 20:54:30 +00:00
for (auto const& error: compiler.errors())
2015-10-15 09:50:25 +00:00
{
auto err = dynamic_pointer_cast<Error const>(error);
errors.append(SourceReferenceFormatter::formatExceptionInformation(
2015-10-02 20:54:30 +00:00
*error,
2015-10-15 09:50:25 +00:00
(err->type() == Error::Type::Warning) ? "Warning" : "Error",
2016-02-22 01:13:41 +00:00
scannerFromSourceName
2015-10-02 20:54:30 +00:00
));
2015-10-15 09:50:25 +00:00
}
2015-10-02 20:54:30 +00:00
success = succ; // keep success false on exception
2015-01-28 07:50:53 +00:00
}
catch (Error const& error)
2015-01-28 07:50:53 +00:00
{
errors.append(SourceReferenceFormatter::formatExceptionInformation(error, error.typeName(), scannerFromSourceName));
2015-01-28 07:50:53 +00:00
}
catch (CompilerError const& exception)
{
errors.append(SourceReferenceFormatter::formatExceptionInformation(exception, "Compiler error (" + exception.lineInfo() + ")", scannerFromSourceName));
2015-01-28 07:50:53 +00:00
}
catch (InternalCompilerError const& exception)
{
errors.append(SourceReferenceFormatter::formatExceptionInformation(exception, "Internal compiler error (" + exception.lineInfo() + ")", scannerFromSourceName));
2015-01-28 07:50:53 +00:00
}
catch (UnimplementedFeatureError const& exception)
{
errors.append(SourceReferenceFormatter::formatExceptionInformation(exception, "Unimplemented feature (" + exception.lineInfo() + ")", scannerFromSourceName));
}
2015-01-28 07:50:53 +00:00
catch (Exception const& exception)
{
2015-09-21 17:43:56 +00:00
errors.append("Exception during compilation: " + boost::diagnostic_information(exception));
2015-01-28 07:50:53 +00:00
}
catch (...)
{
2015-09-21 17:43:56 +00:00
errors.append("Unknown exception during compilation.");
}
if (errors.size() > 0)
output["errors"] = errors;
2015-01-28 07:50:53 +00:00
2015-10-02 20:54:30 +00:00
if (success)
2015-01-28 07:50:53 +00:00
{
2016-09-01 18:14:00 +00:00
try
2015-10-02 20:54:30 +00:00
{
2016-09-01 18:14:00 +00:00
output["contracts"] = Json::Value(Json::objectValue);
for (string const& contractName: compiler.contractNames())
{
Json::Value contractData(Json::objectValue);
2016-11-15 17:33:28 +00:00
contractData["interface"] = dev::jsonCompactPrint(compiler.interface(contractName));
2016-09-01 18:14:00 +00:00
contractData["bytecode"] = compiler.object(contractName).toHex();
contractData["runtimeBytecode"] = compiler.runtimeObject(contractName).toHex();
contractData["opcodes"] = solidity::disassemble(compiler.object(contractName).bytecode);
2016-11-14 10:46:43 +00:00
contractData["metadata"] = compiler.onChainMetadata(contractName);
2016-09-01 18:14:00 +00:00
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.");
}
2016-09-01 18:14:00 +00:00
try
{
2016-09-01 18:14:00 +00:00
// 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(
2016-09-01 18:14:00 +00:00
*error,
(error->type() == Error::Type::Warning) ? "Warning" : "Error",
scannerFromSourceName
));
output["formal"]["errors"] = errors;
}
}
catch (...)
{
output["errors"].append("Unknown exception while generating formal method output.");
}
2015-01-28 07:50:53 +00:00
try
{
2016-09-01 18:14:00 +00:00
// 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 (...)
2016-07-18 16:16:22 +00:00
{
output["errors"].append("Unknown exception while generating source name output.");
2016-07-18 16:16:22 +00:00
}
2015-10-02 20:54:30 +00:00
}
2015-01-28 07:50:53 +00:00
2016-09-01 18:14:00 +00:00
try
{
2016-11-15 17:33:28 +00:00
return dev::jsonCompactPrint(output);
2016-09-01 18:14:00 +00:00
}
catch (...)
{
return "{\"errors\":[\"Unknown error while generating JSON.\"]}";
}
2015-01-28 07:50:53 +00:00
}
2016-01-26 15:25:36 +00:00
string compileMulti(string const& _input, bool _optimize, CStyleReadFileCallback _readCallback = nullptr)
2015-10-15 15:12:22 +00:00
{
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;
2016-11-15 17:33:28 +00:00
return dev::jsonCompactPrint(output);
2015-10-15 15:12:22 +00:00
}
else
{
StringMap sources;
Json::Value jsonSources = input["sources"];
if (jsonSources.isObject())
for (auto const& sourceName: jsonSources.getMemberNames())
sources[sourceName] = jsonSources[sourceName].asString();
2016-01-26 15:25:36 +00:00
return compile(sources, _optimize, _readCallback);
2015-10-15 15:12:22 +00:00
}
}
string compileSingle(string const& _input, bool _optimize)
{
StringMap sources;
sources[""] = _input;
2016-01-26 15:25:36 +00:00
return compile(sources, _optimize, nullptr);
2015-10-15 15:12:22 +00:00
}
2015-10-15 16:29:47 +00:00
static string s_outputBuffer;
2015-01-28 07:50:53 +00:00
extern "C"
{
2015-09-15 10:27:29 +00:00
extern char const* version()
{
return VersionString.c_str();
}
2015-04-22 09:33:02 +00:00
extern char const* compileJSON(char const* _input, bool _optimize)
2015-01-28 07:50:53 +00:00
{
2015-10-15 16:29:47 +00:00
s_outputBuffer = compileSingle(_input, _optimize);
return s_outputBuffer.c_str();
2015-10-15 15:12:22 +00:00
}
extern char const* compileJSONMulti(char const* _input, bool _optimize)
{
2015-10-15 16:29:47 +00:00
s_outputBuffer = compileMulti(_input, _optimize);
return s_outputBuffer.c_str();
2015-01-28 07:50:53 +00:00
}
2016-01-26 15:25:36 +00:00
extern char const* compileJSONCallback(char const* _input, bool _optimize, CStyleReadFileCallback _readCallback)
{
s_outputBuffer = compileMulti(_input, _optimize, _readCallback);
return s_outputBuffer.c_str();
}
2015-01-28 07:50:53 +00:00
}