Skip to content
This repository has been archived by the owner on Oct 28, 2021. It is now read-only.

Commit

Permalink
add validation schemes
Browse files Browse the repository at this point in the history
  • Loading branch information
winsvega committed Jun 7, 2018
1 parent 49e1e24 commit 41dc392
Show file tree
Hide file tree
Showing 7 changed files with 278 additions and 146 deletions.
35 changes: 22 additions & 13 deletions libdevcore/JsonUtils.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -56,13 +56,12 @@ std::string dev::jsonTypeAsString(json_spirit::Value_type _type)
}

void dev::requireJsonFields(json_spirit::mObject const& _o, std::string const& _config,
std::map<std::string, possibleJsonType> const& _validationMap,
std::set<std::string> const& _ignoreFields)
std::map<std::string, jsonType> const& _validationMap)
{
// check for unexpected fiedls
for (auto const& field : _o)
{
if (!_validationMap.count(field.first) && !_ignoreFields.count(field.first))
if (!_validationMap.count(field.first))
{
std::string const comment =
"Unexpected field '" + field.first + "' in config: " + _config;
Expand All @@ -75,27 +74,37 @@ void dev::requireJsonFields(json_spirit::mObject const& _o, std::string const& _
// check field types with validation map
for (auto const vmap : _validationMap)
{
// check that all required fields are in the object
if (!_o.count(vmap.first))
{
std::string const comment =
"Expected field '" + vmap.first + "' not found in config: " + _config;
std::cerr << comment << "\n"
<< json_spirit::write_string((json_spirit::mValue)_o, true) << "\n";
BOOST_THROW_EXCEPTION(MissingField() << errinfo_comment(comment));
if (vmap.second.second == jsonField::Required)
{
std::string const comment =
"Expected field '" + vmap.first + "' not found in config: " + _config;
std::cerr << comment << "\n"
<< json_spirit::write_string((json_spirit::mValue)_o, true) << "\n";
BOOST_THROW_EXCEPTION(MissingField() << errinfo_comment(comment));
}
else if (vmap.second.second == jsonField::Optional)
continue;
}

// check that field type is one of allowed field types
bool matched = false;
std::string sTypes;
for (auto const& type : vmap.second)
for (auto const& type : vmap.second.first)
{
if (sTypes.size())
sTypes += ", or ";
sTypes += jsonTypeAsString(type);
if (_o.at(vmap.first).type() == type)
matched = true;
}
if (matched == false)
{
std::string sTypes;
for (auto const& type : vmap.second.first)
{
if (sTypes.size())
sTypes += ", or ";
sTypes += jsonTypeAsString(type);
}
std::string const comment =
"Field '" + vmap.first + "' expected to be " + sTypes + ", but set to " +
jsonTypeAsString(_o.at(vmap.first).type()) + " in " + _config;
Expand Down
23 changes: 17 additions & 6 deletions libdevcore/JsonUtils.h
Original file line number Diff line number Diff line change
Expand Up @@ -28,10 +28,21 @@ void validateFieldNames(json_spirit::mObject const& _obj, std::set<std::string>
// Converts json value type to string
std::string jsonTypeAsString(json_spirit::Value_type _type);

// Check json _o with validation map that reuires certain field of certain type to be present in
// json
typedef std::set<json_spirit::Value_type> possibleJsonType;
void requireJsonFields(json_spirit::mObject const& _o, std::string const& _config,
std::map<std::string, possibleJsonType> const& _validationMap,
std::set<std::string> const& _ignoreFields = {});
enum jsonField
{
Required,
Optional
};
using jsonTypeSet = std::set<json_spirit::Value_type>;
using jsonType = std::pair<jsonTypeSet, jsonField>;
//! Check the json object with validation map that reuires certain field of certain type to be
//! present in json
/*!
\param _o a json object to check
\param _configName a string with json object name. Will apper in error message.
\param _validationMap a map with json objects that would be checked. "objName" -> {js::str_type,
jsonField::Required}
*/
void requireJsonFields(json_spirit::mObject const& _o, std::string const& _configName,
std::map<std::string, jsonType> const& _validationMap);
}
31 changes: 6 additions & 25 deletions libethereum/Account.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -20,13 +20,15 @@
*/

#include "Account.h"
#include "ValidationSchemes.h"
#include <libdevcore/JsonUtils.h>
#include <libethcore/ChainOperationParams.h>
#include <libethcore/Precompiled.h>

using namespace std;
using namespace dev;
using namespace dev::eth;
using namespace dev::eth::validation;

namespace fs = boost::filesystem;

Expand Down Expand Up @@ -81,29 +83,8 @@ PrecompiledContract createPrecompiledContract(js::mObject& _precompiled)
throw;
}
}

}
namespace
{
string const c_wei = "wei";
string const c_finney = "finney";
string const c_balance = "balance";
string const c_nonce = "nonce";
string const c_code = "code";
string const c_codeFromFile = "codeFromFile"; ///< A file containg a code as bytes.
string const c_storage = "storage";
string const c_shouldnotexist = "shouldnotexist";
string const c_precompiled = "precompiled";
std::set<string> const c_knownAccountFields = {
c_wei, c_finney, c_balance, c_nonce, c_code, c_codeFromFile, c_storage, c_shouldnotexist,
c_code, c_precompiled
};
void validateAccountMapObj(js::mObject const& _o)
{
for (auto const& field: _o)
validateFieldNames(field.second.get_obj(), c_knownAccountFields);
}
}

AccountMap dev::eth::jsonToAccountMap(std::string const& _json, u256 const& _defaultNonce,
AccountMaskMap* o_mask, PrecompiledContractMap* o_precompiled, const fs::path& _configPath)
{
Expand All @@ -119,12 +100,12 @@ AccountMap dev::eth::jsonToAccountMap(std::string const& _json, u256 const& _def
js::mValue val;
json_spirit::read_string_or_throw(_json, val);
js::mObject o = val.get_obj();
validateAccountMapObj(o);
for (auto const& account: o)
for (auto const& account: o)
{
Address a(fromHex(account.first));
Address a(fromHex(account.first));
// FIXME: Do not copy every account object.
auto o = account.second.get_obj();
validateAccountMapObj(o);

bool haveBalance = (o.count(c_wei) || o.count(c_finney) || o.count(c_balance));
bool haveNonce = o.count(c_nonce);
Expand Down
110 changes: 10 additions & 100 deletions libethereum/ChainParams.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -27,15 +27,16 @@
#include <libethcore/SealEngine.h>
#include <libethcore/BlockHeader.h>
#include <libethcore/Precompiled.h>
#include "ValidationSchemes.h"
#include "GenesisInfo.h"
#include "State.h"
#include "Account.h"
using namespace std;
using namespace dev;
using namespace eth;
using namespace eth::validation;
namespace js = json_spirit;


ChainParams::ChainParams()
{
for (unsigned i = 1; i <= 4; ++i)
Expand All @@ -52,79 +53,6 @@ ChainParams::ChainParams(string const& _json, h256 const& _stateRoot)
*this = loadConfig(_json, _stateRoot);
}

namespace
{
string const c_sealEngine = "sealEngine";
string const c_params = "params";
string const c_genesis = "genesis";
string const c_accounts = "accounts";

set<string> const c_knownChainConfigFields =
{c_sealEngine, c_params, c_genesis, c_accounts};

string const c_minGasLimit = "minGasLimit";
string const c_maxGasLimit = "maxGasLimit";
string const c_gasLimitBoundDivisor = "gasLimitBoundDivisor";
string const c_homesteadForkBlock = "homesteadForkBlock";
string const c_daoHardforkBlock = "daoHardforkBlock";
string const c_EIP150ForkBlock = "EIP150ForkBlock";
string const c_EIP158ForkBlock = "EIP158ForkBlock";
string const c_byzantiumForkBlock = "byzantiumForkBlock";
string const c_eWASMForkBlock = "eWASMForkBlock";
string const c_constantinopleForkBlock = "constantinopleForkBlock";
string const c_accountStartNonce = "accountStartNonce";
string const c_maximumExtraDataSize = "maximumExtraDataSize";
string const c_tieBreakingGas = "tieBreakingGas";
string const c_blockReward = "blockReward";
string const c_difficultyBoundDivisor = "difficultyBoundDivisor";
string const c_minimumDifficulty = "minimumDifficulty";
string const c_durationLimit = "durationLimit";
string const c_chainID = "chainID";
string const c_networkID = "networkID";
string const c_allowFutureBlocks = "allowFutureBlocks";

set<string> const c_knownParamNames = {c_minGasLimit, c_maxGasLimit, c_gasLimitBoundDivisor,
c_homesteadForkBlock, c_EIP150ForkBlock, c_EIP158ForkBlock, c_accountStartNonce,
c_maximumExtraDataSize, c_tieBreakingGas, c_blockReward, c_byzantiumForkBlock, c_eWASMForkBlock,
c_constantinopleForkBlock, c_daoHardforkBlock, c_minimumDifficulty, c_difficultyBoundDivisor,
c_durationLimit, c_chainID, c_networkID, c_allowFutureBlocks};
} // anonymous namespace

void validateConfigJson(js::mObject const& _obj)
{
requireJsonFields(_obj, "ChainParams::loadConfig",
{{"sealEngine", {json_spirit::str_type}}, {"params", {json_spirit::obj_type}},
{"genesis", {json_spirit::obj_type}}, {"accounts", {json_spirit::obj_type}}});

requireJsonFields(_obj.at("genesis").get_obj(), "ChainParams::loadConfig",
{{"author", {json_spirit::str_type}}, {"nonce", {json_spirit::str_type}},
{"author", {json_spirit::str_type}}, {"gasLimit", {json_spirit::str_type}},
{"timestamp", {json_spirit::str_type}}, {"difficulty", {json_spirit::str_type}},
{"extraData", {json_spirit::str_type}}},
{"mixHash", "parentHash"});

js::mObject const& accounts = _obj.at("accounts").get_obj();
for (auto const& acc : accounts)
{
js::mObject const& account = acc.second.get_obj();
if (account.count("precompiled"))
{
requireJsonFields(account, "ChainParams::loadConfig",
{{"precompiled", {json_spirit::obj_type}}}, {"wei"});
}
else
{
if (account.count("wei"))
requireJsonFields(account, "ChainParams::loadConfig", {{"wei", {json_spirit::str_type}}});
else
{
requireJsonFields(account, "ChainParams::loadConfig",
{{"balance", {json_spirit::str_type}}}, {"code", "nonce", "storage"});
}
}
}
}

ChainParams ChainParams::loadConfig(
string const& _json, h256 const& _stateRoot, const boost::filesystem::path& _configPath) const
{
Expand All @@ -134,13 +62,10 @@ ChainParams ChainParams::loadConfig(
js::mObject obj = val.get_obj();

validateConfigJson(obj);
validateFieldNames(obj, c_knownChainConfigFields);

cp.sealEngineName = obj[c_sealEngine].get_str();
// params
js::mObject params = obj[c_params].get_obj();
validateFieldNames(params, c_knownParamNames);
cp.accountStartNonce = u256(fromBigEndian<u256>(fromHex(params[c_accountStartNonce].get_str())));
cp.accountStartNonce = u256(fromBigEndian<u256>(fromHex(params[c_accountStartNonce].get_str())));
cp.maximumExtraDataSize = u256(fromBigEndian<u256>(fromHex(params[c_maximumExtraDataSize].get_str())));
cp.tieBreakingGas = params.count(c_tieBreakingGas) ? params[c_tieBreakingGas].get_bool() : true;
cp.setBlockReward(u256(fromBigEndian<u256>(fromHex(params[c_blockReward].get_str()))));
Expand Down Expand Up @@ -178,28 +103,15 @@ ChainParams ChainParams::loadConfig(

cp.genesisState = jsonToAccountMap(
genesisStateStr, cp.accountStartNonce, nullptr, &cp.precompiled, _configPath);
cp.stateRoot = _stateRoot ? _stateRoot : cp.calculateStateRoot(true);

return cp;
}
// Strict account check
json_spirit::read_string_or_throw(genesisStateStr, val);
for (auto const& account: val.get_obj())
validateAccountObj(account.second.get_obj());

namespace
{
string const c_parentHash = "parentHash";
string const c_coinbase = "coinbase";
string const c_author = "author";
string const c_difficulty = "difficulty";
string const c_gasLimit = "gasLimit";
string const c_gasUsed = "gasUsed";
string const c_timestamp = "timestamp";
string const c_extraData = "extraData";
string const c_mixHash = "mixHash";
string const c_nonce = "nonce";
cp.stateRoot = _stateRoot ? _stateRoot : cp.calculateStateRoot(true);

set<string> const c_knownGenesisFields = {
c_parentHash, c_coinbase, c_author, c_difficulty, c_gasLimit, c_gasUsed, c_timestamp,
c_extraData, c_mixHash, c_nonce
};
return cp;
}

ChainParams ChainParams::loadGenesis(string const& _json, h256 const& _stateRoot) const
Expand All @@ -210,9 +122,7 @@ ChainParams ChainParams::loadGenesis(string const& _json, h256 const& _stateRoot
json_spirit::read_string(_json, val);
js::mObject genesis = val.get_obj();

validateFieldNames(genesis, c_knownGenesisFields);

cp.parentHash = h256(genesis[c_parentHash].get_str());
cp.parentHash = h256(0); // required by the YP
cp.author = genesis.count(c_coinbase) ? h160(genesis[c_coinbase].get_str()) : h160(genesis[c_author].get_str());
cp.difficulty = genesis.count(c_difficulty) ? u256(fromBigEndian<u256>(fromHex(genesis[c_difficulty].get_str()))) : 0;
cp.gasLimit = u256(fromBigEndian<u256>(fromHex(genesis[c_gasLimit].get_str())));
Expand Down
Loading

0 comments on commit 41dc392

Please sign in to comment.