aboutsummaryrefslogblamecommitdiffstats
path: root/TestHelper.cpp
blob: aec72285c600a614e1d7194921a4a5db71af4df8 (plain) (tree)




















                                                                             

                       

                 
                                 
                               
                            
                             
                  

                    
                         
 

             












                                                                            

                                           

                 
                                                                            
     




                                        
      
 


                                          
                                                                           
                            














                                                                                          

 
 
 


              
                                           

                                            
                                              
 



                                                                                                                   


                                                     
                                                       



                                                               
                                                                           
         
 
 

                                                                            

                                                                                                                                         
                                                                                                                                                   
 
                         
         

                                          

                                 
                                
                                                     






                                                               
                                                                                                         



                  

                                                    





                                                  
 

















                                                                                                     

                                                                 
 
 


                                                                                                                           
 
                         


                                                            
                                                

                                 
 



                                                                            
                                                                                                                                              
                                                      
                 
 



                                                                          
                                                                                                                                            
                                                  
                 
 
                                                   
 





                                                       
 
                                  
                 
                                                                                                
                                                                         
                 
                    
                                                                                            
 





                                                                                            
 
                                               
                                                    
 
                                                           







                                                                     



                                                                         
                                                                                                                         


         
                                                            
 
                                      
         





                                                 

                                                                   
                                                                                                                                          
                                                                      
                                                                                                                                             
                                                                      
                                                                                                                                             
                                                                   
                                                                                                                                          








                                                                                                                                                                                                          



                                                                                                         
                                        





                                                                                                                                                                               
                                     


                                                                                            
         

 
                                                                                                                                                      
 


                                                                                         
                                                             
                                                                                 
                                                                 
                 

                                                     
         
                                                                                                                     

                                                     






                                                                                           
                                                               
                                 
                                                                                                                                     


                                              
 
                                                         
                                                                                                     
                                                                                                                                                                                        
 
                                                       
                                                                                                                       
                                                                                                                                                                                                        


                                                         
                                                                                                     
                                                                                  
                                                                                  
                                                                                                                                                                                                                               



                                                                                
                                                                                  
                                                                                                                                                                                                                               


                                                      
                                                                                               
                                                                                                                                                                                                       
                 


         
                                                                          
 
                        
 
                                                                                                                                                         
 




                                                                                                                                                                      
                                                                                                                                
                    
                                                                                                                             

                                                                   

         

                                                                                                         
 
                                                 

                                             
                                         
                                                                      

                                                                                                                                           

                                                                
 
                            
                                                             
                                                                               
 
                           
                                                            

                                                                                              

 


                                                              
                                                                          
                                                                 




                                                                                 
                                                                                  
                                                                          


                        

                                                    
                                    
                                               
         
                                       

                                                                                               

                                                   
                                                                    
                                                                                                                            

                                             
                                                                           
                                              
         
                      

 






                                                    
                                                   



                                                      
                                                             
                                                 




                                 

























                                                                          
                                              
 
                                                                                            

 
                                          


                                                       
                                                             


                                                           


                    
                                          


                                                       
                                                         











                                                                       
                                             
 
                              
                               
         
                                                     



                                                                                                              
                                             


                                                              
                                                                



                                          

 
                                                                

                  

                                               
                                                                                               
                                                             

                                                          
                                                                                                               


                                                     
                                                                                    
            
                                                                        



                                                                                                      
                                                                  




                                                                         
                                                                                                    



                                                                            
                                                                                                                                                                                                             

                 
                                                                       

                                                  
                                                                 
                                                                                                          
         

 

                                                               
                                                                       


                                                       


                                                                                     


         

                                                                                                   
                                                                                     


                                                              



                                                                                                                   


         
                                                                             
 
                                       
                       
 
                                                                                           
         


                                                                                                                
 

                                                       
                                     
           
         

                                                                   
                                                    
                                                                                                    

                                                 
 




                                                                                                   
                 



                                                              
                                                                      


                                   
                                                                                           
         

                                        
                                                                          


         
                                                                                                                                                                             



                                        


                                                         
                                     
         
                   
                 


                                                            
                                                                                                                 
                                                                                                                                                  





                                                                                                                 
                                                                                                           


                                                
                                                                                          
                 
         


           
                                                       

                                                                                     
                                                                                                                                                                                                              
                                               
                                                    



                                   
                                                                                           


                                        
                                                                          

         
 
                                                                           
 

                                                

                                           
                                 

                                                              
                                    

                                                                 
                                    

                                                                 
                              






                                                                            
                                 

                                                              
                                

                                               
                             

                                                          
                             

                                                          
                             

                                                           
                                      
                                                                   
 
                         

 
                  



                                                                          
                                       
         
                                                













                                                                                  

                                            

                                              
                                                          
                 
                                     
                                                   
                 









                                                

                                               

                                           

                                              

                                            

                                          






                                           
                                      
                 
                                                               
                 
                                          













                                                                                 
                 

                                               


         





                                







                                                                       




































                                                                                                                                    










                                                    





                                                           











                                                          
                 
