Branch data Line data Source code
1 : : // Copyright (c) 2020-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 : : #ifndef BITCOIN_WALLET_SQLITE_H
6 : : #define BITCOIN_WALLET_SQLITE_H
7 : :
8 : : #include <sync.h>
9 : : #include <wallet/db.h>
10 : :
11 : : struct bilingual_str;
12 : :
13 : : struct sqlite3_stmt;
14 : : struct sqlite3;
15 : :
16 : : namespace wallet {
17 : : class SQLiteDatabase;
18 : :
19 : : /** RAII class that provides a database cursor */
20 : : class SQLiteCursor : public DatabaseCursor
21 : : {
22 : : public:
23 : : sqlite3_stmt* m_cursor_stmt{nullptr};
24 : : // Copies of the prefix things for the prefix cursor.
25 : : // Prevents SQLite from accessing temp variables for the prefix things.
26 : : std::vector<std::byte> m_prefix_range_start;
27 : : std::vector<std::byte> m_prefix_range_end;
28 : :
29 : 5 : explicit SQLiteCursor() = default;
30 : 25242 : explicit SQLiteCursor(std::vector<std::byte> start_range, std::vector<std::byte> end_range)
31 : 25242 : : m_prefix_range_start(std::move(start_range)),
32 : 25242 : m_prefix_range_end(std::move(end_range))
33 : 25242 : {}
34 : : ~SQLiteCursor() override;
35 : :
36 : : Status Next(DataStream& key, DataStream& value) override;
37 : : };
38 : :
39 : : /** Class responsible for executing SQL statements in SQLite databases.
40 : : * Methods are virtual so they can be overridden by unit tests testing unusual database conditions. */
41 [ + - ]: 132276 : class SQliteExecHandler
42 : : {
43 : : public:
44 : 132275 : virtual ~SQliteExecHandler() = default;
45 : : virtual int Exec(SQLiteDatabase& database, const std::string& statement);
46 : : };
47 : :
48 : : /** RAII class that provides access to a WalletDatabase */
49 : : class SQLiteBatch : public DatabaseBatch
50 : : {
51 : : private:
52 : : SQLiteDatabase& m_database;
53 : : std::unique_ptr<SQliteExecHandler> m_exec_handler{std::make_unique<SQliteExecHandler>()};
54 : :
55 : : sqlite3_stmt* m_read_stmt{nullptr};
56 : : sqlite3_stmt* m_insert_stmt{nullptr};
57 : : sqlite3_stmt* m_overwrite_stmt{nullptr};
58 : : sqlite3_stmt* m_delete_stmt{nullptr};
59 : : sqlite3_stmt* m_delete_prefix_stmt{nullptr};
60 : :
61 : : /** Whether this batch has started a database transaction and whether it owns SQLiteDatabase::m_write_semaphore.
62 : : * If the batch starts a db tx, it acquires the semaphore and sets this to true, keeping the semaphore
63 : : * until the transaction ends to prevent other batch objects from writing to the database.
64 : : *
65 : : * If this batch did not start a transaction, the semaphore is acquired transiently when writing and m_txn
66 : : * is not set.
67 : : *
68 : : * m_txn is different from HasActiveTxn() as it is only true when this batch has started the transaction,
69 : : * not just when any batch has started a transaction.
70 : : */
71 : : bool m_txn{false};
72 : :
73 : : void SetupSQLStatements();
74 : : bool ExecStatement(sqlite3_stmt* stmt, std::span<const std::byte> blob);
75 : :
76 : : bool ReadKey(DataStream&& key, DataStream& value) override;
77 : : bool WriteKey(DataStream&& key, DataStream&& value, bool overwrite = true) override;
78 : : bool EraseKey(DataStream&& key) override;
79 : : bool HasKey(DataStream&& key) override;
80 : : bool ErasePrefix(std::span<const std::byte> prefix) override;
81 : :
82 : : public:
83 : : explicit SQLiteBatch(SQLiteDatabase& database);
84 : 264550 : ~SQLiteBatch() override { Close(); }
85 : :
86 : 1 : void SetExecHandler(std::unique_ptr<SQliteExecHandler>&& handler) { m_exec_handler = std::move(handler); }
87 : :
88 : : void Close() override;
89 : :
90 : : std::unique_ptr<DatabaseCursor> GetNewCursor() override;
91 : : std::unique_ptr<DatabaseCursor> GetNewPrefixCursor(std::span<const std::byte> prefix) override;
92 : : bool TxnBegin() override;
93 : : bool TxnCommit() override;
94 : : bool TxnAbort() override;
95 : 19 : bool HasActiveTxn() override { return m_txn; }
96 : : };
97 : :
98 : : /** An instance of this class represents one SQLite3 database.
99 : : **/
100 : : class SQLiteDatabase : public WalletDatabase
101 : : {
102 : : private:
103 : : const bool m_mock{false};
104 : :
105 : : const std::string m_dir_path;
106 : :
107 : : const std::string m_file_path;
108 : :
109 : : /**
110 : : * This mutex protects SQLite initialization and shutdown.
111 : : * sqlite3_config() and sqlite3_shutdown() are not thread-safe (sqlite3_initialize() is).
112 : : * Concurrent threads that execute SQLiteDatabase::SQLiteDatabase() should have just one
113 : : * of them do the init and the rest wait for it to complete before all can proceed.
114 : : */
115 : : static Mutex g_sqlite_mutex;
116 : : static int g_sqlite_count GUARDED_BY(g_sqlite_mutex);
117 : :
118 : : void Cleanup() noexcept EXCLUSIVE_LOCKS_REQUIRED(!g_sqlite_mutex);
119 : :
120 : : public:
121 : : SQLiteDatabase() = delete;
122 : :
123 : : /** Create DB handle to real database */
124 : : SQLiteDatabase(const fs::path& dir_path, const fs::path& file_path, const DatabaseOptions& options, bool mock = false);
125 : :
126 : : ~SQLiteDatabase();
127 : :
128 : : // Batches must acquire this semaphore on writing, and release when done writing.
129 : : // This ensures that only one batch is modifying the database at a time.
130 : : CSemaphore m_write_semaphore;
131 : :
132 : : bool Verify(bilingual_str& error);
133 : :
134 : : /** Open the database if it is not already opened */
135 : : void Open() override;
136 : :
137 : : /** Close the database */
138 : : void Close() override;
139 : :
140 : : /** Rewrite the entire database on disk */
141 : : bool Rewrite(const char* skip = nullptr) override;
142 : :
143 : : /** Back up the entire database to a file.
144 : : */
145 : : bool Backup(const std::string& dest) const override;
146 : :
147 : 369219 : std::string Filename() override { return m_file_path; }
148 : 827 : std::string Format() override { return "sqlite"; }
149 : :
150 : : /** Make a SQLiteBatch connected to this database */
151 : : std::unique_ptr<DatabaseBatch> MakeBatch() override;
152 : :
153 : : /** Return true if there is an on-going txn in this connection */
154 : : bool HasActiveTxn();
155 : :
156 : : sqlite3* m_db{nullptr};
157 : : bool m_use_unsafe_sync;
158 : : };
159 : :
160 : : std::unique_ptr<SQLiteDatabase> MakeSQLiteDatabase(const fs::path& path, const DatabaseOptions& options, DatabaseStatus& status, bilingual_str& error);
161 : :
162 : : std::string SQLiteDatabaseVersion();
163 : : } // namespace wallet
164 : :
165 : : #endif // BITCOIN_WALLET_SQLITE_H
|