// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2023 // // 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/telegram/DialogDb.h" #include "td/telegram/Version.h" #include "td/db/SqliteConnectionSafe.h" #include "td/db/SqliteDb.h" #include "td/db/SqliteKeyValue.h" #include "td/db/SqliteStatement.h" #include "td/actor/actor.h" #include "td/actor/SchedulerLocalStorage.h" #include "td/utils/common.h" #include "td/utils/format.h" #include "td/utils/logging.h" #include "td/utils/misc.h" #include "td/utils/ScopeGuard.h" #include "td/utils/SliceBuilder.h" #include "td/utils/Time.h" namespace td { // NB: must happen inside a transaction Status init_dialog_db(SqliteDb &db, int32 version, KeyValueSyncInterface &binlog_pmc, bool &was_created) { LOG(INFO) << "Init dialog database " << tag("version", version); was_created = false; // Check if database exists TRY_RESULT(has_table, db.has_table("dialogs")); if (!has_table) { version = 0; } else if (version > current_db_version()) { TRY_STATUS(drop_dialog_db(db, version)); version = 0; } auto create_notification_group_table = [&db] { return db.exec( "CREATE TABLE IF NOT EXISTS notification_groups (notification_group_id INT4 PRIMARY KEY, dialog_id " "INT8, last_notification_date INT4)"); }; auto create_last_notification_date_index = [&db] { return db.exec( "CREATE INDEX IF NOT EXISTS notification_group_by_last_notification_date ON notification_groups " "(last_notification_date, dialog_id, notification_group_id) WHERE last_notification_date IS NOT NULL"); }; auto add_dialogs_in_folder_index = [&db] { return db.exec( "CREATE INDEX IF NOT EXISTS dialog_in_folder_by_dialog_order ON dialogs (folder_id, dialog_order, dialog_id) " "WHERE folder_id IS NOT NULL"); }; if (version == 0) { LOG(INFO) << "Create new dialog database"; was_created = true; TRY_STATUS( db.exec("CREATE TABLE IF NOT EXISTS dialogs (dialog_id INT8 PRIMARY KEY, dialog_order INT8, data BLOB, " "folder_id INT4)")); TRY_STATUS(create_notification_group_table()); TRY_STATUS(create_last_notification_date_index()); TRY_STATUS(add_dialogs_in_folder_index()); version = current_db_version(); } if (version < static_cast(DbVersion::AddNotificationsSupport)) { TRY_STATUS(create_notification_group_table()); TRY_STATUS(create_last_notification_date_index()); } if (version < static_cast(DbVersion::AddFolders)) { TRY_STATUS(db.exec("DROP INDEX IF EXISTS dialog_by_dialog_order")); TRY_STATUS(db.exec("ALTER TABLE dialogs ADD COLUMN folder_id INT4")); TRY_STATUS(add_dialogs_in_folder_index()); TRY_STATUS(db.exec("UPDATE dialogs SET folder_id = 0 WHERE dialog_id < -1500000000000 AND dialog_order > 0")); } if (version < static_cast(DbVersion::StorePinnedDialogsInBinlog)) { // 9221294780217032704 == get_dialog_order(Auto(), MIN_PINNED_DIALOG_DATE - 1) TRY_RESULT(get_pinned_dialogs_stmt, db.get_statement("SELECT dialog_id FROM dialogs WHERE folder_id = ?1 AND dialog_order > " "9221294780217032704 ORDER BY dialog_order DESC, dialog_id DESC")); for (auto folder_id = 0; folder_id < 2; folder_id++) { vector pinned_dialog_ids; TRY_STATUS(get_pinned_dialogs_stmt.bind_int32(1, folder_id)); TRY_STATUS(get_pinned_dialogs_stmt.step()); while (get_pinned_dialogs_stmt.has_row()) { pinned_dialog_ids.push_back(PSTRING() << get_pinned_dialogs_stmt.view_int64(0)); TRY_STATUS(get_pinned_dialogs_stmt.step()); } get_pinned_dialogs_stmt.reset(); binlog_pmc.set(PSTRING() << "pinned_dialog_ids" << folder_id, implode(pinned_dialog_ids, ',')); } } return Status::OK(); } // NB: must happen inside a transaction Status drop_dialog_db(SqliteDb &db, int version) { if (version != 0) { LOG(WARNING) << "Drop chat database " << tag("version", version) << tag("current_db_version", current_db_version()); } auto status = db.exec("DROP TABLE IF EXISTS dialogs"); TRY_STATUS(db.exec("DROP TABLE IF EXISTS notification_groups")); return status; } class DialogDbImpl final : public DialogDbSyncInterface { public: explicit DialogDbImpl(SqliteDb db) : db_(std::move(db)) { init().ensure(); } Status init() { TRY_RESULT_ASSIGN(add_dialog_stmt_, db_.get_statement("INSERT OR REPLACE INTO dialogs VALUES(?1, ?2, ?3, ?4)")); TRY_RESULT_ASSIGN(add_notification_group_stmt_, db_.get_statement("INSERT OR REPLACE INTO notification_groups VALUES(?1, ?2, ?3)")); TRY_RESULT_ASSIGN(delete_notification_group_stmt_, db_.get_statement("DELETE FROM notification_groups WHERE notification_group_id = ?1")); TRY_RESULT_ASSIGN(get_dialog_stmt_, db_.get_statement("SELECT data FROM dialogs WHERE dialog_id = ?1")); TRY_RESULT_ASSIGN( get_dialogs_stmt_, db_.get_statement("SELECT data, dialog_id, dialog_order FROM dialogs WHERE " "folder_id = ?1 AND (dialog_order < ?2 OR (dialog_order = ?2 AND dialog_id < ?3)) ORDER " "BY dialog_order DESC, dialog_id DESC LIMIT ?4")); TRY_RESULT_ASSIGN( get_notification_groups_by_last_notification_date_stmt_, db_.get_statement("SELECT notification_group_id, dialog_id, last_notification_date FROM notification_groups " "WHERE last_notification_date < ?1 OR (last_notification_date = ?1 " "AND (dialog_id < ?2 OR (dialog_id = ?2 AND notification_group_id < ?3))) ORDER BY " "last_notification_date DESC, dialog_id DESC LIMIT ?4")); // "WHERE (last_notification_date, dialog_id, notification_group_id) < (?1, ?2, ?3) ORDER BY " // "last_notification_date DESC, dialog_id DESC, notification_group_id DESC LIMIT ?4")); TRY_RESULT_ASSIGN( get_notification_group_stmt_, db_.get_statement( "SELECT dialog_id, last_notification_date FROM notification_groups WHERE notification_group_id = ?1")); TRY_RESULT_ASSIGN( get_secret_chat_count_stmt_, db_.get_statement( "SELECT COUNT(*) FROM dialogs WHERE folder_id = ?1 AND dialog_order > 0 AND dialog_id < -1500000000000")); // LOG(ERROR) << get_dialog_stmt_.explain().ok(); // LOG(ERROR) << get_dialogs_stmt_.explain().ok(); // LOG(ERROR) << get_notification_groups_by_last_notification_date_stmt_.explain().ok(); // LOG(ERROR) << get_notification_group_stmt_.explain().ok(); // LOG(FATAL) << "EXPLAINED"; return Status::OK(); } void add_dialog(DialogId dialog_id, FolderId folder_id, int64 order, BufferSlice data, vector notification_groups) final { SCOPE_EXIT { add_dialog_stmt_.reset(); }; add_dialog_stmt_.bind_int64(1, dialog_id.get()).ensure(); add_dialog_stmt_.bind_int64(2, order).ensure(); add_dialog_stmt_.bind_blob(3, data.as_slice()).ensure(); if (order > 0) { add_dialog_stmt_.bind_int32(4, folder_id.get()).ensure(); } else { add_dialog_stmt_.bind_null(4).ensure(); } add_dialog_stmt_.step().ensure(); for (auto &to_add : notification_groups) { if (to_add.dialog_id.is_valid()) { SCOPE_EXIT { add_notification_group_stmt_.reset(); }; add_notification_group_stmt_.bind_int32(1, to_add.group_id.get()).ensure(); add_notification_group_stmt_.bind_int64(2, to_add.dialog_id.get()).ensure(); if (to_add.last_notification_date != 0) { add_notification_group_stmt_.bind_int32(3, to_add.last_notification_date).ensure(); } else { add_notification_group_stmt_.bind_null(3).ensure(); } add_notification_group_stmt_.step().ensure(); } else { SCOPE_EXIT { delete_notification_group_stmt_.reset(); }; delete_notification_group_stmt_.bind_int32(1, to_add.group_id.get()).ensure(); delete_notification_group_stmt_.step().ensure(); } } } Result get_dialog(DialogId dialog_id) final { SCOPE_EXIT { get_dialog_stmt_.reset(); }; get_dialog_stmt_.bind_int64(1, dialog_id.get()).ensure(); TRY_STATUS(get_dialog_stmt_.step()); if (!get_dialog_stmt_.has_row()) { return Status::Error("Not found"); } return BufferSlice(get_dialog_stmt_.view_blob(0)); } Result get_notification_group(NotificationGroupId notification_group_id) final { SCOPE_EXIT { get_notification_group_stmt_.reset(); }; get_notification_group_stmt_.bind_int32(1, notification_group_id.get()).ensure(); TRY_STATUS(get_notification_group_stmt_.step()); if (!get_notification_group_stmt_.has_row()) { return Status::Error("Not found"); } return NotificationGroupKey(notification_group_id, DialogId(get_notification_group_stmt_.view_int64(0)), get_last_notification_date(get_notification_group_stmt_, 1)); } int32 get_secret_chat_count(FolderId folder_id) final { SCOPE_EXIT { get_secret_chat_count_stmt_.reset(); }; get_secret_chat_count_stmt_.bind_int32(1, folder_id.get()).ensure(); get_secret_chat_count_stmt_.step().ensure(); CHECK(get_secret_chat_count_stmt_.has_row()); return get_secret_chat_count_stmt_.view_int32(0); } DialogDbGetDialogsResult get_dialogs(FolderId folder_id, int64 order, DialogId dialog_id, int32 limit) final { SCOPE_EXIT { get_dialogs_stmt_.reset(); }; get_dialogs_stmt_.bind_int32(1, folder_id.get()).ensure(); get_dialogs_stmt_.bind_int64(2, order).ensure(); get_dialogs_stmt_.bind_int64(3, dialog_id.get()).ensure(); get_dialogs_stmt_.bind_int32(4, limit).ensure(); DialogDbGetDialogsResult result; result.next_dialog_id = dialog_id; result.next_order = order; get_dialogs_stmt_.step().ensure(); while (get_dialogs_stmt_.has_row()) { BufferSlice data(get_dialogs_stmt_.view_blob(0)); result.next_dialog_id = DialogId(get_dialogs_stmt_.view_int64(1)); result.next_order = get_dialogs_stmt_.view_int64(2); LOG(INFO) << "Load " << result.next_dialog_id << " with order " << result.next_order; result.dialogs.emplace_back(std::move(data)); get_dialogs_stmt_.step().ensure(); } return result; } vector get_notification_groups_by_last_notification_date( NotificationGroupKey notification_group_key, int32 limit) final { auto &stmt = get_notification_groups_by_last_notification_date_stmt_; SCOPE_EXIT { stmt.reset(); }; stmt.bind_int32(1, notification_group_key.last_notification_date).ensure(); stmt.bind_int64(2, notification_group_key.dialog_id.get()).ensure(); stmt.bind_int32(3, notification_group_key.group_id.get()).ensure(); stmt.bind_int32(4, limit).ensure(); vector notification_groups; stmt.step().ensure(); while (stmt.has_row()) { notification_groups.emplace_back(NotificationGroupId(stmt.view_int32(0)), DialogId(stmt.view_int64(1)), get_last_notification_date(stmt, 2)); stmt.step().ensure(); } return notification_groups; } Status begin_read_transaction() final { return db_.begin_read_transaction(); } Status begin_write_transaction() final { return db_.begin_write_transaction(); } Status commit_transaction() final { return db_.commit_transaction(); } private: SqliteDb db_; SqliteStatement add_dialog_stmt_; SqliteStatement add_notification_group_stmt_; SqliteStatement delete_notification_group_stmt_; SqliteStatement get_dialog_stmt_; SqliteStatement get_dialogs_stmt_; SqliteStatement get_notification_groups_by_last_notification_date_stmt_; SqliteStatement get_notification_group_stmt_; SqliteStatement get_secret_chat_count_stmt_; static int32 get_last_notification_date(SqliteStatement &stmt, int id) { if (stmt.view_datatype(id) == SqliteStatement::Datatype::Null) { return 0; } return stmt.view_int32(id); } }; std::shared_ptr create_dialog_db_sync( std::shared_ptr sqlite_connection) { class DialogDbSyncSafe final : public DialogDbSyncSafeInterface { public: explicit DialogDbSyncSafe(std::shared_ptr sqlite_connection) : lsls_db_([safe_connection = std::move(sqlite_connection)] { return make_unique(safe_connection->get().clone()); }) { } DialogDbSyncInterface &get() final { return *lsls_db_.get(); } private: LazySchedulerLocalStorage> lsls_db_; }; return std::make_shared(std::move(sqlite_connection)); } class DialogDbAsync final : public DialogDbAsyncInterface { public: DialogDbAsync(std::shared_ptr sync_db, int32 scheduler_id) { impl_ = create_actor_on_scheduler("DialogDbActor", scheduler_id, std::move(sync_db)); } void add_dialog(DialogId dialog_id, FolderId folder_id, int64 order, BufferSlice data, vector notification_groups, Promise promise) final { send_closure(impl_, &Impl::add_dialog, dialog_id, folder_id, order, std::move(data), std::move(notification_groups), std::move(promise)); } void get_notification_groups_by_last_notification_date(NotificationGroupKey notification_group_key, int32 limit, Promise> promise) final { send_closure(impl_, &Impl::get_notification_groups_by_last_notification_date, notification_group_key, limit, std::move(promise)); } void get_notification_group(NotificationGroupId notification_group_id, Promise promise) final { send_closure(impl_, &Impl::get_notification_group, notification_group_id, std::move(promise)); } void get_secret_chat_count(FolderId folder_id, Promise promise) final { send_closure(impl_, &Impl::get_secret_chat_count, folder_id, std::move(promise)); } void get_dialog(DialogId dialog_id, Promise promise) final { send_closure_later(impl_, &Impl::get_dialog, dialog_id, std::move(promise)); } void get_dialogs(FolderId folder_id, int64 order, DialogId dialog_id, int32 limit, Promise promise) final { send_closure_later(impl_, &Impl::get_dialogs, folder_id, order, dialog_id, limit, std::move(promise)); } void close(Promise promise) final { send_closure_later(impl_, &Impl::close, std::move(promise)); } void force_flush() final { send_closure_later(impl_, &Impl::force_flush); } private: class Impl final : public Actor { public: explicit Impl(std::shared_ptr sync_db_safe) : sync_db_safe_(std::move(sync_db_safe)) { } void add_dialog(DialogId dialog_id, FolderId folder_id, int64 order, BufferSlice data, vector notification_groups, Promise promise) { add_write_query([this, dialog_id, folder_id, order, promise = std::move(promise), data = std::move(data), notification_groups = std::move(notification_groups)](Unit) mutable { sync_db_->add_dialog(dialog_id, folder_id, order, std::move(data), std::move(notification_groups)); on_write_result(std::move(promise)); }); } void on_write_result(Promise &&promise) { // We are inside a transaction and don't know how to handle errors finished_writes_.push_back(std::move(promise)); } void get_notification_groups_by_last_notification_date(NotificationGroupKey notification_group_key, int32 limit, Promise> promise) { add_read_query(); promise.set_value(sync_db_->get_notification_groups_by_last_notification_date(notification_group_key, limit)); } void get_notification_group(NotificationGroupId notification_group_id, Promise promise) { add_read_query(); promise.set_result(sync_db_->get_notification_group(notification_group_id)); } void get_secret_chat_count(FolderId folder_id, Promise promise) { add_read_query(); promise.set_value(sync_db_->get_secret_chat_count(folder_id)); } void get_dialog(DialogId dialog_id, Promise promise) { add_read_query(); promise.set_result(sync_db_->get_dialog(dialog_id)); } void get_dialogs(FolderId folder_id, int64 order, DialogId dialog_id, int32 limit, Promise promise) { add_read_query(); promise.set_value(sync_db_->get_dialogs(folder_id, order, dialog_id, limit)); } void close(Promise promise) { do_flush(); sync_db_safe_.reset(); sync_db_ = nullptr; promise.set_value(Unit()); stop(); } void force_flush() { do_flush(); LOG(INFO) << "DialogDb flushed"; } private: std::shared_ptr sync_db_safe_; DialogDbSyncInterface *sync_db_ = nullptr; static constexpr size_t MAX_PENDING_QUERIES_COUNT{50}; static constexpr double MAX_PENDING_QUERIES_DELAY{0.01}; //NB: order is important, destructor of pending_writes_ will change finished_writes_ vector> finished_writes_; vector> pending_writes_; // TODO use Action double wakeup_at_ = 0; template void add_write_query(F &&f) { pending_writes_.push_back(PromiseCreator::lambda(std::forward(f))); if (pending_writes_.size() > MAX_PENDING_QUERIES_COUNT) { do_flush(); wakeup_at_ = 0; } else if (wakeup_at_ == 0) { wakeup_at_ = Time::now_cached() + MAX_PENDING_QUERIES_DELAY; } if (wakeup_at_ != 0) { set_timeout_at(wakeup_at_); } } void add_read_query() { do_flush(); } void do_flush() { if (pending_writes_.empty()) { return; } sync_db_->begin_write_transaction().ensure(); set_promises(pending_writes_); sync_db_->commit_transaction().ensure(); set_promises(finished_writes_); cancel_timeout(); } void timeout_expired() final { do_flush(); } void start_up() final { sync_db_ = &sync_db_safe_->get(); } }; ActorOwn impl_; }; std::shared_ptr create_dialog_db_async(std::shared_ptr sync_db, int32 scheduler_id) { return std::make_shared(std::move(sync_db), scheduler_id); } } // namespace td