You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
343 lines
8.9 KiB
343 lines
8.9 KiB
/*
|
|
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 EthereumHost.cpp
|
|
* @author Gav Wood <i@gavwood.com>
|
|
* @date 2014
|
|
*/
|
|
|
|
#include "EthereumHost.h"
|
|
|
|
#include <chrono>
|
|
#include <thread>
|
|
#include <libdevcore/Common.h>
|
|
#include <libp2p/Host.h>
|
|
#include <libp2p/Session.h>
|
|
#include <libethcore/Exceptions.h>
|
|
#include <libethcore/Params.h>
|
|
#include "BlockChain.h"
|
|
#include "TransactionQueue.h"
|
|
#include "BlockQueue.h"
|
|
#include "EthereumPeer.h"
|
|
#include "DownloadMan.h"
|
|
#include "BlockChainSync.h"
|
|
|
|
using namespace std;
|
|
using namespace dev;
|
|
using namespace dev::eth;
|
|
using namespace p2p;
|
|
|
|
unsigned const EthereumHost::c_oldProtocolVersion = 60; //TODO: remove this once v61+ is common
|
|
unsigned const c_chainReorgSize = 30000;
|
|
|
|
char const* const EthereumHost::s_stateNames[static_cast<int>(SyncState::Size)] = {"Idle", "Waiting", "Hashes", "Blocks", "NewBlocks" };
|
|
|
|
EthereumHost::EthereumHost(BlockChain const& _ch, TransactionQueue& _tq, BlockQueue& _bq, u256 _networkId):
|
|
HostCapability<EthereumPeer>(),
|
|
Worker ("ethsync"),
|
|
m_chain (_ch),
|
|
m_tq (_tq),
|
|
m_bq (_bq),
|
|
m_networkId (_networkId)
|
|
{
|
|
m_latestBlockSent = _ch.currentHash();
|
|
}
|
|
|
|
EthereumHost::~EthereumHost()
|
|
{
|
|
}
|
|
|
|
bool EthereumHost::ensureInitialised()
|
|
{
|
|
if (!m_latestBlockSent)
|
|
{
|
|
// First time - just initialise.
|
|
m_latestBlockSent = m_chain.currentHash();
|
|
clog(NetNote) << "Initialising: latest=" << m_latestBlockSent;
|
|
|
|
for (auto const& i: m_tq.transactions())
|
|
m_transactionsSent.insert(i.first);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void EthereumHost::reset()
|
|
{
|
|
Guard l(x_sync);
|
|
if (m_sync)
|
|
m_sync->abortSync();
|
|
m_sync.reset();
|
|
|
|
m_latestBlockSent = h256();
|
|
m_transactionsSent.clear();
|
|
}
|
|
|
|
void EthereumHost::doWork()
|
|
{
|
|
bool netChange = ensureInitialised();
|
|
auto h = m_chain.currentHash();
|
|
// If we've finished our initial sync (including getting all the blocks into the chain so as to reduce invalid transactions), start trading transactions & blocks
|
|
if (!isSyncing() && m_chain.isKnown(m_latestBlockSent))
|
|
{
|
|
if (m_newTransactions)
|
|
{
|
|
m_newTransactions = false;
|
|
maintainTransactions();
|
|
}
|
|
if (m_newBlocks)
|
|
{
|
|
m_newBlocks = false;
|
|
maintainBlocks(h);
|
|
}
|
|
}
|
|
|
|
foreachPeer([](EthereumPeer* _p) { _p->tick(); return true; });
|
|
|
|
// return netChange;
|
|
// TODO: Figure out what to do with netChange.
|
|
(void)netChange;
|
|
}
|
|
|
|
void EthereumHost::maintainTransactions()
|
|
{
|
|
// Send any new transactions.
|
|
unordered_map<std::shared_ptr<EthereumPeer>, h256s> peerTransactions;
|
|
auto ts = m_tq.transactions();
|
|
for (auto const& i: ts)
|
|
{
|
|
bool unsent = !m_transactionsSent.count(i.first);
|
|
auto peers = get<1>(randomSelection(0, [&](EthereumPeer* p) { return p->m_requireTransactions || (unsent && !p->m_knownTransactions.count(i.first)); }));
|
|
for (auto const& p: peers)
|
|
peerTransactions[p].push_back(i.first);
|
|
}
|
|
for (auto const& t: ts)
|
|
m_transactionsSent.insert(t.first);
|
|
foreachPeerPtr([&](shared_ptr<EthereumPeer> _p)
|
|
{
|
|
bytes b;
|
|
unsigned n = 0;
|
|
for (auto const& h: peerTransactions[_p])
|
|
{
|
|
_p->m_knownTransactions.insert(h);
|
|
b += ts[h].rlp();
|
|
++n;
|
|
}
|
|
|
|
_p->clearKnownTransactions();
|
|
|
|
if (n || _p->m_requireTransactions)
|
|
{
|
|
RLPStream ts;
|
|
_p->prep(ts, TransactionsPacket, n).appendRaw(b, n);
|
|
_p->sealAndSend(ts);
|
|
cnote << "Sent" << n << "transactions to " << _p->session()->info().clientVersion;
|
|
}
|
|
_p->m_requireTransactions = false;
|
|
return true;
|
|
});
|
|
}
|
|
|
|
void EthereumHost::foreachPeer(std::function<bool(EthereumPeer*)> const& _f) const
|
|
{
|
|
foreachPeerPtr([&](std::shared_ptr<EthereumPeer> _p)
|
|
{
|
|
if (_p)
|
|
return _f(_p.get());
|
|
return true;
|
|
});
|
|
}
|
|
|
|
void EthereumHost::foreachPeerPtr(std::function<bool(std::shared_ptr<EthereumPeer>)> const& _f) const
|
|
{
|
|
for (auto s: peerSessions())
|
|
if (!_f(s.first->cap<EthereumPeer>()))
|
|
return;
|
|
for (auto s: peerSessions(c_oldProtocolVersion)) //TODO: remove once v61+ is common
|
|
if (!_f(s.first->cap<EthereumPeer>(c_oldProtocolVersion)))
|
|
return;
|
|
}
|
|
|
|
tuple<vector<shared_ptr<EthereumPeer>>, vector<shared_ptr<EthereumPeer>>, vector<shared_ptr<Session>>> EthereumHost::randomSelection(unsigned _percent, std::function<bool(EthereumPeer*)> const& _allow)
|
|
{
|
|
vector<shared_ptr<EthereumPeer>> chosen;
|
|
vector<shared_ptr<EthereumPeer>> allowed;
|
|
vector<shared_ptr<Session>> sessions;
|
|
|
|
auto const& ps = peerSessions();
|
|
allowed.reserve(ps.size());
|
|
for (auto const& j: ps)
|
|
{
|
|
auto pp = j.first->cap<EthereumPeer>();
|
|
if (_allow(pp.get()))
|
|
{
|
|
allowed.push_back(move(pp));
|
|
sessions.push_back(move(j.first));
|
|
}
|
|
}
|
|
|
|
chosen.reserve((ps.size() * _percent + 99) / 100);
|
|
for (unsigned i = (ps.size() * _percent + 99) / 100; i-- && allowed.size();)
|
|
{
|
|
unsigned n = rand() % allowed.size();
|
|
chosen.push_back(std::move(allowed[n]));
|
|
allowed.erase(allowed.begin() + n);
|
|
}
|
|
return make_tuple(move(chosen), move(allowed), move(sessions));
|
|
}
|
|
|
|
void EthereumHost::maintainBlocks(h256 const& _currentHash)
|
|
{
|
|
// Send any new blocks.
|
|
auto detailsFrom = m_chain.details(m_latestBlockSent);
|
|
auto detailsTo = m_chain.details(_currentHash);
|
|
if (detailsFrom.totalDifficulty < detailsTo.totalDifficulty)
|
|
{
|
|
if (diff(detailsFrom.number, detailsTo.number) < 20)
|
|
{
|
|
// don't be sending more than 20 "new" blocks. if there are any more we were probably waaaay behind.
|
|
clog(NetMessageSummary) << "Sending a new block (current is" << _currentHash << ", was" << m_latestBlockSent << ")";
|
|
|
|
h256s blocks = get<0>(m_chain.treeRoute(m_latestBlockSent, _currentHash, false, false, true));
|
|
|
|
auto s = randomSelection(25, [&](EthereumPeer* p){ DEV_GUARDED(p->x_knownBlocks) return !p->m_knownBlocks.count(_currentHash); return false; });
|
|
for (shared_ptr<EthereumPeer> const& p: get<0>(s))
|
|
for (auto const& b: blocks)
|
|
{
|
|
RLPStream ts;
|
|
p->prep(ts, NewBlockPacket, 2).appendRaw(m_chain.block(b), 1).append(m_chain.details(b).totalDifficulty);
|
|
|
|
Guard l(p->x_knownBlocks);
|
|
p->sealAndSend(ts);
|
|
p->m_knownBlocks.clear();
|
|
}
|
|
for (shared_ptr<EthereumPeer> const& p: get<1>(s))
|
|
{
|
|
RLPStream ts;
|
|
p->prep(ts, NewBlockHashesPacket, blocks.size());
|
|
for (auto const& b: blocks)
|
|
ts.append(b);
|
|
|
|
Guard l(p->x_knownBlocks);
|
|
p->sealAndSend(ts);
|
|
p->m_knownBlocks.clear();
|
|
}
|
|
}
|
|
m_latestBlockSent = _currentHash;
|
|
}
|
|
}
|
|
|
|
BlockChainSync& EthereumHost::sync()
|
|
{
|
|
if (m_sync)
|
|
return *m_sync; // We only chose sync strategy once
|
|
|
|
bool pv61 = false;
|
|
foreachPeer([&](EthereumPeer* _p)
|
|
{
|
|
if (_p->m_protocolVersion == protocolVersion())
|
|
pv61 = true;
|
|
return !pv61;
|
|
});
|
|
m_sync.reset(pv61 ? new PV60Sync(*this) : new PV60Sync(*this));
|
|
return *m_sync;
|
|
}
|
|
|
|
void EthereumHost::onPeerStatus(EthereumPeer* _peer)
|
|
{
|
|
Guard l(x_sync);
|
|
sync().onPeerStatus(_peer);
|
|
}
|
|
|
|
void EthereumHost::onPeerHashes(EthereumPeer* _peer, h256s const& _hashes)
|
|
{
|
|
Guard l(x_sync);
|
|
sync().onPeerHashes(_peer, _hashes);
|
|
}
|
|
|
|
void EthereumHost::onPeerBlocks(EthereumPeer* _peer, RLP const& _r)
|
|
{
|
|
Guard l(x_sync);
|
|
sync().onPeerBlocks(_peer, _r);
|
|
}
|
|
|
|
void EthereumHost::onPeerNewHashes(EthereumPeer* _peer, h256s const& _hashes)
|
|
{
|
|
Guard l(x_sync);
|
|
sync().onPeerNewHashes(_peer, _hashes);
|
|
}
|
|
|
|
void EthereumHost::onPeerNewBlock(EthereumPeer* _peer, RLP const& _r)
|
|
{
|
|
Guard l(x_sync);
|
|
sync().onPeerNewBlock(_peer, _r);
|
|
}
|
|
|
|
void EthereumHost::onPeerTransactions(EthereumPeer* _peer, RLP const& _r)
|
|
{
|
|
if (_peer->isCriticalSyncing())
|
|
{
|
|
clog(NetAllDetail) << "Ignoring transaction from peer we are syncing with";
|
|
return;
|
|
}
|
|
unsigned itemCount = _r.itemCount();
|
|
clog(NetAllDetail) << "Transactions (" << dec << itemCount << "entries)";
|
|
Guard l(_peer->x_knownTransactions);
|
|
for (unsigned i = 0; i < itemCount; ++i)
|
|
{
|
|
auto h = sha3(_r[i].data());
|
|
_peer->m_knownTransactions.insert(h);
|
|
ImportResult ir = m_tq.import(_r[i].data());
|
|
switch (ir)
|
|
{
|
|
case ImportResult::Malformed:
|
|
_peer->addRating(-100);
|
|
break;
|
|
case ImportResult::AlreadyKnown:
|
|
// if we already had the transaction, then don't bother sending it on.
|
|
m_transactionsSent.insert(h);
|
|
_peer->addRating(0);
|
|
break;
|
|
case ImportResult::Success:
|
|
_peer->addRating(100);
|
|
break;
|
|
default:;
|
|
}
|
|
}
|
|
}
|
|
|
|
void EthereumHost::onPeerAborting(EthereumPeer* _peer)
|
|
{
|
|
Guard l(x_sync);
|
|
if (m_sync)
|
|
m_sync->onPeerAborting(_peer);
|
|
}
|
|
|
|
bool EthereumHost::isSyncing() const
|
|
{
|
|
Guard l(x_sync);
|
|
if (!m_sync)
|
|
return false;
|
|
return m_sync->isSyncing();
|
|
}
|
|
|
|
SyncStatus EthereumHost::status() const
|
|
{
|
|
Guard l(x_sync);
|
|
if (!m_sync)
|
|
return SyncStatus();
|
|
return m_sync->status();
|
|
}
|
|
|