@ -43,33 +43,31 @@ NodeEntry::NodeEntry(NodeId const& _src, Public const& _pubk, NodeIPEndpoint con
NodeTable : : NodeTable ( ba : : io_service & _io , KeyPair const & _alias , NodeIPEndpoint const & _endpoint , bool _enabled ) :
NodeTable : : NodeTable ( ba : : io_service & _io , KeyPair const & _alias , NodeIPEndpoint const & _endpoint , bool _enabled ) :
m_node ( Node ( _alias . pub ( ) , _endpoint ) ) ,
m_node ( Node ( _alias . pub ( ) , _endpoint ) ) ,
m_secret ( _alias . sec ( ) ) ,
m_secret ( _alias . sec ( ) ) ,
m_io ( _io ) ,
m_socket ( new NodeSocket ( _io , * this , ( bi : : udp : : endpoint ) m_node . endpoint ) ) ,
m_socket ( new NodeSocket ( m_io , * this , ( bi : : udp : : endpoint ) m_node . endpoint ) ) ,
m_socketPointer ( m_socket . get ( ) ) ,
m_socketPointer ( m_socket . get ( ) ) ,
m_bucketRefreshTimer ( m_io ) ,
m_timers ( _io )
m_evictionCheckTimer ( m_io ) ,
m_disabled ( ! _enabled )
{
{
for ( unsigned i = 0 ; i < s_bins ; i + + )
for ( unsigned i = 0 ; i < s_bins ; i + + )
{
m_state [ i ] . distance = i ;
m_state [ i ] . distance = i ;
m_state [ i ] . modified = chrono : : steady_clock : : now ( ) - chrono : : seconds ( 1 ) ;
}
if ( ! m_disabled )
if ( ! _enabled )
return ;
try
{
{
m_socketPointer - > connect ( ) ;
m_socketPointer - > connect ( ) ;
doRefreshBuckets ( boost : : system : : error_code ( ) ) ;
doDiscovery ( ) ;
}
catch ( std : : exception const & _e )
{
clog ( NetWarn ) < < " Exception connecting NodeTable socket: " < < _e . what ( ) ;
clog ( NetWarn ) < < " Discovery disabled. " ;
}
}
}
}
NodeTable : : ~ NodeTable ( )
NodeTable : : ~ NodeTable ( )
{
{
// Cancel scheduled tasks to ensure.
m_timers . stop ( ) ;
m_evictionCheckTimer . cancel ( ) ;
m_bucketRefreshTimer . cancel ( ) ;
// Disconnect socket so that deallocation is safe.
m_socketPointer - > disconnect ( ) ;
m_socketPointer - > disconnect ( ) ;
}
}
@ -117,16 +115,6 @@ shared_ptr<NodeEntry> NodeTable::addNode(Node const& _node, NodeRelation _relati
return ret ;
return ret ;
}
}
void NodeTable : : discover ( )
{
static chrono : : steady_clock : : time_point s_lastDiscover = chrono : : steady_clock : : now ( ) - std : : chrono : : seconds ( 30 ) ;
if ( chrono : : steady_clock : : now ( ) > s_lastDiscover + std : : chrono : : seconds ( 30 ) )
{
s_lastDiscover = chrono : : steady_clock : : now ( ) ;
discover ( m_node . id ) ;
}
}
list < NodeId > NodeTable : : nodes ( ) const
list < NodeId > NodeTable : : nodes ( ) const
{
{
list < NodeId > nodes ;
list < NodeId > nodes ;
@ -164,14 +152,17 @@ shared_ptr<NodeEntry> NodeTable::nodeEntry(NodeId _id)
return m_nodes . count ( _id ) ? m_nodes [ _id ] : shared_ptr < NodeEntry > ( ) ;
return m_nodes . count ( _id ) ? m_nodes [ _id ] : shared_ptr < NodeEntry > ( ) ;
}
}
void NodeTable : : discover ( NodeId _node , unsigned _round , shared_ptr < set < shared_ptr < NodeEntry > > > _tried )
void NodeTable : : doD iscover ( NodeId _node , unsigned _round , shared_ptr < set < shared_ptr < NodeEntry > > > _tried )
{
{
if ( ! m_socketPointer - > isOpen ( ) | | _round = = s_maxSteps )
// NOTE: ONLY called by doDiscovery!
if ( ! m_socketPointer - > isOpen ( ) )
return ;
return ;
if ( _round = = s_maxSteps )
if ( _round = = s_maxSteps )
{
{
clog ( NodeTableEvent ) < < " Terminating discover after " < < _round < < " rounds. " ;
clog ( NodeTableEvent ) < < " Terminating discover after " < < _round < < " rounds. " ;
doDiscovery ( ) ;
return ;
return ;
}
}
else if ( ! _round & & ! _tried )
else if ( ! _round & & ! _tried )
@ -195,6 +186,7 @@ void NodeTable::discover(NodeId _node, unsigned _round, shared_ptr<set<shared_pt
if ( tried . empty ( ) )
if ( tried . empty ( ) )
{
{
clog ( NodeTableEvent ) < < " Terminating discover after " < < _round < < " rounds. " ;
clog ( NodeTableEvent ) < < " Terminating discover after " < < _round < < " rounds. " ;
doDiscovery ( ) ;
return ;
return ;
}
}
@ -204,13 +196,11 @@ void NodeTable::discover(NodeId _node, unsigned _round, shared_ptr<set<shared_pt
tried . pop_front ( ) ;
tried . pop_front ( ) ;
}
}
auto self ( shared_from_this ( ) ) ;
m_timers . schedule ( c_reqTimeout . count ( ) * 2 , [ this , _node , _round , _tried ] ( boost : : system : : error_code const & _ec )
m_evictionCheckTimer . expires_from_now ( boost : : posix_time : : milliseconds ( c_reqTimeout . count ( ) * 2 ) ) ;
m_evictionCheckTimer . async_wait ( [ this , self , _node , _round , _tried ] ( boost : : system : : error_code const & _ec )
{
{
if ( _ec )
if ( _ec )
return ;
clog ( NodeTableWarn ) < < " Discovery timer canceled! " ;
discover ( _node , _round + 1 , _tried ) ;
doD iscover ( _node , _round + 1 , _tried ) ;
} ) ;
} ) ;
}
}
@ -310,15 +300,15 @@ void NodeTable::evict(shared_ptr<NodeEntry> _leastSeen, shared_ptr<NodeEntry> _n
if ( ! m_socketPointer - > isOpen ( ) )
if ( ! m_socketPointer - > isOpen ( ) )
return ;
return ;
unsigned ec ;
unsigned evi cts ;
DEV_GUARDED ( x_evictions )
DEV_GUARDED ( x_evictions )
{
{
m_evictions . push_back ( EvictionTimeout ( make_pair ( _leastSeen - > id , chrono : : steady_clock : : now ( ) ) , _new - > id ) ) ;
m_evictions . push_back ( EvictionTimeout ( make_pair ( _leastSeen - > id , chrono : : steady_clock : : now ( ) ) , _new - > id ) ) ;
ec = m_evictions . size ( ) ;
evi cts = m_evictions . size ( ) ;
}
}
if ( ec = = 1 )
if ( evi cts = = 1 )
doCheckEvictions ( boost : : system : : error_code ( ) ) ;
doCheckEvictions ( ) ;
ping ( _leastSeen . get ( ) ) ;
ping ( _leastSeen . get ( ) ) ;
}
}
@ -348,14 +338,15 @@ void NodeTable::noteActiveNode(Public const& _pubk, bi::udp::endpoint const& _en
if ( s . nodes . size ( ) > = s_bucketSize )
if ( s . nodes . size ( ) > = s_bucketSize )
{
{
if ( removed )
clog ( NodeTableWarn ) < < " DANGER: Bucket overflow when swapping node position. " ;
// It's only contested iff nodeentry exists
// It's only contested iff nodeentry exists
contested = s . nodes . front ( ) . lock ( ) ;
contested = s . nodes . front ( ) . lock ( ) ;
if ( ! contested )
if ( ! contested )
{
{
s . nodes . pop_front ( ) ;
s . nodes . pop_front ( ) ;
s . nodes . push_back ( node ) ;
s . nodes . push_back ( node ) ;
s . touch ( ) ;
if ( ! removed & & m_nodeEventHandler )
if ( ! removed & & m_nodeEventHandler )
m_nodeEventHandler - > appendEvent ( node - > id , NodeEntryAdded ) ;
m_nodeEventHandler - > appendEvent ( node - > id , NodeEntryAdded ) ;
}
}
@ -363,8 +354,6 @@ void NodeTable::noteActiveNode(Public const& _pubk, bi::udp::endpoint const& _en
else
else
{
{
s . nodes . push_back ( node ) ;
s . nodes . push_back ( node ) ;
s . touch ( ) ;
if ( ! removed & & m_nodeEventHandler )
if ( ! removed & & m_nodeEventHandler )
m_nodeEventHandler - > appendEvent ( node - > id , NodeEntryAdded ) ;
m_nodeEventHandler - > appendEvent ( node - > id , NodeEntryAdded ) ;
}
}
@ -576,14 +565,9 @@ void NodeTable::onReceived(UDPSocketFace*, bi::udp::endpoint const& _from, bytes
}
}
}
}
void NodeTable : : doCheckEvictions ( boost : : system : : error_code const & _ec )
void NodeTable : : doCheckEvictions ( )
{
{
if ( _ec | | ! m_socketPointer - > isOpen ( ) )
m_timers . schedule ( c_evictionCheckInterval . count ( ) , [ this ] ( boost : : system : : error_code const & _ec )
return ;
auto self ( shared_from_this ( ) ) ;
m_evictionCheckTimer . expires_from_now ( c_evictionCheckInterval ) ;
m_evictionCheckTimer . async_wait ( [ this , self ] ( boost : : system : : error_code const & _ec )
{
{
if ( _ec )
if ( _ec )
return ;
return ;
@ -605,28 +589,23 @@ void NodeTable::doCheckEvictions(boost::system::error_code const& _ec)
dropNode ( n ) ;
dropNode ( n ) ;
if ( evictionsRemain )
if ( evictionsRemain )
doCheckEvictions ( boost : : system : : error_code ( ) ) ;
doCheckEvictions ( ) ;
} ) ;
} ) ;
}
}
void NodeTable : : doRefreshBuckets ( boost : : system : : error_code const & _ec )
void NodeTable : : doDiscovery ( )
{
{
if ( _ec )
m_timers . schedule ( c_bucketRefresh . count ( ) , [ this ] ( boost : : system : : error_code const & ec )
{
if ( ec )
return ;
return ;
clog ( NodeTableEvent ) < < " refreshing buckets " ;
clog ( NodeTableEvent ) < < " performing random discovery " ;
bool connected = m_socketPointer - > isOpen ( ) ;
if ( connected )
{
NodeId randNodeId ;
NodeId randNodeId ;
crypto : : Nonce : : get ( ) . ref ( ) . copyTo ( randNodeId . ref ( ) . cropped ( 0 , h256 : : size ) ) ;
crypto : : Nonce : : get ( ) . ref ( ) . copyTo ( randNodeId . ref ( ) . cropped ( 0 , h256 : : size ) ) ;
crypto : : Nonce : : get ( ) . ref ( ) . copyTo ( randNodeId . ref ( ) . cropped ( h256 : : size , h256 : : size ) ) ;
crypto : : Nonce : : get ( ) . ref ( ) . copyTo ( randNodeId . ref ( ) . cropped ( h256 : : size , h256 : : size ) ) ;
discover ( randNodeId ) ;
doDiscover ( randNodeId ) ;
}
} ) ;
auto runcb = [ this ] ( boost : : system : : error_code const & error ) { doRefreshBuckets ( error ) ; } ;
m_bucketRefreshTimer . expires_from_now ( boost : : posix_time : : milliseconds ( c_bucketRefresh . count ( ) ) ) ;
m_bucketRefreshTimer . async_wait ( runcb ) ;
}
}
void PingNode : : streamRLP ( RLPStream & _s ) const
void PingNode : : streamRLP ( RLPStream & _s ) const