26class DbNotFoundError :
public std::exception {
27 using std::exception::exception;
30template <
typename Stream,
typename Data>
31bool SerializeDB(
const CChainParams &chainParams, Stream &stream,
35 CHashWriter hasher(stream.GetType(), stream.GetVersion());
36 stream << chainParams.
DiskMagic() << data;
37 hasher << chainParams.
DiskMagic() << data;
38 stream << hasher.GetHash();
39 }
catch (
const std::exception &e) {
40 return error(
"%s: Serialize or I/O error - %s", __func__, e.what());
46template <
typename Data>
48 const fs::path &path,
const Data &data,
int version) {
50 const uint16_t randv{GetRand<uint16_t>()};
57 if (fileout.IsNull()) {
60 return error(
"%s: Failed to open file %s", __func__,
65 if (!SerializeDB(chainParams, fileout, data)) {
73 return error(
"%s: Failed to flush file %s", __func__,
81 return error(
"%s: Rename-into-place failed", __func__);
87template <
typename Stream,
typename Data>
88void DeserializeDB(
const CChainParams &chainParams, Stream &stream, Data &data,
89 bool fCheckSum =
true) {
93 verifier >> pchMsgTmp;
95 if (memcmp(pchMsgTmp, std::begin(chainParams.
DiskMagic()),
97 throw std::runtime_error{
"Invalid network magic number"};
107 if (hashTmp != verifier.GetHash()) {
108 throw std::runtime_error{
"Checksum mismatch, data corrupted"};
113template <
typename Data>
115 Data &data,
int version) {
119 if (filein.IsNull()) {
120 throw DbNotFoundError{};
123 DeserializeDB(chainParams, filein, data);
129 : m_ban_list_path(
std::move(ban_list_path)), chainParams(_chainParams) {}
143 }
catch (
const std::exception &) {
144 LogPrintf(
"Missing or invalid file %s\n",
155 return SerializeFileDB(chainParams,
"peers", pathAddr, addr,
161 DeserializeDB(chainParams, ssPeers, addr,
false);
167 auto check_addrman = std::clamp<int32_t>(
170 auto addrman{std::make_unique<AddrMan>(
171 asmap, check_addrman)};
176 DeserializeFileDB(chainparams, path_addr, *addrman,
CLIENT_VERSION);
177 LogPrintf(
"Loaded %i addresses from peers.dat %dms\n", addrman->size(),
179 }
catch (
const DbNotFoundError &) {
181 addrman = std::make_unique<AddrMan>(
182 asmap, check_addrman);
183 LogPrintf(
"Creating peers.dat because the file was not found (%s)\n",
189 strprintf(
_(
"Failed to rename invalid peers.dat file. "
190 "Please move or delete it and try again."))};
193 addrman = std::make_unique<AddrMan>(
194 asmap, check_addrman);
195 LogPrintf(
"Creating new peers.dat because the file version was not "
196 "compatible (%s). Original backed up to peers.dat.bak\n",
199 }
catch (
const std::exception &e) {
201 _(
"Invalid or corrupt peers.dat (%s). If you believe this is a "
202 "bug, please report it to %s. As a workaround, you can move the "
203 "file (%s) out of the way (rename, move, or delete) to have a "
204 "new one created on the next start."),
205 e.what(), PACKAGE_BUGREPORT,
212 return {std::move(addrman)};
217 const std::vector<CAddress> &anchors) {
219 "Flush %d outbound block-relay-only peer addresses to anchors.dat",
221 SerializeFileDB(chainParams,
"anchors", anchors_db_path, anchors,
227 std::vector<CAddress> anchors;
229 DeserializeFileDB(chainParams, anchors_db_path, anchors,
231 LogPrintf(
"Loaded %i addresses from %s\n", anchors.size(),
233 }
catch (
const std::exception &) {
237 fs::remove(anchors_db_path);
std::vector< CAddress > ReadAnchors(const CChainParams &chainParams, const fs::path &anchors_db_path)
Read the anchor IP address database (anchors.dat)
util::Result< std::unique_ptr< AddrMan > > LoadAddrman(const CChainParams &chainparams, const std::vector< bool > &asmap, const ArgsManager &args)
Returns an error string on failure.
void ReadFromStream(const CChainParams &chainParams, AddrMan &addr, CDataStream &ssPeers)
Only used by tests.
bool DumpPeerAddresses(const CChainParams &chainParams, const ArgsManager &args, const AddrMan &addr)
void DumpAnchors(const CChainParams &chainParams, const fs::path &anchors_db_path, const std::vector< CAddress > &anchors)
Dump the anchor IP address database (anchors.dat)
static constexpr int32_t DEFAULT_ADDRMAN_CONSISTENCY_CHECKS
Default for -checkaddrman.
Stochastic address manager.
fs::path GetDataDirNet() const
Get data directory path with appended network identifier.
int64_t GetIntArg(const std::string &strArg, int64_t nDefault) const
Return integer argument or default value.
bool Write(const banmap_t &banSet)
const fs::path m_ban_list_path
CBanDB(fs::path ban_list_path, const CChainParams &_chainParams)
bool Read(banmap_t &banSet)
const CChainParams & chainParams
CChainParams defines various tweakable parameters of a given instance of the Bitcoin system.
const CMessageHeader::MessageMagic & DiskMagic() const
Double ended buffer combining vector and stream-like interfaces.
Reads data from an underlying stream, while hashing the read data.
Path class wrapper to block calls to the fs::path(std::string) implicit constructor and the fs::path:...
static constexpr int CLIENT_VERSION
bitcoind-res.rc includes this file, but it cannot cope with real c++ code.
bool RenameOver(fs::path src, fs::path dest)
bool FileCommit(FILE *file)
Ensure file contents are fully committed to disk, using a platform-specific feature analogous to fsyn...
bool error(const char *fmt, const Args &...args)
static auto quoted(const std::string &s)
static std::string PathToString(const path &path)
Convert path object to byte string.
FILE * fopen(const fs::path &p, const char *mode)
Implement std::hash so RCUPtr can be used as a key for maps or sets.
std::map< CSubNet, CBanEntry > banmap_t
static constexpr int ADDRV2_FORMAT
A flag that is ORed into the protocol version to designate that addresses should be serialized in (un...
int64_t GetTimeMillis()
Returns the system time (not mockable)
#define LOG_TIME_SECONDS(end_msg)
bilingual_str _(const char *psz)
Translation function.