2016-04-16 00:13:15 -04:00
// Copyright (c) 2009-2010 Satoshi Nakamoto
2022-12-24 23:49:50 +00:00
// Copyright (c) 2009-2022 The Bitcoin Core developers
2016-04-16 00:13:15 -04:00
// Distributed under the MIT software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2017-11-10 13:57:53 +13:00
# include <addrdb.h>
2016-04-16 00:13:15 -04:00
2017-11-10 13:57:53 +13:00
# include <addrman.h>
# include <chainparams.h>
# include <clientversion.h>
2019-12-29 13:04:02 -08:00
# include <cstdint>
2020-06-11 08:58:46 +02:00
# include <fs.h>
2017-11-10 13:57:53 +13:00
# include <hash.h>
2020-09-12 17:59:09 +03:00
# include <logging/timer.h>
2021-01-14 09:33:04 +01:00
# include <netbase.h>
2021-08-31 18:40:18 +01:00
# include <netgroup.h>
2017-11-10 13:57:53 +13:00
# include <random.h>
# include <streams.h>
# include <tinyformat.h>
2021-01-14 09:33:04 +01:00
# include <univalue.h>
# include <util/settings.h>
2018-10-22 15:51:11 -07:00
# include <util/system.h>
2021-08-21 13:49:16 +02:00
# include <util/translation.h>
2016-04-16 00:13:15 -04:00
2017-04-21 04:50:55 -07:00
namespace {
2021-08-21 16:17:52 +02:00
class DbNotFoundError : public std : : exception
{
using std : : exception : : exception ;
} ;
2017-04-21 04:50:55 -07:00
template < typename Stream , typename Data >
bool SerializeDB ( Stream & stream , const Data & data )
2016-04-16 00:13:15 -04:00
{
2017-04-21 04:50:55 -07:00
// Write and commit header, data
try {
2020-11-26 13:59:44 -08:00
CHashWriter hasher ( stream . GetType ( ) , stream . GetVersion ( ) ) ;
2018-03-20 17:37:32 -07:00
stream < < Params ( ) . MessageStart ( ) < < data ;
hasher < < Params ( ) . MessageStart ( ) < < data ;
2017-04-21 04:50:55 -07:00
stream < < hasher . GetHash ( ) ;
} catch ( const std : : exception & e ) {
return error ( " %s: Serialize or I/O error - %s " , __func__ , e . what ( ) ) ;
}
return true ;
2016-04-16 00:13:15 -04:00
}
2017-04-21 04:50:55 -07:00
template < typename Data >
2020-11-26 13:59:44 -08:00
bool SerializeFileDB ( const std : : string & prefix , const fs : : path & path , const Data & data , int version )
2016-04-16 00:13:15 -04:00
{
// Generate random temporary filename
2022-01-31 19:32:59 +07:00
const uint16_t randv { GetRand < uint16_t > ( ) } ;
2017-04-21 04:50:55 -07:00
std : : string tmpfn = strprintf ( " %s.%04x " , prefix , randv ) ;
2016-04-16 00:13:15 -04:00
// open temp output file, and associate with CAutoFile
2022-03-03 14:40:18 -05:00
fs : : path pathTmp = gArgs . GetDataDirNet ( ) / fs : : u8path ( tmpfn ) ;
2017-03-01 16:28:39 +00:00
FILE * file = fsbridge : : fopen ( pathTmp , " wb " ) ;
2020-11-26 13:59:44 -08:00
CAutoFile fileout ( file , SER_DISK , version ) ;
2019-06-14 08:30:43 +02:00
if ( fileout . IsNull ( ) ) {
fileout . fclose ( ) ;
remove ( pathTmp ) ;
2021-09-10 00:17:20 -04:00
return error ( " %s: Failed to open file %s " , __func__ , fs : : PathToString ( pathTmp ) ) ;
2019-06-14 08:30:43 +02:00
}
2016-04-16 00:13:15 -04:00
2017-04-21 04:50:55 -07:00
// Serialize
2019-06-14 08:30:43 +02:00
if ( ! SerializeDB ( fileout , data ) ) {
fileout . fclose ( ) ;
remove ( pathTmp ) ;
return false ;
}
if ( ! FileCommit ( fileout . Get ( ) ) ) {
fileout . fclose ( ) ;
remove ( pathTmp ) ;
2021-09-10 00:17:20 -04:00
return error ( " %s: Failed to flush file %s " , __func__ , fs : : PathToString ( pathTmp ) ) ;
2019-06-14 08:30:43 +02:00
}
2016-04-16 00:13:15 -04:00
fileout . fclose ( ) ;
2017-04-21 04:50:55 -07:00
// replace existing file, if any, with new file
2019-06-14 08:30:43 +02:00
if ( ! RenameOver ( pathTmp , path ) ) {
remove ( pathTmp ) ;
2016-04-16 00:13:15 -04:00
return error ( " %s: Rename-into-place failed " , __func__ ) ;
2019-06-14 08:30:43 +02:00
}
2016-04-16 00:13:15 -04:00
return true ;
}
2017-04-21 04:50:55 -07:00
template < typename Stream , typename Data >
2021-08-21 16:17:52 +02:00
void DeserializeDB ( Stream & stream , Data & data , bool fCheckSum = true )
2016-04-16 00:13:15 -04:00
{
2021-08-21 16:17:52 +02:00
CHashVerifier < Stream > verifier ( & stream ) ;
// de-serialize file header (network specific magic number) and ..
unsigned char pchMsgTmp [ 4 ] ;
verifier > > pchMsgTmp ;
// ... verify the network matches ours
if ( memcmp ( pchMsgTmp , Params ( ) . MessageStart ( ) , sizeof ( pchMsgTmp ) ) ) {
throw std : : runtime_error { " Invalid network magic number " } ;
}
// de-serialize data
verifier > > data ;
// verify checksum
if ( fCheckSum ) {
uint256 hashTmp ;
stream > > hashTmp ;
if ( hashTmp ! = verifier . GetHash ( ) ) {
throw std : : runtime_error { " Checksum mismatch, data corrupted " } ;
2017-04-21 04:50:55 -07:00
}
2016-04-16 00:13:15 -04:00
}
}
2017-04-21 04:50:55 -07:00
template < typename Data >
2021-08-21 16:17:52 +02:00
void DeserializeFileDB ( const fs : : path & path , Data & data , int version )
2016-04-16 00:13:15 -04:00
{
2017-04-21 04:50:55 -07:00
// open input file, and associate with CAutoFile
2021-02-18 13:34:18 +01:00
FILE * file = fsbridge : : fopen ( path , " rb " ) ;
2020-11-26 13:59:44 -08:00
CAutoFile filein ( file , SER_DISK , version ) ;
2021-02-18 13:34:18 +01:00
if ( filein . IsNull ( ) ) {
2021-08-21 16:17:52 +02:00
throw DbNotFoundError { } ;
2021-02-18 13:34:18 +01:00
}
2021-08-21 16:17:52 +02:00
DeserializeDB ( filein , data ) ;
2016-04-16 00:13:15 -04:00
}
2021-02-18 13:34:18 +01:00
} // namespace
2016-04-16 00:13:15 -04:00
2021-01-14 09:33:04 +01:00
CBanDB : : CBanDB ( fs : : path ban_list_path )
2021-09-10 00:17:20 -04:00
: m_banlist_dat ( ban_list_path + " .dat " ) ,
m_banlist_json ( ban_list_path + " .json " )
2017-04-21 04:50:55 -07:00
{
}
2016-04-16 00:13:15 -04:00
2017-04-21 04:50:55 -07:00
bool CBanDB : : Write ( const banmap_t & banSet )
{
2021-01-14 09:33:04 +01:00
std : : vector < std : : string > errors ;
if ( util : : WriteSettings ( m_banlist_json , { { JSON_KEY , BanMapToJson ( banSet ) } } , errors ) ) {
return true ;
}
for ( const auto & err : errors ) {
error ( " %s " , err ) ;
}
return false ;
2017-04-21 04:50:55 -07:00
}
2016-04-16 00:13:15 -04:00
2021-07-28 20:00:23 +02:00
bool CBanDB : : Read ( banmap_t & banSet )
2017-04-21 04:50:55 -07:00
{
2021-07-28 20:00:23 +02:00
if ( fs : : exists ( m_banlist_dat ) ) {
2021-09-10 00:17:20 -04:00
LogPrintf ( " banlist.dat ignored because it can only be read by " PACKAGE_NAME " version 22.x. Remove %s to silence this warning. \n " , fs : : quoted ( fs : : PathToString ( m_banlist_dat ) ) ) ;
2021-07-28 20:00:23 +02:00
}
// If the JSON banlist does not exist, then recreate it
2021-01-14 09:33:04 +01:00
if ( ! fs : : exists ( m_banlist_json ) ) {
2021-07-28 20:00:23 +02:00
return false ;
2021-01-14 09:33:04 +01:00
}
std : : map < std : : string , util : : SettingsValue > settings ;
std : : vector < std : : string > errors ;
if ( ! util : : ReadSettings ( m_banlist_json , settings , errors ) ) {
for ( const auto & err : errors ) {
2021-09-10 00:17:20 -04:00
LogPrintf ( " Cannot load banlist %s: %s \n " , fs : : PathToString ( m_banlist_json ) , err ) ;
2021-01-14 09:33:04 +01:00
}
return false ;
}
try {
BanMapFromJson ( settings [ JSON_KEY ] , banSet ) ;
} catch ( const std : : runtime_error & e ) {
2021-09-10 00:17:20 -04:00
LogPrintf ( " Cannot parse banlist %s: %s \n " , fs : : PathToString ( m_banlist_json ) , e . what ( ) ) ;
2021-01-14 09:33:04 +01:00
return false ;
}
return true ;
2017-04-21 04:50:55 -07:00
}
2016-04-16 00:13:15 -04:00
2021-09-10 18:16:37 -06:00
bool DumpPeerAddresses ( const ArgsManager & args , const AddrMan & addr )
2017-04-21 04:50:55 -07:00
{
2021-08-21 11:22:21 +02:00
const auto pathAddr = args . GetDataDirNet ( ) / " peers.dat " ;
2020-11-26 13:59:44 -08:00
return SerializeFileDB ( " peers " , pathAddr , addr , CLIENT_VERSION ) ;
2016-04-16 00:13:15 -04:00
}
2021-09-10 18:16:37 -06:00
void ReadFromStream ( AddrMan & addr , CDataStream & ssPeers )
2016-04-16 00:13:15 -04:00
{
2021-08-21 16:17:52 +02:00
DeserializeDB ( ssPeers , addr , false ) ;
2016-04-16 00:13:15 -04:00
}
2020-09-12 17:59:09 +03:00
2021-08-31 18:40:18 +01:00
std : : optional < bilingual_str > LoadAddrman ( const NetGroupManager & netgroupman , const ArgsManager & args , std : : unique_ptr < AddrMan > & addrman )
2021-08-21 13:49:16 +02:00
{
2019-08-22 21:40:41 -04:00
auto check_addrman = std : : clamp < int32_t > ( args . GetIntArg ( " -checkaddrman " , DEFAULT_ADDRMAN_CONSISTENCY_CHECKS ) , 0 , 1000000 ) ;
2021-08-31 18:40:18 +01:00
addrman = std : : make_unique < AddrMan > ( netgroupman , /*deterministic=*/ false , /*consistency_check_ratio=*/ check_addrman ) ;
2021-08-21 13:49:16 +02:00
2022-06-28 17:50:53 +02:00
const auto start { SteadyClock : : now ( ) } ;
2021-08-21 15:20:00 +02:00
const auto path_addr { args . GetDataDirNet ( ) / " peers.dat " } ;
2021-08-21 16:17:52 +02:00
try {
DeserializeFileDB ( path_addr , * addrman , CLIENT_VERSION ) ;
2022-06-28 17:50:53 +02:00
LogPrintf ( " Loaded %i addresses from peers.dat %dms \n " , addrman - > size ( ) , Ticks < std : : chrono : : milliseconds > ( SteadyClock : : now ( ) - start ) ) ;
2021-08-21 16:17:52 +02:00
} catch ( const DbNotFoundError & ) {
2021-08-21 13:49:16 +02:00
// Addrman can be in an inconsistent state after failure, reset it
2021-08-31 18:40:18 +01:00
addrman = std : : make_unique < AddrMan > ( netgroupman , /*deterministic=*/ false , /*consistency_check_ratio=*/ check_addrman ) ;
2021-09-10 00:17:20 -04:00
LogPrintf ( " Creating peers.dat because the file was not found (%s) \n " , fs : : quoted ( fs : : PathToString ( path_addr ) ) ) ;
2021-08-21 13:49:16 +02:00
DumpPeerAddresses ( args , * addrman ) ;
2022-01-29 21:58:59 +09:00
} catch ( const InvalidAddrManVersionError & ) {
if ( ! RenameOver ( path_addr , ( fs : : path ) path_addr + " .bak " ) ) {
addrman = nullptr ;
return strprintf ( _ ( " Failed to rename invalid peers.dat file. Please move or delete it and try again. " ) ) ;
}
// Addrman can be in an inconsistent state after failure, reset it
2021-08-31 18:40:18 +01:00
addrman = std : : make_unique < AddrMan > ( netgroupman , /*deterministic=*/ false , /*consistency_check_ratio=*/ check_addrman ) ;
2022-01-29 21:58:59 +09:00
LogPrintf ( " Creating new peers.dat because the file version was not compatible (%s). Original backed up to peers.dat.bak \n " , fs : : quoted ( fs : : PathToString ( path_addr ) ) ) ;
DumpPeerAddresses ( args , * addrman ) ;
2021-08-21 16:17:52 +02:00
} catch ( const std : : exception & e ) {
addrman = nullptr ;
return strprintf ( _ ( " Invalid or corrupt peers.dat (%s). If you believe this is a bug, please report it to %s. As a workaround, you can move the file (%s) out of the way (rename, move, or delete) to have a new one created on the next start. " ) ,
2021-09-10 00:17:20 -04:00
e . what ( ) , PACKAGE_BUGREPORT , fs : : quoted ( fs : : PathToString ( path_addr ) ) ) ;
2021-08-21 13:49:16 +02:00
}
return std : : nullopt ;
}
2020-09-12 17:59:09 +03:00
void DumpAnchors ( const fs : : path & anchors_db_path , const std : : vector < CAddress > & anchors )
{
LOG_TIME_SECONDS ( strprintf ( " Flush %d outbound block-relay-only peer addresses to anchors.dat " , anchors . size ( ) ) ) ;
2020-11-26 13:59:44 -08:00
SerializeFileDB ( " anchors " , anchors_db_path , anchors , CLIENT_VERSION | ADDRV2_FORMAT ) ;
2020-09-12 17:59:09 +03:00
}
2020-09-12 18:01:19 +03:00
std : : vector < CAddress > ReadAnchors ( const fs : : path & anchors_db_path )
{
std : : vector < CAddress > anchors ;
2021-08-21 16:17:52 +02:00
try {
DeserializeFileDB ( anchors_db_path , anchors , CLIENT_VERSION | ADDRV2_FORMAT ) ;
2021-09-10 00:17:20 -04:00
LogPrintf ( " Loaded %i addresses from %s \n " , anchors . size ( ) , fs : : quoted ( fs : : PathToString ( anchors_db_path . filename ( ) ) ) ) ;
2021-08-21 16:17:52 +02:00
} catch ( const std : : exception & ) {
2020-09-12 18:01:19 +03:00
anchors . clear ( ) ;
}
fs : : remove ( anchors_db_path ) ;
return anchors ;
}