Raven Core  3.0.0
P2P Digital Currency
db.h
Go to the documentation of this file.
1 // Copyright (c) 2009-2010 Satoshi Nakamoto
2 // Copyright (c) 2009-2016 The Bitcoin Core developers
3 // Copyright (c) 2017-2019 The Raven Core developers
4 // Distributed under the MIT software license, see the accompanying
5 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
6 
7 #ifndef RAVEN_WALLET_DB_H
8 #define RAVEN_WALLET_DB_H
9 
10 #include "clientversion.h"
11 #include "fs.h"
12 #include "serialize.h"
13 #include "streams.h"
14 #include "sync.h"
15 #include "version.h"
16 
17 #include <atomic>
18 #include <map>
19 #include <string>
20 #include <vector>
21 
22 #include <db_cxx.h>
23 
24 static const unsigned int DEFAULT_WALLET_DBLOGSIZE = 100;
25 static const bool DEFAULT_WALLET_PRIVDB = true;
26 
27 class CDBEnv
28 {
29 private:
30  bool fDbEnvInit;
31  bool fMockDb;
32  // Don't change into fs::path, as that can result in
33  // shutdown problems/crashes caused by a static initialized internal pointer.
34  std::string strPath;
35 
36  void EnvShutdown();
37 
38 public:
40  DbEnv *dbenv;
41  std::map<std::string, int> mapFileUseCount;
42  std::map<std::string, Db*> mapDb;
43 
44  CDBEnv();
45  ~CDBEnv();
46  void Reset();
47 
48  void MakeMock();
49  bool IsMock() const { return fMockDb; }
50 
60  typedef bool (*recoverFunc_type)(const std::string& strFile, std::string& out_backup_filename);
61  VerifyResult Verify(const std::string& strFile, recoverFunc_type recoverFunc, std::string& out_backup_filename);
69  typedef std::pair<std::vector<unsigned char>, std::vector<unsigned char> > KeyValPair;
70  bool Salvage(const std::string& strFile, bool fAggressive, std::vector<KeyValPair>& vResult);
71 
72  bool Open(const fs::path& path);
73  void Close();
74  void Flush(bool fShutdown);
75  void CheckpointLSN(const std::string& strFile);
76 
77  void CloseDb(const std::string& strFile);
78 
79  DbTxn* TxnBegin(int flags = DB_TXN_WRITE_NOSYNC)
80  {
81  DbTxn* ptxn = nullptr;
82  int ret = dbenv->txn_begin(nullptr, &ptxn, flags);
83  if (!ptxn || ret != 0)
84  return nullptr;
85  return ptxn;
86  }
87 };
88 
89 extern CDBEnv bitdb;
90 
95 {
96  friend class CDB;
97 public:
99  CWalletDBWrapper() : nUpdateCounter(0), nLastSeen(0), nLastFlushed(0), nLastWalletUpdate(0), env(nullptr)
100  {
101  }
102 
104  CWalletDBWrapper(CDBEnv *env_in, const std::string &strFile_in) :
105  nUpdateCounter(0), nLastSeen(0), nLastFlushed(0), nLastWalletUpdate(0), env(env_in), strFile(strFile_in)
106  {
107  }
108 
111  bool Rewrite(const char* pszSkip=nullptr);
112 
115  bool Backup(const std::string& strDest);
116 
119  std::string GetName() const { return strFile; }
120 
123  void Flush(bool shutdown);
124 
125  void IncrementUpdateCounter();
126 
127  std::atomic<unsigned int> nUpdateCounter;
128  unsigned int nLastSeen;
129  unsigned int nLastFlushed;
131 
132 private:
135  std::string strFile;
136 
141  bool IsDummy() { return env == nullptr; }
142 };
143 
144 
146 class CDB
147 {
148 protected:
149  Db* pdb;
150  std::string strFile;
151  DbTxn* activeTxn;
152  bool fReadOnly;
155 
156 public:
157  explicit CDB(CWalletDBWrapper& dbw, const char* pszMode = "r+", bool fFlushOnCloseIn=true);
158  ~CDB() { Close(); }
159 
160  CDB(const CDB&) = delete;
161  CDB& operator=(const CDB&) = delete;
162 
163  void Flush();
164  void Close();
165  static bool Recover(const std::string& filename, void *callbackDataIn, bool (*recoverKVcallback)(void* callbackData, CDataStream ssKey, CDataStream ssValue), std::string& out_backup_filename);
166 
167  /* flush the wallet passively (TRY_LOCK)
168  ideal to be called periodically */
169  static bool PeriodicFlush(CWalletDBWrapper& dbw);
170  /* verifies the database environment */
171  static bool VerifyEnvironment(const std::string& walletFile, const fs::path& dataDir, std::string& errorStr);
172  /* verifies the database file */
173  static bool VerifyDatabaseFile(const std::string& walletFile, const fs::path& dataDir, std::string& warningStr, std::string& errorStr, CDBEnv::recoverFunc_type recoverFunc);
174 
175 public:
176  template <typename K, typename T>
177  bool Read(const K& key, T& value)
178  {
179  if (!pdb)
180  return false;
181 
182  // Key
183  CDataStream ssKey(SER_DISK, CLIENT_VERSION);
184  ssKey.reserve(1000);
185  ssKey << key;
186  Dbt datKey(ssKey.data(), ssKey.size());
187 
188  // Read
189  Dbt datValue;
190  datValue.set_flags(DB_DBT_MALLOC);
191  int ret = pdb->get(activeTxn, &datKey, &datValue, 0);
192  memory_cleanse(datKey.get_data(), datKey.get_size());
193  bool success = false;
194  if (datValue.get_data() != nullptr) {
195  // Unserialize value
196  try {
197  CDataStream ssValue((char*)datValue.get_data(), (char*)datValue.get_data() + datValue.get_size(), SER_DISK, CLIENT_VERSION);
198  ssValue >> value;
199  success = true;
200  } catch (const std::exception&) {
201  // In this case success remains 'false'
202  }
203 
204  // Clear and free memory
205  memory_cleanse(datValue.get_data(), datValue.get_size());
206  free(datValue.get_data());
207  }
208  return ret == 0 && success;
209  }
210 
211  template <typename K, typename T>
212  bool Write(const K& key, const T& value, bool fOverwrite = true)
213  {
214  if (!pdb)
215  return true;
216  if (fReadOnly)
217  assert(!"Write called on database in read-only mode");
218 
219  // Key
220  CDataStream ssKey(SER_DISK, CLIENT_VERSION);
221  ssKey.reserve(1000);
222  ssKey << key;
223  Dbt datKey(ssKey.data(), ssKey.size());
224 
225  // Value
226  CDataStream ssValue(SER_DISK, CLIENT_VERSION);
227  ssValue.reserve(10000);
228  ssValue << value;
229  Dbt datValue(ssValue.data(), ssValue.size());
230 
231  // Write
232  int ret = pdb->put(activeTxn, &datKey, &datValue, (fOverwrite ? 0 : DB_NOOVERWRITE));
233 
234  // Clear memory in case it was a private key
235  memory_cleanse(datKey.get_data(), datKey.get_size());
236  memory_cleanse(datValue.get_data(), datValue.get_size());
237  return (ret == 0);
238  }
239 
240  template <typename K>
241  bool Erase(const K& key)
242  {
243  if (!pdb)
244  return false;
245  if (fReadOnly)
246  assert(!"Erase called on database in read-only mode");
247 
248  // Key
249  CDataStream ssKey(SER_DISK, CLIENT_VERSION);
250  ssKey.reserve(1000);
251  ssKey << key;
252  Dbt datKey(ssKey.data(), ssKey.size());
253 
254  // Erase
255  int ret = pdb->del(activeTxn, &datKey, 0);
256 
257  // Clear memory
258  memory_cleanse(datKey.get_data(), datKey.get_size());
259  return (ret == 0 || ret == DB_NOTFOUND);
260  }
261 
262  template <typename K>
263  bool Exists(const K& key)
264  {
265  if (!pdb)
266  return false;
267 
268  // Key
269  CDataStream ssKey(SER_DISK, CLIENT_VERSION);
270  ssKey.reserve(1000);
271  ssKey << key;
272  Dbt datKey(ssKey.data(), ssKey.size());
273 
274  // Exists
275  int ret = pdb->exists(activeTxn, &datKey, 0);
276 
277  // Clear memory
278  memory_cleanse(datKey.get_data(), datKey.get_size());
279  return (ret == 0);
280  }
281 
282  Dbc* GetCursor()
283  {
284  if (!pdb)
285  return nullptr;
286  Dbc* pcursor = nullptr;
287  int ret = pdb->cursor(nullptr, &pcursor, 0);
288  if (ret != 0)
289  return nullptr;
290  return pcursor;
291  }
292 
293  int ReadAtCursor(Dbc* pcursor, CDataStream& ssKey, CDataStream& ssValue, bool setRange = false)
294  {
295  // Read at cursor
296  Dbt datKey;
297  unsigned int fFlags = DB_NEXT;
298  if (setRange) {
299  datKey.set_data(ssKey.data());
300  datKey.set_size(ssKey.size());
301  fFlags = DB_SET_RANGE;
302  }
303  Dbt datValue;
304  datKey.set_flags(DB_DBT_MALLOC);
305  datValue.set_flags(DB_DBT_MALLOC);
306  int ret = pcursor->get(&datKey, &datValue, fFlags);
307  if (ret != 0)
308  return ret;
309  else if (datKey.get_data() == nullptr || datValue.get_data() == nullptr)
310  return 99999;
311 
312  // Convert to streams
313  ssKey.SetType(SER_DISK);
314  ssKey.clear();
315  ssKey.write((char*)datKey.get_data(), datKey.get_size());
316  ssValue.SetType(SER_DISK);
317  ssValue.clear();
318  ssValue.write((char*)datValue.get_data(), datValue.get_size());
319 
320  // Clear and free memory
321  memory_cleanse(datKey.get_data(), datKey.get_size());
322  memory_cleanse(datValue.get_data(), datValue.get_size());
323  free(datKey.get_data());
324  free(datValue.get_data());
325  return 0;
326  }
327 
328 public:
329  bool TxnBegin()
330  {
331  if (!pdb || activeTxn)
332  return false;
333  DbTxn* ptxn = bitdb.TxnBegin();
334  if (!ptxn)
335  return false;
336  activeTxn = ptxn;
337  return true;
338  }
339 
340  bool TxnCommit()
341  {
342  if (!pdb || !activeTxn)
343  return false;
344  int ret = activeTxn->commit(0);
345  activeTxn = nullptr;
346  return (ret == 0);
347  }
348 
349  bool TxnAbort()
350  {
351  if (!pdb || !activeTxn)
352  return false;
353  int ret = activeTxn->abort();
354  activeTxn = nullptr;
355  return (ret == 0);
356  }
357 
358  bool ReadVersion(int& nVersion)
359  {
360  nVersion = 0;
361  return Read(std::string("version"), nVersion);
362  }
363 
364  bool WriteVersion(int nVersion)
365  {
366  return Write(std::string("version"), nVersion);
367  }
368 
369  bool static Rewrite(CWalletDBWrapper& dbw, const char* pszSkip = nullptr);
370 };
371 
372 #endif // RAVEN_WALLET_DB_H
bool Erase(const K &key)
Definition: db.h:241
std::map< std::string, int > mapFileUseCount
Definition: db.h:41
DbTxn * activeTxn
Definition: db.h:151
unsigned int nLastSeen
Definition: db.h:128
DbTxn * TxnBegin(int flags=DB_TXN_WRITE_NOSYNC)
Definition: db.h:79
int flags
Definition: raven-tx.cpp:500
void Reset()
Definition: db.cpp:77
std::string strFile
Definition: db.h:150
bool Write(const K &key, const T &value, bool fOverwrite=true)
Definition: db.h:212
bool fReadOnly
Definition: db.h:152
value_type * data()
Definition: streams.h:247
Dbc * GetCursor()
Definition: db.h:282
void Flush(bool fShutdown)
Definition: db.cpp:598
CDBEnv bitdb
Definition: db.cpp:62
void EnvShutdown()
Definition: db.cpp:64
bool Exists(const K &key)
Definition: db.h:263
Double ended buffer combining vector and stream-like interfaces.
Definition: streams.h:147
void write(const char *pch, size_t nSize)
Definition: streams.h:382
bool Salvage(const std::string &strFile, bool fAggressive, std::vector< KeyValPair > &vResult)
Definition: db.cpp:327
~CDBEnv()
Definition: db.cpp:90
CDBEnv * env
Definition: db.h:154
CWalletDBWrapper(CDBEnv *env_in, const std::string &strFile_in)
Create DB handle to real database.
Definition: db.h:104
std::string strPath
Definition: db.h:34
bool fDbEnvInit
Definition: db.h:30
bool TxnBegin()
Definition: db.h:329
std::atomic< unsigned int > nUpdateCounter
Definition: db.h:127
bool IsDummy()
Return whether this database handle is a dummy for testing.
Definition: db.h:141
void MakeMock()
Definition: db.cpp:149
VerifyResult
Verify that database file strFile is OK.
Definition: db.h:57
void memory_cleanse(void *ptr, size_t len)
Definition: cleanse.cpp:28
An instance of this class represents one database.
Definition: db.h:94
size_type size() const
Definition: streams.h:238
std::map< std::string, Db * > mapDb
Definition: db.h:42
bool ReadVersion(int &nVersion)
Definition: db.h:358
bool TxnCommit()
Definition: db.h:340
unsigned int nLastFlushed
Definition: db.h:129
bool TxnAbort()
Definition: db.h:349
RAII class that provides access to a Berkeley database.
Definition: db.h:146
bool IsMock() const
Definition: db.h:49
CCriticalSection cs_db
Definition: db.h:39
int64_t nLastWalletUpdate
Definition: db.h:130
bool(* recoverFunc_type)(const std::string &strFile, std::string &out_backup_filename)
Definition: db.h:60
bool Read(const K &key, T &value)
Definition: db.h:177
void Close()
Definition: db.cpp:97
static bool Rewrite(CWalletDBWrapper &dbw, const char *pszSkip=nullptr)
Definition: db.cpp:508
void reserve(size_type n)
Definition: streams.h:241
void CloseDb(const std::string &strFile)
Definition: db.cpp:494
bool WriteVersion(int nVersion)
Definition: db.h:364
Definition: db.h:27
bool fMockDb
Definition: db.h:31
int ReadAtCursor(Dbc *pcursor, CDataStream &ssKey, CDataStream &ssValue, bool setRange=false)
Definition: db.h:293
bool Open(const fs::path &path)
Definition: db.cpp:102
void CheckpointLSN(const std::string &strFile)
Definition: db.cpp:388
void clear()
Definition: streams.h:244
std::string strFile
Definition: db.h:135
CWalletDBWrapper()
Create dummy DB handle.
Definition: db.h:99
Db * pdb
Definition: db.h:149
std::pair< std::vector< unsigned char >, std::vector< unsigned char > > KeyValPair
Salvage data from a file that Verify says is bad.
Definition: db.h:69
CDBEnv()
Definition: db.cpp:85
CDBEnv * env
BerkeleyDB specific.
Definition: db.h:134
~CDB()
Definition: db.h:158
VerifyResult Verify(const std::string &strFile, recoverFunc_type recoverFunc, std::string &out_backup_filename)
Definition: db.cpp:181
bool fFlushOnClose
Definition: db.h:153
Wrapped boost mutex: supports recursive locking, but no waiting TODO: We should move away from using ...
Definition: sync.h:92
DbEnv * dbenv
Definition: db.h:40
std::string GetName() const
Get a name for this database, for debugging etc.
Definition: db.h:119
void SetType(int n)
Definition: streams.h:338