mirror of
				https://github.com/ton-blockchain/ton
				synced 2025-03-09 15:40:10 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			237 lines
		
	
	
	
		
			7.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			237 lines
		
	
	
	
		
			7.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
    This file is part of TON Blockchain Library.
 | 
						|
 | 
						|
    TON Blockchain Library is free software: you can redistribute it and/or modify
 | 
						|
    it under the terms of the GNU Lesser General Public License as published by
 | 
						|
    the Free Software Foundation, either version 2 of the License, or
 | 
						|
    (at your option) any later version.
 | 
						|
 | 
						|
    TON Blockchain Library is distributed in the hope that it will be useful,
 | 
						|
    but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
    GNU Lesser General Public License for more details.
 | 
						|
 | 
						|
    You should have received a copy of the GNU Lesser General Public License
 | 
						|
    along with TON Blockchain Library.  If not, see <http://www.gnu.org/licenses/>.
 | 
						|
 | 
						|
    Copyright 2017-2020 Telegram Systems LLP
 | 
						|
*/
 | 
						|
#pragma once
 | 
						|
 | 
						|
#include "validator/interfaces/db.h"
 | 
						|
#include "package.hpp"
 | 
						|
#include "fileref.hpp"
 | 
						|
#include "td/db/RocksDb.h"
 | 
						|
#include <map>
 | 
						|
 | 
						|
namespace rocksdb {
 | 
						|
class Statistics;
 | 
						|
}
 | 
						|
 | 
						|
