/root/bitcoin/src/dbwrapper.cpp
Line | Count | Source |
1 | | // Copyright (c) 2012-present The Bitcoin Core developers |
2 | | // Distributed under the MIT software license, see the accompanying |
3 | | // file COPYING or http://www.opensource.org/licenses/mit-license.php. |
4 | | |
5 | | #include <dbwrapper.h> |
6 | | |
7 | | #include <leveldb/cache.h> |
8 | | #include <leveldb/db.h> |
9 | | #include <leveldb/env.h> |
10 | | #include <leveldb/filter_policy.h> |
11 | | #include <leveldb/helpers/memenv/memenv.h> |
12 | | #include <leveldb/iterator.h> |
13 | | #include <leveldb/options.h> |
14 | | #include <leveldb/slice.h> |
15 | | #include <leveldb/status.h> |
16 | | #include <leveldb/write_batch.h> |
17 | | #include <logging.h> |
18 | | #include <random.h> |
19 | | #include <serialize.h> |
20 | | #include <span.h> |
21 | | #include <streams.h> |
22 | | #include <util/fs.h> |
23 | | #include <util/fs_helpers.h> |
24 | | #include <util/log.h> |
25 | | #include <util/obfuscation.h> |
26 | | #include <util/strencodings.h> |
27 | | |
28 | | #include <algorithm> |
29 | | #include <cassert> |
30 | | #include <cstdarg> |
31 | | #include <cstdint> |
32 | | #include <cstdio> |
33 | | #include <memory> |
34 | | #include <optional> |
35 | | #include <utility> |
36 | | |
37 | 0 | static auto CharCast(const std::byte* data) { return reinterpret_cast<const char*>(data); } |
38 | | |
39 | | bool DestroyDB(const std::string& path_str) |
40 | 0 | { |
41 | 0 | return leveldb::DestroyDB(path_str, {}).ok(); |
42 | 0 | } |
43 | | |
44 | | /** Handle database error by throwing dbwrapper_error exception. |
45 | | */ |
46 | | static void HandleError(const leveldb::Status& status) |
47 | 0 | { |
48 | 0 | if (status.ok()) |
49 | 0 | return; |
50 | 0 | const std::string errmsg = "Fatal LevelDB error: " + status.ToString(); |
51 | 0 | LogError("%s", errmsg);Line | Count | Source | 97 | 0 | #define LogError(...) LogPrintLevel_(BCLog::LogFlags::ALL, BCLog::Level::Error, /*should_ratelimit=*/true, __VA_ARGS__) Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
|
|
52 | 0 | LogInfo("You can use -debug=leveldb to get more complete diagnostic messages");Line | Count | Source | 95 | 0 | #define LogInfo(...) LogPrintLevel_(BCLog::LogFlags::ALL, BCLog::Level::Info, /*should_ratelimit=*/true, __VA_ARGS__) Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
|
|
53 | 0 | throw dbwrapper_error(errmsg); |
54 | 0 | } |
55 | | |
56 | | class CBitcoinLevelDBLogger : public leveldb::Logger { |
57 | | public: |
58 | | // This code is adapted from posix_logger.h, which is why it is using vsprintf. |
59 | | // Please do not do this in normal code |
60 | 0 | void Logv(const char * format, va_list ap) override { |
61 | 0 | if (!LogAcceptCategory(BCLog::LEVELDB, util::log::Level::Debug)) { |
62 | 0 | return; |
63 | 0 | } |
64 | 0 | char buffer[500]; |
65 | 0 | for (int iter = 0; iter < 2; iter++) { |
66 | 0 | char* base; |
67 | 0 | int bufsize; |
68 | 0 | if (iter == 0) { |
69 | 0 | bufsize = sizeof(buffer); |
70 | 0 | base = buffer; |
71 | 0 | } |
72 | 0 | else { |
73 | 0 | bufsize = 30000; |
74 | 0 | base = new char[bufsize]; |
75 | 0 | } |
76 | 0 | char* p = base; |
77 | 0 | char* limit = base + bufsize; |
78 | | |
79 | | // Print the message |
80 | 0 | if (p < limit) { |
81 | 0 | va_list backup_ap; |
82 | 0 | va_copy(backup_ap, ap); |
83 | | // Do not use vsnprintf elsewhere in bitcoin source code, see above. |
84 | 0 | p += vsnprintf(p, limit - p, format, backup_ap); |
85 | 0 | va_end(backup_ap); |
86 | 0 | } |
87 | | |
88 | | // Truncate to available space if necessary |
89 | 0 | if (p >= limit) { |
90 | 0 | if (iter == 0) { |
91 | 0 | continue; // Try again with larger buffer |
92 | 0 | } |
93 | 0 | else { |
94 | 0 | p = limit - 1; |
95 | 0 | } |
96 | 0 | } |
97 | | |
98 | | // Add newline if necessary |
99 | 0 | if (p == base || p[-1] != '\n') { |
100 | 0 | *p++ = '\n'; |
101 | 0 | } |
102 | |
|
103 | 0 | assert(p <= limit); |
104 | 0 | base[std::min(bufsize - 1, (int)(p - base))] = '\0'; |
105 | 0 | LogDebug(BCLog::LEVELDB, "%s\n", util::RemoveSuffixView(base, "\n")); Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
106 | 0 | if (base != buffer) { |
107 | 0 | delete[] base; |
108 | 0 | } |
109 | 0 | break; |
110 | 0 | } |
111 | 0 | } |
112 | | }; |
113 | | |
114 | 0 | static void SetMaxOpenFiles(leveldb::Options *options) { |
115 | | // On most platforms the default setting of max_open_files (which is 1000) |
116 | | // is optimal. On Windows using a large file count is OK because the handles |
117 | | // do not interfere with select() loops. On 64-bit Unix hosts this value is |
118 | | // also OK, because up to that amount LevelDB will use an mmap |
119 | | // implementation that does not use extra file descriptors (the fds are |
120 | | // closed after being mmap'ed). |
121 | | // |
122 | | // Increasing the value beyond the default is dangerous because LevelDB will |
123 | | // fall back to a non-mmap implementation when the file count is too large. |
124 | | // On 32-bit Unix host we should decrease the value because the handles use |
125 | | // up real fds, and we want to avoid fd exhaustion issues. |
126 | | // |
127 | | // See PR #12495 for further discussion. |
128 | |
|
129 | 0 | int default_open_files = options->max_open_files; |
130 | 0 | #ifndef WIN32 |
131 | 0 | if (sizeof(void*) < 8) { |
132 | 0 | options->max_open_files = 64; |
133 | 0 | } |
134 | 0 | #endif |
135 | 0 | LogDebug(BCLog::LEVELDB, "LevelDB using max_open_files=%d (default=%d)\n", Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
136 | 0 | options->max_open_files, default_open_files); |
137 | 0 | } |
138 | | |
139 | | static leveldb::Options GetOptions(size_t nCacheSize) |
140 | 0 | { |
141 | 0 | leveldb::Options options; |
142 | 0 | options.block_cache = leveldb::NewLRUCache(nCacheSize / 2); |
143 | 0 | options.write_buffer_size = nCacheSize / 4; // up to two write buffers may be held in memory simultaneously |
144 | 0 | options.filter_policy = leveldb::NewBloomFilterPolicy(10); |
145 | 0 | options.compression = leveldb::kNoCompression; |
146 | 0 | options.info_log = new CBitcoinLevelDBLogger(); |
147 | 0 | if (leveldb::kMajorVersion > 1 || (leveldb::kMajorVersion == 1 && leveldb::kMinorVersion >= 16)) { |
148 | | // LevelDB versions before 1.16 consider short writes to be corruption. Only trigger error |
149 | | // on corruption in later versions. |
150 | 0 | options.paranoid_checks = true; |
151 | 0 | } |
152 | 0 | options.max_file_size = std::max(options.max_file_size, DBWRAPPER_MAX_FILE_SIZE); |
153 | 0 | SetMaxOpenFiles(&options); |
154 | 0 | return options; |
155 | 0 | } |
156 | | |
157 | | struct CDBBatch::WriteBatchImpl { |
158 | | leveldb::WriteBatch batch; |
159 | | }; |
160 | | |
161 | | CDBBatch::CDBBatch(const CDBWrapper& _parent) |
162 | 0 | : parent{_parent}, |
163 | 0 | m_impl_batch{std::make_unique<CDBBatch::WriteBatchImpl>()} |
164 | 0 | { |
165 | 0 | Clear(); |
166 | 0 | }; |
167 | | |
168 | 0 | CDBBatch::~CDBBatch() = default; |
169 | | |
170 | | void CDBBatch::Clear() |
171 | 0 | { |
172 | 0 | m_impl_batch->batch.Clear(); |
173 | 0 | } |
174 | | |
175 | | void CDBBatch::WriteImpl(std::span<const std::byte> key, DataStream& ssValue) |
176 | 0 | { |
177 | 0 | leveldb::Slice slKey(CharCast(key.data()), key.size()); |
178 | 0 | dbwrapper_private::GetObfuscation(parent)(ssValue); |
179 | 0 | leveldb::Slice slValue(CharCast(ssValue.data()), ssValue.size()); |
180 | 0 | m_impl_batch->batch.Put(slKey, slValue); |
181 | 0 | } |
182 | | |
183 | | void CDBBatch::EraseImpl(std::span<const std::byte> key) |
184 | 0 | { |
185 | 0 | leveldb::Slice slKey(CharCast(key.data()), key.size()); |
186 | 0 | m_impl_batch->batch.Delete(slKey); |
187 | 0 | } |
188 | | |
189 | | size_t CDBBatch::ApproximateSize() const |
190 | 0 | { |
191 | 0 | return m_impl_batch->batch.ApproximateSize(); |
192 | 0 | } |
193 | | |
194 | | struct LevelDBContext { |
195 | | //! custom environment this database is using (may be nullptr in case of default environment) |
196 | | leveldb::Env* penv; |
197 | | |
198 | | //! database options used |
199 | | leveldb::Options options; |
200 | | |
201 | | //! options used when reading from the database |
202 | | leveldb::ReadOptions readoptions; |
203 | | |
204 | | //! options used when iterating over values of the database |
205 | | leveldb::ReadOptions iteroptions; |
206 | | |
207 | | //! options used when writing to the database |
208 | | leveldb::WriteOptions writeoptions; |
209 | | |
210 | | //! options used when sync writing to the database |
211 | | leveldb::WriteOptions syncoptions; |
212 | | |
213 | | //! the database itself |
214 | | leveldb::DB* pdb; |
215 | | }; |
216 | | |
217 | | CDBWrapper::CDBWrapper(const DBParams& params) |
218 | 0 | : m_db_context{std::make_unique<LevelDBContext>()}, m_name{fs::PathToString(params.path.stem())} |
219 | 0 | { |
220 | 0 | DBContext().penv = nullptr; |
221 | 0 | DBContext().readoptions.verify_checksums = true; |
222 | 0 | DBContext().iteroptions.verify_checksums = true; |
223 | 0 | DBContext().iteroptions.fill_cache = false; |
224 | 0 | DBContext().syncoptions.sync = true; |
225 | 0 | DBContext().options = GetOptions(params.cache_bytes); |
226 | 0 | DBContext().options.create_if_missing = true; |
227 | 0 | if (params.memory_only) { |
228 | 0 | DBContext().penv = leveldb::NewMemEnv(leveldb::Env::Default()); |
229 | 0 | DBContext().options.env = DBContext().penv; |
230 | 0 | } else { |
231 | 0 | if (params.wipe_data) { |
232 | 0 | LogInfo("Wiping LevelDB in %s", fs::PathToString(params.path));Line | Count | Source | 95 | 0 | #define LogInfo(...) LogPrintLevel_(BCLog::LogFlags::ALL, BCLog::Level::Info, /*should_ratelimit=*/true, __VA_ARGS__) Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
|
|
233 | 0 | leveldb::Status result = leveldb::DestroyDB(fs::PathToString(params.path), DBContext().options); |
234 | 0 | HandleError(result); |
235 | 0 | } |
236 | 0 | TryCreateDirectories(params.path); |
237 | 0 | LogInfo("Opening LevelDB in %s", fs::PathToString(params.path));Line | Count | Source | 95 | 0 | #define LogInfo(...) LogPrintLevel_(BCLog::LogFlags::ALL, BCLog::Level::Info, /*should_ratelimit=*/true, __VA_ARGS__) Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
|
|
238 | 0 | } |
239 | | // PathToString() return value is safe to pass to leveldb open function, |
240 | | // because on POSIX leveldb passes the byte string directly to ::open(), and |
241 | | // on Windows it converts from UTF-8 to UTF-16 before calling ::CreateFileW |
242 | | // (see env_posix.cc and env_windows.cc). |
243 | 0 | leveldb::Status status = leveldb::DB::Open(DBContext().options, fs::PathToString(params.path), &DBContext().pdb); |
244 | 0 | HandleError(status); |
245 | 0 | LogInfo("Opened LevelDB successfully");Line | Count | Source | 95 | 0 | #define LogInfo(...) LogPrintLevel_(BCLog::LogFlags::ALL, BCLog::Level::Info, /*should_ratelimit=*/true, __VA_ARGS__) Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
|
|
246 | |
|
247 | 0 | if (params.options.force_compact) { |
248 | 0 | LogInfo("Starting database compaction of %s", fs::PathToString(params.path));Line | Count | Source | 95 | 0 | #define LogInfo(...) LogPrintLevel_(BCLog::LogFlags::ALL, BCLog::Level::Info, /*should_ratelimit=*/true, __VA_ARGS__) Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
|
|
249 | 0 | DBContext().pdb->CompactRange(nullptr, nullptr); |
250 | 0 | LogInfo("Finished database compaction of %s", fs::PathToString(params.path));Line | Count | Source | 95 | 0 | #define LogInfo(...) LogPrintLevel_(BCLog::LogFlags::ALL, BCLog::Level::Info, /*should_ratelimit=*/true, __VA_ARGS__) Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
|
|
251 | 0 | } |
252 | |
|
253 | 0 | if (!Read(OBFUSCATION_KEY, m_obfuscation) && params.obfuscate && IsEmpty()) { |
254 | | // Generate and write the new obfuscation key. |
255 | 0 | const Obfuscation obfuscation{FastRandomContext{}.randbytes<Obfuscation::KEY_SIZE>()}; |
256 | 0 | assert(!m_obfuscation); // Make sure the key is written without obfuscation. |
257 | 0 | Write(OBFUSCATION_KEY, obfuscation); |
258 | 0 | m_obfuscation = obfuscation; |
259 | 0 | LogInfo("Wrote new obfuscation key for %s: %s", fs::PathToString(params.path), m_obfuscation.HexKey());Line | Count | Source | 95 | 0 | #define LogInfo(...) LogPrintLevel_(BCLog::LogFlags::ALL, BCLog::Level::Info, /*should_ratelimit=*/true, __VA_ARGS__) Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
|
|
260 | 0 | } |
261 | 0 | LogInfo("Using obfuscation key for %s: %s", fs::PathToString(params.path), m_obfuscation.HexKey());Line | Count | Source | 95 | 0 | #define LogInfo(...) LogPrintLevel_(BCLog::LogFlags::ALL, BCLog::Level::Info, /*should_ratelimit=*/true, __VA_ARGS__) Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
|
|
262 | 0 | } |
263 | | |
264 | | CDBWrapper::~CDBWrapper() |
265 | 0 | { |
266 | 0 | delete DBContext().pdb; |
267 | 0 | DBContext().pdb = nullptr; |
268 | 0 | delete DBContext().options.filter_policy; |
269 | 0 | DBContext().options.filter_policy = nullptr; |
270 | 0 | delete DBContext().options.info_log; |
271 | 0 | DBContext().options.info_log = nullptr; |
272 | 0 | delete DBContext().options.block_cache; |
273 | 0 | DBContext().options.block_cache = nullptr; |
274 | 0 | delete DBContext().penv; |
275 | 0 | DBContext().options.env = nullptr; |
276 | 0 | } |
277 | | |
278 | | void CDBWrapper::WriteBatch(CDBBatch& batch, bool fSync) |
279 | 0 | { |
280 | 0 | const bool log_memory = LogAcceptCategory(BCLog::LEVELDB, util::log::Level::Debug); |
281 | 0 | double mem_before = 0; |
282 | 0 | if (log_memory) { |
283 | 0 | mem_before = DynamicMemoryUsage() / 1024.0 / 1024; |
284 | 0 | } |
285 | 0 | leveldb::Status status = DBContext().pdb->Write(fSync ? DBContext().syncoptions : DBContext().writeoptions, &batch.m_impl_batch->batch); |
286 | 0 | HandleError(status); |
287 | 0 | if (log_memory) { |
288 | 0 | double mem_after = DynamicMemoryUsage() / 1024.0 / 1024; |
289 | 0 | LogDebug(BCLog::LEVELDB, "WriteBatch memory usage: db=%s, before=%.1fMiB, after=%.1fMiB\n", Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
290 | 0 | m_name, mem_before, mem_after); |
291 | 0 | } |
292 | 0 | } |
293 | | |
294 | | size_t CDBWrapper::DynamicMemoryUsage() const |
295 | 0 | { |
296 | 0 | std::string memory; |
297 | 0 | std::optional<size_t> parsed; |
298 | 0 | if (!DBContext().pdb->GetProperty("leveldb.approximate-memory-usage", &memory) || !(parsed = ToIntegral<size_t>(memory))) { |
299 | 0 | LogDebug(BCLog::LEVELDB, "Failed to get approximate-memory-usage property\n"); Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
300 | 0 | return 0; |
301 | 0 | } |
302 | 0 | return parsed.value(); |
303 | 0 | } |
304 | | |
305 | | std::optional<std::string> CDBWrapper::ReadImpl(std::span<const std::byte> key) const |
306 | 0 | { |
307 | 0 | leveldb::Slice slKey(CharCast(key.data()), key.size()); |
308 | 0 | std::string strValue; |
309 | 0 | leveldb::Status status = DBContext().pdb->Get(DBContext().readoptions, slKey, &strValue); |
310 | 0 | if (!status.ok()) { |
311 | 0 | if (status.IsNotFound()) |
312 | 0 | return std::nullopt; |
313 | 0 | LogError("LevelDB read failure: %s", status.ToString());Line | Count | Source | 97 | 0 | #define LogError(...) LogPrintLevel_(BCLog::LogFlags::ALL, BCLog::Level::Error, /*should_ratelimit=*/true, __VA_ARGS__) Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
|
|
314 | 0 | HandleError(status); |
315 | 0 | } |
316 | 0 | return strValue; |
317 | 0 | } |
318 | | |
319 | | bool CDBWrapper::ExistsImpl(std::span<const std::byte> key) const |
320 | 0 | { |
321 | 0 | leveldb::Slice slKey(CharCast(key.data()), key.size()); |
322 | |
|
323 | 0 | std::string strValue; |
324 | 0 | leveldb::Status status = DBContext().pdb->Get(DBContext().readoptions, slKey, &strValue); |
325 | 0 | if (!status.ok()) { |
326 | 0 | if (status.IsNotFound()) |
327 | 0 | return false; |
328 | 0 | LogError("LevelDB read failure: %s", status.ToString());Line | Count | Source | 97 | 0 | #define LogError(...) LogPrintLevel_(BCLog::LogFlags::ALL, BCLog::Level::Error, /*should_ratelimit=*/true, __VA_ARGS__) Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
|
|
329 | 0 | HandleError(status); |
330 | 0 | } |
331 | 0 | return true; |
332 | 0 | } |
333 | | |
334 | | size_t CDBWrapper::EstimateSizeImpl(std::span<const std::byte> key1, std::span<const std::byte> key2) const |
335 | 0 | { |
336 | 0 | leveldb::Slice slKey1(CharCast(key1.data()), key1.size()); |
337 | 0 | leveldb::Slice slKey2(CharCast(key2.data()), key2.size()); |
338 | 0 | uint64_t size = 0; |
339 | 0 | leveldb::Range range(slKey1, slKey2); |
340 | 0 | DBContext().pdb->GetApproximateSizes(&range, 1, &size); |
341 | 0 | return size; |
342 | 0 | } |
343 | | |
344 | | bool CDBWrapper::IsEmpty() |
345 | 0 | { |
346 | 0 | std::unique_ptr<CDBIterator> it(NewIterator()); |
347 | 0 | it->SeekToFirst(); |
348 | 0 | return !(it->Valid()); |
349 | 0 | } |
350 | | |
351 | | struct CDBIterator::IteratorImpl { |
352 | | const std::unique_ptr<leveldb::Iterator> iter; |
353 | | |
354 | 0 | explicit IteratorImpl(leveldb::Iterator* _iter) : iter{_iter} {} |
355 | | }; |
356 | | |
357 | 0 | CDBIterator::CDBIterator(const CDBWrapper& _parent, std::unique_ptr<IteratorImpl> _piter) : parent(_parent), |
358 | 0 | m_impl_iter(std::move(_piter)) {} |
359 | | |
360 | | CDBIterator* CDBWrapper::NewIterator() |
361 | 0 | { |
362 | 0 | return new CDBIterator{*this, std::make_unique<CDBIterator::IteratorImpl>(DBContext().pdb->NewIterator(DBContext().iteroptions))}; |
363 | 0 | } |
364 | | |
365 | | void CDBIterator::SeekImpl(std::span<const std::byte> key) |
366 | 0 | { |
367 | 0 | leveldb::Slice slKey(CharCast(key.data()), key.size()); |
368 | 0 | m_impl_iter->iter->Seek(slKey); |
369 | 0 | } |
370 | | |
371 | | std::span<const std::byte> CDBIterator::GetKeyImpl() const |
372 | 0 | { |
373 | 0 | return MakeByteSpan(m_impl_iter->iter->key()); |
374 | 0 | } |
375 | | |
376 | | std::span<const std::byte> CDBIterator::GetValueImpl() const |
377 | 0 | { |
378 | 0 | return MakeByteSpan(m_impl_iter->iter->value()); |
379 | 0 | } |
380 | | |
381 | 0 | CDBIterator::~CDBIterator() = default; |
382 | 0 | bool CDBIterator::Valid() const { return m_impl_iter->iter->Valid(); } |
383 | 0 | void CDBIterator::SeekToFirst() { m_impl_iter->iter->SeekToFirst(); } |
384 | 0 | void CDBIterator::Next() { m_impl_iter->iter->Next(); } |
385 | | |
386 | | namespace dbwrapper_private { |
387 | | |
388 | | const Obfuscation& GetObfuscation(const CDBWrapper& w) |
389 | 0 | { |
390 | 0 | return w.m_obfuscation; |
391 | 0 | } |
392 | | |
393 | | } // namespace dbwrapper_private |