2018-12-31 22:04:05 +03:00
|
|
|
//
|
2020-01-01 04:23:48 +03:00
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2020
|
2018-12-31 22:04:05 +03:00
|
|
|
//
|
|
|
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
|
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
|
|
//
|
|
|
|
#include "td/db/SqliteStatement.h"
|
|
|
|
|
|
|
|
#include "td/utils/format.h"
|
|
|
|
#include "td/utils/logging.h"
|
|
|
|
#include "td/utils/StackAllocator.h"
|
|
|
|
#include "td/utils/StringBuilder.h"
|
|
|
|
|
|
|
|
#include "sqlite/sqlite3.h"
|
|
|
|
|
|
|
|
namespace td {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
int printExplainQueryPlan(StringBuilder &sb, sqlite3_stmt *pStmt) {
|
|
|
|
const char *zSql = sqlite3_sql(pStmt);
|
|
|
|
if (zSql == nullptr) {
|
|
|
|
return SQLITE_ERROR;
|
|
|
|
}
|
|
|
|
|
2018-12-14 01:48:36 +03:00
|
|
|
sb << "Explain query " << zSql;
|
2018-12-31 22:04:05 +03:00
|
|
|
char *zExplain = sqlite3_mprintf("EXPLAIN QUERY PLAN %s", zSql);
|
|
|
|
if (zExplain == nullptr) {
|
|
|
|
return SQLITE_NOMEM;
|
|
|
|
}
|
|
|
|
|
|
|
|
sqlite3_stmt *pExplain; /* Compiled EXPLAIN QUERY PLAN command */
|
|
|
|
int rc = sqlite3_prepare_v2(sqlite3_db_handle(pStmt), zExplain, -1, &pExplain, nullptr);
|
|
|
|
sqlite3_free(zExplain);
|
|
|
|
if (rc != SQLITE_OK) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (SQLITE_ROW == sqlite3_step(pExplain)) {
|
|
|
|
int iSelectid = sqlite3_column_int(pExplain, 0);
|
|
|
|
int iOrder = sqlite3_column_int(pExplain, 1);
|
|
|
|
int iFrom = sqlite3_column_int(pExplain, 2);
|
|
|
|
const char *zDetail = reinterpret_cast<const char *>(sqlite3_column_text(pExplain, 3));
|
|
|
|
|
2018-12-14 01:48:36 +03:00
|
|
|
sb << '\n' << iSelectid << ' ' << iOrder << ' ' << iFrom << ' ' << zDetail;
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return sqlite3_finalize(pExplain);
|
|
|
|
}
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
SqliteStatement::SqliteStatement(sqlite3_stmt *stmt, std::shared_ptr<detail::RawSqliteDb> db)
|
|
|
|
: stmt_(stmt), db_(std::move(db)) {
|
|
|
|
CHECK(stmt != nullptr);
|
|
|
|
}
|
|
|
|
SqliteStatement::~SqliteStatement() = default;
|
|
|
|
|
|
|
|
Result<string> SqliteStatement::explain() {
|
|
|
|
if (empty()) {
|
|
|
|
return Status::Error("No statement");
|
|
|
|
}
|
|
|
|
auto tmp = StackAllocator::alloc(10000);
|
|
|
|
StringBuilder sb(tmp.as_slice());
|
|
|
|
auto code = printExplainQueryPlan(sb, stmt_.get());
|
|
|
|
if (code != SQLITE_OK) {
|
|
|
|
return last_error();
|
|
|
|
}
|
|
|
|
if (sb.is_error()) {
|
|
|
|
return Status::Error("StringBuilder buffer overflow");
|
|
|
|
}
|
|
|
|
return sb.as_cslice().str();
|
|
|
|
}
|
|
|
|
Status SqliteStatement::bind_blob(int id, Slice blob) {
|
|
|
|
auto rc = sqlite3_bind_blob(stmt_.get(), id, blob.data(), static_cast<int>(blob.size()), nullptr);
|
|
|
|
if (rc != SQLITE_OK) {
|
|
|
|
return last_error();
|
|
|
|
}
|
|
|
|
return Status::OK();
|
|
|
|
}
|
|
|
|
Status SqliteStatement::bind_string(int id, Slice str) {
|
|
|
|
auto rc = sqlite3_bind_text(stmt_.get(), id, str.data(), static_cast<int>(str.size()), nullptr);
|
|
|
|
if (rc != SQLITE_OK) {
|
|
|
|
return last_error();
|
|
|
|
}
|
|
|
|
return Status::OK();
|
|
|
|
}
|
|
|
|
|
|
|
|
Status SqliteStatement::bind_int32(int id, int32 value) {
|
|
|
|
auto rc = sqlite3_bind_int(stmt_.get(), id, value);
|
|
|
|
if (rc != SQLITE_OK) {
|
|
|
|
return last_error();
|
|
|
|
}
|
|
|
|
return Status::OK();
|
|
|
|
}
|
|
|
|
Status SqliteStatement::bind_int64(int id, int64 value) {
|
|
|
|
auto rc = sqlite3_bind_int64(stmt_.get(), id, value);
|
|
|
|
if (rc != SQLITE_OK) {
|
|
|
|
return last_error();
|
|
|
|
}
|
|
|
|
return Status::OK();
|
|
|
|
}
|
|
|
|
Status SqliteStatement::bind_null(int id) {
|
|
|
|
auto rc = sqlite3_bind_null(stmt_.get(), id);
|
|
|
|
if (rc != SQLITE_OK) {
|
|
|
|
return last_error();
|
|
|
|
}
|
|
|
|
return Status::OK();
|
|
|
|
}
|
|
|
|
|
|
|
|
StringBuilder &operator<<(StringBuilder &sb, SqliteStatement::Datatype type) {
|
|
|
|
using Datatype = SqliteStatement::Datatype;
|
|
|
|
switch (type) {
|
|
|
|
case Datatype::Integer:
|
|
|
|
return sb << "Integer";
|
|
|
|
case Datatype::Float:
|
|
|
|
return sb << "Float";
|
|
|
|
case Datatype::Blob:
|
|
|
|
return sb << "Blob";
|
|
|
|
case Datatype::Null:
|
|
|
|
return sb << "Null";
|
|
|
|
case Datatype::Text:
|
|
|
|
return sb << "Text";
|
|
|
|
}
|
|
|
|
UNREACHABLE();
|
|
|
|
return sb;
|
|
|
|
}
|
|
|
|
Slice SqliteStatement::view_blob(int id) {
|
|
|
|
LOG_IF(ERROR, view_datatype(id) != Datatype::Blob) << view_datatype(id);
|
|
|
|
auto *data = sqlite3_column_blob(stmt_.get(), id);
|
|
|
|
auto size = sqlite3_column_bytes(stmt_.get(), id);
|
|
|
|
if (data == nullptr) {
|
|
|
|
return Slice();
|
|
|
|
}
|
|
|
|
return Slice(static_cast<const char *>(data), size);
|
|
|
|
}
|
|
|
|
Slice SqliteStatement::view_string(int id) {
|
|
|
|
LOG_IF(ERROR, view_datatype(id) != Datatype::Text) << view_datatype(id);
|
|
|
|
auto *data = sqlite3_column_text(stmt_.get(), id);
|
|
|
|
auto size = sqlite3_column_bytes(stmt_.get(), id);
|
|
|
|
if (data == nullptr) {
|
|
|
|
return Slice();
|
|
|
|
}
|
|
|
|
return Slice(data, size);
|
|
|
|
}
|
|
|
|
int32 SqliteStatement::view_int32(int id) {
|
|
|
|
LOG_IF(ERROR, view_datatype(id) != Datatype::Integer) << view_datatype(id);
|
|
|
|
return sqlite3_column_int(stmt_.get(), id);
|
|
|
|
}
|
|
|
|
int64 SqliteStatement::view_int64(int id) {
|
|
|
|
LOG_IF(ERROR, view_datatype(id) != Datatype::Integer) << view_datatype(id);
|
|
|
|
return sqlite3_column_int64(stmt_.get(), id);
|
|
|
|
}
|
|
|
|
SqliteStatement::Datatype SqliteStatement::view_datatype(int id) {
|
|
|
|
auto type = sqlite3_column_type(stmt_.get(), id);
|
|
|
|
switch (type) {
|
|
|
|
case SQLITE_INTEGER:
|
|
|
|
return Datatype::Integer;
|
|
|
|
case SQLITE_FLOAT:
|
|
|
|
return Datatype::Float;
|
|
|
|
case SQLITE_BLOB:
|
|
|
|
return Datatype::Blob;
|
|
|
|
case SQLITE_NULL:
|
|
|
|
return Datatype::Null;
|
|
|
|
case SQLITE3_TEXT:
|
|
|
|
return Datatype::Text;
|
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SqliteStatement::reset() {
|
|
|
|
sqlite3_reset(stmt_.get());
|
|
|
|
state_ = Start;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status SqliteStatement::step() {
|
|
|
|
if (state_ == Finish) {
|
|
|
|
return Status::Error("One has to reset statement");
|
|
|
|
}
|
2018-12-14 01:48:36 +03:00
|
|
|
VLOG(sqlite) << "Start step " << tag("query", sqlite3_sql(stmt_.get())) << tag("statement", stmt_.get())
|
|
|
|
<< tag("database", db_.get());
|
2018-12-31 22:04:05 +03:00
|
|
|
auto rc = sqlite3_step(stmt_.get());
|
2018-12-14 01:48:36 +03:00
|
|
|
VLOG(sqlite) << "Finish step " << tag("query", sqlite3_sql(stmt_.get())) << tag("statement", stmt_.get())
|
|
|
|
<< tag("database", db_.get());
|
2018-12-31 22:04:05 +03:00
|
|
|
if (rc == SQLITE_ROW) {
|
|
|
|
state_ = GotRow;
|
|
|
|
return Status::OK();
|
|
|
|
}
|
|
|
|
if (rc == SQLITE_DONE) {
|
|
|
|
state_ = Finish;
|
|
|
|
return Status::OK();
|
|
|
|
}
|
|
|
|
state_ = Finish;
|
|
|
|
return last_error();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SqliteStatement::StmtDeleter::operator()(sqlite3_stmt *stmt) {
|
|
|
|
sqlite3_finalize(stmt);
|
|
|
|
}
|
|
|
|
|
|
|
|
Status SqliteStatement::last_error() {
|
|
|
|
return db_->last_error();
|
|
|
|
}
|
2018-07-18 00:41:26 +03:00
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
} // namespace td
|