| Directory: | cvmfs/ |
|---|---|
| File: | cvmfs/sql.h |
| Date: | 2025-11-09 02:35:23 |
| Exec | Total | Coverage | |
|---|---|---|---|
| Lines: | 104 | 109 | 95.4% |
| Branches: | 18 | 24 | 75.0% |
| Line | Branch | Exec | Source |
|---|---|---|---|
| 1 | /** | ||
| 2 | * This file is part of the CernVM file system. | ||
| 3 | */ | ||
| 4 | |||
| 5 | #ifndef CVMFS_SQL_H_ | ||
| 6 | #define CVMFS_SQL_H_ | ||
| 7 | |||
| 8 | #include <cassert> | ||
| 9 | #include <string> | ||
| 10 | |||
| 11 | #include "duplex_sqlite3.h" | ||
| 12 | #include "util/file_guard.h" | ||
| 13 | #include "util/pointer.h" | ||
| 14 | |||
| 15 | namespace sqlite { | ||
| 16 | |||
| 17 | struct MemStatistics { | ||
| 18 | 7 | MemStatistics() | |
| 19 | 7 | : lookaside_slots_used(-1) | |
| 20 | 7 | , lookaside_slots_max(-1) | |
| 21 | 7 | , lookaside_hit(-1) | |
| 22 | 7 | , lookaside_miss_size(-1) | |
| 23 | 7 | , lookaside_miss_full(-1) | |
| 24 | 7 | , page_cache_used(-1) | |
| 25 | 7 | , page_cache_hit(-1) | |
| 26 | 7 | , page_cache_miss(-1) | |
| 27 | 7 | , schema_used(-1) | |
| 28 | 7 | , stmt_used(-1) { } | |
| 29 | int lookaside_slots_used; | ||
| 30 | int lookaside_slots_max; | ||
| 31 | int lookaside_hit; | ||
| 32 | int lookaside_miss_size; | ||
| 33 | int lookaside_miss_full; | ||
| 34 | int page_cache_used; ///< Bytes used for caching pages | ||
| 35 | int page_cache_hit; | ||
| 36 | int page_cache_miss; | ||
| 37 | int schema_used; ///< Bytes used to store db schema | ||
| 38 | int stmt_used; ///< Bytes used for prepared statmements (lookaside + heap) | ||
| 39 | }; | ||
| 40 | |||
| 41 | class Sql; | ||
| 42 | |||
| 43 | /** | ||
| 44 | * Encapsulates an SQlite connection. | ||
| 45 | * | ||
| 46 | * This is an abstract base class for different SQLite database flavours used | ||
| 47 | * throughout CernVM-FS. It provides a general interface for creating, opening, | ||
| 48 | * compacting and migrating an SQLite database. | ||
| 49 | * Furthermore it manages a `properties` table in each database, to store simple | ||
| 50 | * key-value style information in a common fashion. For that, it offers the | ||
| 51 | * templated methods SetProperty(), GetProperty<>() and HasProperty() that take | ||
| 52 | * all common data types and persist it in the database. | ||
| 53 | * | ||
| 54 | * By default Database<> objects do not take ownership of the underlying SQLite | ||
| 55 | * database file and hence do not unlink it on database closure. If the using | ||
| 56 | * code calls Database<>::TakeFileOwnership() the SQLite file will be unlinked | ||
| 57 | * in the destructor of the Database<> object. | ||
| 58 | * | ||
| 59 | * Note: This implements a Curiously Recurring Template Pattern in order to | ||
| 60 | * implement Database::Create and Database::Open as a static polymorphism | ||
| 61 | * | ||
| 62 | * The following methods need to be implemented by each subclass: | ||
| 63 | * (Database<> assumes 'true' as a return value on success) | ||
| 64 | * | ||
| 65 | * -> bool CreateEmptyDatabase() | ||
| 66 | * creates all necessary SQLite tables for the concrete Database | ||
| 67 | * implementation. Furthermore it can insert default data into the | ||
| 68 | * newly created database tables. | ||
| 69 | * -> bool CheckSchemaCompatibility() | ||
| 70 | * checks a database for compatibility directly after opening it. | ||
| 71 | * Database<> provides schema_version() and schema_revision() to | ||
| 72 | * access the compatibility stored in the `properties` table | ||
| 73 | * -> bool LiveSchemaUpgradeIfNecessary() | ||
| 74 | * this allows for on-the-fly schema updates and is always called | ||
| 75 | * when a database is opened read/write. It assumes 'true' both on | ||
| 76 | * successful migration and if no migration was necessary | ||
| 77 | * -> bool CompactDatabase() | ||
| 78 | * here implementation specific cleanup actions can take place on | ||
| 79 | * databases opened as read/write. It is invoked by the `Vacuum()` | ||
| 80 | * method, that can be used by higher level user code | ||
| 81 | * | ||
| 82 | * Furthermore Database<> expects two static constants to be defined: | ||
| 83 | * | ||
| 84 | * -> kLatestSchema - the newest schema version generated by invoking | ||
| 85 | * DerivedT::CreateEmptyDatabase() | ||
| 86 | * -> kLatestSchemaRevision - same as kLatestSchema, however different schema | ||
| 87 | * revisions are supposed to be backward compatible | ||
| 88 | * or on-the-fly updateable by | ||
| 89 | * DerivedT::LiveSchemaUpgradeIfNecessary() | ||
| 90 | * | ||
| 91 | * @param DerivedT the name of the inheriting Database implementation class | ||
| 92 | * (Curiously Recurring Template Pattern) | ||
| 93 | * | ||
| 94 | * TODO(rmeusel): C++11 Move Constructors to allow for stack allocated databases | ||
| 95 | */ | ||
| 96 | template<class DerivedT> | ||
| 97 | class Database : SingleCopy { | ||
| 98 | public: | ||
| 99 | enum OpenMode { | ||
| 100 | kOpenReadOnly, | ||
| 101 | kOpenReadWrite, | ||
| 102 | }; | ||
| 103 | |||
| 104 | static const float kSchemaEpsilon; // floats get imprecise in SQlite | ||
| 105 | |||
| 106 | /** | ||
| 107 | * Creates a new database file of the type implemented by DerivedT. During the | ||
| 108 | * invocation of this static method DerivedT::CreateEmptyDatabase() is called. | ||
| 109 | * | ||
| 110 | * @param filename the file location of the newly created database | ||
| 111 | * (file does not need to exist) | ||
| 112 | * @return an empty database of type DerivedT (or NULL on failure) | ||
| 113 | */ | ||
| 114 | static DerivedT *Create(const std::string &filename); | ||
| 115 | |||
| 116 | /** | ||
| 117 | * Opens a database file and assumes it to be of type DerivedT. This method | ||
| 118 | * will call DerivedT::CheckSchemaCompatibility() to figure out readability of | ||
| 119 | * the contained schema revision. Furthermore, if the database was opened in | ||
| 120 | * read/write mode, it calls DerivedT::LiveSchemaUpgradeIfNecessary() to allow | ||
| 121 | * for on-the-fly schema upgrades of the underlying database file. | ||
| 122 | * | ||
| 123 | * @param filename path to the SQLite file to be opened as DerivedT | ||
| 124 | * @param open_mode kOpenReadOnly or kOpenReadWrite open modes | ||
| 125 | * @return a database of type DerivedT (or NULL on failure) | ||
| 126 | */ | ||
| 127 | static DerivedT *Open(const std::string &filename, const OpenMode open_mode); | ||
| 128 | |||
| 129 | 76280 | bool IsEqualSchema(const float value, const float compare) const { | |
| 130 | 76280 | return (value > compare - kSchemaEpsilon | |
| 131 |
4/4✓ Branch 0 taken 75623 times.
✓ Branch 1 taken 657 times.
✓ Branch 2 taken 71501 times.
✓ Branch 3 taken 4122 times.
|
76280 | && value < compare + kSchemaEpsilon); |
| 132 | } | ||
| 133 | |||
| 134 | bool BeginTransaction() const; | ||
| 135 | bool CommitTransaction() const; | ||
| 136 | |||
| 137 | template<typename T> | ||
| 138 | T GetProperty(const std::string &key) const; | ||
| 139 | template<typename T> | ||
| 140 | T GetPropertyDefault(const std::string &key, const T default_value) const; | ||
| 141 | template<typename T> | ||
| 142 | bool SetProperty(const std::string &key, const T value); | ||
| 143 | bool HasProperty(const std::string &key) const; | ||
| 144 | |||
| 145 | 233134 | sqlite3 *sqlite_db() const { return database_.database(); } | |
| 146 | 53893 | const std::string &filename() const { return database_.filename(); } | |
| 147 | 146983 | float schema_version() const { return schema_version_; } | |
| 148 | 142210 | unsigned schema_revision() const { return schema_revision_; } | |
| 149 | 32993 | bool read_write() const { return read_write_; } | |
| 150 | |||
| 151 | /** | ||
| 152 | * Provides the number of rows modified by INSERT, UPDATE or DELETE statements | ||
| 153 | * that have been run against this database since it was opened. | ||
| 154 | * @return number of rows modified by all executed manipulating statements | ||
| 155 | */ | ||
| 156 | unsigned GetModifiedRowCount() const; | ||
| 157 | |||
| 158 | /** | ||
| 159 | * Figures out the ratio of free SQLite memory pages in the SQLite database | ||
| 160 | * file. A high ratio can be an indication of a necessary call to Vacuum(). | ||
| 161 | * Note: This is not done automatically and the decision is left to the using | ||
| 162 | * code! | ||
| 163 | * @return the free-page ratio in the opened database file (free pages/pages) | ||
| 164 | */ | ||
| 165 | double GetFreePageRatio() const; | ||
| 166 | |||
| 167 | /** | ||
| 168 | * Retrieves the per-connection memory statistics from SQlite | ||
| 169 | */ | ||
| 170 | void GetMemStatistics(MemStatistics *stats) const; | ||
| 171 | |||
| 172 | /** | ||
| 173 | * Performs a VACUUM call on the opened database file to compacts the | ||
| 174 | * database. As a first step it runs DerivedT::CompactDatabase() to allow for | ||
| 175 | * implement- ation dependent cleanup actions. Vacuum() assumes that the | ||
| 176 | * SQLite database was opened in read/write mode. | ||
| 177 | * @return true on success | ||
| 178 | */ | ||
| 179 | bool Vacuum() const; | ||
| 180 | |||
| 181 | /** | ||
| 182 | * Prints the given error message, together with the last encountered SQLite | ||
| 183 | * error of this database. | ||
| 184 | * @param error_msg an error message to be printed along with the SQL error | ||
| 185 | */ | ||
| 186 | void PrintSqlError(const std::string &error_msg); | ||
| 187 | |||
| 188 | /** | ||
| 189 | * Returns the english language error description of the last error | ||
| 190 | * happened in the context of the encapsulated sqlite3 database object. | ||
| 191 | * Note: In a multithreaded context it might be unpredictable which | ||
| 192 | * the actual last error is. | ||
| 193 | * @return english language error description of last error | ||
| 194 | */ | ||
| 195 | std::string GetLastErrorMsg() const; | ||
| 196 | |||
| 197 | |||
| 198 | /** | ||
| 199 | * Transfers the ownership of the SQLite database file to the Database<> ob- | ||
| 200 | * ject. Hence, it will automatically unlink the file, once the Database<> | ||
| 201 | * object goes out of scope or is deleted. | ||
| 202 | */ | ||
| 203 | void TakeFileOwnership(); | ||
| 204 | |||
| 205 | /** | ||
| 206 | * Resigns from the ownership of the SQLite database file underlying this | ||
| 207 | * Database<> object. After calling this the using code is responsible of | ||
| 208 | * managing the database file. | ||
| 209 | */ | ||
| 210 | void DropFileOwnership(); | ||
| 211 | |||
| 212 | /** | ||
| 213 | * Check if the SQLite database file is managed by the Database<> object | ||
| 214 | * Note: unmanaged means, that the using code needs to take care of the file | ||
| 215 | * management (i.e. delete the file after usage) | ||
| 216 | * | ||
| 217 | * @return false if file is unmanaged | ||
| 218 | */ | ||
| 219 | 1161 | bool OwnsFile() const { return database_.OwnsFile(); } | |
| 220 | |||
| 221 | /** | ||
| 222 | * Used when attaching legacy catalogs to set 0.9 schema where mistakenly 1.0 | ||
| 223 | * was used. | ||
| 224 | */ | ||
| 225 | 9 | void EnforceSchema(float version, unsigned revision) { | |
| 226 | 9 | schema_version_ = version; | |
| 227 | 9 | schema_revision_ = revision; | |
| 228 | 9 | } | |
| 229 | |||
| 230 | protected: | ||
| 231 | /** | ||
| 232 | * Private constructor! Use the factory methods DerivedT::Create() or | ||
| 233 | * DerivedT::Open() to instantiate a database object of type DerivedT. | ||
| 234 | */ | ||
| 235 | Database(const std::string &filename, const OpenMode open_mode); | ||
| 236 | |||
| 237 | bool Initialize(); | ||
| 238 | |||
| 239 | bool CreatePropertiesTable(); | ||
| 240 | bool PrepareCommonQueries(); | ||
| 241 | |||
| 242 | bool OpenDatabase(const int sqlite_open_flags); | ||
| 243 | bool Configure(); | ||
| 244 | bool FileReadAhead(); | ||
| 245 | |||
| 246 | void ReadSchemaRevision(); | ||
| 247 | bool StoreSchemaRevision(); | ||
| 248 | |||
| 249 | 7888 | void set_schema_version(const float ver) { schema_version_ = ver; } | |
| 250 | 8635 | void set_schema_revision(const unsigned rev) { schema_revision_ = rev; } | |
| 251 | |||
| 252 | private: | ||
| 253 | /** | ||
| 254 | * This wraps the opaque SQLite database object along with a file unlink guard | ||
| 255 | * to control the life time of the database connection and the database file | ||
| 256 | * in an RAII fashion. | ||
| 257 | */ | ||
| 258 | struct DatabaseRaiiWrapper { | ||
| 259 | 14775 | DatabaseRaiiWrapper(const std::string &filename, | |
| 260 | Database<DerivedT> *delegate) | ||
| 261 | 14775 | : sqlite_db(NULL) | |
| 262 | 14775 | , lookaside_buffer(NULL) | |
| 263 | 14775 | , db_file_guard(filename, UnlinkGuard::kDisabled) | |
| 264 | 14775 | , delegate_(delegate) { } | |
| 265 | ~DatabaseRaiiWrapper(); | ||
| 266 | |||
| 267 | 233134 | sqlite3 *database() const { return sqlite_db; } | |
| 268 | 72926 | const std::string &filename() const { return db_file_guard.path(); } | |
| 269 | |||
| 270 | bool Close(); | ||
| 271 | |||
| 272 | 4152 | void TakeFileOwnership() { db_file_guard.Enable(); } | |
| 273 | 156 | void DropFileOwnership() { db_file_guard.Disable(); } | |
| 274 | 1161 | bool OwnsFile() const { return db_file_guard.IsEnabled(); } | |
| 275 | |||
| 276 | sqlite3 *sqlite_db; | ||
| 277 | void *lookaside_buffer; | ||
| 278 | UnlinkGuard db_file_guard; | ||
| 279 | Database<DerivedT> *delegate_; | ||
| 280 | }; | ||
| 281 | |||
| 282 | static const char *kSchemaVersionKey; | ||
| 283 | static const char *kSchemaRevisionKey; | ||
| 284 | |||
| 285 | DatabaseRaiiWrapper database_; | ||
| 286 | |||
| 287 | const bool read_write_; | ||
| 288 | float schema_version_; | ||
| 289 | unsigned schema_revision_; | ||
| 290 | |||
| 291 | UniquePtr<Sql> begin_transaction_; | ||
| 292 | UniquePtr<Sql> commit_transaction_; | ||
| 293 | |||
| 294 | UniquePtr<Sql> has_property_; | ||
| 295 | UniquePtr<Sql> set_property_; | ||
| 296 | UniquePtr<Sql> get_property_; | ||
| 297 | }; | ||
| 298 | |||
| 299 | |||
| 300 | // | ||
| 301 | // # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # # | ||
| 302 | // | ||
| 303 | |||
| 304 | |||
| 305 | /** | ||
| 306 | * Base class for all SQL statement classes. It wraps a single SQL statement | ||
| 307 | * and all necessary calls of the sqlite3 API to deal with this statement. | ||
| 308 | * | ||
| 309 | * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * | ||
| 310 | * NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE * | ||
| 311 | * NOTE This base class implements a lazy initialization of the SQLite * | ||
| 312 | * NOTE prepared statement. Therefore it is strongly discouraged to use * | ||
| 313 | * NOTE any sqlite3_***() functions directly in the subclasses. Instead * | ||
| 314 | * NOTE one must wrap them in this base class and implement the lazy * | ||
| 315 | * NOTE initialization scheme as seen below. * | ||
| 316 | * NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE NOTE * | ||
| 317 | * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * | ||
| 318 | * | ||
| 319 | * Derived classes can decide if their statement should be prepared immediately | ||
| 320 | * or on first use (aka lazily). The derived constructor must call Sql::Init() | ||
| 321 | * or Sql::DeferredInit() accordingly. | ||
| 322 | * | ||
| 323 | * Sql objects created via the public constructor rather than by the constructor | ||
| 324 | * of a derived class are prepared immediately by default. | ||
| 325 | */ | ||
| 326 | class Sql { | ||
| 327 | public: | ||
| 328 | /** | ||
| 329 | * Basic constructor to use this class for a specific statement. | ||
| 330 | * @param database the database to use the query on | ||
| 331 | * @param statement the statement to prepare | ||
| 332 | */ | ||
| 333 | Sql(sqlite3 *sqlite_db, const std::string &statement); | ||
| 334 | virtual ~Sql(); | ||
| 335 | |||
| 336 | bool Execute(); | ||
| 337 | bool FetchRow(); | ||
| 338 | std::string DebugResultTable(); | ||
| 339 | bool Reset(); | ||
| 340 | 276 | inline int GetLastError() const { return last_error_code_; } | |
| 341 | |||
| 342 | /** | ||
| 343 | * Returns the english language error description of the last error | ||
| 344 | * happened in the context of the sqlite3 database object this statement is | ||
| 345 | * registered to. | ||
| 346 | * Note: In a multithreaded context it might be unpredictable which | ||
| 347 | * the actual last error is. | ||
| 348 | * @return english language error description of last error | ||
| 349 | */ | ||
| 350 | std::string GetLastErrorMsg() const; | ||
| 351 | |||
| 352 | 9975 | bool BindBlob(const int index, const void *value, const unsigned size) { | |
| 353 | 9975 | LazyInit(); | |
| 354 | 9975 | last_error_code_ = sqlite3_bind_blob(statement_, index, value, | |
| 355 | static_cast<int>(size), SQLITE_STATIC); | ||
| 356 | 9975 | return Successful(); | |
| 357 | } | ||
| 358 | ✗ | bool BindBlobTransient(const int index, const void *value, | |
| 359 | const unsigned size) { | ||
| 360 | ✗ | LazyInit(); | |
| 361 | ✗ | last_error_code_ = sqlite3_bind_blob(statement_, index, value, | |
| 362 | static_cast<int>(size), | ||
| 363 | SQLITE_TRANSIENT); // NOLINT | ||
| 364 | ✗ | return Successful(); | |
| 365 | } | ||
| 366 | 8693 | bool BindDouble(const int index, const double value) { | |
| 367 | 8693 | LazyInit(); | |
| 368 | 8693 | last_error_code_ = sqlite3_bind_double(statement_, index, value); | |
| 369 | 8693 | return Successful(); | |
| 370 | } | ||
| 371 | 52300 | bool BindInt(const int index, const int value) { | |
| 372 | 52300 | LazyInit(); | |
| 373 | 52300 | last_error_code_ = sqlite3_bind_int(statement_, index, value); | |
| 374 | 52300 | return Successful(); | |
| 375 | } | ||
| 376 | 1237859 | bool BindInt64(const int index, const sqlite3_int64 value) { | |
| 377 | 1237859 | LazyInit(); | |
| 378 | 1237859 | last_error_code_ = sqlite3_bind_int64(statement_, index, value); | |
| 379 | 1237859 | return Successful(); | |
| 380 | } | ||
| 381 | 58471 | bool BindNull(const int index) { | |
| 382 | 58471 | LazyInit(); | |
| 383 | 58471 | last_error_code_ = sqlite3_bind_null(statement_, index); | |
| 384 | 58471 | return Successful(); | |
| 385 | } | ||
| 386 | 7556970 | bool BindTextTransient(const int index, const std::string &value) { | |
| 387 | 7556970 | return BindTextTransient(index, value.data(), | |
| 388 | 15113940 | static_cast<int>(value.length())); | |
| 389 | } | ||
| 390 | 7556970 | bool BindTextTransient(const int index, const char *value, const int size) { | |
| 391 | // NOLINTNEXTLINE(performance-no-int-to-ptr) | ||
| 392 | 7556970 | return BindText(index, value, size, SQLITE_TRANSIENT); | |
| 393 | } | ||
| 394 | 8360085 | bool BindText(const int index, const std::string &value) { | |
| 395 | 8360085 | return BindText(index, value.data(), static_cast<int>(value.length()), | |
| 396 | 8360067 | SQLITE_STATIC); | |
| 397 | } | ||
| 398 | 15971844 | bool BindText(const int index, | |
| 399 | const char *value, | ||
| 400 | const int size, | ||
| 401 | void (*dtor)(void *) = SQLITE_STATIC) { | ||
| 402 | 15971844 | LazyInit(); | |
| 403 | 15971844 | last_error_code_ = sqlite3_bind_text(statement_, index, value, size, dtor); | |
| 404 | 15971826 | return Successful(); | |
| 405 | } | ||
| 406 | |||
| 407 | /** | ||
| 408 | * Figures out the type to be bound by template parameter deduction | ||
| 409 | * NOTE: For strings this is suboptimal, since it needs to assume that the | ||
| 410 | * provided buffer is transient and copy it to be sure. | ||
| 411 | */ | ||
| 412 | template<typename T> | ||
| 413 | inline bool Bind(const int index, const T &value); | ||
| 414 | |||
| 415 | |||
| 416 | 1503 | int RetrieveType(const int idx_column) const { | |
| 417 | 1503 | return sqlite3_column_type(statement_, idx_column); | |
| 418 | } | ||
| 419 | |||
| 420 | /** | ||
| 421 | * Determines the number of bytes necessary to store the column's data as a | ||
| 422 | * string. This might involve type conversions and depends on which other | ||
| 423 | * RetrieveXXX methods were called on the same column index before! | ||
| 424 | * | ||
| 425 | * See SQLite documentation for sqlite_column_bytes() for details: | ||
| 426 | * https://www.sqlite.org/c3ref/column_blob.html | ||
| 427 | */ | ||
| 428 | 30067 | int RetrieveBytes(const int idx_column) const { | |
| 429 | 30067 | return sqlite3_column_bytes(statement_, idx_column); | |
| 430 | } | ||
| 431 | 30074 | const void *RetrieveBlob(const int idx_column) const { | |
| 432 | 30074 | return sqlite3_column_blob(statement_, idx_column); | |
| 433 | } | ||
| 434 | 9077 | double RetrieveDouble(const int idx_column) const { | |
| 435 | 9077 | return sqlite3_column_double(statement_, idx_column); | |
| 436 | } | ||
| 437 | 29860 | int RetrieveNullableInt(const int idx_column, const int val_null) const { | |
| 438 |
1/2✓ Branch 1 taken 29860 times.
✗ Branch 2 not taken.
|
29860 | if (sqlite3_column_type(statement_, idx_column) == SQLITE_NULL) |
| 439 | 29860 | return val_null; | |
| 440 | ✗ | return sqlite3_column_int(statement_, idx_column); | |
| 441 | } | ||
| 442 | 90637 | int RetrieveInt(const int idx_column) const { | |
| 443 | 90637 | return sqlite3_column_int(statement_, idx_column); | |
| 444 | } | ||
| 445 | 583496 | sqlite3_int64 RetrieveInt64(const int idx_column) const { | |
| 446 | 583496 | return sqlite3_column_int64(statement_, idx_column); | |
| 447 | } | ||
| 448 | 392373 | const unsigned char *RetrieveText(const int idx_column) const { | |
| 449 | 392373 | return sqlite3_column_text(statement_, idx_column); | |
| 450 | } | ||
| 451 | 325999 | std::string RetrieveString(const int idx_column) const { | |
| 452 |
2/4✓ Branch 2 taken 325999 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 325999 times.
✗ Branch 6 not taken.
|
325999 | return reinterpret_cast<const char *>(RetrieveText(idx_column)); |
| 453 | } | ||
| 454 | template<typename T> | ||
| 455 | inline T Retrieve(const int index); | ||
| 456 | |||
| 457 | protected: | ||
| 458 | 167098 | Sql() | |
| 459 | 167098 | : database_(NULL) | |
| 460 | 167098 | , statement_(NULL) | |
| 461 | 167098 | , query_string_(NULL) | |
| 462 | 167098 | , last_error_code_(0) { } | |
| 463 | |||
| 464 | 25619462 | bool IsInitialized() const { return statement_ != NULL; } | |
| 465 | |||
| 466 | /** | ||
| 467 | * Initializes the prepared statement immediately. | ||
| 468 | * | ||
| 469 | * @param database the sqlite database pointer to be query against | ||
| 470 | * @param statement the query string to be prepared for execution | ||
| 471 | * @return true on successful statement preparation | ||
| 472 | */ | ||
| 473 | bool Init(const sqlite3 *database, const std::string &statement); | ||
| 474 | |||
| 475 | /** | ||
| 476 | * Defers the initialization of the prepared statement to the first usage to | ||
| 477 | * safe memory and CPU cycles for statements that are defined but never used. | ||
| 478 | * Typically this method is used in constructors of derived classes. | ||
| 479 | * | ||
| 480 | * @param database the sqlite database pointer to be query against | ||
| 481 | * @param statement the query string to be prepared for execution | ||
| 482 | */ | ||
| 483 | void DeferredInit(const sqlite3 *database, const char *statement); | ||
| 484 | |||
| 485 | /** | ||
| 486 | * Checks the last action for success | ||
| 487 | * @return true if last action succeeded otherwise false | ||
| 488 | */ | ||
| 489 | 41528515 | inline bool Successful() const { | |
| 490 |
2/2✓ Branch 0 taken 15670489 times.
✓ Branch 1 taken 5602 times.
|
15676091 | return SQLITE_OK == last_error_code_ || SQLITE_ROW == last_error_code_ |
| 491 |
4/4✓ Branch 0 taken 15676091 times.
✓ Branch 1 taken 25852424 times.
✓ Branch 2 taken 15669606 times.
✓ Branch 3 taken 883 times.
|
57204606 | || SQLITE_DONE == last_error_code_; |
| 492 | } | ||
| 493 | |||
| 494 | private: | ||
| 495 | bool Init(const char *statement); | ||
| 496 | 25619462 | void LazyInit() { | |
| 497 |
2/2✓ Branch 1 taken 36489 times.
✓ Branch 2 taken 25582973 times.
|
25619462 | if (!IsInitialized()) { |
| 498 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 36489 times.
|
36489 | assert(NULL != database_); |
| 499 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 36489 times.
|
36489 | assert(NULL != query_string_); |
| 500 | 36489 | const bool success = Init(query_string_); | |
| 501 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 36489 times.
|
36489 | assert(success); |
| 502 | } | ||
| 503 | 25619462 | } | |
| 504 | |||
| 505 | sqlite3 *database_; | ||
| 506 | sqlite3_stmt *statement_; | ||
| 507 | const char *query_string_; | ||
| 508 | int last_error_code_; | ||
| 509 | }; | ||
| 510 | |||
| 511 | } // namespace sqlite | ||
| 512 | |||
| 513 | #include "sql_impl.h" | ||
| 514 | |||
| 515 | #endif // CVMFS_SQL_H_ | ||
| 516 |