/*
    This file is part of cpp-ethereum.

    cpp-ethereum 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.

    cpp-ethereum 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 cpp-ethereum.  If not, see <http://www.gnu.org/licenses/>.
*/
/** @file TestHelper.cpp
 * @author Marko Simovic <markobarko@gmail.com>
 * @date 2014
 */

#include "TestHelper.h"

#include <thread>
#include <chrono>
#include <libethcore/EthashAux.h>
#include <libethereum/Client.h>
#include <liblll/Compiler.h>
#include <libevm/VMFactory.h>
#include "Stats.h"

using namespace std;
using namespace dev::eth;

namespace dev
{
namespace eth
{

void mine(Client& c, int numBlocks)
{
    auto startBlock = c.blockChain().details().number;

    c.startMining();
    while(c.blockChain().details().number < startBlock + numBlocks)
        std::this_thread::sleep_for(std::chrono::milliseconds(100));
    c.stopMining();
}

void connectClients(Client& c1, Client& c2)
{
    (void)c1;
    (void)c2;
    // TODO: Move to WebThree. eth::Client no longer handles networking.
#if 0
    short c1Port = 20000;
    short c2Port = 21000;
    c1.startNetwork(c1Port);
    c2.startNetwork(c2Port);
    c2.connect("127.0.0.1", c1Port);
#endif
}

void mine(State& s, BlockChain const& _bc)
{
    std::unique_ptr<SealEngineFace> sealer(Ethash::createSealEngine());
    s.commitToMine(_bc);
    Notified<bytes> sealed;
    sealer->onSealGenerated([&](bytes const& sealedHeader){ sealed = sealedHeader; });
    sealer->generateSeal(s.info());
    sealed.waitNot({});
    s.sealBlock(sealed);
}

void mine(Ethash::BlockHeader& _bi)
{
    std::unique_ptr<SealEngineFace> sealer(Ethash::createSealEngine());
    Notified<bytes> sealed;
    sealer->onSealGenerated([&](bytes const& sealedHeader){ sealed = sealedHeader; });
    sealer->generateSeal(_bi);
    sealed.waitNot({});
    _bi = Ethash::BlockHeader(sealed);
}

}

namespace test
{

struct ValueTooLarge: virtual Exception {};
struct MissingFields : virtual Exception {};

bigint const c_max256plus1 = bigint(1) << 256;

ImportTest::ImportTest(json_spirit::mObject& _o, bool isFiller):
    m_statePre(OverlayDB(), eth::BaseState::Empty, Address(_o["env"].get_obj()["currentCoinbase"].get_str())),
    m_statePost(OverlayDB(), eth::BaseState::Empty, Address(_o["env"].get_obj()["currentCoinbase"].get_str())),
    m_TestObject(_o)
{
    importEnv(_o["env"].get_obj());
    importState(_o["pre"].get_obj(), m_statePre);
    importTransaction(_o["transaction"].get_obj());

    if (!isFiller)
    {
        importState(_o["post"].get_obj(), m_statePost);
        m_environment.sub.logs = importLog(_o["logs"].get_array());
    }
}

json_spirit::mObject& ImportTest::makeAllFieldsHex(json_spirit::mObject& _o)
{
    static const set<string> hashes {"bloom" , "coinbase", "hash", "mixHash", "parentHash", "receiptTrie",
                                     "stateRoot", "transactionsTrie", "uncleHash", "currentCoinbase",
                                     "previousHash", "to", "address", "caller", "origin", "secretKey", "data"};

    for (auto& i: _o)
    {
        std::string key = i.first;
        if (hashes.count(key))
            continue;

        std::string str;
        json_spirit::mValue value = i.second;

        if (value.type() == json_spirit::int_type)
            str = toString(value.get_int());
        else if (value.type() == json_spirit::str_type)
            str = value.get_str();
        else continue;

        _o[key] = (str.substr(0, 2) == "0x") ? str : toCompactHex(toInt(str), HexPrefix::Add, 1);
    }
    return _o;
}

void ImportTest::importEnv(json_spirit::mObject& _o)
{
    assert(_o.count("previousHash") > 0);
    assert(_o.count("currentGasLimit") > 0);
    assert(_o.count("currentDifficulty") > 0);
    assert(_o.count("currentTimestamp") > 0);
    assert(_o.count("currentCoinbase") > 0);
    assert(_o.count("currentNumber") > 0);

    RLPStream rlpStream;
    rlpStream.appendList(BlockInfo::BasicFields);

    rlpStream << h256(_o["previousHash"].get_str());
    rlpStream << EmptyListSHA3;
    rlpStream << Address(_o["currentCoinbase"].get_str());
    rlpStream << h256(); // stateRoot
    rlpStream << EmptyTrie; // transactionTrie
    rlpStream << EmptyTrie; // receiptTrie
    rlpStream << LogBloom(); // bloom
    rlpStream << toInt(_o["currentDifficulty"]);
    rlpStream << toInt(_o["currentNumber"]);
    rlpStream << toInt(_o["currentGasLimit"]);
    rlpStream << 0; //gasUsed
    rlpStream << toInt(_o["currentTimestamp"]);
    rlpStream << std::string(); //extra data

    m_environment.currentBlock = BlockInfo(rlpStream.out(), CheckEverything, h256{}, HeaderData);
    m_statePre.m_previousBlock = m_environment.previousBlock;
    m_statePre.m_currentBlock = m_environment.currentBlock;
}

// import state from not fully declared json_spirit::mObject, writing to _stateOptionsMap which fields were defined in json

void ImportTest::importState(json_spirit::mObject& _o, State& _state, stateOptionsMap& _stateOptionsMap)
{
    for (auto& i: _o)
    {
        json_spirit::mObject o = i.second.get_obj();

        ImportStateOptions stateOptions;
        u256 balance = 0;
        u256 nonce = 0;

        if (o.count("balance") > 0)
        {
            stateOptions.m_bHasBalance = true;
            if (bigint(o["balance"].get_str()) >= c_max256plus1)
                BOOST_THROW_EXCEPTION(ValueTooLarge() << errinfo_comment("State 'balance' is equal or greater than 2**256") );
            balance = toInt(o["balance"]);
        }

        if (o.count("nonce") > 0)
        {
            stateOptions.m_bHasNonce = true;
            if (bigint(o["nonce"].get_str()) >= c_max256plus1)
                BOOST_THROW_EXCEPTION(ValueTooLarge() << errinfo_comment("State 'nonce' is equal or greater than 2**256") );
            nonce = toInt(o["nonce"]);
        }

        Address address = Address(i.first);

        bytes code;
        if (o.count("code") > 0)
        {
            code = importCode(o);
            stateOptions.m_bHasCode = true;
        }

        if (!code.empty())
        {
            _state.m_cache[address] = Account(balance, Account::ContractConception);
            _state.m_cache[address].setCode(std::move(code));
        }
        else
            _state.m_cache[address] = Account(balance, Account::NormalCreation);

        if (o.count("storage") > 0)
        {
            stateOptions.m_bHasStorage = true;
            for (auto const& j: o["storage"].get_obj())
                _state.setStorage(address, toInt(j.first), toInt(j.second));
        }

        for (int i = 0; i < nonce; ++i)
            _state.noteSending(address);

        _state.ensureCached(address, false, false);
        _stateOptionsMap[address] = stateOptions;
    }
}

void ImportTest::importState(json_spirit::mObject& _o, State& _state)
{
    stateOptionsMap importedMap;
    importState(_o, _state, importedMap);
    for (auto& stateOptionMap : importedMap)
    {
        //check that every parameter was declared in state object
        if (!stateOptionMap.second.isAllSet())
            BOOST_THROW_EXCEPTION(MissingFields() << errinfo_comment("Import State: Missing state fields!"));
    }
}

void ImportTest::importTransaction(json_spirit::mObject& _o)
{
    if (_o.count("secretKey") > 0)
    {
        assert(_o.count("nonce") > 0);
        assert(_o.count("gasPrice") > 0);
        assert(_o.count("gasLimit") > 0);
        assert(_o.count("to") > 0);
        assert(_o.count("value") > 0);
        assert(_o.count("data") > 0);

        if (bigint(_o["nonce"].get_str()) >= c_max256plus1)
            BOOST_THROW_EXCEPTION(ValueTooLarge() << errinfo_comment("Transaction 'nonce' is equal or greater than 2**256") );
        if (bigint(_o["gasPrice"].get_str()) >= c_max256plus1)
            BOOST_THROW_EXCEPTION(ValueTooLarge() << errinfo_comment("Transaction 'gasPrice' is equal or greater than 2**256") );
        if (bigint(_o["gasLimit"].get_str()) >= c_max256plus1)
            BOOST_THROW_EXCEPTION(ValueTooLarge() << errinfo_comment("Transaction 'gasLimit' is equal or greater than 2**256") );
        if (bigint(_o["value"].get_str()) >= c_max256plus1)
            BOOST_THROW_EXCEPTION(ValueTooLarge() << errinfo_comment("Transaction 'value' is equal or greater than 2**256") );

        m_transaction = _o["to"].get_str().empty() ?
            Transaction(toInt(_o["value"]), toInt(_o["gasPrice"]), toInt(_o["gasLimit"]), importData(_o), toInt(_o["nonce"]), Secret(_o["secretKey"].get_str())) :
            Transaction(toInt(_o["value"]), toInt(_o["gasPrice"]), toInt(_o["gasLimit"]), Address(_o["to"].get_str()), importData(_o), toInt(_o["nonce"]), Secret(_o["secretKey"].get_str()));
    }
    else
    {
        RLPStream transactionRLPStream = createRLPStreamFromTransactionFields(_o);
        RLP transactionRLP(transactionRLPStream.out());
        try
        {
            m_transaction = Transaction(transactionRLP.data(), CheckTransaction::Everything);
        }
        catch (InvalidSignature)
        {
            // create unsigned transaction
            m_transaction = _o["to"].get_str().empty() ?
                Transaction(toInt(_o["value"]), toInt(_o["gasPrice"]), toInt(_o["gasLimit"]), importData(_o), toInt(_o["nonce"])) :
                Transaction(toInt(_o["value"]), toInt(_o["gasPrice"]), toInt(_o["gasLimit"]), Address(_o["to"].get_str()), importData(_o), toInt(_o["nonce"]));
        }
        catch (Exception& _e)
        {
            cnote << "invalid transaction" << boost::diagnostic_information(_e);
        }
    }
}

void ImportTest::checkExpectedState(State const& _stateExpect, State const& _statePost, stateOptionsMap const _expectedStateOptions, WhenError _throw)
{
    #define CHECK(a,b)                      \
        {                                   \
            if (_throw == WhenError::Throw) \
                {TBOOST_CHECK_MESSAGE(a,b);}\
            else                            \
                {TBOOST_WARN_MESSAGE(a,b);} \
        }

    for (auto const& a: _stateExpect.addresses())
    {
        CHECK(_statePost.addressInUse(a.first), "Filling Test: " << a.first << " missing expected address!");
        if (_statePost.addressInUse(a.first))
        {
            ImportStateOptions addressOptions(true);
            if(_expectedStateOptions.size())
            {
                try
                {
                    addressOptions = _expectedStateOptions.at(a.first);
                }
                catch(std::out_of_range const&)
                {
                    TBOOST_ERROR("expectedStateOptions map does not match expectedState in checkExpectedState!");
                    break;
                }
            }

            if (addressOptions.m_bHasBalance)
                CHECK((_stateExpect.balance(a.first) == _statePost.balance(a.first)),
                        "Check State: " << a.first <<  ": incorrect balance " << _statePost.balance(a.first) << ", expected " << _stateExpect.balance(a.first));

            if (addressOptions.m_bHasNonce)
                CHECK((_stateExpect.transactionsFrom(a.first) == _statePost.transactionsFrom(a.first)),
                        "Check State: " << a.first <<  ": incorrect nonce " << _statePost.transactionsFrom(a.first) << ", expected " << _stateExpect.transactionsFrom(a.first));

            if (addressOptions.m_bHasStorage)
            {
                unordered_map<u256, u256> stateStorage = _statePost.storage(a.first);
                for (auto const& s: _stateExpect.storage(a.first))
                    CHECK((stateStorage[s.first] == s.second),
                            "Check State: " << a.first <<  ": incorrect storage [" << s.first << "] = " << toHex(stateStorage[s.first]) << ", expected [" << s.first << "] = " << toHex(s.second));

                //Check for unexpected storage values
                stateStorage = _stateExpect.storage(a.first);
                for (auto const& s: _statePost.storage(a.first))
                    CHECK((stateStorage[s.first] == s.second),
                            "Check State: " << a.first <<  ": incorrect storage [" << s.first << "] = " << toHex(s.second) << ", expected [" << s.first << "] = " << toHex(stateStorage[s.first]));
            }

            if (addressOptions.m_bHasCode)
                CHECK((_stateExpect.code(a.first) == _statePost.code(a.first)),
                        "Check State: " << a.first <<  ": incorrect code '" << toHex(_statePost.code(a.first)) << "', expected '" << toHex(_stateExpect.code(a.first)) << "'");
        }
    }
}

void ImportTest::exportTest(bytes const& _output, State const& _statePost)
{
    // export output

    m_TestObject["out"] = (_output.size() > 4096 && !Options::get().fulloutput) ? "#" + toString(_output.size()) : toHex(_output, 2, HexPrefix::Add);

    // compare expected output with post output
    if (m_TestObject.count("expectOut") > 0)
    {
        std::string warning = "Check State: Error! Unexpected output: " + m_TestObject["out"].get_str() + " Expected: " + m_TestObject["expectOut"].get_str();
        if (Options::get().checkState)
            {TBOOST_CHECK_MESSAGE((m_TestObject["out"].get_str() == m_TestObject["expectOut"].get_str()), warning);}
        else
            TBOOST_WARN_MESSAGE((m_TestObject["out"].get_str() == m_TestObject["expectOut"].get_str()), warning);

        m_TestObject.erase(m_TestObject.find("expectOut"));
    }

    // export logs
    m_TestObject["logs"] = exportLog(_statePost.pending().size() ? _statePost.log(0) : LogEntries());

    // compare expected state with post state
    if (m_TestObject.count("expect") > 0)
    {
        stateOptionsMap stateMap;
        State expectState(OverlayDB(), eth::BaseState::Empty);
        importState(m_TestObject["expect"].get_obj(), expectState, stateMap);
        checkExpectedState(expectState, _statePost, stateMap, Options::get().checkState ? WhenError::Throw : WhenError::DontThrow);
        m_TestObject.erase(m_TestObject.find("expect"));
    }

    // export post state
    m_TestObject["post"] = fillJsonWithState(_statePost);
    m_TestObject["postStateRoot"] = toHex(_statePost.rootHash().asBytes());

    // export pre state
    m_TestObject["pre"] = fillJsonWithState(m_statePre);
    m_TestObject["env"] = makeAllFieldsHex(m_TestObject["env"].get_obj());
    m_TestObject["transaction"] = makeAllFieldsHex(m_TestObject["transaction"].get_obj());
}

json_spirit::mObject fillJsonWithTransaction(Transaction _txn)
{
    json_spirit::mObject txObject;
    txObject["nonce"] = toCompactHex(_txn.nonce(), HexPrefix::Add, 1);
    txObject["data"] = toHex(_txn.data(), 2, HexPrefix::Add);
    txObject["gasLimit"] = toCompactHex(_txn.gas(), HexPrefix::Add, 1);
    txObject["gasPrice"] = toCompactHex(_txn.gasPrice(), HexPrefix::Add, 1);
    txObject["r"] = toCompactHex(_txn.signature().r, HexPrefix::Add, 1);
    txObject["s"] = toCompactHex(_txn.signature().s, HexPrefix::Add, 1);
    txObject["v"] = toCompactHex(_txn.signature().v + 27, HexPrefix::Add, 1);
    txObject["to"] = _txn.isCreation() ? "" : toString(_txn.receiveAddress());
    txObject["value"] = toCompactHex(_txn.value(), HexPrefix::Add, 1);
    return txObject;
}

json_spirit::mObject fillJsonWithState(State _state)
{
    json_spirit::mObject oState;
    for (auto const& a: _state.addresses())
    {
        json_spirit::mObject o;
        o["balance"] = toCompactHex(_state.balance(a.first), HexPrefix::Add, 1);
        o["nonce"] = toCompactHex(_state.transactionsFrom(a.first), HexPrefix::Add, 1);
        {
            json_spirit::mObject store;
            for (auto const& s: _state.storage(a.first))
                store[toCompactHex(s.first, HexPrefix::Add, 1)] = toCompactHex(s.second, HexPrefix::Add, 1);
            o["storage"] = store;
        }
        o["code"] = toHex(_state.code(a.first), 2, HexPrefix::Add);
        oState[toString(a.first)] = o;
    }
    return oState;
}

json_spirit::mArray exportLog(eth::LogEntries _logs)
{
    json_spirit::mArray ret;
    if (_logs.size() == 0) return ret;
    for (LogEntry const& l: _logs)
    {
        json_spirit::mObject o;
        o["address"] = toString(l.address);
        json_spirit::mArray topics;
        for (auto const& t: l.topics)
            topics.push_back(toString(t));
        o["topics"] = topics;
        o["data"] = toHex(l.data, 2, HexPrefix::Add);
        o["bloom"] = toString(l.bloom());
        ret.push_back(o);
    }
    return ret;
}

u256 toInt(json_spirit::mValue const& _v)
{
    switch (_v.type())
    {
    case json_spirit::str_type: return u256(_v.get_str());
    case json_spirit::int_type: return (u256)_v.get_uint64();
    case json_spirit::bool_type: return (u256)(uint64_t)_v.get_bool();
    case json_spirit::real_type: return (u256)(uint64_t)_v.get_real();
    default: cwarn << "Bad type for scalar: " << _v.type();
    }
    return 0;
}

byte toByte(json_spirit::mValue const& _v)
{
    switch (_v.type())
    {
    case json_spirit::str_type: return (byte)stoi(_v.get_str());
    case json_spirit::int_type: return (byte)_v.get_uint64();
    case json_spirit::bool_type: return (byte)_v.get_bool();
    case json_spirit::real_type: return (byte)_v.get_real();
    default: cwarn << "Bad type for scalar: " << _v.type();
    }
    return 0;
}

bytes importByteArray(std::string const& _str)
{
    return fromHex(_str.substr(0, 2) == "0x" ? _str.substr(2) : _str, WhenError::Throw);
}

bytes importData(json_spirit::mObject& _o)
{
    bytes data;
    if (_o["data"].type() == json_spirit::str_type)
        data = importByteArray(_o["data"].get_str());
    else
        for (auto const& j: _o["data"].get_array())
            data.push_back(toByte(j));
    return data;
}

bytes importCode(json_spirit::mObject& _o)
{
    bytes code;
    if (_o["code"].type() == json_spirit::str_type)
        if (_o["code"].get_str().find("0x") != 0)
            code = compileLLL(_o["code"].get_str(), false);
        else
            code = fromHex(_o["code"].get_str().substr(2));
    else if (_o["code"].type() == json_spirit::array_type)
    {
        code.clear();
        for (auto const& j: _o["code"].get_array())
            code.push_back(toByte(j));
    }
    return code;
}

LogEntries importLog(json_spirit::mArray& _a)
{
    LogEntries logEntries;
    for (auto const& l: _a)
    {
        json_spirit::mObject o = l.get_obj();
        // cant use BOOST_REQUIRE, because this function is used outside boost test (createRandomTest)
        assert(o.count("address") > 0);
        assert(o.count("topics") > 0);
        assert(o.count("data") > 0);
        assert(o.count("bloom") > 0);
        LogEntry log;
        log.address = Address(o["address"].get_str());
        for (auto const& t: o["topics"].get_array())
            log.topics.push_back(h256(t.get_str()));
        log.data = importData(o);
        logEntries.push_back(log);
    }
    return logEntries;
}

void checkOutput(bytes const& _output, json_spirit::mObject& _o)
{
    int j = 0;

    if (_o["out"].get_str().find("#") == 0)
        {TBOOST_CHECK(((u256)_output.size() == toInt(_o["out"].get_str().substr(1))));}
    else if (_o["out"].type() == json_spirit::array_type)
        for (auto const& d: _o["out"].get_array())
        {
            TBOOST_CHECK_MESSAGE((_output[j] == toInt(d)), "Output byte [" << j << "] different!");
            ++j;
        }
    else if (_o["out"].get_str().find("0x") == 0)
        {TBOOST_CHECK((_output == fromHex(_o["out"].get_str().substr(2))));}
    else
        TBOOST_CHECK((_output == fromHex(_o["out"].get_str())));
}

void checkStorage(map<u256, u256> _expectedStore, map<u256, u256> _resultStore, Address _expectedAddr)
{
    _expectedAddr = _expectedAddr; //unsed parametr when macro
    for (auto&& expectedStorePair : _expectedStore)
    {
        auto& expectedStoreKey = expectedStorePair.first;
        auto resultStoreIt = _resultStore.find(expectedStoreKey);
        if (resultStoreIt == _resultStore.end())
            {TBOOST_ERROR(_expectedAddr << ": missing store key " << expectedStoreKey);}
        else
        {
            auto& expectedStoreValue = expectedStorePair.second;
            auto& resultStoreValue = resultStoreIt->second;
            TBOOST_CHECK_MESSAGE((expectedStoreValue == resultStoreValue), _expectedAddr << ": store[" << expectedStoreKey << "] = " << resultStoreValue << ", expected " << expectedStoreValue);
        }
    }
    TBOOST_CHECK_EQUAL(_resultStore.size(), _expectedStore.size());
    for (auto&& resultStorePair: _resultStore)
    {
        if (!_expectedStore.count(resultStorePair.first))
            TBOOST_ERROR(_expectedAddr << ": unexpected store key " << resultStorePair.first);
    }
}

void checkLog(LogEntries _resultLogs, LogEntries _expectedLogs)
{
    TBOOST_REQUIRE_EQUAL(_resultLogs.size(), _expectedLogs.size());

    for (size_t i = 0; i < _resultLogs.size(); ++i)
    {
        TBOOST_CHECK_EQUAL(_resultLogs[i].address, _expectedLogs[i].address);
        TBOOST_CHECK_EQUAL(_resultLogs[i].topics, _expectedLogs[i].topics);
        TBOOST_CHECK((_resultLogs[i].data == _expectedLogs[i].data));
    }
}

void checkCallCreates(eth::Transactions _resultCallCreates, eth::Transactions _expectedCallCreates)
{
    TBOOST_REQUIRE_EQUAL(_resultCallCreates.size(), _expectedCallCreates.size());

    for (size_t i = 0; i < _resultCallCreates.size(); ++i)
    {
        TBOOST_CHECK((_resultCallCreates[i].data() == _expectedCallCreates[i].data()));
        TBOOST_CHECK((_resultCallCreates[i].receiveAddress() == _expectedCallCreates[i].receiveAddress()));
        TBOOST_CHECK((_resultCallCreates[i].gas() == _expectedCallCreates[i].gas()));
        TBOOST_CHECK((_resultCallCreates[i].value() == _expectedCallCreates[i].value()));
    }
}

void userDefinedTest(std::function<void(json_spirit::mValue&, bool)> doTests)
{
    if (!Options::get().singleTest)
        return;

    if (Options::get().singleTestFile.empty() || Options::get().singleTestName.empty())
    {
        cnote << "Missing user test specification\nUsage: testeth --singletest <filename> <testname>\n";
        return;
    }

    auto& filename = Options::get().singleTestFile;
    auto& testname = Options::get().singleTestName;
    VerbosityHolder sentinel(12);
    try
    {
        cnote << "Testing user defined test: " << filename;
        json_spirit::mValue v;
        string s = contentsString(filename);
        TBOOST_REQUIRE_MESSAGE((s.length() > 0), "Contents of " + filename + " is empty. ");
        json_spirit::read_string(s, v);
        json_spirit::mObject oSingleTest;

        json_spirit::mObject::const_iterator pos = v.get_obj().find(testname);
        if (pos == v.get_obj().end())
        {
            cnote << "Could not find test: " << testname << " in " << filename << "\n";
            return;
        }
        else
            oSingleTest[pos->first] = pos->second;

        json_spirit::mValue v_singleTest(oSingleTest);
        doTests(v_singleTest, test::Options::get().fillTests);
    }
    catch (Exception const& _e)
    {
        TBOOST_ERROR("Failed Test with Exception: " << diagnostic_information(_e));
    }
    catch (std::exception const& _e)
    {
        TBOOST_ERROR("Failed Test with Exception: " << _e.what());
    }
}

void executeTests(const string& _name, const string& _testPathAppendix, const boost::filesystem::path _pathToFiller, std::function<void(json_spirit::mValue&, bool)> doTests)
{
    string testPath = getTestPath();
    testPath += _testPathAppendix;

    if (Options::get().stats)
        Listener::registerListener(Stats::get());

    if (Options::get().fillTests)
    {
        try
        {
            cnote << "Populating tests...";
            json_spirit::mValue v;
            boost::filesystem::path p(__FILE__);
            string s = asString(dev::contents(_pathToFiller.string() + "/" + _name + "Filler.json"));
            TBOOST_REQUIRE_MESSAGE((s.length() > 0), "Contents of " + _pathToFiller.string() + "/" + _name + "Filler.json is empty.");
            json_spirit::read_string(s, v);
            doTests(v, true);
            writeFile(testPath + "/" + _name + ".json", asBytes(json_spirit::write_string(v, true)));
        }
        catch (Exception const& _e)
        {
            TBOOST_ERROR("Failed filling test with Exception: " << diagnostic_information(_e));
        }
        catch (std::exception const& _e)
        {
            TBOOST_ERROR("Failed filling test with Exception: " << _e.what());
        }
    }

    try
    {
        std::cout << "TEST " << _name << ":\n";
        json_spirit::mValue v;
        string s = asString(dev::contents(testPath + "/" + _name + ".json"));
        TBOOST_REQUIRE_MESSAGE((s.length() > 0), "Contents of " + testPath + "/" + _name + ".json is empty. Have you cloned the 'tests' repo branch develop and set ETHEREUM_TEST_PATH to its path?");
        json_spirit::read_string(s, v);
        Listener::notifySuiteStarted(_name);
        doTests(v, false);
    }
    catch (Exception const& _e)
    {
        TBOOST_ERROR("Failed test with Exception: " << diagnostic_information(_e));
    }
    catch (std::exception const& _e)
    {
        TBOOST_ERROR("Failed test with Exception: " << _e.what());
    }
}

RLPStream createRLPStreamFromTransactionFields(json_spirit::mObject& _tObj)
{
    //Construct Rlp of the given transaction
    RLPStream rlpStream;
    rlpStream.appendList(_tObj.size());

    if (_tObj.count("nonce"))
        rlpStream << bigint(_tObj["nonce"].get_str());

    if (_tObj.count("gasPrice"))
        rlpStream << bigint(_tObj["gasPrice"].get_str());

    if (_tObj.count("gasLimit"))
        rlpStream << bigint(_tObj["gasLimit"].get_str());

    if (_tObj.count("to"))
    {
        if (_tObj["to"].get_str().empty())
            rlpStream << "";
        else
            rlpStream << importByteArray(_tObj["to"].get_str());
    }

    if (_tObj.count("value"))
        rlpStream << bigint(_tObj["value"].get_str());

    if (_tObj.count("data"))
        rlpStream << importData(_tObj);

    if (_tObj.count("v"))
        rlpStream << bigint(_tObj["v"].get_str());

    if (_tObj.count("r"))
        rlpStream << bigint(_tObj["r"].get_str());

    if (_tObj.count("s"))
        rlpStream <<  bigint(_tObj["s"].get_str());

    if (_tObj.count("extrafield"))
        rlpStream << bigint(_tObj["extrafield"].get_str());

    return rlpStream;
}

Options::Options()
{
    auto argc = boost::unit_test::framework::master_test_suite().argc;
    auto argv = boost::unit_test::framework::master_test_suite().argv;

    for (auto i = 0; i < argc; ++i)
    {
        auto arg = std::string{argv[i]};
        if (arg == "--vm" && i + 1 < argc)
        {
            string vmKind = argv[++i];
            if (vmKind == "interpreter")
                VMFactory::setKind(VMKind::Interpreter);
            else if (vmKind == "jit")
                VMFactory::setKind(VMKind::JIT);
            else if (vmKind == "smart")
                VMFactory::setKind(VMKind::Smart);
            else
                cerr << "Unknown VM kind: " << vmKind << endl;
        }
        else if (arg == "--jit") // TODO: Remove deprecated option "--jit"
            VMFactory::setKind(VMKind::JIT);
        else if (arg == "--vmtrace")
            vmtrace = true;
        else if (arg == "--filltests")
            fillTests = true;
        else if (arg == "--stats" && i + 1 < argc)
        {
            stats = true;
            statsOutFile = argv[i + 1];
        }
        else if (arg == "--performance")
            performance = true;
        else if (arg == "--quadratic")
            quadratic = true;
        else if (arg == "--memory")
            memory = true;
        else if (arg == "--inputlimits")
            inputLimits = true;
        else if (arg == "--bigdata")
            bigData = true;
        else if (arg == "--checkstate")
            checkState = true;
        else if (arg == "--wallet")
            wallet = true;
        else if (arg == "--nonetwork")
            nonetwork = true;
        else if (arg == "--network")
            nonetwork = false;
        else if (arg == "--nodag")
            nodag = true;
        else if (arg == "--all")
        {
            performance = true;
            quadratic = true;
            memory = true;
            inputLimits = true;
            bigData = true;
            wallet = true;
        }
        else if (arg == "--singletest" && i + 1 < argc)
        {
            singleTest = true;
            auto name1 = std::string{argv[i + 1]};
            if (i + 1 < argc) // two params
            {
                auto name2 = std::string{argv[i + 2]};
                if (name2[0] == '-') // not param, another option
                    singleTestName = std::move(name1);
                else
                {
                    singleTestFile = std::move(name1);
                    singleTestName = std::move(name2);
                }
            }
            else
                singleTestName = std::move(name1);
        }
        else if (arg == "--fulloutput")
            fulloutput = true;
    }
}

Options const& Options::get()
{
    static Options instance;
    return instance;
}

LastHashes lastHashes(u256 _currentBlockNumber)
{
    LastHashes ret;
    for (u256 i = 1; i <= 256 && i <= _currentBlockNumber; ++i)
        ret.push_back(sha3(toString(_currentBlockNumber - i)));
    return ret;
}

dev::eth::Ethash::BlockHeader constructHeader(
    h256 const& _parentHash,
    h256 const& _sha3Uncles,
    Address const& _coinbaseAddress,
    h256 const& _stateRoot,
    h256 const& _transactionsRoot,
    h256 const& _receiptsRoot,
    dev::eth::LogBloom const& _logBloom,
    u256 const& _difficulty,
    u256 const& _number,
    u256 const& _gasLimit,
    u256 const& _gasUsed,
    u256 const& _timestamp,
    bytes const& _extraData)
{
    RLPStream rlpStream;
    rlpStream.appendList(Ethash::BlockHeader::Fields);

    rlpStream << _parentHash << _sha3Uncles << _coinbaseAddress << _stateRoot << _transactionsRoot << _receiptsRoot << _logBloom
        << _difficulty << _number << _gasLimit << _gasUsed << _timestamp << _extraData << h256{} << Nonce{};

    return Ethash::BlockHeader(rlpStream.out());
}

void updateEthashSeal(dev::eth::Ethash::BlockHeader& _header, h256 const& _mixHash, dev::eth::Nonce const& _nonce)
{
    RLPStream source;
    _header.streamRLP(source);
    RLP sourceRlp(source.out());
    RLPStream header;
    header.appendList(Ethash::BlockHeader::Fields);
    for (size_t i = 0; i < BlockInfo::BasicFields; i++)
        header << sourceRlp[i];

    header << _mixHash << _nonce;
    _header = Ethash::BlockHeader(header.out());
}

namespace
{
    Listener* g_listener;
}

void Listener::registerListener(Listener& _listener)
{
    g_listener = &_listener;
}

void Listener::notifySuiteStarted(std::string const& _name)
{
    if (g_listener)
        g_listener->suiteStarted(_name);
}

void Listener::notifyTestStarted(std::string const& _name)
{
    if (g_listener)
        g_listener->testStarted(_name);
}

void Listener::notifyTestFinished()
{
    if (g_listener)
        g_listener->testFinished();
}

} } // namespaces