2019-07-15 18:57:42 +00:00
/*
2019-08-29 17:58:47 +00:00
* Copyright ( c ) 2019 ZeroTier , Inc .
2019-07-15 18:57:42 +00:00
*
2019-08-29 17:58:47 +00:00
* Use of this software is governed by the Business Source License included
* in the LICENSE . TXT file in the project ' s root directory .
2019-07-15 18:57:42 +00:00
*
2019-08-29 17:58:47 +00:00
* Change Date : 2023 - 01 - 01
2019-07-15 18:57:42 +00:00
*
2019-08-29 17:58:47 +00:00
* On the date above , in accordance with the Business Source License , use
* of this software will be governed by version 2.0 of the Apache License .
2019-07-15 18:57:42 +00:00
*/
2019-08-29 17:58:47 +00:00
/****/
2019-07-15 18:57:42 +00:00
2019-09-03 19:21:57 +00:00
/*
* This is a high - throughput minimal root server . It implements only
* those functions of a ZT node that a root must perform and does so
* using highly efficient multithreaded I / O code . It ' s only been
* thoroughly tested on Linux but should also run on BSDs .
*
* Root configuration file format ( JSON ) :
*
* {
* " name " : Name of this root for documentation / UI purposes ( string )
* " port " : UDP port ( int )
* " httpPort " : Local HTTP port for basic stats ( int )
2019-09-03 20:17:12 +00:00
* " relayMaxHops " : Max hops ( up to 7 )
2019-09-03 21:44:13 +00:00
* " planetFile " : Location of planet file for pre - 2. x peers ( string )
2019-09-03 19:21:57 +00:00
* " statsRoot " : If present , path to periodically save stats files ( string )
2019-09-03 21:57:02 +00:00
* " s_siblings " : [
2019-09-03 19:21:57 +00:00
* {
* " name " : Sibling name for UI / documentation purposes ( string )
* " id " : Full public identity of subling ( string )
* " ip " : IP address of sibling ( string )
* " port " : port of subling ( for ZeroTier UDP ) ( int )
* } , . . .
* ]
* }
*
* The only required field is port . If statsRoot is present then files
* are periodically written there containing the root ' s current state .
* It should be a memory filesystem like / dev / shm on Linux as these
* files are large and rewritten frequently and do not need to be
* persisted .
*
2019-09-03 21:57:02 +00:00
* s_siblings are other root servers that should receive packets to peers
2019-09-03 19:21:57 +00:00
* that we can ' t find . This can occur due to e . g . network topology
2019-09-03 21:57:02 +00:00
* hiccups , IP blockages , etc . s_siblings are used in the order in which
2019-09-03 19:21:57 +00:00
* they appear with the first alive sibling being used .
*/
2019-08-30 16:08:39 +00:00
# include <Constants.hpp>
2019-07-15 18:57:42 +00:00
# include <stdio.h>
# include <stdlib.h>
# include <unistd.h>
# include <string.h>
# include <fcntl.h>
# include <signal.h>
2019-08-29 21:27:41 +00:00
# include <errno.h>
2019-07-15 18:57:42 +00:00
# include <sys/stat.h>
# include <sys/types.h>
# include <sys/socket.h>
# include <sys/select.h>
# include <sys/time.h>
# include <sys/un.h>
# include <sys/ioctl.h>
# include <arpa/inet.h>
# include <netinet/in.h>
# include <netinet/ip.h>
# include <netinet/ip6.h>
# include <netinet/tcp.h>
2019-08-30 16:08:39 +00:00
# include <netinet/udp.h>
# include <json.hpp>
# include <httplib.h>
# include <Packet.hpp>
# include <Utils.hpp>
# include <Address.hpp>
# include <Identity.hpp>
# include <InetAddress.hpp>
# include <Mutex.hpp>
# include <SharedPtr.hpp>
# include <MulticastGroup.hpp>
# include <CertificateOfMembership.hpp>
# include <OSUtils.hpp>
2019-09-03 19:21:57 +00:00
# include <Meter.hpp>
2019-08-28 18:49:45 +00:00
2019-07-15 18:57:42 +00:00
# include <string>
# include <thread>
# include <map>
2019-08-28 20:04:58 +00:00
# include <set>
2019-07-15 18:57:42 +00:00
# include <vector>
# include <iostream>
2019-08-28 18:49:45 +00:00
# include <unordered_map>
2019-08-28 22:07:38 +00:00
# include <unordered_set>
2019-08-28 18:49:45 +00:00
# include <vector>
2019-08-28 18:59:13 +00:00
# include <atomic>
2019-08-28 20:04:58 +00:00
# include <mutex>
2019-08-30 16:08:39 +00:00
# include <sstream>
2019-08-28 18:49:45 +00:00
2019-09-06 14:37:44 +00:00
# include "geoip-html.h"
2019-08-28 18:49:45 +00:00
using namespace ZeroTier ;
2019-08-29 17:58:47 +00:00
using json = nlohmann : : json ;
2019-08-29 21:27:41 +00:00
# ifdef MSG_DONTWAIT
# define SENDTO_FLAGS MSG_DONTWAIT
# else
# define SENDTO_FLAGS 0
# endif
2019-08-29 17:58:47 +00:00
//////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////
2019-08-28 18:49:45 +00:00
2019-09-03 19:21:57 +00:00
/**
* RootPeer is a normal peer known to this root
*
* This can also be a sibling root , which is itself a peer . Sibling roots
* are sent HELLO while for other peers we only listen for HELLO .
*/
2019-08-29 14:17:18 +00:00
struct RootPeer
2019-08-28 18:49:45 +00:00
{
2019-09-06 14:37:44 +00:00
ZT_ALWAYS_INLINE RootPeer ( ) : lastSend ( 0 ) , lastReceive ( 0 ) , lastSync ( 0 ) , lastEcho ( 0 ) , lastHello ( 0 ) , vProto ( - 1 ) , vMajor ( - 1 ) , vMinor ( - 1 ) , vRev ( - 1 ) { }
2019-08-29 21:27:41 +00:00
ZT_ALWAYS_INLINE ~ RootPeer ( ) { Utils : : burn ( key , sizeof ( key ) ) ; }
2019-09-03 19:21:57 +00:00
Identity id ; // Identity
uint8_t key [ 32 ] ; // Shared secret key
InetAddress ip4 , ip6 ; // IPv4 and IPv6 addresses
int64_t lastSend ; // Time of last send (any packet)
int64_t lastReceive ; // Time of last receive (any packet)
int64_t lastSync ; // Time of last data synchronization with LF or other root state backend (currently unused)
int64_t lastEcho ; // Time of last received ECHO
int64_t lastHello ; // Time of last received HELLO
2019-09-03 21:44:13 +00:00
int vProto ; // Protocol version
2019-09-03 19:21:57 +00:00
int vMajor , vMinor , vRev ; // Peer version or -1,-1,-1 if unknown
2019-08-29 21:27:41 +00:00
std : : mutex lock ;
2019-08-28 20:38:52 +00:00
2019-08-28 20:04:58 +00:00
AtomicCounter __refCount ;
2019-08-28 18:49:45 +00:00
} ;
2019-07-15 18:57:42 +00:00
2019-09-06 14:37:44 +00:00
// Hashers for std::unordered_map
struct IdentityHasher { ZT_ALWAYS_INLINE std : : size_t operator ( ) ( const Identity & id ) const { return ( std : : size_t ) id . hashCode ( ) ; } } ;
struct AddressHasher { ZT_ALWAYS_INLINE std : : size_t operator ( ) ( const Address & a ) const { return ( std : : size_t ) a . toInt ( ) ; } } ;
struct InetAddressHasher { ZT_ALWAYS_INLINE std : : size_t operator ( ) ( const InetAddress & ip ) const { return ( std : : size_t ) ip . hashCode ( ) ; } } ;
struct MulticastGroupHasher { ZT_ALWAYS_INLINE std : : size_t operator ( ) ( const MulticastGroup & mg ) const { return ( std : : size_t ) mg . hashCode ( ) ; } } ;
// An ordered tuple key representing an introduction of one peer to another
struct RendezvousKey
{
RendezvousKey ( const Address & aa , const Address & bb )
{
if ( aa > bb ) {
a = aa ;
b = bb ;
} else {
a = bb ;
b = aa ;
}
}
Address a , b ;
ZT_ALWAYS_INLINE bool operator = = ( const RendezvousKey & k ) const { return ( ( a = = k . a ) & & ( b = = k . b ) ) ; }
ZT_ALWAYS_INLINE bool operator ! = ( const RendezvousKey & k ) const { return ( ( a ! = k . a ) | | ( b ! = k . b ) ) ; }
struct Hasher { ZT_ALWAYS_INLINE std : : size_t operator ( ) ( const RendezvousKey & k ) const { return ( std : : size_t ) ( k . a . toInt ( ) ^ k . b . toInt ( ) ) ; } } ;
} ;
2019-09-03 22:19:17 +00:00
struct RendezvousStats
{
RendezvousStats ( ) : count ( 0 ) , ts ( 0 ) { }
int64_t count ;
int64_t ts ;
} ;
struct ForwardingStats
{
ForwardingStats ( ) : bytes ( 0 ) , ts ( 0 ) , bps ( ) { }
uint64_t bytes ;
int64_t ts ;
Meter bps ;
} ;
2019-09-06 14:37:44 +00:00
static int64_t s_startTime ; // Time service was started
static std : : vector < int > s_ports ; // Ports to bind for UDP traffic
static int s_relayMaxHops = 0 ; // Max relay hops
static Identity s_self ; // My identity (including secret)
static std : : atomic_bool s_run ; // Remains true until shutdown is ordered
static json s_config ; // JSON config file contents
static std : : string s_statsRoot ; // Root to write stats, peers, etc.
static std : : atomic_bool s_geoInit ; // True if geoIP data is initialized
2019-08-28 20:51:13 +00:00
2019-09-03 21:57:02 +00:00
static Meter s_inputRate ;
static Meter s_outputRate ;
static Meter s_forwardRate ;
static Meter s_discardedForwardRate ;
2019-09-03 19:21:57 +00:00
2019-09-03 21:57:02 +00:00
static std : : string s_planet ;
static std : : unordered_map < uint64_t , std : : unordered_map < MulticastGroup , std : : unordered_map < Address , int64_t , AddressHasher > , MulticastGroupHasher > > s_multicastSubscriptions ;
static std : : unordered_map < Identity , SharedPtr < RootPeer > , IdentityHasher > s_peersByIdentity ;
static std : : unordered_map < Address , std : : set < SharedPtr < RootPeer > > , AddressHasher > s_peersByVirtAddr ;
static std : : unordered_map < InetAddress , std : : set < SharedPtr < RootPeer > > , InetAddressHasher > s_peersByPhysAddr ;
2019-09-06 14:37:44 +00:00
static std : : unordered_map < RendezvousKey , RendezvousStats , RendezvousKey : : Hasher > s_rendezvousTracking ;
2019-09-03 22:19:17 +00:00
static std : : unordered_map < Address , ForwardingStats , AddressHasher > s_lastForwardedTo ;
2019-09-06 14:37:44 +00:00
static std : : map < std : : pair < uint32_t , uint32_t > , std : : pair < float , float > > s_geoIp4 ;
static std : : map < std : : pair < std : : array < uint64_t , 2 > , std : : array < uint64_t , 2 > > , std : : pair < float , float > > s_geoIp6 ;
2019-08-28 20:51:13 +00:00
2019-09-03 21:44:13 +00:00
static std : : mutex s_planet_l ;
2019-09-03 21:57:02 +00:00
static std : : mutex s_siblings_l ;
static std : : mutex s_multicastSubscriptions_l ;
static std : : mutex s_peersByIdentity_l ;
static std : : mutex s_peersByVirtAddr_l ;
static std : : mutex s_peersByPhysAddr_l ;
2019-09-06 14:37:44 +00:00
static std : : mutex s_rendezvousTracking_l ;
2019-09-03 22:19:17 +00:00
static std : : mutex s_lastForwardedTo_l ;
2019-08-28 18:49:45 +00:00
2019-08-29 17:58:47 +00:00
//////////////////////////////////////////////////////////////////////////////
2019-08-29 15:00:54 +00:00
//////////////////////////////////////////////////////////////////////////////
2019-09-06 14:37:44 +00:00
static uint32_t ip4ToH32 ( const InetAddress & ip )
{
return Utils : : ntoh ( ( uint32_t ) ( ( ( const struct sockaddr_in * ) & ip ) - > sin_addr . s_addr ) ) ;
}
static std : : array < uint64_t , 2 > ip6ToH128 ( const InetAddress & ip )
{
std : : array < uint64_t , 2 > i128 ;
memcpy ( i128 . data ( ) , ip . rawIpData ( ) , 16 ) ;
i128 [ 0 ] = Utils : : ntoh ( i128 [ 0 ] ) ;
i128 [ 1 ] = Utils : : ntoh ( i128 [ 1 ] ) ;
return i128 ;
}
2019-08-29 17:58:47 +00:00
static void handlePacket ( const int v4s , const int v6s , const InetAddress * const ip , Packet & pkt )
2019-08-28 18:49:45 +00:00
{
2019-08-29 14:17:18 +00:00
char ipstr [ 128 ] , ipstr2 [ 128 ] , astr [ 32 ] , astr2 [ 32 ] , tmpstr [ 256 ] ;
2019-08-28 20:04:58 +00:00
const bool fragment = pkt [ ZT_PACKET_FRAGMENT_IDX_FRAGMENT_INDICATOR ] = = ZT_PACKET_FRAGMENT_INDICATOR ;
2019-08-29 14:17:18 +00:00
const Address source ( pkt . source ( ) ) ;
const Address dest ( pkt . destination ( ) ) ;
const int64_t now = OSUtils : : now ( ) ;
2019-08-28 20:04:58 +00:00
2019-09-03 21:57:02 +00:00
s_inputRate . log ( now , pkt . size ( ) ) ;
2019-09-03 19:21:57 +00:00
2019-09-03 21:44:13 +00:00
if ( ( ! fragment ) & & ( ! pkt . fragmented ( ) ) & & ( dest = = s_self . address ( ) ) ) {
2019-08-29 14:17:18 +00:00
SharedPtr < RootPeer > peer ;
2019-08-28 20:04:58 +00:00
// If this is an un-encrypted HELLO, either learn a new peer or verify
// that this is a peer we already know.
if ( ( pkt . cipher ( ) = = ZT_PROTO_CIPHER_SUITE__POLY1305_NONE ) & & ( pkt . verb ( ) = = Packet : : VERB_HELLO ) ) {
2019-09-03 21:57:02 +00:00
std : : lock_guard < std : : mutex > pbi_l ( s_peersByIdentity_l ) ;
std : : lock_guard < std : : mutex > pbv_l ( s_peersByVirtAddr_l ) ;
2019-08-29 15:00:54 +00:00
2019-08-28 20:04:58 +00:00
Identity id ;
if ( id . deserialize ( pkt , ZT_PROTO_VERB_HELLO_IDX_IDENTITY ) ) {
{
2019-09-03 21:57:02 +00:00
auto pById = s_peersByIdentity . find ( id ) ;
if ( pById ! = s_peersByIdentity . end ( ) ) {
2019-08-28 20:04:58 +00:00
peer = pById - > second ;
2019-08-29 14:17:18 +00:00
//printf("%s has %s (known (1))" ZT_EOL_S,ip->toString(ipstr),source().toString(astr));
2019-08-28 20:04:58 +00:00
}
}
if ( peer ) {
if ( ! pkt . dearmor ( peer - > key ) ) {
printf ( " %s HELLO rejected: packet authentication failed " ZT_EOL_S , ip - > toString ( ipstr ) ) ;
return ;
}
} else {
2019-08-29 14:17:18 +00:00
peer . set ( new RootPeer ) ;
2019-09-03 21:44:13 +00:00
if ( s_self . agree ( id , peer - > key ) ) {
2019-08-28 20:04:58 +00:00
if ( pkt . dearmor ( peer - > key ) ) {
2019-08-29 18:40:31 +00:00
if ( ! pkt . uncompress ( ) ) {
printf ( " %s HELLO rejected: decompression failed " ZT_EOL_S , ip - > toString ( ipstr ) ) ;
return ;
}
2019-08-28 20:51:13 +00:00
peer - > id = id ;
2019-08-29 18:23:23 +00:00
peer - > lastReceive = now ;
2019-09-03 21:57:02 +00:00
s_peersByIdentity . emplace ( id , peer ) ;
s_peersByVirtAddr [ id . address ( ) ] . emplace ( peer ) ;
2019-08-28 20:04:58 +00:00
} else {
printf ( " %s HELLO rejected: packet authentication failed " ZT_EOL_S , ip - > toString ( ipstr ) ) ;
return ;
}
} else {
printf ( " %s HELLO rejected: key agreement failed " ZT_EOL_S , ip - > toString ( ipstr ) ) ;
return ;
}
}
}
}
// If it wasn't a HELLO, check to see if any known identities for the sender's
// short ZT address successfully decrypt the packet.
if ( ! peer ) {
2019-09-03 21:57:02 +00:00
std : : lock_guard < std : : mutex > pbv_l ( s_peersByVirtAddr_l ) ;
auto peers = s_peersByVirtAddr . find ( source ) ;
if ( peers ! = s_peersByVirtAddr . end ( ) ) {
2019-08-28 20:04:58 +00:00
for ( auto p = peers - > second . begin ( ) ; p ! = peers - > second . end ( ) ; + + p ) {
if ( pkt . dearmor ( ( * p ) - > key ) ) {
2019-08-29 18:40:31 +00:00
if ( ! pkt . uncompress ( ) ) {
printf ( " %s packet rejected: decompression failed " ZT_EOL_S , ip - > toString ( ipstr ) ) ;
return ;
}
2019-08-28 20:04:58 +00:00
peer = ( * p ) ;
2019-08-29 14:17:18 +00:00
//printf("%s has %s (known (2))" ZT_EOL_S,ip->toString(ipstr),source().toString(astr));
2019-08-28 20:04:58 +00:00
break ;
}
}
}
}
2019-08-29 17:58:47 +00:00
// If we found the peer, update IP and/or time and handle certain key packet types that the
// root must concern itself with.
2019-08-28 20:04:58 +00:00
if ( peer ) {
2019-08-29 21:27:41 +00:00
std : : lock_guard < std : : mutex > pl ( peer - > lock ) ;
2019-08-29 17:58:47 +00:00
InetAddress * const peerIp = ip - > isV4 ( ) ? & ( peer - > ip4 ) : & ( peer - > ip6 ) ;
2019-08-28 20:04:58 +00:00
if ( * peerIp ! = ip ) {
2019-09-03 21:57:02 +00:00
std : : lock_guard < std : : mutex > pbp_l ( s_peersByPhysAddr_l ) ;
2019-08-28 20:04:58 +00:00
if ( * peerIp ) {
2019-09-03 21:57:02 +00:00
auto prev = s_peersByPhysAddr . find ( * peerIp ) ;
if ( prev ! = s_peersByPhysAddr . end ( ) ) {
2019-08-28 20:04:58 +00:00
prev - > second . erase ( peer ) ;
if ( prev - > second . empty ( ) )
2019-09-03 21:57:02 +00:00
s_peersByPhysAddr . erase ( prev ) ;
2019-08-28 20:04:58 +00:00
}
}
* peerIp = ip ;
2019-09-03 21:57:02 +00:00
s_peersByPhysAddr [ ip ] . emplace ( peer ) ;
2019-08-28 20:04:58 +00:00
}
2019-08-28 20:38:52 +00:00
const int64_t now = OSUtils : : now ( ) ;
peer - > lastReceive = now ;
switch ( pkt . verb ( ) ) {
2019-08-28 22:20:34 +00:00
case Packet : : VERB_HELLO :
try {
2019-09-04 15:19:12 +00:00
if ( ( now - peer - > lastHello ) > 500 ) {
2019-08-29 21:27:41 +00:00
peer - > lastHello = now ;
2019-09-04 15:19:12 +00:00
2019-09-03 21:44:13 +00:00
peer - > vProto = ( int ) pkt [ ZT_PROTO_VERB_HELLO_IDX_PROTOCOL_VERSION ] ;
2019-08-30 16:08:39 +00:00
peer - > vMajor = ( int ) pkt [ ZT_PROTO_VERB_HELLO_IDX_MAJOR_VERSION ] ;
peer - > vMinor = ( int ) pkt [ ZT_PROTO_VERB_HELLO_IDX_MINOR_VERSION ] ;
peer - > vRev = ( int ) pkt . template at < uint16_t > ( ZT_PROTO_VERB_HELLO_IDX_REVISION ) ;
2019-08-29 21:27:41 +00:00
const uint64_t origId = pkt . packetId ( ) ;
const uint64_t ts = pkt . template at < uint64_t > ( ZT_PROTO_VERB_HELLO_IDX_TIMESTAMP ) ;
2019-08-30 16:08:39 +00:00
2019-09-03 21:44:13 +00:00
pkt . reset ( source , s_self . address ( ) , Packet : : VERB_OK ) ;
2019-08-29 21:27:41 +00:00
pkt . append ( ( uint8_t ) Packet : : VERB_HELLO ) ;
pkt . append ( origId ) ;
pkt . append ( ts ) ;
pkt . append ( ( uint8_t ) ZT_PROTO_VERSION ) ;
pkt . append ( ( uint8_t ) 0 ) ;
pkt . append ( ( uint8_t ) 0 ) ;
pkt . append ( ( uint16_t ) 0 ) ;
ip - > serialize ( pkt ) ;
2019-09-03 21:44:13 +00:00
if ( peer - > vProto < 11 ) { // send planet file for pre-2.x peers
std : : lock_guard < std : : mutex > pl ( s_planet_l ) ;
if ( s_planet . length ( ) > 0 ) {
pkt . append ( ( uint16_t ) s_planet . size ( ) ) ;
pkt . append ( ( const uint8_t * ) s_planet . data ( ) , s_planet . size ( ) ) ;
}
}
2019-08-29 21:27:41 +00:00
pkt . armor ( peer - > key , true ) ;
sendto ( ip - > isV4 ( ) ? v4s : v6s , pkt . data ( ) , pkt . size ( ) , SENDTO_FLAGS , ( const struct sockaddr * ) ip , ( socklen_t ) ( ( ip - > ss_family = = AF_INET ) ? sizeof ( struct sockaddr_in ) : sizeof ( struct sockaddr_in6 ) ) ) ;
2019-08-30 16:08:39 +00:00
2019-09-03 21:57:02 +00:00
s_outputRate . log ( now , pkt . size ( ) ) ;
2019-08-30 16:08:39 +00:00
peer - > lastSend = now ;
2019-08-29 21:27:41 +00:00
}
2019-08-28 22:20:34 +00:00
} catch ( . . . ) {
printf ( " * unexpected exception handling HELLO from %s " ZT_EOL_S , ip - > toString ( ipstr ) ) ;
}
break ;
2019-08-28 20:38:52 +00:00
2019-08-29 21:27:41 +00:00
case Packet : : VERB_ECHO :
try {
2019-09-04 15:19:12 +00:00
if ( ( now - peer - > lastEcho ) > 500 ) {
2019-08-29 21:27:41 +00:00
peer - > lastEcho = now ;
2019-09-03 19:21:57 +00:00
2019-09-03 21:44:13 +00:00
Packet outp ( source , s_self . address ( ) , Packet : : VERB_OK ) ;
2019-08-29 21:27:41 +00:00
outp . append ( ( uint8_t ) Packet : : VERB_ECHO ) ;
outp . append ( pkt . packetId ( ) ) ;
outp . append ( ( ( const uint8_t * ) pkt . data ( ) ) + ZT_PACKET_IDX_PAYLOAD , pkt . size ( ) - ZT_PACKET_IDX_PAYLOAD ) ;
outp . compress ( ) ;
outp . armor ( peer - > key , true ) ;
sendto ( ip - > isV4 ( ) ? v4s : v6s , outp . data ( ) , outp . size ( ) , SENDTO_FLAGS , ( const struct sockaddr * ) ip , ( socklen_t ) ( ( ip - > ss_family = = AF_INET ) ? sizeof ( struct sockaddr_in ) : sizeof ( struct sockaddr_in6 ) ) ) ;
2019-09-03 19:21:57 +00:00
2019-09-03 21:57:02 +00:00
s_outputRate . log ( now , outp . size ( ) ) ;
2019-08-30 16:08:39 +00:00
peer - > lastSend = now ;
2019-08-29 21:27:41 +00:00
}
} catch ( . . . ) {
printf ( " * unexpected exception handling ECHO from %s " ZT_EOL_S , ip - > toString ( ipstr ) ) ;
}
case Packet : : VERB_WHOIS :
try {
std : : vector < SharedPtr < RootPeer > > results ;
{
2019-09-03 21:57:02 +00:00
std : : lock_guard < std : : mutex > l ( s_peersByVirtAddr_l ) ;
2019-08-29 21:27:41 +00:00
for ( unsigned int ptr = ZT_PACKET_IDX_PAYLOAD ; ( ptr + ZT_ADDRESS_LENGTH ) < = pkt . size ( ) ; ptr + = ZT_ADDRESS_LENGTH ) {
2019-09-03 21:57:02 +00:00
auto peers = s_peersByVirtAddr . find ( Address ( pkt . field ( ptr , ZT_ADDRESS_LENGTH ) , ZT_ADDRESS_LENGTH ) ) ;
if ( peers ! = s_peersByVirtAddr . end ( ) ) {
2019-08-29 21:27:41 +00:00
for ( auto p = peers - > second . begin ( ) ; p ! = peers - > second . end ( ) ; + + p )
results . push_back ( * p ) ;
}
}
}
if ( ! results . empty ( ) ) {
const uint64_t origId = pkt . packetId ( ) ;
2019-09-03 21:44:13 +00:00
pkt . reset ( source , s_self . address ( ) , Packet : : VERB_OK ) ;
2019-08-29 21:27:41 +00:00
pkt . append ( ( uint8_t ) Packet : : VERB_WHOIS ) ;
pkt . append ( origId ) ;
for ( auto p = results . begin ( ) ; p ! = results . end ( ) ; + + p )
( * p ) - > id . serialize ( pkt , false ) ;
pkt . armor ( peer - > key , true ) ;
sendto ( ip - > isV4 ( ) ? v4s : v6s , pkt . data ( ) , pkt . size ( ) , SENDTO_FLAGS , ( const struct sockaddr * ) ip , ( socklen_t ) ( ( ip - > ss_family = = AF_INET ) ? sizeof ( struct sockaddr_in ) : sizeof ( struct sockaddr_in6 ) ) ) ;
2019-09-03 19:21:57 +00:00
2019-09-03 21:57:02 +00:00
s_outputRate . log ( now , pkt . size ( ) ) ;
2019-08-30 16:08:39 +00:00
peer - > lastSend = now ;
2019-08-29 21:27:41 +00:00
}
} catch ( . . . ) {
printf ( " * unexpected exception handling ECHO from %s " ZT_EOL_S , ip - > toString ( ipstr ) ) ;
}
2019-08-28 22:20:34 +00:00
case Packet : : VERB_MULTICAST_LIKE :
try {
2019-09-03 21:57:02 +00:00
std : : lock_guard < std : : mutex > l ( s_multicastSubscriptions_l ) ;
2019-08-28 22:20:34 +00:00
for ( unsigned int ptr = ZT_PACKET_IDX_PAYLOAD ; ( ptr + 18 ) < = pkt . size ( ) ; ptr + = 18 ) {
const uint64_t nwid = pkt . template at < uint64_t > ( ptr ) ;
const MulticastGroup mg ( MAC ( pkt . field ( ptr + 8 , 6 ) , 6 ) , pkt . template at < uint32_t > ( ptr + 14 ) ) ;
2019-09-03 21:57:02 +00:00
s_multicastSubscriptions [ nwid ] [ mg ] [ source ] = now ;
2019-08-29 18:40:31 +00:00
//printf("%s %s subscribes to %s/%.8lx on network %.16llx" ZT_EOL_S,ip->toString(ipstr),source.toString(astr),mg.mac().toString(tmpstr),(unsigned long)mg.adi(),(unsigned long long)nwid);
2019-08-28 22:20:34 +00:00
}
} catch ( . . . ) {
printf ( " * unexpected exception handling MULTICAST_LIKE from %s " ZT_EOL_S , ip - > toString ( ipstr ) ) ;
2019-08-28 20:38:52 +00:00
}
2019-08-28 22:20:34 +00:00
break ;
2019-08-28 20:38:52 +00:00
2019-08-28 22:20:34 +00:00
case Packet : : VERB_MULTICAST_GATHER :
try {
const uint64_t nwid = pkt . template at < uint64_t > ( ZT_PROTO_VERB_MULTICAST_GATHER_IDX_NETWORK_ID ) ;
2019-09-03 19:21:57 +00:00
//const unsigned int flags = pkt[ZT_PROTO_VERB_MULTICAST_GATHER_IDX_FLAGS];
2019-08-28 22:20:34 +00:00
const MulticastGroup mg ( MAC ( pkt . field ( ZT_PROTO_VERB_MULTICAST_GATHER_IDX_MAC , 6 ) , 6 ) , pkt . template at < uint32_t > ( ZT_PROTO_VERB_MULTICAST_GATHER_IDX_ADI ) ) ;
unsigned int gatherLimit = pkt . template at < uint32_t > ( ZT_PROTO_VERB_MULTICAST_GATHER_IDX_GATHER_LIMIT ) ;
if ( gatherLimit > 255 )
gatherLimit = 255 ;
2019-08-28 22:07:38 +00:00
2019-08-28 22:20:34 +00:00
const uint64_t origId = pkt . packetId ( ) ;
2019-09-03 21:44:13 +00:00
pkt . reset ( source , s_self . address ( ) , Packet : : VERB_OK ) ;
2019-08-28 22:20:34 +00:00
pkt . append ( ( uint8_t ) Packet : : VERB_MULTICAST_GATHER ) ;
pkt . append ( origId ) ;
pkt . append ( nwid ) ;
mg . mac ( ) . appendTo ( pkt ) ;
pkt . append ( ( uint32_t ) mg . adi ( ) ) ;
2019-08-28 22:07:38 +00:00
2019-08-28 22:20:34 +00:00
{
2019-09-03 21:57:02 +00:00
std : : lock_guard < std : : mutex > l ( s_multicastSubscriptions_l ) ;
auto forNet = s_multicastSubscriptions . find ( nwid ) ;
if ( forNet ! = s_multicastSubscriptions . end ( ) ) {
2019-08-28 22:20:34 +00:00
auto forGroup = forNet - > second . find ( mg ) ;
if ( forGroup ! = forNet - > second . end ( ) ) {
pkt . append ( ( uint32_t ) forGroup - > second . size ( ) ) ;
2019-08-29 19:02:37 +00:00
const unsigned int countAt = pkt . size ( ) ;
pkt . addSize ( 2 ) ;
2019-08-28 22:20:34 +00:00
unsigned int l = 0 ;
2019-08-31 00:34:05 +00:00
for ( auto g = forGroup - > second . begin ( ) ; ( ( l < gatherLimit ) & & ( g ! = forGroup - > second . end ( ) ) ) ; + + g ) {
if ( g - > first ! = source ) {
+ + l ;
2019-08-29 19:02:37 +00:00
g - > first . appendTo ( pkt ) ;
2019-08-31 00:34:05 +00:00
}
2019-08-29 19:02:37 +00:00
}
2019-08-28 22:20:34 +00:00
if ( l > 0 ) {
2019-08-29 19:02:37 +00:00
pkt . setAt < uint16_t > ( countAt , ( uint16_t ) l ) ;
2019-08-29 18:40:31 +00:00
pkt . armor ( peer - > key , true ) ;
2019-08-29 21:27:41 +00:00
sendto ( ip - > isV4 ( ) ? v4s : v6s , pkt . data ( ) , pkt . size ( ) , SENDTO_FLAGS , ( const struct sockaddr * ) ip , ( socklen_t ) ( ip - > isV4 ( ) ? sizeof ( struct sockaddr_in ) : sizeof ( struct sockaddr_in6 ) ) ) ;
2019-09-03 19:21:57 +00:00
2019-09-03 21:57:02 +00:00
s_outputRate . log ( now , pkt . size ( ) ) ;
2019-08-30 16:08:39 +00:00
peer - > lastSend = now ;
2019-08-29 18:40:31 +00:00
//printf("%s %s gathered %u subscribers to %s/%.8lx on network %.16llx" ZT_EOL_S,ip->toString(ipstr),source.toString(astr),l,mg.mac().toString(tmpstr),(unsigned long)mg.adi(),(unsigned long long)nwid);
2019-08-28 22:20:34 +00:00
}
2019-08-28 22:07:38 +00:00
}
}
}
2019-08-28 22:20:34 +00:00
} catch ( . . . ) {
printf ( " * unexpected exception handling MULTICAST_GATHER from %s " ZT_EOL_S , ip - > toString ( ipstr ) ) ;
2019-08-28 22:07:38 +00:00
}
2019-08-28 22:20:34 +00:00
break ;
2019-08-28 20:38:52 +00:00
default :
break ;
}
2019-08-28 20:04:58 +00:00
return ;
}
}
2019-08-29 14:17:18 +00:00
// If we made it here, we are forwarding this packet to someone else and also possibly
// sending a RENDEZVOUS message.
2019-09-03 21:44:13 +00:00
int hops = 0 ;
2019-08-29 14:17:18 +00:00
bool introduce = false ;
2019-09-03 20:17:12 +00:00
if ( fragment ) {
2019-09-03 21:44:13 +00:00
if ( ( hops = ( int ) reinterpret_cast < Packet : : Fragment * > ( & pkt ) - > incrementHops ( ) ) > s_relayMaxHops ) {
2019-09-03 20:17:12 +00:00
//printf("%s refused to forward to %s: max hop count exceeded" ZT_EOL_S,ip->toString(ipstr),dest.toString(astr));
2019-09-03 21:57:02 +00:00
s_discardedForwardRate . log ( now , pkt . size ( ) ) ;
2019-09-03 20:17:12 +00:00
return ;
}
} else {
2019-09-03 21:44:13 +00:00
if ( ( hops = ( int ) pkt . incrementHops ( ) ) > s_relayMaxHops ) {
2019-09-03 20:17:12 +00:00
//printf("%s refused to forward to %s: max hop count exceeded" ZT_EOL_S,ip->toString(ipstr),dest.toString(astr));
2019-09-03 21:57:02 +00:00
s_discardedForwardRate . log ( now , pkt . size ( ) ) ;
2019-09-03 20:17:12 +00:00
return ;
}
2019-09-03 21:44:13 +00:00
if ( hops = = 1 ) {
RendezvousKey rk ( source , dest ) ;
2019-09-06 14:37:44 +00:00
std : : lock_guard < std : : mutex > l ( s_rendezvousTracking_l ) ;
RendezvousStats & lr = s_rendezvousTracking [ rk ] ;
2019-09-03 22:19:17 +00:00
if ( ( now - lr . ts ) > = 30000 ) {
+ + lr . count ;
lr . ts = now ;
2019-09-03 21:44:13 +00:00
introduce = true ;
}
2019-08-29 14:17:18 +00:00
}
}
std : : vector < std : : pair < InetAddress * , SharedPtr < RootPeer > > > toAddrs ;
2019-08-28 20:04:58 +00:00
{
2019-09-03 21:57:02 +00:00
std : : lock_guard < std : : mutex > pbv_l ( s_peersByVirtAddr_l ) ;
auto peers = s_peersByVirtAddr . find ( dest ) ;
if ( peers ! = s_peersByVirtAddr . end ( ) ) {
2019-08-28 20:04:58 +00:00
for ( auto p = peers - > second . begin ( ) ; p ! = peers - > second . end ( ) ; + + p ) {
2019-08-29 21:27:41 +00:00
if ( ( * p ) - > ip4 ) {
2019-08-29 18:23:23 +00:00
toAddrs . push_back ( std : : pair < InetAddress * , SharedPtr < RootPeer > > ( & ( ( * p ) - > ip4 ) , * p ) ) ;
2019-08-29 21:27:41 +00:00
} else if ( ( * p ) - > ip6 ) {
toAddrs . push_back ( std : : pair < InetAddress * , SharedPtr < RootPeer > > ( & ( ( * p ) - > ip6 ) , * p ) ) ;
2019-08-28 22:15:21 +00:00
}
2019-08-28 20:04:58 +00:00
}
}
}
2019-09-03 20:01:01 +00:00
if ( toAddrs . empty ( ) ) {
2019-09-03 21:57:02 +00:00
s_discardedForwardRate . log ( now , pkt . size ( ) ) ;
2019-09-03 19:21:57 +00:00
return ;
2019-09-03 20:01:01 +00:00
}
2019-08-28 20:04:58 +00:00
2019-09-03 22:19:17 +00:00
{
std : : lock_guard < std : : mutex > l ( s_lastForwardedTo_l ) ;
ForwardingStats & fs = s_lastForwardedTo [ dest ] ;
fs . bytes + = ( uint64_t ) pkt . size ( ) ;
fs . ts = now ;
fs . bps . log ( now , pkt . size ( ) ) ;
}
2019-08-29 14:17:18 +00:00
if ( introduce ) {
2019-09-03 21:57:02 +00:00
std : : lock_guard < std : : mutex > l ( s_peersByVirtAddr_l ) ;
auto sources = s_peersByVirtAddr . find ( source ) ;
if ( sources ! = s_peersByVirtAddr . end ( ) ) {
2019-08-29 14:25:16 +00:00
for ( auto a = sources - > second . begin ( ) ; a ! = sources - > second . end ( ) ; + + a ) {
for ( auto b = toAddrs . begin ( ) ; b ! = toAddrs . end ( ) ; + + b ) {
2019-08-29 17:58:47 +00:00
if ( ( ( * a ) - > ip6 ) & & ( b - > second - > ip6 ) ) {
2019-08-29 15:00:54 +00:00
//printf("* introducing %s(%s) to %s(%s)" ZT_EOL_S,ip->toString(ipstr),source.toString(astr),b->second->ip6.toString(ipstr2),dest.toString(astr2));
2019-08-29 14:25:16 +00:00
2019-08-29 17:58:47 +00:00
// Introduce source to destination (V6)
2019-09-03 21:44:13 +00:00
Packet outp ( source , s_self . address ( ) , Packet : : VERB_RENDEZVOUS ) ;
2019-08-29 14:25:16 +00:00
outp . append ( ( uint8_t ) 0 ) ;
dest . appendTo ( outp ) ;
outp . append ( ( uint16_t ) b - > second - > ip6 . port ( ) ) ;
outp . append ( ( uint8_t ) 16 ) ;
outp . append ( ( const uint8_t * ) b - > second - > ip6 . rawIpData ( ) , 16 ) ;
outp . armor ( ( * a ) - > key , true ) ;
2019-08-31 04:17:34 +00:00
sendto ( v6s , outp . data ( ) , outp . size ( ) , SENDTO_FLAGS , ( const struct sockaddr * ) & ( ( * a ) - > ip6 ) , ( socklen_t ) sizeof ( struct sockaddr_in6 ) ) ;
2019-09-03 19:21:57 +00:00
2019-09-03 21:57:02 +00:00
s_outputRate . log ( now , outp . size ( ) ) ;
2019-08-30 16:08:39 +00:00
( * a ) - > lastSend = now ;
2019-08-29 14:25:16 +00:00
2019-08-29 17:58:47 +00:00
// Introduce destination to source (V6)
2019-09-03 21:44:13 +00:00
outp . reset ( dest , s_self . address ( ) , Packet : : VERB_RENDEZVOUS ) ;
2019-08-29 14:25:16 +00:00
outp . append ( ( uint8_t ) 0 ) ;
source . appendTo ( outp ) ;
outp . append ( ( uint16_t ) ip - > port ( ) ) ;
outp . append ( ( uint8_t ) 16 ) ;
outp . append ( ( const uint8_t * ) ip - > rawIpData ( ) , 16 ) ;
outp . armor ( b - > second - > key , true ) ;
2019-08-31 04:17:34 +00:00
sendto ( v6s , outp . data ( ) , outp . size ( ) , SENDTO_FLAGS , ( const struct sockaddr * ) & ( b - > second - > ip6 ) , ( socklen_t ) sizeof ( struct sockaddr_in6 ) ) ;
2019-09-03 19:21:57 +00:00
2019-09-03 21:57:02 +00:00
s_outputRate . log ( now , outp . size ( ) ) ;
2019-08-30 16:08:39 +00:00
b - > second - > lastSend = now ;
2019-08-29 21:27:41 +00:00
}
if ( ( ( * a ) - > ip4 ) & & ( b - > second - > ip4 ) ) {
2019-08-29 15:00:54 +00:00
//printf("* introducing %s(%s) to %s(%s)" ZT_EOL_S,ip->toString(ipstr),source.toString(astr),b->second->ip4.toString(ipstr2),dest.toString(astr2));
2019-08-29 14:25:16 +00:00
2019-08-29 17:58:47 +00:00
// Introduce source to destination (V4)
2019-09-03 21:44:13 +00:00
Packet outp ( source , s_self . address ( ) , Packet : : VERB_RENDEZVOUS ) ;
2019-08-29 14:25:16 +00:00
outp . append ( ( uint8_t ) 0 ) ;
dest . appendTo ( outp ) ;
outp . append ( ( uint16_t ) b - > second - > ip4 . port ( ) ) ;
outp . append ( ( uint8_t ) 4 ) ;
outp . append ( ( const uint8_t * ) b - > second - > ip4 . rawIpData ( ) , 4 ) ;
outp . armor ( ( * a ) - > key , true ) ;
2019-08-31 04:17:34 +00:00
sendto ( v4s , outp . data ( ) , outp . size ( ) , SENDTO_FLAGS , ( const struct sockaddr * ) & ( ( * a ) - > ip4 ) , ( socklen_t ) sizeof ( struct sockaddr_in ) ) ;
2019-09-03 19:21:57 +00:00
2019-09-03 21:57:02 +00:00
s_outputRate . log ( now , outp . size ( ) ) ;
2019-08-30 16:08:39 +00:00
( * a ) - > lastSend = now ;
2019-08-29 14:25:16 +00:00
2019-08-29 17:58:47 +00:00
// Introduce destination to source (V4)
2019-09-03 21:44:13 +00:00
outp . reset ( dest , s_self . address ( ) , Packet : : VERB_RENDEZVOUS ) ;
2019-08-29 14:25:16 +00:00
outp . append ( ( uint8_t ) 0 ) ;
source . appendTo ( outp ) ;
outp . append ( ( uint16_t ) ip - > port ( ) ) ;
outp . append ( ( uint8_t ) 4 ) ;
outp . append ( ( const uint8_t * ) ip - > rawIpData ( ) , 4 ) ;
outp . armor ( b - > second - > key , true ) ;
2019-08-31 04:17:34 +00:00
sendto ( v4s , outp . data ( ) , outp . size ( ) , SENDTO_FLAGS , ( const struct sockaddr * ) & ( b - > second - > ip4 ) , ( socklen_t ) sizeof ( struct sockaddr_in ) ) ;
2019-09-03 19:21:57 +00:00
2019-09-03 21:57:02 +00:00
s_outputRate . log ( now , outp . size ( ) ) ;
2019-08-30 16:08:39 +00:00
b - > second - > lastSend = now ;
2019-08-29 14:25:16 +00:00
}
2019-08-29 14:17:18 +00:00
}
}
}
}
2019-08-28 20:04:58 +00:00
for ( auto i = toAddrs . begin ( ) ; i ! = toAddrs . end ( ) ; + + i ) {
2019-08-29 21:27:41 +00:00
//printf("%s -> %s for %s -> %s (%u bytes)" ZT_EOL_S,ip->toString(ipstr),i->first->toString(ipstr2),source.toString(astr),dest.toString(astr2),pkt.size());
2019-09-03 20:01:01 +00:00
if ( sendto ( i - > first - > isV4 ( ) ? v4s : v6s , pkt . data ( ) , pkt . size ( ) , SENDTO_FLAGS , ( const struct sockaddr * ) i - > first , ( socklen_t ) ( i - > first - > isV4 ( ) ? sizeof ( struct sockaddr_in ) : sizeof ( struct sockaddr_in6 ) ) ) > 0 ) {
2019-09-03 21:57:02 +00:00
s_outputRate . log ( now , pkt . size ( ) ) ;
s_forwardRate . log ( now , pkt . size ( ) ) ;
2019-08-30 16:08:39 +00:00
i - > second - > lastSend = now ;
2019-08-29 21:27:41 +00:00
}
2019-08-28 20:04:58 +00:00
}
2019-08-28 18:49:45 +00:00
}
2019-07-15 18:57:42 +00:00
2019-08-29 17:58:47 +00:00
//////////////////////////////////////////////////////////////////////////////
2019-08-29 15:00:54 +00:00
//////////////////////////////////////////////////////////////////////////////
2019-08-30 16:08:39 +00:00
static int bindSocket ( struct sockaddr * const bindAddr )
2019-07-15 18:57:42 +00:00
{
2019-08-30 16:08:39 +00:00
const int s = socket ( bindAddr - > sa_family , SOCK_DGRAM , 0 ) ;
2019-07-15 18:57:42 +00:00
if ( s < 0 ) {
close ( s ) ;
return - 1 ;
}
2019-09-06 14:37:44 +00:00
int f = 4194304 ;
while ( f > 131072 ) {
2019-08-29 18:23:23 +00:00
if ( setsockopt ( s , SOL_SOCKET , SO_RCVBUF , ( const char * ) & f , sizeof ( f ) ) = = 0 )
break ;
2019-09-06 14:37:44 +00:00
f - = 131072 ;
2019-08-29 18:23:23 +00:00
}
2019-09-06 14:37:44 +00:00
f = 4194304 ;
while ( f > 131072 ) {
2019-08-29 18:23:23 +00:00
if ( setsockopt ( s , SOL_SOCKET , SO_SNDBUF , ( const char * ) & f , sizeof ( f ) ) = = 0 )
break ;
2019-09-06 14:37:44 +00:00
f - = 131072 ;
2019-08-29 18:23:23 +00:00
}
2019-07-15 18:57:42 +00:00
if ( bindAddr - > sa_family = = AF_INET6 ) {
f = 1 ; setsockopt ( s , IPPROTO_IPV6 , IPV6_V6ONLY , ( void * ) & f , sizeof ( f ) ) ;
# ifdef IPV6_MTU_DISCOVER
f = 0 ; setsockopt ( s , IPPROTO_IPV6 , IPV6_MTU_DISCOVER , & f , sizeof ( f ) ) ;
# endif
# ifdef IPV6_DONTFRAG
f = 0 ; setsockopt ( s , IPPROTO_IPV6 , IPV6_DONTFRAG , & f , sizeof ( f ) ) ;
# endif
}
# ifdef IP_DONTFRAG
f = 0 ; setsockopt ( s , IPPROTO_IP , IP_DONTFRAG , & f , sizeof ( f ) ) ;
# endif
# ifdef IP_MTU_DISCOVER
f = IP_PMTUDISC_DONT ; setsockopt ( s , IPPROTO_IP , IP_MTU_DISCOVER , & f , sizeof ( f ) ) ;
# endif
2019-08-30 16:08:39 +00:00
2019-07-15 18:57:42 +00:00
# ifdef SO_NO_CHECK
if ( bindAddr - > sa_family = = AF_INET ) {
f = 1 ; setsockopt ( s , SOL_SOCKET , SO_NO_CHECK , ( void * ) & f , sizeof ( f ) ) ;
}
# endif
2019-08-30 16:08:39 +00:00
# if defined(SO_REUSEPORT)
f = 1 ; setsockopt ( s , SOL_SOCKET , SO_REUSEPORT , ( void * ) & f , sizeof ( f ) ) ;
# endif
# ifndef __LINUX__ // linux wants just SO_REUSEPORT
f = 1 ; setsockopt ( s , SOL_SOCKET , SO_REUSEADDR , ( void * ) & f , sizeof ( f ) ) ;
# endif
2019-07-15 18:57:42 +00:00
if ( bind ( s , bindAddr , ( bindAddr - > sa_family = = AF_INET ) ? sizeof ( struct sockaddr_in ) : sizeof ( struct sockaddr_in6 ) ) ) {
close ( s ) ;
2019-08-30 16:08:39 +00:00
//printf("%s\n",strerror(errno));
2019-07-15 18:57:42 +00:00
return - 1 ;
}
return s ;
}
2019-09-03 21:44:13 +00:00
static void shutdownSigHandler ( int sig ) { s_run = false ; }
2019-08-28 21:04:42 +00:00
2019-07-15 18:57:42 +00:00
int main ( int argc , char * * argv )
{
2019-09-06 14:37:44 +00:00
std : : vector < std : : thread > threads ;
std : : vector < int > sockets ;
int v4Sock = - 1 , v6Sock = - 1 ;
2019-08-28 21:04:42 +00:00
signal ( SIGTERM , shutdownSigHandler ) ;
signal ( SIGINT , shutdownSigHandler ) ;
signal ( SIGQUIT , shutdownSigHandler ) ;
signal ( SIGPIPE , SIG_IGN ) ;
signal ( SIGUSR1 , SIG_IGN ) ;
signal ( SIGUSR2 , SIG_IGN ) ;
2019-08-29 17:58:47 +00:00
signal ( SIGCHLD , SIG_IGN ) ;
2019-08-28 21:04:42 +00:00
2019-09-03 21:44:13 +00:00
s_startTime = OSUtils : : now ( ) ;
2019-09-06 14:37:44 +00:00
s_geoInit = false ;
2019-09-03 19:21:57 +00:00
2019-08-29 17:58:47 +00:00
if ( argc < 3 ) {
printf ( " Usage: zerotier-root <identity.secret> <config path> " ZT_EOL_S ) ;
2019-08-28 20:09:14 +00:00
return 1 ;
}
2019-08-29 17:58:47 +00:00
{
std : : string myIdStr ;
if ( ! OSUtils : : readFile ( argv [ 1 ] , myIdStr ) ) {
printf ( " FATAL: cannot read identity.secret at %s " ZT_EOL_S , argv [ 1 ] ) ;
return 1 ;
}
2019-09-03 21:44:13 +00:00
if ( ! s_self . fromString ( myIdStr . c_str ( ) ) ) {
2019-08-29 17:58:47 +00:00
printf ( " FATAL: cannot read identity.secret at %s (invalid identity) " ZT_EOL_S , argv [ 1 ] ) ;
return 1 ;
}
2019-09-03 21:44:13 +00:00
if ( ! s_self . hasPrivate ( ) ) {
2019-08-29 17:58:47 +00:00
printf ( " FATAL: cannot read identity.secret at %s (missing secret key) " ZT_EOL_S , argv [ 1 ] ) ;
return 1 ;
}
2019-08-28 20:09:14 +00:00
}
2019-08-29 17:58:47 +00:00
{
std : : string configStr ;
if ( ! OSUtils : : readFile ( argv [ 2 ] , configStr ) ) {
printf ( " FATAL: cannot read config file at %s " ZT_EOL_S , argv [ 2 ] ) ;
return 1 ;
}
try {
2019-09-03 21:44:13 +00:00
s_config = json : : parse ( configStr ) ;
2019-08-29 17:58:47 +00:00
} catch ( std : : exception & exc ) {
printf ( " FATAL: config file at %s invalid: %s " ZT_EOL_S , argv [ 2 ] , exc . what ( ) ) ;
return 1 ;
} catch ( . . . ) {
printf ( " FATAL: config file at %s invalid: unknown exception " ZT_EOL_S , argv [ 2 ] ) ;
return 1 ;
}
2019-09-03 21:44:13 +00:00
if ( ! s_config . is_object ( ) ) {
2019-08-29 17:58:47 +00:00
printf ( " FATAL: config file at %s invalid: does not contain a JSON object " ZT_EOL_S , argv [ 2 ] ) ;
return 1 ;
}
2019-08-28 20:09:14 +00:00
}
2019-08-29 17:58:47 +00:00
try {
2019-09-03 21:44:13 +00:00
auto jport = s_config [ " port " ] ;
2019-09-03 19:21:57 +00:00
if ( jport . is_array ( ) ) {
for ( long i = 0 ; i < ( long ) jport . size ( ) ; + + i ) {
int port = jport [ i ] ;
if ( ( port < = 0 ) | | ( port > 65535 ) ) {
printf ( " FATAL: invalid port in config file %d " ZT_EOL_S , port ) ;
return 1 ;
}
2019-09-03 21:44:13 +00:00
s_ports . push_back ( port ) ;
2019-09-03 19:21:57 +00:00
}
} else {
int port = jport ;
if ( ( port < = 0 ) | | ( port > 65535 ) ) {
printf ( " FATAL: invalid port in config file %d " ZT_EOL_S , port ) ;
return 1 ;
}
2019-09-03 21:44:13 +00:00
s_ports . push_back ( port ) ;
2019-08-29 17:58:47 +00:00
}
2019-09-03 19:21:57 +00:00
} catch ( . . . ) { }
2019-09-03 21:44:13 +00:00
if ( s_ports . empty ( ) )
s_ports . push_back ( ZT_DEFAULT_PORT ) ;
std : : sort ( s_ports . begin ( ) , s_ports . end ( ) ) ;
2019-09-03 19:21:57 +00:00
int httpPort = ZT_DEFAULT_PORT ;
2019-08-30 16:08:39 +00:00
try {
2019-09-03 21:44:13 +00:00
httpPort = s_config [ " httpPort " ] ;
2019-08-30 16:08:39 +00:00
if ( ( httpPort < = 0 ) | | ( httpPort > 65535 ) ) {
printf ( " FATAL: invalid HTTP port in config file %d " ZT_EOL_S , httpPort ) ;
return 1 ;
}
} catch ( . . . ) {
httpPort = ZT_DEFAULT_PORT ;
}
2019-09-03 19:21:57 +00:00
2019-09-03 21:44:13 +00:00
std : : string planetFilePath ;
2019-08-29 21:27:41 +00:00
try {
2019-09-03 21:44:13 +00:00
planetFilePath = s_config [ " planetFile " ] ;
2019-08-29 21:27:41 +00:00
} catch ( . . . ) {
2019-09-03 21:44:13 +00:00
planetFilePath = " " ;
2019-08-29 21:27:41 +00:00
}
2019-09-03 20:17:12 +00:00
try {
2019-09-03 21:44:13 +00:00
s_statsRoot = s_config [ " statsRoot " ] ;
while ( ( s_statsRoot . length ( ) > 0 ) & & ( s_statsRoot [ s_statsRoot . length ( ) - 1 ] = = ZT_PATH_SEPARATOR ) )
s_statsRoot = s_statsRoot . substr ( 0 , s_statsRoot . length ( ) - 1 ) ;
if ( s_statsRoot . length ( ) > 0 )
OSUtils : : mkdir ( s_statsRoot ) ;
2019-09-03 20:17:12 +00:00
} catch ( . . . ) {
2019-09-03 21:44:13 +00:00
s_statsRoot = " " ;
2019-09-03 20:17:12 +00:00
}
2019-09-03 21:44:13 +00:00
s_relayMaxHops = ZT_RELAY_MAX_HOPS ;
2019-09-03 19:21:57 +00:00
try {
2019-09-06 14:37:44 +00:00
s_relayMaxHops = s_config [ " relayMaxHops " ] ;
2019-09-03 21:44:13 +00:00
if ( s_relayMaxHops > ZT_PROTO_MAX_HOPS )
s_relayMaxHops = ZT_PROTO_MAX_HOPS ;
else if ( s_relayMaxHops < 0 )
s_relayMaxHops = 0 ;
} catch ( . . . ) {
s_relayMaxHops = ZT_RELAY_MAX_HOPS ;
}
try {
2019-09-06 14:37:44 +00:00
std : : string geoIpPath = s_config [ " geoIp " ] ;
if ( geoIpPath . length ( ) > 0 ) {
FILE * gf = fopen ( geoIpPath . c_str ( ) , " rb " ) ;
if ( gf ) {
threads . emplace_back ( std : : thread ( [ gf ] ( ) {
try {
char line [ 1024 ] ;
2019-09-06 14:55:36 +00:00
line [ 1023 ] = 0 ;
while ( fgets ( line , sizeof ( line ) - 1 , gf ) ) {
2019-09-06 14:37:44 +00:00
InetAddress start , end ;
2019-09-06 14:55:36 +00:00
float lat = 0.0F , lon = 0.0F ;
2019-09-06 14:37:44 +00:00
int field = 0 ;
for ( char * saveptr = nullptr , * f = Utils : : stok ( line , " , \r \n " , & saveptr ) ; ( f ) ; f = Utils : : stok ( nullptr , " , \r \n " , & saveptr ) ) {
switch ( field + + ) {
case 0 :
start . fromString ( f ) ;
break ;
case 1 :
end . fromString ( f ) ;
break ;
case 2 :
lat = strtof ( f , nullptr ) ;
break ;
case 3 :
lon = strtof ( f , nullptr ) ;
break ;
}
}
if ( ( start ) & & ( end ) & & ( start . ss_family = = end . ss_family ) & & ( lat > = - 90.0F ) & & ( lat < = 90.0F ) & & ( lon > = - 180.0F ) & & ( lon < = 180.0F ) ) {
if ( start . ss_family = = AF_INET ) {
s_geoIp4 [ std : : pair < uint32_t , uint32_t > ( ip4ToH32 ( start ) , ip4ToH32 ( end ) ) ] = std : : pair < float , float > ( lat , lon ) ;
} else if ( start . ss_family = = AF_INET6 ) {
s_geoIp6 [ std : : pair < std : : array < uint64_t , 2 > , std : : array < uint64_t , 2 > > ( ip6ToH128 ( start ) , ip6ToH128 ( end ) ) ] = std : : pair < float , float > ( lat , lon ) ;
}
}
2019-09-03 21:57:54 +00:00
}
2019-09-06 14:37:44 +00:00
s_geoInit = true ;
} catch ( . . . ) { }
fclose ( gf ) ;
} ) ) ;
2019-09-03 19:21:57 +00:00
}
}
2019-09-06 14:37:44 +00:00
} catch ( . . . ) { }
2019-08-28 20:09:14 +00:00
2019-07-15 18:57:42 +00:00
unsigned int ncores = std : : thread : : hardware_concurrency ( ) ;
if ( ncores = = 0 ) ncores = 1 ;
2019-09-03 21:44:13 +00:00
s_run = true ;
2019-08-28 18:59:13 +00:00
2019-09-03 21:44:13 +00:00
for ( auto port = s_ports . begin ( ) ; port ! = s_ports . end ( ) ; + + port ) {
2019-09-03 19:21:57 +00:00
for ( unsigned int tn = 0 ; tn < ncores ; + + tn ) {
2019-07-15 18:57:42 +00:00
struct sockaddr_in6 in6 ;
memset ( & in6 , 0 , sizeof ( in6 ) ) ;
2019-09-03 19:21:57 +00:00
in6 . sin6_family = AF_INET6 ;
in6 . sin6_port = htons ( ( uint16_t ) * port ) ;
const int s6 = bindSocket ( ( struct sockaddr * ) & in6 ) ;
if ( s6 < 0 ) {
std : : cout < < " ERROR: unable to bind to port " < < * port < < ZT_EOL_S ;
exit ( 1 ) ;
2019-07-15 18:57:42 +00:00
}
struct sockaddr_in in4 ;
memset ( & in4 , 0 , sizeof ( in4 ) ) ;
2019-09-03 19:21:57 +00:00
in4 . sin_family = AF_INET ;
in4 . sin_port = htons ( ( uint16_t ) * port ) ;
const int s4 = bindSocket ( ( struct sockaddr * ) & in4 ) ;
if ( s4 < 0 ) {
std : : cout < < " ERROR: unable to bind to port " < < * port < < ZT_EOL_S ;
exit ( 1 ) ;
}
sockets . push_back ( s6 ) ;
sockets . push_back ( s4 ) ;
if ( v4Sock < 0 ) v4Sock = s4 ;
if ( v6Sock < 0 ) v6Sock = s6 ;
threads . push_back ( std : : thread ( [ s6 , s4 ] ( ) {
struct sockaddr_in6 in6 ;
Packet pkt ;
memset ( & in6 , 0 , sizeof ( in6 ) ) ;
for ( ; ; ) {
socklen_t sl = sizeof ( in6 ) ;
const int pl = ( int ) recvfrom ( s6 , pkt . unsafeData ( ) , pkt . capacity ( ) , 0 , ( struct sockaddr * ) & in6 , & sl ) ;
if ( pl > 0 ) {
if ( pl > = ZT_PROTO_MIN_FRAGMENT_LENGTH ) {
try {
pkt . setSize ( ( unsigned int ) pl ) ;
handlePacket ( s4 , s6 , reinterpret_cast < const InetAddress * > ( & in6 ) , pkt ) ;
} catch ( . . . ) {
char ipstr [ 128 ] ;
printf ( " * unexpected exception handling packet from %s " ZT_EOL_S , reinterpret_cast < const InetAddress * > ( & in6 ) - > toString ( ipstr ) ) ;
}
2019-08-28 22:22:28 +00:00
}
2019-09-03 19:21:57 +00:00
} else {
break ;
2019-08-28 18:49:45 +00:00
}
}
2019-09-03 19:21:57 +00:00
} ) ) ;
threads . push_back ( std : : thread ( [ s6 , s4 ] ( ) {
struct sockaddr_in in4 ;
Packet pkt ;
memset ( & in4 , 0 , sizeof ( in4 ) ) ;
for ( ; ; ) {
socklen_t sl = sizeof ( in4 ) ;
const int pl = ( int ) recvfrom ( s4 , pkt . unsafeData ( ) , pkt . capacity ( ) , 0 , ( struct sockaddr * ) & in4 , & sl ) ;
if ( pl > 0 ) {
if ( pl > = ZT_PROTO_MIN_FRAGMENT_LENGTH ) {
try {
pkt . setSize ( ( unsigned int ) pl ) ;
handlePacket ( s4 , s6 , reinterpret_cast < const InetAddress * > ( & in4 ) , pkt ) ;
} catch ( . . . ) {
char ipstr [ 128 ] ;
printf ( " * unexpected exception handling packet from %s " ZT_EOL_S , reinterpret_cast < const InetAddress * > ( & in4 ) - > toString ( ipstr ) ) ;
}
}
} else {
break ;
}
}
} ) ) ;
}
2019-07-15 18:57:42 +00:00
}
2019-09-03 19:21:57 +00:00
// Minimal local API for use with monitoring clients, etc.
2019-08-30 16:08:39 +00:00
httplib : : Server apiServ ;
threads . push_back ( std : : thread ( [ & apiServ , httpPort ] ( ) {
2019-08-30 16:29:06 +00:00
apiServ . Get ( " / " , [ ] ( const httplib : : Request & req , httplib : : Response & res ) {
std : : ostringstream o ;
2019-09-03 21:57:02 +00:00
std : : lock_guard < std : : mutex > l0 ( s_peersByIdentity_l ) ;
std : : lock_guard < std : : mutex > l1 ( s_peersByPhysAddr_l ) ;
2019-08-30 16:29:06 +00:00
o < < " ZeroTier Root Server " < < ZEROTIER_ONE_VERSION_MAJOR < < ' . ' < < ZEROTIER_ONE_VERSION_MINOR < < ' . ' < < ZEROTIER_ONE_VERSION_REVISION < < ZT_EOL_S ;
o < < " (c)2019 ZeroTier, Inc. " ZT_EOL_S " Licensed under the ZeroTier BSL 1.1 " ZT_EOL_S ZT_EOL_S ;
2019-09-03 21:57:02 +00:00
o < < " Peers Online: " < < s_peersByIdentity . size ( ) < < ZT_EOL_S ;
o < < " Physical Addresses: " < < s_peersByPhysAddr . size ( ) < < ZT_EOL_S ;
2019-08-30 16:29:06 +00:00
res . set_content ( o . str ( ) , " text/plain " ) ;
} ) ;
2019-09-06 14:37:44 +00:00
2019-08-30 16:08:39 +00:00
apiServ . Get ( " /peer " , [ ] ( const httplib : : Request & req , httplib : : Response & res ) {
char tmp [ 256 ] ;
std : : ostringstream o ;
o < < ' [ ' ;
{
2019-08-30 16:38:18 +00:00
bool first = true ;
2019-09-03 21:57:02 +00:00
std : : lock_guard < std : : mutex > l ( s_peersByIdentity_l ) ;
for ( auto p = s_peersByIdentity . begin ( ) ; p ! = s_peersByIdentity . end ( ) ; + + p ) {
2019-08-30 16:39:46 +00:00
if ( first )
2019-08-30 16:38:18 +00:00
first = false ;
2019-08-30 16:39:46 +00:00
else o < < ' , ' ;
2019-08-30 16:08:39 +00:00
o < <
" { \" address \" : \" " < < p - > first . address ( ) . toString ( tmp ) < < " \" "
" , \" latency \" :-1 "
" , \" paths \" :[ " ;
if ( p - > second - > ip4 ) {
o < <
" { \" active \" :true "
" , \" address \" : \" " < < p - > second - > ip4 . toIpString ( tmp ) < < " \\ / " < < p - > second - > ip4 . port ( ) < < " \" "
" , \" expired \" :false "
" , \" lastReceive \" : " < < p - > second - > lastReceive < <
" , \" lastSend \" : " < < p - > second - > lastSend < <
" , \" preferred \" :true "
" , \" trustedPathId \" :0} " ;
}
if ( p - > second - > ip6 ) {
2019-08-30 16:38:18 +00:00
if ( p - > second - > ip4 )
o < < ' , ' ;
2019-08-30 16:08:39 +00:00
o < <
" { \" active \" :true "
" , \" address \" : \" " < < p - > second - > ip6 . toIpString ( tmp ) < < " \\ / " < < p - > second - > ip6 . port ( ) < < " \" "
" , \" expired \" :false "
" , \" lastReceive \" : " < < p - > second - > lastReceive < <
" , \" lastSend \" : " < < p - > second - > lastSend < <
" , \" preferred \" : " < < ( ( p - > second - > ip4 ) ? " false " : " true " ) < <
" , \" trustedPathId \" :0} " ;
}
o < < " ] "
" , \" role \" : \" LEAF \" "
" , \" version \" : \" " < < p - > second - > vMajor < < ' . ' < < p - > second - > vMinor < < ' . ' < < p - > second - > vRev < < " \" "
" , \" versionMajor \" : " < < p - > second - > vMajor < <
" , \" versionMinor \" : " < < p - > second - > vMinor < <
" , \" versionRev \" : " < < p - > second - > vRev < < " } " ;
}
}
o < < ' ] ' ;
res . set_content ( o . str ( ) , " application/json " ) ;
} ) ;
2019-09-06 14:37:44 +00:00
apiServ . Get ( " /map " , [ ] ( const httplib : : Request & req , httplib : : Response & res ) {
char tmp [ 128 ] ;
if ( ! s_geoInit ) {
res . set_content ( " Not enabled or GeoIP CSV file not finished reading. " , " text/plain " ) ;
return ;
}
std : : ostringstream o ;
o < < ZT_GEOIP_HTML_HEAD ;
{
bool firstCoord = true ;
2019-09-06 14:55:36 +00:00
std : : pair < uint32_t , uint32_t > k4 ( 0 , 0xffffffff ) ;
2019-09-06 14:37:44 +00:00
std : : pair < std : : array < uint64_t , 2 > , std : : array < uint64_t , 2 > > k6 ;
2019-09-06 14:55:36 +00:00
k6 . second [ 0 ] = 0xffffffffffffffffULL ; k6 . second [ 1 ] = 0xffffffffffffffffULL ;
2019-09-06 14:37:44 +00:00
std : : lock_guard < std : : mutex > l ( s_peersByPhysAddr_l ) ;
for ( auto p = s_peersByPhysAddr . begin ( ) ; p ! = s_peersByPhysAddr . end ( ) ; + + p ) {
if ( ! p - > second . empty ( ) ) {
2019-09-06 14:55:36 +00:00
if ( p - > first . isV4 ( ) ) {
k4 . first = ip4ToH32 ( p - > first ) ;
auto geo = std : : make_reverse_iterator ( s_geoIp4 . upper_bound ( k4 ) ) ;
while ( geo ! = s_geoIp4 . rend ( ) ) {
if ( ( geo - > first . first < = k4 . first ) & & ( geo - > first . second > = k4 . first ) ) {
if ( ! firstCoord )
2019-09-06 14:37:44 +00:00
o < < ' , ' ;
2019-09-06 14:55:36 +00:00
firstCoord = false ;
o < < " {lat: " < < geo - > second . first < < " ,lng: " < < geo - > second . second < < " ,_l: \" " ;
bool firstAddr = true ;
for ( auto a = p - > second . begin ( ) ; a ! = p - > second . end ( ) ; + + a ) {
if ( ! firstAddr )
o < < ' , ' ;
o < < ( * a ) - > id . address ( ) . toString ( tmp ) ;
firstAddr = false ;
}
o < < " \" } " ;
break ;
} else if ( ( geo - > first . first < k4 . first ) & & ( geo - > first . second < k4 . first ) ) {
break ;
}
+ + geo ;
}
} else if ( p - > first . isV6 ( ) ) {
k6 . first = ip6ToH128 ( p - > first ) ;
auto geo = std : : make_reverse_iterator ( s_geoIp6 . upper_bound ( k6 ) ) ;
while ( geo ! = s_geoIp6 . rend ( ) ) {
if ( ( geo - > first . first < = k6 . first ) & & ( geo - > first . second > = k6 . first ) ) {
if ( ! firstCoord )
o < < ' , ' ;
firstCoord = false ;
o < < " {lat: " < < geo - > second . first < < " ,lng: " < < geo - > second . second < < " ,_l: \" " ;
bool firstAddr = true ;
for ( auto a = p - > second . begin ( ) ; a ! = p - > second . end ( ) ; + + a ) {
if ( ! firstAddr )
o < < ' , ' ;
o < < ( * a ) - > id . address ( ) . toString ( tmp ) ;
firstAddr = false ;
}
o < < " \" } " ;
break ;
} else if ( ( geo - > first . first < k6 . first ) & & ( geo - > first . second < k6 . first ) ) {
break ;
2019-09-06 14:37:44 +00:00
}
2019-09-06 14:55:36 +00:00
+ + geo ;
2019-09-06 14:37:44 +00:00
}
}
}
}
}
o < < ZT_GEOIP_HTML_TAIL ;
res . set_content ( o . str ( ) , " text/html " ) ;
} ) ;
2019-08-30 16:08:39 +00:00
apiServ . listen ( " 127.0.0.1 " , httpPort , 0 ) ;
} ) ) ;
2019-09-03 19:21:57 +00:00
// In the main thread periodically clean stuff up
int64_t lastCleaned = 0 ;
2019-08-29 21:27:41 +00:00
int64_t lastWroteStats = 0 ;
2019-09-03 21:44:13 +00:00
while ( s_run ) {
2019-09-03 21:57:02 +00:00
//s_peersByIdentity_l.lock();
//s_peersByPhysAddr_l.lock();
//printf("*** have %lu peers at %lu physical endpoints" ZT_EOL_S,(unsigned long)s_peersByIdentity.size(),(unsigned long)s_peersByPhysAddr.size());
//s_peersByPhysAddr_l.unlock();
//s_peersByIdentity_l.unlock();
2019-08-28 18:59:13 +00:00
sleep ( 1 ) ;
2019-08-28 22:15:21 +00:00
const int64_t now = OSUtils : : now ( ) ;
2019-08-28 22:52:18 +00:00
2019-09-03 19:21:57 +00:00
if ( ( now - lastCleaned ) > 120000 ) {
lastCleaned = now ;
2019-08-28 22:52:18 +00:00
2019-09-03 19:21:57 +00:00
// Old multicast subscription cleanup
{
2019-09-03 21:57:02 +00:00
std : : lock_guard < std : : mutex > l ( s_multicastSubscriptions_l ) ;
for ( auto a = s_multicastSubscriptions . begin ( ) ; a ! = s_multicastSubscriptions . end ( ) ; ) {
2019-09-03 19:21:57 +00:00
for ( auto b = a - > second . begin ( ) ; b ! = a - > second . end ( ) ; ) {
for ( auto c = b - > second . begin ( ) ; c ! = b - > second . end ( ) ; ) {
if ( ( now - c - > second ) > ZT_MULTICAST_LIKE_EXPIRE )
b - > second . erase ( c + + ) ;
else + + c ;
}
if ( b - > second . empty ( ) )
a - > second . erase ( b + + ) ;
else + + b ;
}
if ( a - > second . empty ( ) )
2019-09-03 21:57:02 +00:00
s_multicastSubscriptions . erase ( a + + ) ;
2019-09-03 19:21:57 +00:00
else + + a ;
}
}
// Remove expired peers
2019-08-29 21:27:41 +00:00
{
2019-09-03 21:57:02 +00:00
std : : lock_guard < std : : mutex > pbi_l ( s_peersByIdentity_l ) ;
for ( auto p = s_peersByIdentity . begin ( ) ; p ! = s_peersByIdentity . end ( ) ; ) {
2019-09-06 14:37:44 +00:00
if ( ( now - p - > second - > lastReceive ) > ZT_PEER_ACTIVITY_TIMEOUT ) {
2019-09-03 21:57:02 +00:00
std : : lock_guard < std : : mutex > pbv_l ( s_peersByVirtAddr_l ) ;
std : : lock_guard < std : : mutex > pbp_l ( s_peersByPhysAddr_l ) ;
2019-08-29 21:27:41 +00:00
2019-09-03 21:57:02 +00:00
auto pbv = s_peersByVirtAddr . find ( p - > second - > id . address ( ) ) ;
if ( pbv ! = s_peersByVirtAddr . end ( ) ) {
2019-08-29 21:27:41 +00:00
pbv - > second . erase ( p - > second ) ;
if ( pbv - > second . empty ( ) )
2019-09-03 21:57:02 +00:00
s_peersByVirtAddr . erase ( pbv ) ;
2019-08-29 21:27:41 +00:00
}
2019-08-28 22:52:18 +00:00
2019-08-29 21:27:41 +00:00
if ( p - > second - > ip4 ) {
2019-09-03 21:57:02 +00:00
auto pbp = s_peersByPhysAddr . find ( p - > second - > ip4 ) ;
if ( pbp ! = s_peersByPhysAddr . end ( ) ) {
2019-08-29 21:27:41 +00:00
pbp - > second . erase ( p - > second ) ;
if ( pbp - > second . empty ( ) )
2019-09-03 21:57:02 +00:00
s_peersByPhysAddr . erase ( pbp ) ;
2019-08-29 21:27:41 +00:00
}
2019-08-28 22:52:18 +00:00
}
2019-08-29 21:27:41 +00:00
if ( p - > second - > ip6 ) {
2019-09-03 21:57:02 +00:00
auto pbp = s_peersByPhysAddr . find ( p - > second - > ip6 ) ;
if ( pbp ! = s_peersByPhysAddr . end ( ) ) {
2019-08-29 21:27:41 +00:00
pbp - > second . erase ( p - > second ) ;
if ( pbp - > second . empty ( ) )
2019-09-03 21:57:02 +00:00
s_peersByPhysAddr . erase ( pbp ) ;
2019-08-29 21:27:41 +00:00
}
2019-08-28 22:52:18 +00:00
}
2019-08-29 21:27:41 +00:00
2019-09-03 21:57:02 +00:00
s_peersByIdentity . erase ( p + + ) ;
2019-08-29 21:27:41 +00:00
} else + + p ;
}
}
2019-09-03 22:19:17 +00:00
// Remove old rendezvous and last forwarded tracking entries
2019-08-29 21:27:41 +00:00
{
2019-09-06 14:37:44 +00:00
std : : lock_guard < std : : mutex > l ( s_rendezvousTracking_l ) ;
for ( auto lr = s_rendezvousTracking . begin ( ) ; lr ! = s_rendezvousTracking . end ( ) ; ) {
2019-09-03 22:19:17 +00:00
if ( ( now - lr - > second . ts ) > ZT_PEER_ACTIVITY_TIMEOUT )
2019-09-06 14:37:44 +00:00
s_rendezvousTracking . erase ( lr + + ) ;
2019-08-29 21:27:41 +00:00
else + + lr ;
}
}
2019-09-03 22:19:17 +00:00
// Remove old last forwarded tracking entries
{
std : : lock_guard < std : : mutex > l ( s_lastForwardedTo_l ) ;
for ( auto lf = s_lastForwardedTo . begin ( ) ; lf ! = s_lastForwardedTo . end ( ) ; ) {
if ( ( now - lf - > second . ts ) > ZT_PEER_ACTIVITY_TIMEOUT )
s_lastForwardedTo . erase ( lf + + ) ;
else + + lf ;
}
}
2019-08-29 21:27:41 +00:00
}
2019-09-03 21:44:13 +00:00
// Write stats if configured to do so, and periodically refresh planet file (if any)
if ( ( ( now - lastWroteStats ) > 15000 ) & & ( s_statsRoot . length ( ) > 0 ) ) {
2019-08-29 21:27:41 +00:00
lastWroteStats = now ;
2019-09-03 21:44:13 +00:00
try {
if ( planetFilePath . length ( ) > 0 ) {
std : : string planetData ;
if ( ( OSUtils : : readFile ( planetFilePath . c_str ( ) , planetData ) ) & & ( planetData . length ( ) > 0 ) ) {
std : : lock_guard < std : : mutex > pl ( s_planet_l ) ;
s_planet = planetData ;
}
}
} catch ( . . . ) {
std : : lock_guard < std : : mutex > pl ( s_planet_l ) ;
s_planet . clear ( ) ;
}
std : : string peersFilePath ( s_statsRoot ) ;
2019-09-03 19:21:57 +00:00
peersFilePath . append ( " /.peers.tmp " ) ;
2019-08-29 21:27:41 +00:00
FILE * pf = fopen ( peersFilePath . c_str ( ) , " wb " ) ;
if ( pf ) {
std : : vector < SharedPtr < RootPeer > > sp ;
{
2019-09-03 21:57:02 +00:00
std : : lock_guard < std : : mutex > pbi_l ( s_peersByIdentity_l ) ;
sp . reserve ( s_peersByIdentity . size ( ) ) ;
for ( auto p = s_peersByIdentity . begin ( ) ; p ! = s_peersByIdentity . end ( ) ; + + p ) {
2019-08-29 21:27:41 +00:00
sp . push_back ( p - > second ) ;
2019-08-28 22:52:18 +00:00
}
2019-08-29 21:27:41 +00:00
}
std : : sort ( sp . begin ( ) , sp . end ( ) , [ ] ( const SharedPtr < RootPeer > & a , const SharedPtr < RootPeer > & b ) { return ( a - > id < b - > id ) ; } ) ;
2019-09-03 22:59:34 +00:00
fprintf ( pf , " Address %21s %45s %10s %6s %10s " ZT_EOL_S , " IPv4 " , " IPv6 " , " Age(sec) " , " Vers " , " Fwd(KiB/s) " ) ;
2019-09-03 22:48:10 +00:00
{
std : : lock_guard < std : : mutex > lf_l ( s_lastForwardedTo_l ) ;
char ip4 [ 128 ] , ip6 [ 128 ] , ver [ 128 ] ;
for ( auto p = sp . begin ( ) ; p ! = sp . end ( ) ; + + p ) {
if ( ( * p ) - > ip4 ) {
( * p ) - > ip4 . toString ( ip4 ) ;
} else {
ip4 [ 0 ] = ' - ' ;
ip4 [ 1 ] = 0 ;
}
if ( ( * p ) - > ip6 ) {
( * p ) - > ip6 . toString ( ip6 ) ;
} else {
ip6 [ 0 ] = ' - ' ;
ip6 [ 1 ] = 0 ;
}
OSUtils : : ztsnprintf ( ver , sizeof ( ver ) , " %d.%d.%d " , ( * p ) - > vMajor , ( * p ) - > vMinor , ( * p ) - > vRev ) ;
double forwardingSpeed = 0.0 ;
auto lft = s_lastForwardedTo . find ( ( * p ) - > id . address ( ) ) ;
if ( lft ! = s_lastForwardedTo . end ( ) )
forwardingSpeed = lft - > second . bps . perSecond ( now ) / 1024.0 ;
2019-09-03 22:55:23 +00:00
fprintf ( pf , " %.10llx %21s %45s %10.4f %6s %10.4f " ZT_EOL_S ,
2019-09-03 22:48:10 +00:00
( unsigned long long ) ( * p ) - > id . address ( ) . toInt ( ) ,
ip4 ,
ip6 ,
fabs ( ( double ) ( now - ( * p ) - > lastReceive ) / 1000.0 ) ,
ver ,
forwardingSpeed ) ;
2019-08-29 21:27:41 +00:00
}
}
2019-08-28 22:52:18 +00:00
2019-09-03 19:21:57 +00:00
fclose ( pf ) ;
2019-09-03 21:44:13 +00:00
std : : string peersFilePath2 ( s_statsRoot ) ;
2019-08-29 21:27:41 +00:00
peersFilePath2 . append ( " /peers " ) ;
OSUtils : : rm ( peersFilePath2 ) ;
OSUtils : : rename ( peersFilePath . c_str ( ) , peersFilePath2 . c_str ( ) ) ;
2019-08-28 22:52:18 +00:00
}
2019-09-03 19:21:57 +00:00
2019-09-03 21:44:13 +00:00
std : : string statsFilePath ( s_statsRoot ) ;
2019-09-03 19:21:57 +00:00
statsFilePath . append ( " /.stats.tmp " ) ;
FILE * sf = fopen ( statsFilePath . c_str ( ) , " wb " ) ;
if ( sf ) {
2019-09-03 21:44:13 +00:00
fprintf ( sf , " Uptime (seconds) : %ld " ZT_EOL_S , ( long ) ( ( now - s_startTime ) / 1000 ) ) ;
2019-09-03 21:57:02 +00:00
s_peersByIdentity_l . lock ( ) ;
fprintf ( sf , " Peers : %llu " ZT_EOL_S , ( unsigned long long ) s_peersByIdentity . size ( ) ) ;
s_peersByVirtAddr_l . lock ( ) ;
fprintf ( sf , " Virtual Address Collisions : %llu " ZT_EOL_S , ( unsigned long long ) ( s_peersByIdentity . size ( ) - s_peersByVirtAddr . size ( ) ) ) ;
s_peersByVirtAddr_l . unlock ( ) ;
s_peersByIdentity_l . unlock ( ) ;
s_peersByPhysAddr_l . lock ( ) ;
fprintf ( sf , " Physical Endpoints : %llu " ZT_EOL_S , ( unsigned long long ) s_peersByPhysAddr . size ( ) ) ;
s_peersByPhysAddr_l . unlock ( ) ;
2019-09-06 14:37:44 +00:00
s_rendezvousTracking_l . lock ( ) ;
2019-09-03 22:19:17 +00:00
uint64_t unsuccessfulp2p = 0 ;
2019-09-06 14:37:44 +00:00
for ( auto lr = s_rendezvousTracking . begin ( ) ; lr ! = s_rendezvousTracking . end ( ) ; + + lr ) {
2019-09-03 22:48:10 +00:00
if ( lr - > second . count > 6 ) // 6 == two attempts per edge, one for each direction
2019-09-03 22:19:17 +00:00
+ + unsuccessfulp2p ;
}
2019-09-06 14:37:44 +00:00
fprintf ( sf , " Recent P2P Graph Edges : %llu " ZT_EOL_S , ( unsigned long long ) s_rendezvousTracking . size ( ) ) ;
if ( s_rendezvousTracking . empty ( ) ) {
fprintf ( sf , " Recent P2P Success Rate : 100.0000%% " ZT_EOL_S ) ;
2019-09-03 22:19:17 +00:00
} else {
2019-09-06 14:37:44 +00:00
fprintf ( sf , " Recent P2P Success Rate : %.4f%% " ZT_EOL_S , ( 1.0 - ( ( double ) unsuccessfulp2p / ( double ) s_rendezvousTracking . size ( ) ) ) * 100.0 ) ;
2019-09-03 22:19:17 +00:00
}
2019-09-06 14:37:44 +00:00
s_rendezvousTracking_l . unlock ( ) ;
2019-09-03 22:19:17 +00:00
fprintf ( sf , " Input (MiB/s) : %.4f " ZT_EOL_S , s_inputRate . perSecond ( now ) / 1048576.0 ) ;
fprintf ( sf , " Output (MiB/s) : %.4f " ZT_EOL_S , s_outputRate . perSecond ( now ) / 1048576.0 ) ;
fprintf ( sf , " Forwarded (MiB/s) : %.4f " ZT_EOL_S , s_forwardRate . perSecond ( now ) / 1048576.0 ) ;
fprintf ( sf , " Discarded Forward (MiB/s) : %.4f " ZT_EOL_S , s_discardedForwardRate . perSecond ( now ) / 1048576.0 ) ;
2019-09-03 19:21:57 +00:00
fclose ( sf ) ;
2019-09-03 21:44:13 +00:00
std : : string statsFilePath2 ( s_statsRoot ) ;
2019-09-03 19:21:57 +00:00
statsFilePath2 . append ( " /stats " ) ;
OSUtils : : rm ( statsFilePath2 ) ;
OSUtils : : rename ( statsFilePath . c_str ( ) , statsFilePath2 . c_str ( ) ) ;
}
2019-08-28 22:52:18 +00:00
}
2019-08-28 18:59:13 +00:00
}
2019-09-03 19:21:57 +00:00
// If we received a kill signal, close everything and wait
// for threads to die before exiting.
2019-08-30 16:08:39 +00:00
apiServ . stop ( ) ;
2019-08-28 21:06:45 +00:00
for ( auto s = sockets . begin ( ) ; s ! = sockets . end ( ) ; + + s ) {
shutdown ( * s , SHUT_RDWR ) ;
close ( * s ) ;
}
for ( auto t = threads . begin ( ) ; t ! = threads . end ( ) ; + + t )
t - > join ( ) ;
2019-07-15 18:57:42 +00:00
return 0 ;
}