// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2021 // // 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) // #pragma once #include "td/db/binlog/BinlogEvent.h" #include "td/db/DbKey.h" #include "td/actor/PromiseFuture.h" #include "td/utils/AesCtrByteFlow.h" #include "td/utils/buffer.h" #include "td/utils/BufferedFd.h" #include "td/utils/ByteFlow.h" #include "td/utils/common.h" #include "td/utils/crypto.h" #include "td/utils/logging.h" #include "td/utils/port/FileFd.h" #include "td/utils/Slice.h" #include "td/utils/Status.h" #include "td/utils/StorerBase.h" #include "td/utils/UInt.h" #include namespace td { extern int32 VERBOSITY_NAME(binlog); struct BinlogInfo { bool was_created{false}; uint64 last_id{0}; bool is_encrypted{false}; bool wrong_password{false}; bool is_opened{false}; }; namespace detail { class BinlogReader; class BinlogEventsProcessor; class BinlogEventsBuffer; } // namespace detail class Binlog { public: enum Error : int { WrongPassword = -1 }; Binlog(); Binlog(const Binlog &other) = delete; Binlog &operator=(const Binlog &other) = delete; Binlog(Binlog &&other) = delete; Binlog &operator=(Binlog &&other) = delete; ~Binlog(); using Callback = std::function; Status init(string path, const Callback &callback, DbKey db_key = DbKey::empty(), DbKey old_db_key = DbKey::empty(), int32 dummy = -1, const Callback &debug_callback = Callback()) TD_WARN_UNUSED_RESULT; uint64 next_id() { return ++last_id_; } uint64 next_id(int32 shift) { auto res = last_id_ + 1; last_id_ += shift; return res; } uint64 peek_next_id() const { return last_id_ + 1; } bool empty() const { return fd_.empty(); } uint64 add(int32 type, const Storer &storer) { auto log_event_id = next_id(); add_raw_event(BinlogEvent::create_raw(log_event_id, type, 0, storer), {}); return log_event_id; } uint64 rewrite(uint64 log_event_id, int32 type, const Storer &storer) { auto seq_no = next_id(); add_raw_event(BinlogEvent::create_raw(log_event_id, type, BinlogEvent::Flags::Rewrite, storer), {}); return seq_no; } uint64 erase(uint64 log_event_id) { auto seq_no = next_id(); add_raw_event(BinlogEvent::create_raw(log_event_id, BinlogEvent::ServiceTypes::Empty, BinlogEvent::Flags::Rewrite, EmptyStorer()), {}); return seq_no; } void add_raw_event(BufferSlice &&raw_event, BinlogDebugInfo info) { add_event(BinlogEvent(std::move(raw_event), info)); } void add_event(BinlogEvent &&event); void sync(); void flush(); void lazy_flush(); double need_flush_since() const { return need_flush_since_; } void change_key(DbKey new_db_key); Status close(bool need_sync = true) TD_WARN_UNUSED_RESULT; void close(Promise<> promise); Status close_and_destroy() TD_WARN_UNUSED_RESULT; static Status destroy(Slice path) TD_WARN_UNUSED_RESULT; CSlice get_path() const { return path_; } BinlogInfo get_info() const { // works even after binlog was closed return info_; } private: BufferedFdBase fd_; ChainBufferWriter buffer_writer_; ChainBufferReader buffer_reader_; detail::BinlogReader *binlog_reader_ptr_ = nullptr; BinlogInfo info_; DbKey db_key_; bool db_key_used_ = false; DbKey old_db_key_; enum class EncryptionType { None, AesCtr } encryption_type_ = EncryptionType::None; // AesCtrEncryption BufferSlice aes_ctr_key_salt_; UInt256 aes_ctr_key_; AesCtrState aes_ctr_state_; bool byte_flow_flag_ = false; ByteFlowSource byte_flow_source_; ByteFlowSink byte_flow_sink_; AesCtrByteFlow aes_xcode_byte_flow_; int64 fd_size_{0}; uint64 fd_events_{0}; string path_; std::vector pending_events_; unique_ptr processor_; unique_ptr events_buffer_; bool in_flush_events_buffer_{false}; uint64 last_id_{0}; double need_flush_since_ = 0; bool need_sync_{false}; enum class State { Empty, Load, Reindex, Run } state_{State::Empty}; static Result open_binlog(const string &path, int32 flags); size_t flush_events_buffer(bool force); void do_add_event(BinlogEvent &&event); void do_event(BinlogEvent &&event); Status load_binlog(const Callback &callback, const Callback &debug_callback = Callback()) TD_WARN_UNUSED_RESULT; void do_reindex(); void update_encryption(Slice key, Slice iv); void reset_encryption(); void update_read_encryption(); void update_write_encryption(); string debug_get_binlog_data(int64 begin_offset, int64 end_offset); }; } // namespace td