namespace ton {
 | 
						|
 | 
						|
namespace validator {
 | 
						|
 | 
						|
struct PackageId {
 | 
						|
  td::uint32 id;
 | 
						|
  bool key;
 | 
						|
  bool temp;
 | 
						|
 | 
						|
  explicit PackageId(td::uint32 id, bool key, bool temp) : id(id), key(key), temp(temp) {
 | 
						|
  }
 | 
						|
 | 
						|
  bool operator<(const PackageId &with) const {
 | 
						|
    return id < with.id;
 | 
						|
  }
 | 
						|
  bool operator==(const PackageId &with) const {
 | 
						|
    return id == with.id;
 | 
						|
  }
 | 
						|
 | 
						|
  std::string path() const;
 | 
						|
  std::string name() const;
 | 
						|
 | 
						|
  bool is_empty() const {
 | 
						|
    return id == std::numeric_limits<td::uint32>::max();
 | 
						|
  }
 | 
						|
  static PackageId empty(bool key, bool temp) {
 | 
						|
    return PackageId(std::numeric_limits<td::uint32>::max(), key, temp);
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
class PackageStatistics;
 | 
						|
 | 
						|
struct DbStatistics {
 | 
						|
  void init();
 | 
						|
  std::string to_string_and_reset();
 | 
						|
 | 
						|
  std::shared_ptr<PackageStatistics> pack_statistics;
 | 
						|
  std::shared_ptr<rocksdb::Statistics> rocksdb_statistics;
 | 
						|
};
 | 
						|
 | 
						|
class PackageWriter : public td::actor::Actor {
 | 
						|
 public:
 | 
						|
  PackageWriter(std::weak_ptr<Package> package, bool async_mode = false, std::shared_ptr<PackageStatistics> statistics = nullptr)
 | 
						|
      : package_(std::move(package)), async_mode_(async_mode), statistics_(std::move(statistics)) {
 | 
						|
  }
 | 
						|
 | 
						|
  void append(std::string filename, td::BufferSlice data, td::Promise<std::pair<td::uint64, td::uint64>> promise);
 | 
						|
  void set_async_mode(bool mode, td::Promise<td::Unit> promise) {
 | 
						|
    async_mode_ = mode;
 | 
						|
    if (!async_mode_) {
 | 
						|
      auto p = package_.lock();
 | 
						|
      if (p) {
 | 
						|
        p->sync();
 | 
						|
      }
 | 
						|
    }
 | 
						|
    promise.set_value(td::Unit());
 | 
						|
  }
 | 
						|
 | 
						|
 private:
 | 
						|
  std::weak_ptr<Package> package_;
 | 
						|
  bool async_mode_ = false;
 | 
						|
  std::shared_ptr<PackageStatistics> statistics_;
 | 
						|
};
 | 
						|
 | 
						|
class ArchiveLru;
 | 
						|
 | 
						|
class ArchiveSlice : public td::actor::Actor {
 | 
						|
 public:
 | 
						|
  ArchiveSlice(td::uint32 archive_id, bool key_blocks_only, bool temp, bool finalized, std::string db_root,
 | 
						|
               td::actor::ActorId<ArchiveLru> archive_lru, DbStatistics statistics = {});
 | 
						|
 | 
						|
  void get_archive_id(BlockSeqno masterchain_seqno, td::Promise<td::uint64> promise);
 | 
						|
 | 
						|
  void add_handle(BlockHandle handle, td::Promise<td::Unit> promise);
 | 
						|
  void update_handle(BlockHandle handle, td::Promise<td::Unit> promise);
 | 
						|
  void add_file(BlockHandle handle, FileReference ref_id, td::BufferSlice data, td::Promise<td::Unit> promise);
 | 
						|
  void get_handle(BlockIdExt block_id, td::Promise<BlockHandle> promise);
 | 
						|
  void get_temp_handle(BlockIdExt block_id, td::Promise<ConstBlockHandle> promise);
 | 
						|
  void get_file(ConstBlockHandle handle, FileReference ref_id, td::Promise<td::BufferSlice> promise);
 | 
						|
 | 
						|
  /* from LTDB */
 | 
						|
  void get_block_by_unix_time(AccountIdPrefixFull account_id, UnixTime ts, td::Promise<ConstBlockHandle> promise);
 | 
						|
  void get_block_by_lt(AccountIdPrefixFull account_id, LogicalTime lt, td::Promise<ConstBlockHandle> promise);
 | 
						|
  void get_block_by_seqno(AccountIdPrefixFull account_id, BlockSeqno seqno, td::Promise<ConstBlockHandle> promise);
 | 
						|
  void get_block_common(AccountIdPrefixFull account_id,
 | 
						|
                        std::function<td::int32(ton_api::db_lt_desc_value &)> compare_desc,
 | 
						|
                        std::function<td::int32(ton_api::db_lt_el_value &)> compare, bool exact,
 | 
						|
                        td::Promise<ConstBlockHandle> promise);
 | 
						|
 | 
						|
  void get_slice(td::uint64 archive_id, td::uint64 offset, td::uint32 limit, td::Promise<td::BufferSlice> promise);
 | 
						|
 | 
						|
  void destroy(td::Promise<td::Unit> promise);
 | 
						|
  void truncate(BlockSeqno masterchain_seqno, ConstBlockHandle handle, td::Promise<td::Unit> promise);
 | 
						|
 | 
						|
  void set_async_mode(bool mode, td::Promise<td::Unit> promise);
 | 
						|
 | 
						|
  void open_files();
 | 
						|
  void close_files();
 | 
						|
 | 
						|
 private:
 | 
						|
  void before_query();
 | 
						|
  void do_close();
 | 
						|
  template<typename T>
 | 
						|
  td::Promise<T> begin_async_query(td::Promise<T> promise);
 | 
						|
  void end_async_query();
 | 
						|
 | 
						|
  void begin_transaction();
 | 
						|
  void commit_transaction();
 | 
						|
 | 
						|
  void add_file_cont(size_t idx, FileReference ref_id, td::uint64 offset, td::uint64 size,
 | 
						|
                     td::Promise<td::Unit> promise);
 | 
						|
 | 
						|
  /* ltdb */
 | 
						|
  td::BufferSlice get_db_key_lt_desc(ShardIdFull shard);
 | 
						|
  td::BufferSlice get_db_key_lt_el(ShardIdFull shard, td::uint32 idx);
 | 
						|
  td::BufferSlice get_db_key_block_info(BlockIdExt block_id);
 | 
						|
 | 
						|
  td::uint32 archive_id_;
 | 
						|
 | 
						|
  bool key_blocks_only_;
 | 
						|
  bool temp_;
 | 
						|
  bool finalized_;
 | 
						|
  PackageId p_id_;
 | 
						|
  std::string db_path_;
 | 
						|
 | 
						|
  bool destroyed_ = false;
 | 
						|
  bool async_mode_ = false;
 | 
						|
  bool huge_transaction_started_ = false;
 | 
						|
  bool sliced_mode_{false};
 | 
						|
  td::uint32 huge_transaction_size_ = 0;
 | 
						|
  td::uint32 slice_size_{100};
 | 
						|
 | 
						|
  enum Status {
 | 
						|
    st_closed, st_open, st_want_close
 | 
						|
  } status_ = st_closed;
 | 
						|
  size_t active_queries_ = 0;
 | 
						|
 | 
						|
  std::string db_root_;
 | 
						|
  td::actor::ActorId<ArchiveLru> archive_lru_;
 | 
						|
  DbStatistics statistics_;
 | 
						|
  std::unique_ptr<td::KeyValue> kv_;
 | 
						|
 | 
						|
  struct PackageInfo {
 | 
						|
    PackageInfo(std::shared_ptr<Package> package, td::actor::ActorOwn<PackageWriter> writer, BlockSeqno id,
 | 
						|
                std::string path, td::uint32 idx, td::uint32 version)
 | 
						|
        : package(std::move(package))
 | 
						|
        , writer(std ::move(writer))
 | 
						|
        , id(id)
 | 
						|
        , path(std::move(path))
 | 
						|
        , idx(idx)
 | 
						|
        , version(version) {
 | 
						|
    }
 | 
						|
    std::shared_ptr<Package> package;
 | 
						|
    td::actor::ActorOwn<PackageWriter> writer;
 | 
						|
    BlockSeqno id;
 | 
						|
    std::string path;
 | 
						|
    td::uint32 idx;
 | 
						|
    td::uint32 version;
 | 
						|
  };
 | 
						|
  std::vector<PackageInfo> packages_;
 | 
						|
 | 
						|
  td::Result<PackageInfo *> choose_package(BlockSeqno masterchain_seqno, bool force);
 | 
						|
  void add_package(BlockSeqno masterchain_seqno, td::uint64 size, td::uint32 version);
 | 
						|
  void truncate_shard(BlockSeqno masterchain_seqno, ShardIdFull shard, td::uint32 cutoff_idx, Package *pack);
 | 
						|
  bool truncate_block(BlockSeqno masterchain_seqno, BlockIdExt block_id, td::uint32 cutoff_idx, Package *pack);
 | 
						|
 | 
						|
  void delete_handle(ConstBlockHandle handle);
 | 
						|
  void delete_file(FileReference ref_id);
 | 
						|
  void move_handle(ConstBlockHandle handle, Package *old_pack, Package *pack);
 | 
						|
  void move_file(FileReference ref_id, Package *old_pack, Package *pack);
 | 
						|
 | 
						|
  BlockSeqno max_masterchain_seqno();
 | 
						|
 | 
						|
  static constexpr td::uint32 default_package_version() {
 | 
						|
    return 1;
 | 
						|
  }
 | 
						|
 | 
						|
  static const size_t ESTIMATED_DB_OPEN_FILES = 5;
 | 
						|
};
 | 
						|
 | 
						|
class ArchiveLru : public td::actor::Actor {
 | 
						|
 public:
 | 
						|
  explicit ArchiveLru(size_t max_total_files) : max_total_files_(max_total_files) {
 | 
						|
    CHECK(max_total_files_ > 0);
 | 
						|
  }
 | 
						|
  void on_query(td::actor::ActorId<ArchiveSlice> slice, PackageId id, size_t files_count);
 | 
						|
  void set_permanent_slices(std::vector<PackageId> ids);
 | 
						|
 private:
 | 
						|
  size_t current_idx_ = 1;
 | 
						|
  struct SliceInfo {
 | 
						|
    td::actor::ActorId<ArchiveSlice> actor;
 | 
						|
    size_t files_count = 0;
 | 
						|
    size_t opened_idx = 0;  // 0 - not opened
 | 
						|
    bool is_permanent = false;
 | 
						|
  };
 | 
						|
  std::map<std::tuple<td::uint32, bool, bool>, SliceInfo> slices_;
 | 
						|
  std::map<size_t, PackageId> lru_;
 | 
						|
  size_t total_files_ = 0;
 | 
						|
  size_t max_total_files_ = 0;
 | 
						|
  std::vector<PackageId> permanent_slices_;
 | 
						|
 | 
						|
  void enforce_limit();
 | 
						|
};
 | 
						|
 | 
						|
}  // namespace validator
 | 
						|
 | 
						|
}  // namespace ton
 |