2018-08-31 21:58:15 +00:00
/*
* ZeroTier One - Network Virtualization Everywhere
* Copyright ( C ) 2011 - 2018 ZeroTier , Inc .
*
* This program 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 .
*
* This program 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 this program . If not , see < http : //www.gnu.org/licenses/>.
*/
# ifdef ZT_CONTROLLER_USE_LIBPQ
# include "PostgreSQL.hpp"
# include "EmbeddedNetworkController.hpp"
# include "../version.h"
# include <libpq-fe.h>
2018-09-28 17:55:39 +00:00
# include <sstream>
2018-08-31 21:58:15 +00:00
using json = nlohmann : : json ;
namespace {
static const char * _timestr ( )
{
time_t t = time ( 0 ) ;
char * ts = ctime ( & t ) ;
char * p = ts ;
if ( ! p )
return " " ;
while ( * p ) {
if ( * p = = ' \n ' ) {
* p = ( char ) 0 ;
break ;
}
+ + p ;
}
return ts ;
}
2018-09-06 22:14:16 +00:00
std : : string join ( const std : : vector < std : : string > & elements , const char * const separator )
{
switch ( elements . size ( ) ) {
case 0 :
return " " ;
case 1 :
return elements [ 0 ] ;
default :
std : : ostringstream os ;
std : : copy ( elements . begin ( ) , elements . end ( ) - 1 , std : : ostream_iterator < std : : string > ( os , separator ) ) ;
os < < * elements . rbegin ( ) ;
return os . str ( ) ;
}
}
2018-08-31 21:58:15 +00:00
}
using namespace ZeroTier ;
PostgreSQL : : PostgreSQL ( EmbeddedNetworkController * const nc , const Identity & myId , const char * path )
: DB ( nc , myId , path )
, _ready ( 0 )
, _connected ( 1 )
, _run ( 1 )
, _waitNoticePrinted ( false )
{
_connString = std : : string ( path ) ;
_readyLock . lock ( ) ;
_heartbeatThread = std : : thread ( & PostgreSQL : : heartbeat , this ) ;
_membersDbWatcher = std : : thread ( & PostgreSQL : : membersDbWatcher , this ) ;
_networksDbWatcher = std : : thread ( & PostgreSQL : : networksDbWatcher , this ) ;
for ( int i = 0 ; i < ZT_CONTROLLER_RETHINKDB_COMMIT_THREADS ; + + i ) {
_commitThread [ i ] = std : : thread ( & PostgreSQL : : commitThread , this ) ;
}
_onlineNotificationThread = std : : thread ( & PostgreSQL : : onlineNotificationThread , this ) ;
}
PostgreSQL : : ~ PostgreSQL ( )
{
_run = 0 ;
std : : this_thread : : sleep_for ( std : : chrono : : milliseconds ( 100 ) ) ;
_heartbeatThread . join ( ) ;
_membersDbWatcher . join ( ) ;
_networksDbWatcher . join ( ) ;
for ( int i = 0 ; i < ZT_CONTROLLER_RETHINKDB_COMMIT_THREADS ; + + i ) {
_commitThread [ i ] . join ( ) ;
}
_onlineNotificationThread . join ( ) ;
}
bool PostgreSQL : : waitForReady ( )
{
while ( _ready < 2 ) {
if ( ! _waitNoticePrinted ) {
_waitNoticePrinted = true ;
fprintf ( stderr , " [%s] NOTICE: %.10llx controller PostgreSQL waiting for initial data download... " ZT_EOL_S , : : _timestr ( ) , ( unsigned long long ) _myAddress . toInt ( ) ) ;
}
_readyLock . lock ( ) ;
_readyLock . unlock ( ) ;
}
return true ;
}
bool PostgreSQL : : isReady ( )
{
return ( ( _ready = = 2 ) & & ( _connected ) ) ;
}
void PostgreSQL : : save ( nlohmann : : json * orig , nlohmann : : json & record )
{
2018-10-24 19:06:17 +00:00
try {
if ( ! record . is_object ( ) ) {
return ;
}
waitForReady ( ) ;
if ( orig ) {
if ( * orig ! = record ) {
record [ " revision " ] = OSUtils : : jsonInt ( record [ " revision " ] , 0ULL ) + 1 ;
_commitQueue . post ( new nlohmann : : json ( record ) ) ;
}
} else {
record [ " revision " ] = 1 ;
2018-08-31 21:58:15 +00:00
_commitQueue . post ( new nlohmann : : json ( record ) ) ;
}
2018-10-24 19:06:17 +00:00
} catch ( std : : exception & e ) {
fprintf ( stderr , " Error on PostgreSQL::save: %s \n " , e . what ( ) ) ;
} catch ( . . . ) {
fprintf ( stderr , " Unknown error on PostgreSQL::save \n " ) ;
2018-08-31 21:58:15 +00:00
}
}
void PostgreSQL : : eraseNetwork ( const uint64_t networkId )
{
char tmp2 [ 24 ] ;
waitForReady ( ) ;
Utils : : hex ( networkId , tmp2 ) ;
json * tmp = new json ( ) ;
( * tmp ) [ " id " ] = tmp2 ;
( * tmp ) [ " objtype " ] = " _delete_network " ;
_commitQueue . post ( tmp ) ;
}
void PostgreSQL : : eraseMember ( const uint64_t networkId , const uint64_t memberId )
{
char tmp2 [ 24 ] ;
json * tmp = new json ( ) ;
Utils : : hex ( networkId , tmp2 ) ;
( * tmp ) [ " nwid " ] = tmp2 ;
Utils : : hex ( memberId , tmp2 ) ;
( * tmp ) [ " id " ] = tmp2 ;
( * tmp ) [ " objtype " ] = " _delete_member " ;
_commitQueue . post ( tmp ) ;
}
void PostgreSQL : : nodeIsOnline ( const uint64_t networkId , const uint64_t memberId , const InetAddress & physicalAddress )
{
std : : lock_guard < std : : mutex > l ( _lastOnline_l ) ;
std : : pair < int64_t , InetAddress > & i = _lastOnline [ std : : pair < uint64_t , uint64_t > ( networkId , memberId ) ] ;
i . first = OSUtils : : now ( ) ;
if ( physicalAddress ) {
i . second = physicalAddress ;
}
}
void PostgreSQL : : initializeNetworks ( PGconn * conn )
{
2018-09-05 18:49:07 +00:00
try {
if ( PQstatus ( conn ) ! = CONNECTION_OK ) {
fprintf ( stderr , " Bad Database Connection: %s " , PQerrorMessage ( conn ) ) ;
exit ( 1 ) ;
}
2018-09-04 21:00:02 +00:00
2018-09-05 18:49:07 +00:00
const char * params [ 1 ] = {
_myAddressStr . c_str ( )
} ;
2018-09-04 21:00:02 +00:00
2018-09-05 18:49:07 +00:00
PGresult * res = PQexecParams ( conn , " SELECT id, EXTRACT(EPOCH FROM creation_time AT TIME ZONE 'UTC')*1000, capabilities, "
" enable_broadcast, EXTRACT(EPOCH FROM last_modified AT TIME ZONE 'UTC')*1000, mtu, multicast_limit, name, private, remote_trace_level, "
" remote_trace_target, revision, rules, tags, v4_assign_mode, v6_assign_mode FROM ztc_network "
" WHERE deleted = false AND controller_id = $1 " ,
2018-09-04 23:05:34 +00:00
1 ,
NULL ,
params ,
NULL ,
NULL ,
0 ) ;
2018-09-05 18:49:07 +00:00
if ( PQresultStatus ( res ) ! = PGRES_TUPLES_OK ) {
fprintf ( stderr , " Networks Initialization Failed: %s " , PQerrorMessage ( conn ) ) ;
2018-09-04 23:05:34 +00:00
PQclear ( res ) ;
exit ( 1 ) ;
}
2018-09-05 18:49:07 +00:00
int numRows = PQntuples ( res ) ;
for ( int i = 0 ; i < numRows ; + + i ) {
json empty ;
json config ;
2018-09-05 23:54:23 +00:00
2018-09-28 17:55:39 +00:00
const char * nwidparam [ 1 ] = {
PQgetvalue ( res , i , 0 )
} ;
2018-09-05 23:08:06 +00:00
config [ " id " ] = PQgetvalue ( res , i , 0 ) ;
2018-09-05 18:49:07 +00:00
config [ " nwid " ] = PQgetvalue ( res , i , 0 ) ;
2018-11-12 05:08:51 +00:00
try {
config [ " creationTime " ] = std : : stoull ( PQgetvalue ( res , i , 1 ) ) ;
} catch ( std : : exception & e ) {
config [ " creationTime " ] = 0ULL ;
//fprintf(stderr, "Error converting creation time: %s\n", PQgetvalue(res, i, 1));
}
2018-09-05 18:49:07 +00:00
config [ " capabilities " ] = json : : parse ( PQgetvalue ( res , i , 2 ) ) ;
2018-09-05 23:54:23 +00:00
config [ " enableBroadcast " ] = ( strcmp ( PQgetvalue ( res , i , 3 ) , " t " ) = = 0 ) ;
2018-11-12 05:08:51 +00:00
try {
config [ " lastModified " ] = std : : stoull ( PQgetvalue ( res , i , 4 ) ) ;
} catch ( std : : exception & e ) {
config [ " lastModified " ] = 0ULL ;
//fprintf(stderr, "Error converting last modified: %s\n", PQgetvalue(res, i, 4));
}
try {
config [ " mtu " ] = std : : stoi ( PQgetvalue ( res , i , 5 ) ) ;
} catch ( std : : exception & e ) {
config [ " mtu " ] = 2800 ;
}
try {
config [ " multicastLimit " ] = std : : stoi ( PQgetvalue ( res , i , 6 ) ) ;
} catch ( std : : exception & e ) {
config [ " multicastLimit " ] = 64 ;
}
2018-09-05 18:49:07 +00:00
config [ " name " ] = PQgetvalue ( res , i , 7 ) ;
2018-09-05 23:54:23 +00:00
config [ " private " ] = ( strcmp ( PQgetvalue ( res , i , 8 ) , " t " ) = = 0 ) ;
2018-11-12 05:08:51 +00:00
try {
config [ " remoteTraceLevel " ] = std : : stoi ( PQgetvalue ( res , i , 9 ) ) ;
} catch ( std : : exception & e ) {
config [ " remoteTraceLevel " ] = 0 ;
}
2018-09-05 18:49:07 +00:00
config [ " remoteTraceTarget " ] = PQgetvalue ( res , i , 10 ) ;
2018-11-12 05:08:51 +00:00
try {
config [ " revision " ] = std : : stoull ( PQgetvalue ( res , i , 11 ) ) ;
} catch ( std : : exception & e ) {
config [ " revision " ] = 0ULL ;
//fprintf(stderr, "Error converting revision: %s\n", PQgetvalue(res, i, 11));
}
2018-09-05 18:49:07 +00:00
config [ " rules " ] = json : : parse ( PQgetvalue ( res , i , 12 ) ) ;
config [ " tags " ] = json : : parse ( PQgetvalue ( res , i , 13 ) ) ;
config [ " v4AssignMode " ] = json : : parse ( PQgetvalue ( res , i , 14 ) ) ;
config [ " v6AssignMode " ] = json : : parse ( PQgetvalue ( res , i , 15 ) ) ;
config [ " objtype " ] = " network " ;
config [ " ipAssignmentPools " ] = json : : array ( ) ;
config [ " routes " ] = json : : array ( ) ;
PGresult * r2 = PQexecParams ( conn ,
" SELECT host(ip_range_start), host(ip_range_end) FROM ztc_network_assignment_pool WHERE network_id = $1 " ,
1 ,
NULL ,
2018-09-28 17:55:39 +00:00
nwidparam ,
2018-09-05 18:49:07 +00:00
NULL ,
NULL ,
0 ) ;
if ( PQresultStatus ( r2 ) ! = PGRES_TUPLES_OK ) {
fprintf ( stderr , " ERROR: Error retreiving IP pools for network: %s \n " , PQresultErrorMessage ( r2 ) ) ;
PQclear ( r2 ) ;
PQclear ( res ) ;
exit ( 1 ) ;
}
2018-09-04 23:05:34 +00:00
2018-09-05 18:49:07 +00:00
int n = PQntuples ( r2 ) ;
for ( int j = 0 ; j < n ; + + j ) {
json ip ;
ip [ " ipRangeStart " ] = PQgetvalue ( r2 , j , 0 ) ;
ip [ " ipRangeEnd " ] = PQgetvalue ( r2 , j , 1 ) ;
2018-09-04 23:05:34 +00:00
2018-09-05 18:49:07 +00:00
config [ " ipAssignmentPools " ] . push_back ( ip ) ;
}
2018-09-04 23:05:34 +00:00
PQclear ( r2 ) ;
2018-09-05 18:49:07 +00:00
r2 = PQexecParams ( conn ,
" SELECT host(address), bits, host(via) FROM ztc_network_route WHERE network_id = $1 " ,
1 ,
NULL ,
2018-09-28 17:55:39 +00:00
nwidparam ,
2018-09-05 18:49:07 +00:00
NULL ,
NULL ,
0 ) ;
2018-09-04 23:05:34 +00:00
2018-09-05 18:49:07 +00:00
if ( PQresultStatus ( r2 ) ! = PGRES_TUPLES_OK ) {
fprintf ( stderr , " ERROR: Error retreiving routes for network: %s \n " , PQresultErrorMessage ( r2 ) ) ;
PQclear ( r2 ) ;
PQclear ( res ) ;
exit ( 1 ) ;
}
2018-09-04 23:05:34 +00:00
2018-09-05 18:49:07 +00:00
n = PQntuples ( r2 ) ;
for ( int j = 0 ; j < n ; + + j ) {
std : : string addr = PQgetvalue ( r2 , j , 0 ) ;
std : : string bits = PQgetvalue ( r2 , j , 1 ) ;
std : : string via = PQgetvalue ( r2 , j , 2 ) ;
json route ;
route [ " target " ] = addr + " / " + bits ;
if ( via = = " NULL " ) {
route [ " via " ] = nullptr ;
} else {
route [ " via " ] = via ;
}
config [ " routes " ] . push_back ( route ) ;
2018-09-04 23:05:34 +00:00
}
2018-09-05 18:49:07 +00:00
PQclear ( r2 ) ;
_networkChanged ( empty , config , false ) ;
}
2018-09-04 21:00:02 +00:00
2018-09-05 18:49:07 +00:00
PQclear ( res ) ;
2018-08-31 21:58:15 +00:00
2018-09-05 18:49:07 +00:00
if ( + + this - > _ready = = 2 ) {
if ( _waitNoticePrinted ) {
fprintf ( stderr , " [%s] NOTICE: %.10llx controller PostgreSQL data download complete. " ZT_EOL_S , _timestr ( ) , ( unsigned long long ) _myAddress . toInt ( ) ) ;
}
_readyLock . unlock ( ) ;
2018-08-31 21:58:15 +00:00
}
2018-09-05 18:49:07 +00:00
} catch ( std : : exception & e ) {
fprintf ( stderr , " ERROR: Error initializing networks: %s " , e . what ( ) ) ;
exit ( - 1 ) ;
2018-08-31 21:58:15 +00:00
}
}
void PostgreSQL : : initializeMembers ( PGconn * conn )
{
2018-09-05 18:49:07 +00:00
try {
if ( PQstatus ( conn ) ! = CONNECTION_OK ) {
fprintf ( stderr , " Bad Database Connection: %s " , PQerrorMessage ( conn ) ) ;
exit ( 1 ) ;
}
2018-09-04 21:51:35 +00:00
2018-09-05 18:49:07 +00:00
const char * params [ 1 ] = {
_myAddressStr . c_str ( )
2018-09-04 21:51:35 +00:00
} ;
2018-09-05 18:49:07 +00:00
PGresult * res = PQexecParams ( conn ,
" SELECT m.id, m.network_id, m.active_bridge, m.authorized, m.capabilities, EXTRACT(EPOCH FROM m.creation_time AT TIME ZONE 'UTC')*1000, m.identity, "
" EXTRACT(EPOCH FROM m.last_authorized_time AT TIME ZONE 'UTC')*1000, "
" EXTRACT(EPOCH FROM m.last_deauthorized_time AT TIME ZONE 'UTC')*1000, "
" m.remote_trace_level, m.remote_trace_target, m.tags, m.v_major, m.v_minor, m.v_rev, m.v_proto, "
" m.no_auto_assign_ips, m.revision "
" FROM ztc_member m "
" INNER JOIN ztc_network n "
" ON n.id = m.network_id "
" WHERE n.controller_id = $1 AND m.deleted = false " ,
1 ,
2018-09-04 21:51:35 +00:00
NULL ,
2018-09-05 18:49:07 +00:00
params ,
2018-09-04 21:51:35 +00:00
NULL ,
NULL ,
0 ) ;
2018-09-05 18:49:07 +00:00
if ( PQresultStatus ( res ) ! = PGRES_TUPLES_OK ) {
2018-09-04 21:51:35 +00:00
fprintf ( stderr , " Member Initialization Failed: %s " , PQerrorMessage ( conn ) ) ;
PQclear ( res ) ;
exit ( 1 ) ;
}
2018-09-05 18:49:07 +00:00
int numRows = PQntuples ( res ) ;
for ( int i = 0 ; i < numRows ; + + i ) {
json empty ;
json config ;
std : : string memberId ( PQgetvalue ( res , i , 0 ) ) ;
std : : string networkId ( PQgetvalue ( res , i , 1 ) ) ;
2018-09-05 23:08:06 +00:00
std : : string ctime = PQgetvalue ( res , i , 5 ) ;
2018-09-05 18:49:07 +00:00
config [ " id " ] = memberId ;
config [ " nwid " ] = networkId ;
2018-09-05 23:54:23 +00:00
config [ " activeBridge " ] = ( strcmp ( PQgetvalue ( res , i , 2 ) , " t " ) = = 0 ) ;
config [ " authorized " ] = ( strcmp ( PQgetvalue ( res , i , 3 ) , " t " ) = = 0 ) ;
2018-09-05 23:08:06 +00:00
config [ " capabilities " ] = json : : parse ( PQgetvalue ( res , i , 4 ) ) ;
2018-11-12 05:08:51 +00:00
try {
config [ " creationTime " ] = std : : stoull ( PQgetvalue ( res , i , 5 ) ) ;
} catch ( std : : exception & e ) {
config [ " creationTime " ] = 0ULL ;
//fprintf(stderr, "Error upding creation time (member): %s\n", PQgetvalue(res, i, 5));
}
2018-09-05 23:08:06 +00:00
config [ " identity " ] = PQgetvalue ( res , i , 6 ) ;
2018-11-12 05:08:51 +00:00
try {
config [ " lastAuthorizedTime " ] = std : : stoull ( PQgetvalue ( res , i , 7 ) ) ;
} catch ( std : : exception & e ) {
config [ " lastAuthorizedTime " ] = 0ULL ;
//fprintf(stderr, "Error updating last auth time (member): %s\n", PQgetvalue(res, i, 7));
}
try {
config [ " lastDeauthorizedTime " ] = std : : stoull ( PQgetvalue ( res , i , 8 ) ) ;
} catch ( std : : exception & e ) {
config [ " lastDeauthorizedTime " ] = 0ULL ;
//fprintf(stderr, "Error updating last deauth time (member): %s\n", PQgetvalue(res, i, 8));
}
try {
config [ " remoteTraceLevel " ] = std : : stoi ( PQgetvalue ( res , i , 9 ) ) ;
} catch ( std : : exception & e ) {
config [ " remoteTraceLevel " ] = 0 ;
}
2018-09-05 23:08:06 +00:00
config [ " remoteTraceTarget " ] = PQgetvalue ( res , i , 10 ) ;
config [ " tags " ] = json : : parse ( PQgetvalue ( res , i , 11 ) ) ;
2018-11-12 05:08:51 +00:00
try {
config [ " vMajor " ] = std : : stoi ( PQgetvalue ( res , i , 12 ) ) ;
} catch ( std : : exception & e ) {
config [ " vMajor " ] = - 1 ;
}
try {
config [ " vMinor " ] = std : : stoi ( PQgetvalue ( res , i , 13 ) ) ;
} catch ( std : : exception & e ) {
config [ " vMinor " ] = - 1 ;
}
try {
config [ " vRev " ] = std : : stoi ( PQgetvalue ( res , i , 14 ) ) ;
} catch ( std : : exception & e ) {
config [ " vRev " ] = - 1 ;
}
try {
config [ " vProto " ] = std : : stoi ( PQgetvalue ( res , i , 15 ) ) ;
} catch ( std : : exception & e ) {
config [ " vProto " ] = - 1 ;
}
2018-09-05 23:54:23 +00:00
config [ " noAutoAssignIps " ] = ( strcmp ( PQgetvalue ( res , i , 16 ) , " t " ) = = 0 ) ;
2018-11-12 05:08:51 +00:00
try {
config [ " revision " ] = std : : stoull ( PQgetvalue ( res , i , 17 ) ) ;
} catch ( std : : exception & e ) {
config [ " revision " ] = 0ULL ;
//fprintf(stderr, "Error updating revision (member): %s\n", PQgetvalue(res, i, 17));
}
2018-09-05 18:49:07 +00:00
config [ " objtype " ] = " member " ;
config [ " ipAssignments " ] = json : : array ( ) ;
const char * p2 [ 2 ] = {
memberId . c_str ( ) ,
networkId . c_str ( )
} ;
2018-09-04 21:51:35 +00:00
2018-09-05 18:49:07 +00:00
PGresult * r2 = PQexecParams ( conn ,
" SELECT address FROM ztc_member_ip_assignment WHERE member_id = $1 AND network_id = $2 " ,
2 ,
NULL ,
p2 ,
NULL ,
NULL ,
0 ) ;
2018-09-04 21:51:35 +00:00
2018-09-05 18:49:07 +00:00
if ( PQresultStatus ( r2 ) ! = PGRES_TUPLES_OK ) {
fprintf ( stderr , " Member Initialization Failed: %s " , PQerrorMessage ( conn ) ) ;
PQclear ( r2 ) ;
PQclear ( res ) ;
exit ( 1 ) ;
}
int n = PQntuples ( r2 ) ;
for ( int j = 0 ; j < n ; + + j ) {
config [ " ipAssignments " ] . push_back ( PQgetvalue ( r2 , j , 0 ) ) ;
}
2018-08-31 21:58:15 +00:00
2018-09-05 18:49:07 +00:00
_memberChanged ( empty , config , false ) ;
2018-08-31 21:58:15 +00:00
}
2018-09-05 18:49:07 +00:00
PQclear ( res ) ;
if ( + + this - > _ready = = 2 ) {
if ( _waitNoticePrinted ) {
fprintf ( stderr , " [%s] NOTICE: %.10llx controller PostgreSQL data download complete. " ZT_EOL_S , _timestr ( ) , ( unsigned long long ) _myAddress . toInt ( ) ) ;
}
_readyLock . unlock ( ) ;
}
} catch ( std : : exception & e ) {
fprintf ( stderr , " ERROR: Error initializing members: %s \n " , e . what ( ) ) ;
exit ( - 1 ) ;
2018-08-31 21:58:15 +00:00
}
}
void PostgreSQL : : heartbeat ( )
{
char publicId [ 1024 ] ;
char hostnameTmp [ 1024 ] ;
_myId . toString ( false , publicId ) ;
if ( gethostname ( hostnameTmp , sizeof ( hostnameTmp ) ) ! = 0 ) {
hostnameTmp [ 0 ] = ( char ) 0 ;
} else {
for ( int i = 0 ; i < sizeof ( hostnameTmp ) ; + + i ) {
if ( ( hostnameTmp [ i ] = = ' . ' ) | | ( hostnameTmp [ i ] = = 0 ) ) {
hostnameTmp [ i ] = ( char ) 0 ;
break ;
}
}
}
const char * controllerId = _myAddressStr . c_str ( ) ;
const char * publicIdentity = publicId ;
const char * hostname = hostnameTmp ;
PGconn * conn = PQconnectdb ( _path . c_str ( ) ) ;
if ( PQstatus ( conn ) = = CONNECTION_BAD ) {
fprintf ( stderr , " Connection to database failed: %s \n " , PQerrorMessage ( conn ) ) ;
PQfinish ( conn ) ;
exit ( 1 ) ;
}
while ( _run = = 1 ) {
if ( PQstatus ( conn ) ! = CONNECTION_OK ) {
PQfinish ( conn ) ;
conn = PQconnectdb ( _path . c_str ( ) ) ;
}
if ( conn ) {
2018-10-24 19:06:17 +00:00
std : : string major = std : : to_string ( ZEROTIER_ONE_VERSION_MAJOR ) ;
std : : string minor = std : : to_string ( ZEROTIER_ONE_VERSION_MINOR ) ;
std : : string rev = std : : to_string ( ZEROTIER_ONE_VERSION_REVISION ) ;
std : : string build = std : : to_string ( ZEROTIER_ONE_VERSION_BUILD ) ;
std : : string now = std : : to_string ( OSUtils : : now ( ) ) ;
2018-08-31 21:58:15 +00:00
const char * values [ 8 ] = {
controllerId ,
hostname ,
2018-10-24 19:06:17 +00:00
now . c_str ( ) ,
2018-08-31 21:58:15 +00:00
publicIdentity ,
2018-10-24 19:06:17 +00:00
major . c_str ( ) ,
minor . c_str ( ) ,
rev . c_str ( ) ,
build . c_str ( )
2018-08-31 21:58:15 +00:00
} ;
PGresult * res = PQexecParams ( conn ,
" INSERT INTO ztc_controller (id, cluster_host, last_alive, public_identity, v_major, v_minor, v_rev, v_build) "
" VALUES ($1, $2, TO_TIMESTAMP($3::double precision/1000), $4, $5, $6, $7, $8) "
" ON CONFLICT (id) DO UPDATE SET cluster_host = EXCLUDED.cluster_host, last_alive = EXCLUDED.last_alive, "
" public_identity = EXCLUDED.public_identity, v_major = EXCLUDED.v_major, v_minor = EXCLUDED.v_minor, "
" v_rev = EXCLUDED.v_rev, v_build = EXCLUDED.v_rev " ,
8 , // number of parameters
NULL , // oid field. ignore
values , // values for substitution
2018-10-24 19:06:17 +00:00
NULL , // lengths in bytes of each value
NULL , // binary?
2018-08-31 21:58:15 +00:00
0 ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " Heartbeat Update Failed: %s \n " , PQresultErrorMessage ( res ) ) ;
}
PQclear ( res ) ;
}
std : : this_thread : : sleep_for ( std : : chrono : : milliseconds ( 1000 ) ) ;
}
PQfinish ( conn ) ;
conn = NULL ;
}
void PostgreSQL : : membersDbWatcher ( )
{
PGconn * conn = PQconnectdb ( _path . c_str ( ) ) ;
if ( PQstatus ( conn ) = = CONNECTION_BAD ) {
fprintf ( stderr , " Connection to database failed: %s \n " , PQerrorMessage ( conn ) ) ;
PQfinish ( conn ) ;
exit ( 1 ) ;
}
initializeMembers ( conn ) ;
char buf [ 11 ] = { 0 } ;
std : : string cmd = " LISTEN member_ " + std : : string ( _myAddress . toString ( buf ) ) ;
PGresult * res = PQexec ( conn , cmd . c_str ( ) ) ;
if ( ! res | | PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " LISTEN command failed: %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
PQfinish ( conn ) ;
exit ( 1 ) ;
}
2018-09-04 21:00:02 +00:00
PQclear ( res ) ; res = NULL ;
2018-08-31 21:58:15 +00:00
while ( _run = = 1 ) {
if ( PQstatus ( conn ) ! = CONNECTION_OK ) {
fprintf ( stderr , " ERROR: Member Watcher lost connection to Postgres. " ) ;
exit ( - 1 ) ;
}
PGnotify * notify = NULL ;
PQconsumeInput ( conn ) ;
while ( ( notify = PQnotifies ( conn ) ) ! = NULL ) {
2018-10-24 19:06:17 +00:00
//fprintf(stderr, "ASYNC NOTIFY of '%s' id:%s received\n", notify->relname, notify->extra);
2018-08-31 21:58:15 +00:00
try {
json tmp ( json : : parse ( notify - > extra ) ) ;
json & ov = tmp [ " old_val " ] ;
json & nv = tmp [ " new_val " ] ;
json oldConfig , newConfig ;
if ( ov . is_object ( ) ) oldConfig = ov ;
if ( nv . is_object ( ) ) newConfig = nv ;
if ( oldConfig . is_object ( ) | | newConfig . is_object ( ) ) {
_memberChanged ( oldConfig , newConfig , ( this - > _ready > = 2 ) ) ;
}
} catch ( . . . ) { } // ignore bad records
free ( notify ) ;
}
std : : this_thread : : sleep_for ( std : : chrono : : milliseconds ( 10 ) ) ;
}
PQfinish ( conn ) ;
conn = NULL ;
}
void PostgreSQL : : networksDbWatcher ( )
{
PGconn * conn = PQconnectdb ( _path . c_str ( ) ) ;
if ( PQstatus ( conn ) = = CONNECTION_BAD ) {
fprintf ( stderr , " Connection to database failed: %s \n " , PQerrorMessage ( conn ) ) ;
PQfinish ( conn ) ;
exit ( 1 ) ;
}
initializeNetworks ( conn ) ;
char buf [ 11 ] = { 0 } ;
std : : string cmd = " LISTEN network_ " + std : : string ( _myAddress . toString ( buf ) ) ;
PGresult * res = PQexec ( conn , cmd . c_str ( ) ) ;
if ( ! res | | PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " LISTEN command failed: %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
PQfinish ( conn ) ;
exit ( 1 ) ;
}
2018-09-04 21:00:02 +00:00
PQclear ( res ) ; res = NULL ;
2018-08-31 21:58:15 +00:00
while ( _run = = 1 ) {
if ( PQstatus ( conn ) ! = CONNECTION_OK ) {
fprintf ( stderr , " ERROR: Network Watcher lost connection to Postgres. " ) ;
exit ( - 1 ) ;
}
PGnotify * notify = NULL ;
PQconsumeInput ( conn ) ;
while ( ( notify = PQnotifies ( conn ) ) ! = NULL ) {
2018-10-24 19:06:17 +00:00
//fprintf(stderr, "ASYNC NOTIFY of '%s' id:%s received\n", notify->relname, notify->extra);
2018-08-31 21:58:15 +00:00
try {
json tmp ( json : : parse ( notify - > extra ) ) ;
json & ov = tmp [ " old_val " ] ;
json & nv = tmp [ " new_val " ] ;
json oldConfig , newConfig ;
if ( ov . is_object ( ) ) oldConfig = ov ;
if ( nv . is_object ( ) ) newConfig = nv ;
if ( oldConfig . is_object ( ) | | newConfig . is_object ( ) ) {
_networkChanged ( oldConfig , newConfig , ( this - > _ready > = 2 ) ) ;
}
} catch ( . . . ) { } // ignore bad records
free ( notify ) ;
}
std : : this_thread : : sleep_for ( std : : chrono : : milliseconds ( 10 ) ) ;
}
PQfinish ( conn ) ;
conn = NULL ;
}
void PostgreSQL : : commitThread ( )
{
2018-09-04 23:05:34 +00:00
PGconn * conn = PQconnectdb ( _path . c_str ( ) ) ;
if ( PQstatus ( conn ) = = CONNECTION_BAD ) {
2018-09-05 18:49:07 +00:00
fprintf ( stderr , " ERROR: Connection to database failed: %s \n " , PQerrorMessage ( conn ) ) ;
2018-09-04 23:05:34 +00:00
PQfinish ( conn ) ;
exit ( 1 ) ;
}
2018-08-31 21:58:15 +00:00
json * config = nullptr ;
while ( _commitQueue . get ( config ) & ( _run = = 1 ) ) {
if ( ! config ) {
continue ;
}
2018-09-05 18:49:07 +00:00
if ( PQstatus ( conn ) = = CONNECTION_BAD ) {
fprintf ( stderr , " ERROR: Connection to database failed: %s \n " , PQerrorMessage ( conn ) ) ;
PQfinish ( conn ) ;
2018-10-24 19:06:17 +00:00
delete config ;
2018-09-05 18:49:07 +00:00
exit ( 1 ) ;
}
try {
const std : : string objtype = ( * config ) [ " objtype " ] ;
if ( objtype = = " member " ) {
try {
std : : string memberId = ( * config ) [ " id " ] ;
std : : string networkId = ( * config ) [ " nwid " ] ;
std : : string identity = ( * config ) [ " identity " ] ;
std : : string target = " NULL " ;
2018-10-24 20:36:07 +00:00
2018-09-05 18:49:07 +00:00
if ( ! ( * config ) [ " remoteTraceTarget " ] . is_null ( ) ) {
target = ( * config ) [ " remoteTraceTarget " ] ;
}
2018-10-24 19:06:17 +00:00
std : : string caps = OSUtils : : jsonDump ( ( * config ) [ " capabilities " ] , - 1 ) ;
std : : string lastAuthTime = std : : to_string ( ( long long ) ( * config ) [ " lastAuthorizedTime " ] ) ;
std : : string lastDeauthTime = std : : to_string ( ( long long ) ( * config ) [ " lastDeauthorizedTime " ] ) ;
std : : string rtraceLevel = std : : to_string ( ( int ) ( * config ) [ " remoteTraceLevel " ] ) ;
std : : string rev = std : : to_string ( ( unsigned long long ) ( * config ) [ " revision " ] ) ;
std : : string tags = OSUtils : : jsonDump ( ( * config ) [ " tags " ] , - 1 ) ;
std : : string vmajor = std : : to_string ( ( int ) ( * config ) [ " vMajor " ] ) ;
std : : string vminor = std : : to_string ( ( int ) ( * config ) [ " vMinor " ] ) ;
std : : string vrev = std : : to_string ( ( int ) ( * config ) [ " vRev " ] ) ;
std : : string vproto = std : : to_string ( ( int ) ( * config ) [ " vProto " ] ) ;
2018-09-05 18:49:07 +00:00
const char * values [ 19 ] = {
memberId . c_str ( ) ,
networkId . c_str ( ) ,
( ( * config ) [ " activeBridge " ] ? " true " : " false " ) ,
( ( * config ) [ " authorized " ] ? " true " : " false " ) ,
2018-10-24 19:06:17 +00:00
caps . c_str ( ) ,
2018-09-05 18:49:07 +00:00
identity . c_str ( ) ,
2018-10-24 19:06:17 +00:00
lastAuthTime . c_str ( ) ,
lastDeauthTime . c_str ( ) ,
2018-09-05 18:49:07 +00:00
( ( * config ) [ " noAutoAssignIps " ] ? " true " : " false " ) ,
2018-10-24 19:06:17 +00:00
rtraceLevel . c_str ( ) ,
2018-09-05 18:49:07 +00:00
( target = = " NULL " ) ? NULL : target . c_str ( ) ,
2018-10-24 19:06:17 +00:00
rev . c_str ( ) ,
tags . c_str ( ) ,
vmajor . c_str ( ) ,
vminor . c_str ( ) ,
vrev . c_str ( ) ,
vproto . c_str ( )
2018-09-05 18:49:07 +00:00
} ;
PGresult * res = PQexecParams ( conn ,
" INSERT INTO ztc_member (id, network_id, active_bridge, authorized, capabilities, "
2018-09-05 23:08:06 +00:00
" identity, last_authorized_time, last_deauthorized_time, no_auto_assign_ips, "
2018-09-05 18:49:07 +00:00
" remote_trace_level, remote_trace_target, revision, tags, v_major, v_minor, v_rev, v_proto) "
2018-09-05 23:08:06 +00:00
" VALUES ($1, $2, $3, $4, $5, $6, "
" TO_TIMESTAMP($7::double precision/1000), TO_TIMESTAMP($8::double precision/1000), "
" $9, $10, $11, $12, $13, $14, $15, $16, $17) ON CONFLICT (network_id, id) DO UPDATE SET "
" active_bridge = EXCLUDED.active_bridge, authorized = EXCLUDED.authorized, capabilities = EXCLUDED.capabilities, "
2018-09-05 18:49:07 +00:00
" identity = EXCLUDED.identity, last_authorized_time = EXCLUDED.last_authorized_time, "
" last_deauthorized_time = EXCLUDED.last_deauthorized_time, no_auto_assign_ips = EXCLUDED.no_auto_assign_ips, "
" remote_trace_level = EXCLUDED.remote_trace_level, remote_trace_target = EXCLUDED.remote_trace_target, "
" revision = EXCLUDED.revision+1, tags = EXCLUDED.tags, v_major = EXCLUDED.v_major, "
" v_minor = EXCLUDED.v_minor, v_rev = EXCLUDED.v_rev, v_proto = EXCLUDED.v_proto " ,
2018-09-05 23:08:06 +00:00
17 ,
2018-09-05 18:49:07 +00:00
NULL ,
values ,
NULL ,
NULL ,
0 ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error updating member: %s \n " , PQresultErrorMessage ( res ) ) ;
2018-09-05 23:08:06 +00:00
fprintf ( stderr , " %s " , OSUtils : : jsonDump ( * config , 2 ) . c_str ( ) ) ;
2018-09-05 18:49:07 +00:00
PQclear ( res ) ;
2018-10-24 19:06:17 +00:00
delete config ;
config = nullptr ;
2018-09-05 18:49:07 +00:00
continue ;
}
2018-09-04 23:05:34 +00:00
2018-09-05 18:49:07 +00:00
PQclear ( res ) ;
2018-09-04 23:05:34 +00:00
2018-09-05 18:49:07 +00:00
res = PQexec ( conn , " BEGIN " ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error beginning transaction: %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
2018-10-24 19:06:17 +00:00
delete config ;
config = nullptr ;
2018-09-05 18:49:07 +00:00
continue ;
}
2018-09-04 23:05:34 +00:00
2018-09-05 18:49:07 +00:00
PQclear ( res ) ;
2018-09-04 23:05:34 +00:00
2018-09-05 18:49:07 +00:00
const char * v2 [ 2 ] = {
memberId . c_str ( ) ,
networkId . c_str ( )
} ;
res = PQexecParams ( conn ,
" DELETE FROM ztc_member_ip_assignment WHERE member_id = $1 AND network_id = $2 " ,
2 ,
NULL ,
v2 ,
NULL ,
NULL ,
0 ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error updating IP address assignments: %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
PQclear ( PQexec ( conn , " ROLLBACK " ) ) ; ;
2018-10-24 19:06:17 +00:00
delete config ;
config = nullptr ;
2018-09-05 18:49:07 +00:00
continue ;
}
2018-09-04 23:05:34 +00:00
2018-09-05 18:49:07 +00:00
PQclear ( res ) ;
2018-09-04 23:05:34 +00:00
2018-09-05 18:49:07 +00:00
for ( auto i = ( * config ) [ " ipAssignments " ] . begin ( ) ; i ! = ( * config ) [ " ipAssignments " ] . end ( ) ; + + i ) {
std : : string addr = * i ;
const char * v3 [ 3 ] = {
memberId . c_str ( ) ,
networkId . c_str ( ) ,
addr . c_str ( )
} ;
res = PQexecParams ( conn ,
" INSERT INTO ztc_member_ip_assignment (member_id, network_id, address) VALUES ($1, $2, $3) " ,
3 ,
NULL ,
v3 ,
NULL ,
NULL ,
0 ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error setting IP addresses for member: %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
PQclear ( PQexec ( conn , " ROLLBACK " ) ) ;
continue ;
}
}
res = PQexec ( conn , " COMMIT " ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error committing ip address data: %s \n " , PQresultErrorMessage ( res ) ) ;
}
2018-09-04 23:05:34 +00:00
PQclear ( res ) ;
2018-10-24 19:06:17 +00:00
const uint64_t nwidInt = OSUtils : : jsonIntHex ( ( * config ) [ " nwid " ] , 0ULL ) ;
const uint64_t memberidInt = OSUtils : : jsonIntHex ( ( * config ) [ " id " ] , 0ULL ) ;
if ( nwidInt & & memberidInt ) {
nlohmann : : json nwOrig ;
nlohmann : : json memOrig ;
nlohmann : : json memNew ( * config ) ;
get ( nwidInt , nwOrig , memberidInt , memOrig ) ;
_memberChanged ( memOrig , memNew , ( this - > _ready > = 2 ) ) ;
} else {
fprintf ( stderr , " Can't notify of change. Error parsing nwid or memberid: %lu-%lu \n " , nwidInt , memberidInt ) ;
}
2018-09-05 18:49:07 +00:00
} catch ( std : : exception & e ) {
fprintf ( stderr , " ERROR: Error updating member: %s \n " , e . what ( ) ) ;
2018-09-04 23:05:34 +00:00
}
2018-09-05 18:49:07 +00:00
} else if ( objtype = = " network " ) {
try {
std : : string id = ( * config ) [ " id " ] ;
std : : string controllerId = _myAddressStr . c_str ( ) ;
std : : string name = ( * config ) [ " name " ] ;
std : : string remoteTraceTarget ( " NULL " ) ;
if ( ! ( * config ) [ " remoteTraceTarget " ] . is_null ( ) ) {
remoteTraceTarget = ( * config ) [ " remoteTraceTarget " ] ;
}
std : : string rulesSource = ( * config ) [ " rulesSource " ] ;
2018-10-24 19:06:17 +00:00
std : : string caps = OSUtils : : jsonDump ( ( * config ) [ " capabilitles " ] , - 1 ) ;
std : : string now = std : : to_string ( OSUtils : : now ( ) ) ;
std : : string mtu = std : : to_string ( ( int ) ( * config ) [ " mtu " ] ) ;
std : : string mcastLimit = std : : to_string ( ( int ) ( * config ) [ " multicastLimit " ] ) ;
std : : string rtraceLevel = std : : to_string ( ( int ) ( * config ) [ " remoteTraceLevel " ] ) ;
std : : string rules = OSUtils : : jsonDump ( ( * config ) [ " rules " ] , - 1 ) ;
std : : string tags = OSUtils : : jsonDump ( ( * config ) [ " tags " ] , - 1 ) ;
std : : string v4mode = OSUtils : : jsonDump ( ( * config ) [ " v4AssignMode " ] , - 1 ) ;
std : : string v6mode = OSUtils : : jsonDump ( ( * config ) [ " v6AssignMode " ] , - 1 ) ;
bool enableBroadcast = ( * config ) [ " enableBroadcast " ] ;
bool isPrivate = ( * config ) [ " private " ] ;
2018-09-05 18:49:07 +00:00
const char * values [ 16 ] = {
id . c_str ( ) ,
controllerId . c_str ( ) ,
2018-10-24 19:06:17 +00:00
caps . c_str ( ) ,
enableBroadcast ? " true " : " false " ,
now . c_str ( ) ,
mtu . c_str ( ) ,
mcastLimit . c_str ( ) ,
2018-09-05 18:49:07 +00:00
name . c_str ( ) ,
2018-10-24 19:06:17 +00:00
isPrivate ? " true " : " false " ,
rtraceLevel . c_str ( ) ,
2018-09-05 18:49:07 +00:00
( remoteTraceTarget = = " NULL " ? NULL : remoteTraceTarget . c_str ( ) ) ,
2018-10-24 19:06:17 +00:00
rules . c_str ( ) ,
2018-09-05 18:49:07 +00:00
rulesSource . c_str ( ) ,
2018-10-24 19:06:17 +00:00
tags . c_str ( ) ,
v4mode . c_str ( ) ,
v6mode . c_str ( ) ,
2018-09-05 18:49:07 +00:00
} ;
PGresult * res = PQexecParams ( conn ,
" UPDATE ztc_network SET controller_id = $2, capabilities = $3, enable_broadcast = $4, "
" last_updated = $5, mtu = $6, multicast_limit = $7, name = $8, private = $9, "
" remote_trace_level = $10, remote_trace_target = $11, rules = $12, rules_source = $13, "
" tags = $14, v4_assign_mode = $15, v6_assign_mode = $16 "
" WHERE id = $1 " ,
16 ,
NULL ,
values ,
NULL ,
NULL ,
0 ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error updating network record: %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
2018-10-24 19:06:17 +00:00
delete config ;
config = nullptr ;
2018-09-05 18:49:07 +00:00
continue ;
}
2018-09-05 18:30:17 +00:00
2018-09-05 18:49:07 +00:00
PQclear ( res ) ;
2018-09-05 18:30:17 +00:00
2018-09-05 18:49:07 +00:00
res = PQexec ( conn , " BEGIN " ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error beginnning transaction: %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
2018-10-24 19:06:17 +00:00
delete config ;
config = nullptr ;
2018-09-05 18:49:07 +00:00
continue ;
}
2018-09-05 18:30:17 +00:00
2018-09-05 18:49:07 +00:00
PQclear ( res ) ;
2018-09-05 18:30:17 +00:00
2018-09-05 18:49:07 +00:00
const char * params [ 1 ] = {
id . c_str ( )
} ;
res = PQexecParams ( conn ,
" DELETE FROM ztc_network_assignment_pool WHERE network_id = $1 " ,
1 ,
NULL ,
params ,
NULL ,
NULL ,
0 ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error updating assignment pool: %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
PQclear ( PQexec ( conn , " ROLLBACK " ) ) ;
2018-10-24 19:06:17 +00:00
delete config ;
config = nullptr ;
2018-09-05 18:49:07 +00:00
continue ;
}
2018-09-05 18:30:17 +00:00
2018-09-05 18:49:07 +00:00
PQclear ( res ) ;
2018-09-05 18:30:17 +00:00
2018-09-05 18:49:07 +00:00
auto pool = ( * config ) [ " ipAssignmentPools " ] ;
bool err = false ;
for ( auto i = pool . begin ( ) ; i ! = pool . end ( ) ; + + i ) {
std : : string start = ( * i ) [ " ipRangeStart " ] ;
std : : string end = ( * i ) [ " ipRangeEnd " ] ;
const char * p [ 3 ] = {
id . c_str ( ) ,
start . c_str ( ) ,
end . c_str ( )
} ;
res = PQexecParams ( conn ,
" INSERT INTO ztc_network_assignment_pool (network_id, ip_range_start, ip_range_end) "
" VALUES ($1, $2, $3) " ,
3 ,
NULL ,
p ,
NULL ,
NULL ,
0 ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error updating assignment pool: %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
err = true ;
break ;
}
PQclear ( res ) ;
}
if ( err ) {
PQclear ( PQexec ( conn , " ROLLBACK " ) ) ;
2018-10-24 19:06:17 +00:00
delete config ;
config = nullptr ;
2018-09-05 18:49:07 +00:00
continue ;
}
res = PQexecParams ( conn ,
" DELETE FROM ztc_network_route WHERE network_id = $1 " ,
1 ,
NULL ,
params ,
NULL ,
NULL ,
0 ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error updating routes: %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
PQclear ( PQexec ( conn , " ROLLBACK " ) ) ;
2018-10-24 19:06:17 +00:00
delete config ;
config = nullptr ;
2018-09-05 18:49:07 +00:00
continue ;
}
auto routes = ( * config ) [ " routes " ] ;
err = false ;
for ( auto i = routes . begin ( ) ; i ! = routes . end ( ) ; + + i ) {
std : : string t = ( * i ) [ " target " ] ;
std : : vector < std : : string > target ;
std : : istringstream f ( t ) ;
std : : string s ;
while ( std : : getline ( f , s , ' / ' ) ) {
target . push_back ( s ) ;
}
if ( target . empty ( ) | | target . size ( ) ! = 2 ) {
continue ;
}
std : : string targetAddr = target [ 0 ] ;
std : : string targetBits = target [ 1 ] ;
std : : string via = " NULL " ;
if ( ! ( * i ) [ " via " ] . is_null ( ) ) {
via = ( * i ) [ " via " ] ;
}
const char * p [ 4 ] = {
id . c_str ( ) ,
targetAddr . c_str ( ) ,
targetBits . c_str ( ) ,
( via = = " NULL " ? NULL : via . c_str ( ) ) ,
} ;
res = PQexecParams ( conn ,
" INSERT INTO ztc_network_route (network_id, address, bits, via) VALUES ($1, $2, $3, $4) " ,
4 ,
NULL ,
p ,
NULL ,
NULL ,
0 ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error updating routes: %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
err = true ;
break ;
}
PQclear ( res ) ;
}
if ( err ) {
2018-10-24 19:06:17 +00:00
PQclear ( PQexec ( conn , " ROLLBACK " ) ) ;
delete config ;
config = nullptr ;
2018-09-05 18:49:07 +00:00
continue ;
}
res = PQexec ( conn , " COMMIT " ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error committing network update: %s \n " , PQresultErrorMessage ( res ) ) ;
}
2018-09-05 18:30:17 +00:00
PQclear ( res ) ;
2018-10-24 19:06:17 +00:00
const uint64_t nwidInt = OSUtils : : jsonIntHex ( ( * config ) [ " nwid " ] , 0ULL ) ;
if ( nwidInt ) {
nlohmann : : json nwOrig ;
nlohmann : : json nwNew ( * config ) ;
get ( nwidInt , nwOrig ) ;
_networkChanged ( nwOrig , nwNew , true ) ;
} else {
fprintf ( stderr , " Can't notify network changed: %lu \n " , nwidInt ) ;
}
2018-09-05 18:49:07 +00:00
} catch ( std : : exception & e ) {
fprintf ( stderr , " ERROR: Error updating member: %s \n " , e . what ( ) ) ;
2018-09-05 18:30:17 +00:00
}
2018-09-05 18:49:07 +00:00
} else if ( objtype = = " trace " ) {
fprintf ( stderr , " ERROR: Trace not yet implemented " ) ;
} else if ( objtype = = " _delete_network " ) {
try {
std : : string networkId = ( * config ) [ " nwid " ] ;
const char * values [ 1 ] = {
networkId . c_str ( )
} ;
PGresult * res = PQexecParams ( conn ,
" UPDATE ztc_network SET deleted = true WHERE id = $1 " ,
1 ,
NULL ,
values ,
NULL ,
NULL ,
0 ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error deleting network: %s \n " , PQresultErrorMessage ( res ) ) ;
}
2018-09-05 18:30:17 +00:00
2018-09-05 18:49:07 +00:00
PQclear ( res ) ;
} catch ( std : : exception & e ) {
fprintf ( stderr , " ERROR: Error deleting network: %s \n " , e . what ( ) ) ;
2018-09-05 18:30:17 +00:00
}
2018-09-05 18:49:07 +00:00
} else if ( objtype = = " _delete_member " ) {
try {
std : : string memberId = ( * config ) [ " id " ] ;
std : : string networkId = ( * config ) [ " nwid " ] ;
const char * values [ 2 ] = {
memberId . c_str ( ) ,
networkId . c_str ( )
} ;
PGresult * res = PQexecParams ( conn ,
" UPDATE ztc_member SET hidden = true, deleted = true WHERE id = $1 AND network_id = $2 " ,
2 ,
NULL ,
values ,
NULL ,
NULL ,
0 ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error deleting member: %s \n " , PQresultErrorMessage ( res ) ) ;
}
2018-09-05 18:30:17 +00:00
PQclear ( res ) ;
2018-09-05 18:49:07 +00:00
} catch ( std : : exception & e ) {
fprintf ( stderr , " ERROR: Error deleting member: %s \n " , e . what ( ) ) ;
2018-09-05 18:30:17 +00:00
}
2018-09-05 18:49:07 +00:00
} else {
fprintf ( stderr , " ERROR: unknown objtype " ) ;
2018-09-04 23:05:34 +00:00
}
2018-09-05 18:49:07 +00:00
} catch ( std : : exception & e ) {
fprintf ( stderr , " ERROR: Error getting objtype: %s \n " , e . what ( ) ) ;
2018-09-04 23:05:34 +00:00
}
delete config ;
2018-10-24 19:06:17 +00:00
config = nullptr ;
2018-08-31 21:58:15 +00:00
std : : this_thread : : sleep_for ( std : : chrono : : milliseconds ( 10 ) ) ;
}
2018-09-04 23:05:34 +00:00
PQfinish ( conn ) ;
2018-08-31 21:58:15 +00:00
}
void PostgreSQL : : onlineNotificationThread ( )
{
2018-09-04 21:00:02 +00:00
PGconn * conn = PQconnectdb ( _path . c_str ( ) ) ;
if ( PQstatus ( conn ) = = CONNECTION_BAD ) {
fprintf ( stderr , " Connection to database failed: %s \n " , PQerrorMessage ( conn ) ) ;
PQfinish ( conn ) ;
exit ( 1 ) ;
}
2018-08-31 21:58:15 +00:00
_connected = 1 ;
2018-09-04 21:00:02 +00:00
int64_t lastUpdatedNetworkStatus = 0 ;
std : : unordered_map < std : : pair < uint64_t , uint64_t > , int64_t , _PairHasher > lastOnlineCumulative ;
2018-09-06 22:14:16 +00:00
2018-08-31 21:58:15 +00:00
while ( _run = = 1 ) {
2018-09-04 21:00:02 +00:00
if ( PQstatus ( conn ) ! = CONNECTION_OK ) {
fprintf ( stderr , " ERROR: Online Notification thread lost connection to Postgres. " ) ;
exit ( - 1 ) ;
}
2018-09-06 22:14:16 +00:00
// map used to send notifications to front end
std : : unordered_map < std : : string , std : : vector < std : : string > > updateMap ;
2018-09-04 21:00:02 +00:00
std : : unordered_map < std : : pair < uint64_t , uint64_t > , std : : pair < int64_t , InetAddress > , _PairHasher > lastOnline ;
{
std : : lock_guard < std : : mutex > l ( _lastOnline_l ) ;
lastOnline . swap ( _lastOnline ) ;
}
PGresult * res = NULL ;
int qCount = 0 ;
2018-09-06 00:03:49 +00:00
res = PQexec ( conn , " BEGIN " ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error on BEGIN command (onlineNotificationThread): %s \n " , PQresultErrorMessage ( res ) ) ;
2018-09-04 21:00:02 +00:00
PQclear ( res ) ;
2018-09-06 00:03:49 +00:00
exit ( 1 ) ;
2018-09-04 21:00:02 +00:00
}
2018-09-06 00:03:49 +00:00
PQclear ( res ) ;
2018-09-04 21:00:02 +00:00
for ( auto i = lastOnline . begin ( ) ; i ! = lastOnline . end ( ) ; + + i ) {
2018-09-05 23:54:23 +00:00
uint64_t nwid_i = i - > first . first ;
2018-09-04 21:00:02 +00:00
char nwidTmp [ 64 ] ;
char memTmp [ 64 ] ;
char ipTmp [ 64 ] ;
2018-09-05 23:54:23 +00:00
OSUtils : : ztsnprintf ( nwidTmp , sizeof ( nwidTmp ) , " %.16llx " , nwid_i ) ;
2018-09-04 21:00:02 +00:00
OSUtils : : ztsnprintf ( memTmp , sizeof ( memTmp ) , " %.10llx " , i - > first . second ) ;
2018-09-06 00:03:49 +00:00
2018-09-05 23:54:23 +00:00
auto found = _networks . find ( nwid_i ) ;
if ( found = = _networks . end ( ) ) {
2018-09-06 00:03:49 +00:00
continue ; // skip members trying to join non-existant networks
2018-09-05 23:54:23 +00:00
}
lastOnlineCumulative [ i - > first ] = i - > second . first ;
2018-09-04 21:00:02 +00:00
std : : string networkId ( nwidTmp ) ;
std : : string memberId ( memTmp ) ;
2018-09-06 22:14:16 +00:00
std : : vector < std : : string > & members = updateMap [ networkId ] ;
members . push_back ( memberId ) ;
2018-09-04 21:00:02 +00:00
int64_t ts = i - > second . first ;
std : : string ipAddr = i - > second . second . toIpString ( ipTmp ) ;
2018-10-24 19:06:17 +00:00
std : : string timestamp = std : : to_string ( ts ) ;
2018-09-04 21:00:02 +00:00
const char * values [ 4 ] = {
networkId . c_str ( ) ,
memberId . c_str ( ) ,
2018-09-05 23:08:06 +00:00
( ipAddr . empty ( ) ? NULL : ipAddr . c_str ( ) ) ,
2018-10-24 19:06:17 +00:00
timestamp . c_str ( ) ,
2018-09-04 21:00:02 +00:00
} ;
res = PQexecParams ( conn ,
2018-09-05 23:08:06 +00:00
" INSERT INTO ztc_member_status (network_id, member_id, address, last_updated) VALUES ($1, $2, $3, TO_TIMESTAMP($4::double precision/1000)) "
2018-09-04 21:00:02 +00:00
" ON CONFLICT (network_id, member_id) DO UPDATE SET address = EXCLUDED.address, last_updated = EXCLUDED.last_updated " ,
2018-09-05 23:08:06 +00:00
4 , // number of parameters
2018-09-04 21:00:02 +00:00
NULL , // oid field. ignore
values , // values for substitution
NULL , // lengths in bytes of each value
NULL ,
0 ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " Error on Member Status upsert: %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
2018-09-05 23:54:23 +00:00
PQclear ( PQexec ( conn , " ROLLBACK " ) ) ;
continue ;
2018-09-04 21:00:02 +00:00
}
PQclear ( res ) ;
if ( ( + + qCount ) = = 1024 ) {
res = PQexec ( conn , " COMMIT " ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error on commit (onlineNotificationThread): %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
2018-09-25 17:17:15 +00:00
PQclear ( PQexec ( conn , " ROLLBACK " ) ) ;
2018-09-04 21:00:02 +00:00
exit ( 1 ) ;
}
PQclear ( res ) ;
res = PQexec ( conn , " BEGIN " ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error on BEGIN (onlineNotificationThread): %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
exit ( 1 ) ;
}
PQclear ( res ) ;
qCount = 0 ;
}
}
2018-09-06 00:03:49 +00:00
res = PQexec ( conn , " COMMIT " ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error on commit (onlineNotificationThread): %s \n " , PQresultErrorMessage ( res ) ) ;
2018-09-04 21:00:02 +00:00
PQclear ( res ) ;
2018-09-25 17:17:15 +00:00
PQclear ( PQexec ( conn , " ROLLBACK " ) ) ;
2018-09-06 00:03:49 +00:00
exit ( 1 ) ;
2018-09-04 21:00:02 +00:00
}
2018-09-06 00:03:49 +00:00
PQclear ( res ) ;
2018-09-04 21:00:02 +00:00
const int64_t now = OSUtils : : now ( ) ;
if ( ( now - lastUpdatedNetworkStatus ) > 10000 ) {
lastUpdatedNetworkStatus = now ;
std : : vector < std : : pair < uint64_t , std : : shared_ptr < _Network > > > networks ;
{
std : : lock_guard < std : : mutex > l ( _networks_l ) ;
for ( auto i = _networks . begin ( ) ; i ! = _networks . end ( ) ; + + i ) {
networks . push_back ( * i ) ;
}
}
int nCount = 0 ;
2018-09-06 00:03:49 +00:00
res = PQexec ( conn , " BEGIN " ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error on BEGIN command (onlineNotificationThread): %s \n " , PQresultErrorMessage ( res ) ) ;
2018-09-04 21:00:02 +00:00
PQclear ( res ) ;
2018-09-06 00:03:49 +00:00
exit ( 1 ) ;
2018-09-04 21:00:02 +00:00
}
2018-09-06 00:03:49 +00:00
PQclear ( res ) ;
2018-09-04 21:00:02 +00:00
for ( auto i = networks . begin ( ) ; i ! = networks . end ( ) ; + + i ) {
char tmp [ 64 ] ;
Utils : : hex ( i - > first , tmp ) ;
std : : string networkId ( tmp ) ;
2018-09-06 22:14:16 +00:00
std : : vector < std : : string > & _notUsed = updateMap [ networkId ] ;
( void ) _notUsed ;
2018-09-04 21:00:02 +00:00
uint64_t authMemberCount = 0 ;
uint64_t totalMemberCount = 0 ;
uint64_t onlineMemberCount = 0 ;
uint64_t bridgeCount = 0 ;
uint64_t ts = now ;
{
std : : lock_guard < std : : mutex > l2 ( i - > second - > lock ) ;
authMemberCount = i - > second - > authorizedMembers . size ( ) ;
totalMemberCount = i - > second - > members . size ( ) ;
bridgeCount = i - > second - > activeBridgeMembers . size ( ) ;
for ( auto m = i - > second - > members . begin ( ) ; m ! = i - > second - > members . end ( ) ; + + m ) {
auto lo = lastOnlineCumulative . find ( std : : pair < uint64_t , uint64_t > ( i - > first , m - > first ) ) ;
if ( lo ! = lastOnlineCumulative . end ( ) ) {
if ( ( now - lo - > second ) < = ( ZT_NETWORK_AUTOCONF_DELAY * 2 ) ) {
+ + onlineMemberCount ;
} else {
lastOnlineCumulative . erase ( lo ) ;
}
}
}
}
2018-10-24 19:06:17 +00:00
std : : string bc = std : : to_string ( bridgeCount ) ;
std : : string amc = std : : to_string ( authMemberCount ) ;
std : : string omc = std : : to_string ( onlineMemberCount ) ;
std : : string tmc = std : : to_string ( totalMemberCount ) ;
std : : string timestamp = std : : to_string ( ts ) ;
2018-09-04 21:00:02 +00:00
const char * values [ 6 ] = {
networkId . c_str ( ) ,
2018-10-24 19:06:17 +00:00
bc . c_str ( ) ,
amc . c_str ( ) ,
omc . c_str ( ) ,
tmc . c_str ( ) ,
timestamp . c_str ( )
2018-09-04 21:00:02 +00:00
} ;
res = PQexecParams ( conn , " INSERT INTO ztc_network_status (network_id, bridge_count, authorized_member_count, "
2018-09-05 23:08:06 +00:00
" online_member_count, total_member_count, last_modified) VALUES ($1, $2, $3, $4, $5, TO_TIMESTAMP($6::double precision/1000)) "
2018-09-04 21:00:02 +00:00
" ON CONFLICT (network_id) DO UPDATE SET bridge_count = EXCLUDED.bridge_count, "
2018-09-05 23:08:06 +00:00
" authorized_member_count = EXCLUDED.authorized_member_count, online_member_count = EXCLUDED.online_member_count, "
2018-09-04 21:00:02 +00:00
" total_member_count = EXCLUDED.total_member_count, last_modified = EXCLUDED.last_modified " ,
6 ,
NULL ,
values ,
NULL ,
NULL ,
0 ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
2018-09-05 23:08:06 +00:00
fprintf ( stderr , " ERROR: Error on Network Status upsert (onlineNotificationThread): %s \n " , PQresultErrorMessage ( res ) ) ;
2018-09-04 21:00:02 +00:00
PQclear ( res ) ;
2018-09-25 17:17:15 +00:00
PQclear ( PQexec ( conn , " ROLLBACK " ) ) ;
2018-09-04 21:00:02 +00:00
exit ( 1 ) ;
}
if ( ( + + nCount ) = = 1024 ) {
res = PQexec ( conn , " COMMIT " ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error on COMMIT (onlineNotificationThread): %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
2018-09-25 17:17:15 +00:00
PQclear ( PQexec ( conn , " ROLLBACK " ) ) ;
2018-09-04 21:00:02 +00:00
exit ( 1 ) ;
}
res = PQexec ( conn , " BEGIN " ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error on BEGIN command (onlineNotificationThread): %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
exit ( 1 ) ;
}
nCount = 0 ;
}
}
2018-09-06 00:03:49 +00:00
res = PQexec ( conn , " COMMIT " ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error on COMMIT (onlineNotificationThread): %s \n " , PQresultErrorMessage ( res ) ) ;
PQclear ( res ) ;
2018-09-25 17:17:15 +00:00
PQclear ( PQexec ( conn , " ROLLBACK " ) ) ;
2018-09-06 00:03:49 +00:00
exit ( 1 ) ;
2018-09-04 21:00:02 +00:00
}
}
2018-09-06 22:14:16 +00:00
for ( auto it = updateMap . begin ( ) ; it ! = updateMap . end ( ) ; + + it ) {
std : : string networkId = it - > first ;
std : : vector < std : : string > members = it - > second ;
std : : stringstream queryBuilder ;
std : : string membersStr = : : join ( members , " , " ) ;
queryBuilder < < " NOTIFY controller, ' " < < networkId < < " : " < < membersStr < < " ' " ;
std : : string query = queryBuilder . str ( ) ;
PGresult * res = PQexec ( conn , query . c_str ( ) ) ;
if ( PQresultStatus ( res ) ! = PGRES_COMMAND_OK ) {
fprintf ( stderr , " ERROR: Error sending NOTIFY: %s \n " , PQresultErrorMessage ( res ) ) ;
}
PQclear ( res ) ;
}
2018-08-31 21:58:15 +00:00
std : : this_thread : : sleep_for ( std : : chrono : : milliseconds ( 250 ) ) ;
}
2018-09-04 21:00:02 +00:00
PQfinish ( conn ) ;
2018-08-31 21:58:15 +00:00
}
2018-09-28 17:55:39 +00:00
# endif //ZT_CONTROLLER_USE_LIBPQ