LCOV - code coverage report
Current view: top level - src - addrdb.cpp (source / functions) Hit Total Coverage
Test: fuzz_coverage.info Lines: 44 172 25.6 %
Date: 2024-05-24 08:22:33 Functions: 6 21 28.6 %
Branches: 40 520 7.7 %

           Branch data     Line data    Source code
       1                 :            : // Copyright (c) 2009-2010 Satoshi Nakamoto
       2                 :            : // Copyright (c) 2009-2022 The Bitcoin Core developers
       3                 :            : // Distributed under the MIT software license, see the accompanying
       4                 :            : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
       5                 :            : 
       6                 :            : #include <config/bitcoin-config.h> // IWYU pragma: keep
       7                 :            : 
       8                 :            : #include <addrdb.h>
       9                 :            : 
      10                 :            : #include <addrman.h>
      11                 :            : #include <chainparams.h>
      12                 :            : #include <clientversion.h>
      13                 :            : #include <common/args.h>
      14                 :            : #include <common/settings.h>
      15                 :            : #include <cstdint>
      16                 :            : #include <hash.h>
      17                 :            : #include <logging.h>
      18                 :            : #include <logging/timer.h>
      19                 :            : #include <netbase.h>
      20                 :            : #include <netgroup.h>
      21                 :            : #include <random.h>
      22                 :            : #include <streams.h>
      23                 :            : #include <tinyformat.h>
      24                 :            : #include <univalue.h>
      25                 :            : #include <util/fs.h>
      26                 :            : #include <util/fs_helpers.h>
      27                 :            : #include <util/translation.h>
      28                 :            : 
      29                 :            : namespace {
      30                 :            : 
      31                 :            : class DbNotFoundError : public std::exception
      32                 :            : {
      33                 :            :     using std::exception::exception;
      34                 :            : };
      35                 :            : 
      36                 :            : template <typename Stream, typename Data>
      37                 :          0 : bool SerializeDB(Stream& stream, const Data& data)
      38                 :            : {
      39                 :            :     // Write and commit header, data
      40                 :            :     try {
      41   [ #  #  #  # ]:          0 :         HashedSourceWriter hashwriter{stream};
      42   [ #  #  #  #  :          0 :         hashwriter << Params().MessageStart() << data;
          #  #  #  #  #  
             #  #  #  #  
                      # ]
      43   [ #  #  #  #  :          0 :         stream << hashwriter.GetHash();
             #  #  #  # ]
      44   [ #  #  #  # ]:          0 :     } catch (const std::exception& e) {
      45   [ #  #  #  #  :          0 :         LogError("%s: Serialize or I/O error - %s\n", __func__, e.what());
          #  #  #  #  #  
                #  #  # ]
      46                 :          0 :         return false;
      47   [ #  #  #  # ]:          0 :     }
      48                 :            : 
      49                 :          0 :     return true;
      50                 :          0 : }
      51                 :            : 
      52                 :            : template <typename Data>
      53                 :          0 : bool SerializeFileDB(const std::string& prefix, const fs::path& path, const Data& data)
      54                 :            : {
      55                 :            :     // Generate random temporary filename
      56                 :          0 :     const uint16_t randv{GetRand<uint16_t>()};
      57                 :          0 :     std::string tmpfn = strprintf("%s.%04x", prefix, randv);
      58                 :            : 
      59                 :            :     // open temp output file
      60   [ #  #  #  #  :          0 :     fs::path pathTmp = gArgs.GetDataDirNet() / fs::u8path(tmpfn);
          #  #  #  #  #  
                #  #  # ]
      61   [ #  #  #  # ]:          0 :     FILE *file = fsbridge::fopen(pathTmp, "wb");
      62   [ #  #  #  # ]:          0 :     AutoFile fileout{file};
      63   [ #  #  #  #  :          0 :     if (fileout.IsNull()) {
                   #  # ]
      64   [ #  #  #  # ]:          0 :         fileout.fclose();
      65   [ #  #  #  # ]:          0 :         remove(pathTmp);
      66   [ #  #  #  #  :          0 :         LogError("%s: Failed to open file %s\n", __func__, fs::PathToString(pathTmp));
          #  #  #  #  #  
          #  #  #  #  #  
                   #  # ]
      67                 :          0 :         return false;
      68                 :            :     }
      69                 :            : 
      70                 :            :     // Serialize
      71   [ #  #  #  #  :          0 :     if (!SerializeDB(fileout, data)) {
             #  #  #  # ]
      72   [ #  #  #  # ]:          0 :         fileout.fclose();
      73   [ #  #  #  # ]:          0 :         remove(pathTmp);
      74                 :          1 :         return false;
      75                 :            :     }
      76   [ #  #  #  #  :          0 :     if (!FileCommit(fileout.Get())) {
          #  #  #  #  #  
                      # ]
      77   [ #  #  #  # ]:          0 :         fileout.fclose();
      78   [ #  #  #  # ]:          0 :         remove(pathTmp);
      79   [ #  #  #  #  :          0 :         LogError("%s: Failed to flush file %s\n", __func__, fs::PathToString(pathTmp));
          #  #  #  #  #  
          #  #  #  #  #  
                   #  # ]
      80                 :          0 :         return false;
      81                 :            :     }
      82   [ #  #  #  # ]:          0 :     fileout.fclose();
      83                 :            : 
      84                 :            :     // replace existing file, if any, with new file
      85   [ #  #  #  #  :          0 :     if (!RenameOver(pathTmp, path)) {
          #  #  #  #  #  
          #  #  #  #  #  
                   #  # ]
      86   [ #  #  #  # ]:          0 :         remove(pathTmp);
      87   [ #  #  #  #  :          0 :         LogError("%s: Rename-into-place failed\n", __func__);
          #  #  #  #  #  
                #  #  # ]
      88                 :          0 :         return false;
      89                 :            :     }
      90                 :            : 
      91                 :          0 :     return true;
      92                 :          0 : }
      93                 :            : 
      94                 :            : template <typename Stream, typename Data>
      95                 :       1739 : void DeserializeDB(Stream& stream, Data&& data, bool fCheckSum = true)
      96                 :            : {
      97                 :       1739 :     HashVerifier verifier{stream};
      98                 :            :     // de-serialize file header (network specific magic number) and ..
      99                 :       1739 :     MessageStartChars pchMsgTmp;
     100                 :       1739 :     verifier >> pchMsgTmp;
     101                 :            :     // ... verify the network matches ours
     102   [ +  +  #  #  :       1739 :     if (pchMsgTmp != Params().MessageStart()) {
                   #  # ]
     103   [ +  -  #  #  :          7 :         throw std::runtime_error{"Invalid network magic number"};
                   #  # ]
     104                 :            :     }
     105                 :            : 
     106                 :            :     // de-serialize data
     107                 :       1732 :     verifier >> data;
     108                 :            : 
     109                 :            :     // verify checksum
     110   [ +  -  #  #  :       1732 :     if (fCheckSum) {
                   #  # ]
     111                 :          0 :         uint256 hashTmp;
     112                 :          0 :         stream >> hashTmp;
     113   [ #  #  #  #  :          0 :         if (hashTmp != verifier.GetHash()) {
                   #  # ]
     114   [ #  #  #  #  :          0 :             throw std::runtime_error{"Checksum mismatch, data corrupted"};
                   #  # ]
     115                 :            :         }
     116                 :          0 :     }
     117                 :       1732 : }
     118                 :            : 
     119                 :            : template <typename Data>
     120                 :          0 : void DeserializeFileDB(const fs::path& path, Data&& data)
     121                 :            : {
     122                 :          0 :     FILE* file = fsbridge::fopen(path, "rb");
     123   [ #  #  #  # ]:          0 :     AutoFile filein{file};
     124   [ #  #  #  # ]:          0 :     if (filein.IsNull()) {
     125   [ #  #  #  # ]:          0 :         throw DbNotFoundError{};
     126                 :            :     }
     127   [ #  #  #  # ]:          0 :     DeserializeDB(filein, data);
     128                 :          0 : }
     129                 :            : } // namespace
     130                 :            : 
     131                 :       4327 : CBanDB::CBanDB(fs::path ban_list_path)
     132         [ +  - ]:       4327 :     : m_banlist_dat(ban_list_path + ".dat"),
     133   [ +  -  -  + ]:       4327 :       m_banlist_json(ban_list_path + ".json")
     134                 :            : {
     135                 :       4327 : }
     136                 :            : 
     137                 :      25048 : bool CBanDB::Write(const banmap_t& banSet)
     138                 :            : {
     139                 :      25048 :     std::vector<std::string> errors;
     140   [ +  -  +  -  :      25048 :     if (common::WriteSettings(m_banlist_json, {{JSON_KEY, BanMapToJson(banSet)}}, errors)) {
          +  -  -  +  +  
                +  #  # ]
     141                 :      20343 :         return true;
     142                 :            :     }
     143                 :            : 
     144         [ +  + ]:       9410 :     for (const auto& err : errors) {
     145   [ +  -  +  -  :       4705 :         LogError("%s\n", err);
                   +  - ]
     146                 :       4705 :     }
     147                 :       4705 :     return false;
     148                 :      25048 : }
     149                 :            : 
     150                 :       4327 : bool CBanDB::Read(banmap_t& banSet)
     151                 :            : {
     152         [ +  - ]:       4327 :     if (fs::exists(m_banlist_dat)) {
     153   [ #  #  #  #  :          0 :         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)));
          #  #  #  #  #  
                      # ]
     154                 :          0 :     }
     155                 :            :     // If the JSON banlist does not exist, then recreate it
     156         [ +  + ]:       4327 :     if (!fs::exists(m_banlist_json)) {
     157                 :       1356 :         return false;
     158                 :            :     }
     159                 :            : 
     160                 :       2971 :     std::map<std::string, common::SettingsValue> settings;
     161                 :       2971 :     std::vector<std::string> errors;
     162                 :            : 
     163   [ +  -  +  + ]:       2971 :     if (!common::ReadSettings(m_banlist_json, settings, errors)) {
     164         [ +  + ]:       1748 :         for (const auto& err : errors) {
     165   [ -  +  -  +  :        874 :             LogPrintf("Cannot load banlist %s: %s\n", fs::PathToString(m_banlist_json), err);
             -  +  +  - ]
     166                 :        874 :         }
     167                 :        874 :         return false;
     168                 :            :     }
     169                 :            : 
     170                 :            :     try {
     171   [ +  -  +  -  :       2097 :         BanMapFromJson(settings[JSON_KEY], banSet);
                   +  + ]
     172         [ -  + ]:       2097 :     } catch (const std::runtime_error& e) {
     173   [ +  -  +  -  :        169 :         LogPrintf("Cannot parse banlist %s: %s\n", fs::PathToString(m_banlist_json), e.what());
             +  -  -  + ]
     174                 :        169 :         return false;
     175   [ +  -  #  # ]:        169 :     }
     176                 :            : 
     177                 :       1928 :     return true;
     178                 :       4496 : }
     179                 :            : 
     180                 :          0 : bool DumpPeerAddresses(const ArgsManager& args, const AddrMan& addr)
     181                 :            : {
     182         [ #  # ]:          0 :     const auto pathAddr = args.GetDataDirNet() / "peers.dat";
     183   [ #  #  #  # ]:          0 :     return SerializeFileDB("peers", pathAddr, addr);
     184                 :          0 : }
     185                 :            : 
     186                 :       1739 : void ReadFromStream(AddrMan& addr, DataStream& ssPeers)
     187                 :            : {
     188                 :       1739 :     DeserializeDB(ssPeers, addr, false);
     189                 :       1739 : }
     190                 :            : 
     191                 :          0 : util::Result<std::unique_ptr<AddrMan>> LoadAddrman(const NetGroupManager& netgroupman, const ArgsManager& args)
     192                 :            : {
     193   [ #  #  #  #  :          0 :     auto check_addrman = std::clamp<int32_t>(args.GetIntArg("-checkaddrman", DEFAULT_ADDRMAN_CONSISTENCY_CHECKS), 0, 1000000);
                   #  # ]
     194   [ #  #  #  # ]:          0 :     bool deterministic = HasTestOption(args, "addrman"); // use a deterministic addrman only for tests
     195                 :            : 
     196                 :          0 :     auto addrman{std::make_unique<AddrMan>(netgroupman, deterministic, /*consistency_check_ratio=*/check_addrman)};
     197                 :            : 
     198                 :          0 :     const auto start{SteadyClock::now()};
     199   [ #  #  #  # ]:          0 :     const auto path_addr{args.GetDataDirNet() / "peers.dat"};
     200                 :            :     try {
     201         [ #  # ]:          0 :         DeserializeFileDB(path_addr, *addrman);
     202   [ #  #  #  #  :          0 :         LogPrintf("Loaded %i addresses from peers.dat  %dms\n", addrman->Size(), Ticks<std::chrono::milliseconds>(SteadyClock::now() - start));
          #  #  #  #  #  
                #  #  # ]
     203   [ #  #  #  #  :          0 :     } catch (const DbNotFoundError&) {
                   #  # ]
     204                 :            :         // Addrman can be in an inconsistent state after failure, reset it
     205         [ #  # ]:          0 :         addrman = std::make_unique<AddrMan>(netgroupman, deterministic, /*consistency_check_ratio=*/check_addrman);
     206   [ #  #  #  #  :          0 :         LogPrintf("Creating peers.dat because the file was not found (%s)\n", fs::quoted(fs::PathToString(path_addr)));
          #  #  #  #  #  
                      # ]
     207         [ #  # ]:          0 :         DumpPeerAddresses(args, *addrman);
     208   [ #  #  #  # ]:          0 :     } catch (const InvalidAddrManVersionError&) {
     209   [ #  #  #  #  :          0 :         if (!RenameOver(path_addr, (fs::path)path_addr + ".bak")) {
          #  #  #  #  #  
                      # ]
     210   [ #  #  #  #  :          0 :             return util::Error{strprintf(_("Failed to rename invalid peers.dat file. Please move or delete it and try again."))};
                   #  # ]
     211                 :            :         }
     212                 :            :         // Addrman can be in an inconsistent state after failure, reset it
     213         [ #  # ]:          0 :         addrman = std::make_unique<AddrMan>(netgroupman, deterministic, /*consistency_check_ratio=*/check_addrman);
     214   [ #  #  #  #  :          0 :         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)));
          #  #  #  #  #  
                      # ]
     215         [ #  # ]:          0 :         DumpPeerAddresses(args, *addrman);
     216   [ #  #  #  #  :          0 :     } catch (const std::exception& e) {
                   #  # ]
     217   [ #  #  #  #  :          0 :         return util::Error{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."),
                   #  # ]
     218   [ #  #  #  # ]:          0 :                                      e.what(), PACKAGE_BUGREPORT, fs::quoted(fs::PathToString(path_addr)))};
     219   [ #  #  #  # ]:          0 :     }
     220         [ #  # ]:          0 :     return addrman;
     221                 :          0 : }
     222                 :            : 
     223                 :          0 : void DumpAnchors(const fs::path& anchors_db_path, const std::vector<CAddress>& anchors)
     224                 :            : {
     225   [ #  #  #  #  :          0 :     LOG_TIME_SECONDS(strprintf("Flush %d outbound block-relay-only peer addresses to anchors.dat", anchors.size()));
                   #  # ]
     226   [ #  #  #  #  :          0 :     SerializeFileDB("anchors", anchors_db_path, CAddress::V2_DISK(anchors));
                   #  # ]
     227                 :          0 : }
     228                 :            : 
     229                 :          0 : std::vector<CAddress> ReadAnchors(const fs::path& anchors_db_path)
     230                 :            : {
     231                 :          0 :     std::vector<CAddress> anchors;
     232                 :            :     try {
     233   [ #  #  #  # ]:          0 :         DeserializeFileDB(anchors_db_path, CAddress::V2_DISK(anchors));
     234   [ #  #  #  #  :          0 :         LogPrintf("Loaded %i addresses from %s\n", anchors.size(), fs::quoted(fs::PathToString(anchors_db_path.filename())));
          #  #  #  #  #  
                #  #  # ]
     235         [ #  # ]:          0 :     } catch (const std::exception&) {
     236                 :          0 :         anchors.clear();
     237         [ #  # ]:          0 :     }
     238                 :            : 
     239         [ #  # ]:          0 :     fs::remove(anchors_db_path);
     240                 :          0 :     return anchors;
     241         [ #  # ]:          0 : }

Generated by: LCOV version 1.16