/*
    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 .
    Copyright 2017-2020 Telegram Systems LLP
*/
#include "fabric.h"
#include "collator-impl.h"
#include "validator/db/rootdb.hpp"
#include "validator/block-handle.hpp"
#include "apply-block.hpp"
#include "accept-block.hpp"
#include "shard.hpp"
#include "block.hpp"
#include "proof.hpp"
#include "signature-set.hpp"
#include "external-message.hpp"
#include "ihr-message.hpp"
#include "validate-query.hpp"
#include "check-proof.hpp"
#include "top-shard-descr.hpp"
#include "ton/ton-io.hpp"
#include "liteserver.hpp"
#include "validator/fabric.h"
namespace ton {
namespace validator {
td::actor::ActorOwn create_db_actor(td::actor::ActorId manager, std::string db_root_) {
  return td::actor::create_actor("db", manager, db_root_);
}
td::actor::ActorOwn create_liteserver_cache_actor(td::actor::ActorId manager,
                                                                   std::string db_root) {
  return td::actor::create_actor("cache");
}
td::Result> create_block(BlockIdExt block_id, td::BufferSlice data) {
  auto res = BlockQ::create(block_id, std::move(data));
  if (res.is_error()) {
    return res.move_as_error();
  } else {
    return td::Ref{res.move_as_ok()};
  }
}
td::Result> create_block(ReceivedBlock data) {
  return create_block(data.id, std::move(data.data));
}
td::Result> create_proof(BlockIdExt masterchain_block_id, td::BufferSlice proof) {
  return Ref{true, masterchain_block_id, std::move(proof)};
}
td::Result> create_proof_link(BlockIdExt block_id, td::BufferSlice proof_link) {
  return Ref{true, block_id, std::move(proof_link)};
}
td::Result> create_signature_set(td::BufferSlice sig_set) {
  return BlockSignatureSetQ::fetch(std::move(sig_set));
}
td::Result> create_shard_state(BlockIdExt block_id, td::BufferSlice data) {
  auto res = ShardStateQ::fetch(block_id, std::move(data));
  if (res.is_error()) {
    return res.move_as_error();
  } else {
    return td::Ref{res.move_as_ok()};
  }
}
td::Result> create_shard_state(BlockIdExt block_id, td::Ref root_cell) {
  auto res = ShardStateQ::fetch(block_id, {}, std::move(root_cell));
  if (res.is_error()) {
    return res.move_as_error();
  } else {
    return td::Ref{res.move_as_ok()};
  }
}
td::Result create_block_handle(td::BufferSlice data) {
  return ton::validator::BlockHandleImpl::create(data.as_slice());
}
td::Result create_block_handle(td::Slice data) {
  return ton::validator::BlockHandleImpl::create(data);
}
td::Result create_temp_block_handle(td::BufferSlice data) {
  return ton::validator::BlockHandleImpl::create(std::move(data));
}
BlockHandle create_empty_block_handle(BlockIdExt id) {
  return ton::validator::BlockHandleImpl::create_empty(id);
}
td::Ref create_signature_set(std::vector sig_set) {
  return td::Ref{true, std::move(sig_set)};
}
td::Result> create_ext_message(td::BufferSlice data) {
  TRY_RESULT(res, ExtMessageQ::create_ext_message(std::move(data)));
  return std::move(res);
}
void run_check_external_message(td::BufferSlice data, td::actor::ActorId manager, td::Promise promise) {
  ExtMessageQ::run_message(std::move(data), std::move(manager), std::move(promise));
}
td::Result> create_ihr_message(td::BufferSlice data) {
  TRY_RESULT(res, IhrMessageQ::create_ihr_message(std::move(data)));
  return std::move(res);
}
void run_accept_block_query(BlockIdExt id, td::Ref data, std::vector prev,
                            td::Ref validator_set, td::Ref signatures,
                            td::Ref approve_signatures, bool send_broadcast, bool apply,
                            td::actor::ActorId manager, td::Promise promise) {
  td::actor::create_actor("accept", id, std::move(data), prev, std::move(validator_set),
                                            std::move(signatures), std::move(approve_signatures), send_broadcast, apply,
                                            manager, std::move(promise))
      .release();
}
void run_fake_accept_block_query(BlockIdExt id, td::Ref data, std::vector prev,
                                 td::Ref validator_set, td::actor::ActorId manager,
                                 td::Promise promise) {
  td::actor::create_actor("fakeaccept", AcceptBlockQuery::IsFake(), id, std::move(data),
                                            std::move(prev), std::move(validator_set), std::move(manager),
                                            std::move(promise))
      .release();
}
void run_hardfork_accept_block_query(BlockIdExt id, td::Ref data,
                                     td::actor::ActorId manager, td::Promise promise) {
  td::actor::create_actor("fork/accept", AcceptBlockQuery::ForceFork(), id, std::move(data),
                                            std::move(manager), std::move(promise))
      .release();
}
void run_apply_block_query(BlockIdExt id, td::Ref block, BlockIdExt masterchain_block_id,
                           td::actor::ActorId manager, td::Timestamp timeout,
                           td::Promise promise) {
  td::actor::create_actor(PSTRING() << "apply " << id, id, std::move(block), masterchain_block_id, manager,
                                      timeout, std::move(promise))
      .release();
}
void run_check_proof_query(BlockIdExt id, td::Ref proof, td::actor::ActorId manager,
                           td::Timestamp timeout, td::Promise promise, bool skip_check_signatures) {
  td::actor::create_actor("checkproof", id, std::move(proof), manager, timeout, std::move(promise),
                                      skip_check_signatures)
      .release();
}
void run_check_proof_query(BlockIdExt id, td::Ref proof, td::actor::ActorId manager,
                           td::Timestamp timeout, td::Promise promise,
                           td::Ref rel_key_block_proof, bool skip_check_signatures) {
  td::actor::create_actor("checkproof/key", id, std::move(proof), manager, timeout, std::move(promise),
                                      skip_check_signatures, std::move(rel_key_block_proof))
      .release();
}
void run_check_proof_query(BlockIdExt id, td::Ref proof, td::actor::ActorId manager,
                           td::Timestamp timeout, td::Promise promise,
                           td::Ref rel_mc_state, bool skip_check_signatures) {
  td::actor::create_actor("checkproof/st", id, std::move(proof), manager, timeout, std::move(promise),
                                      skip_check_signatures, std::move(rel_mc_state))
      .release();
}
void run_check_proof_link_query(BlockIdExt id, td::Ref proof, td::actor::ActorId manager,
                                td::Timestamp timeout, td::Promise promise) {
  td::actor::create_actor("checkprooflink", id, std::move(proof), manager, timeout, std::move(promise))
      .release();
}
void run_validate_query(ShardIdFull shard, BlockIdExt min_masterchain_block_id,
                        std::vector prev, BlockCandidate candidate, td::Ref validator_set,
                        td::actor::ActorId manager, td::Timestamp timeout,
                        td::Promise promise, unsigned mode) {
  BlockSeqno seqno = 0;
  for (auto& p : prev) {
    if (p.seqno() > seqno) {
      seqno = p.seqno();
    }
  }
  bool is_fake = mode & ValidateMode::fake;
  td::actor::create_actor(
      PSTRING() << (is_fake ? "fakevalidate" : "validateblock") << shard.to_str() << ":" << (seqno + 1), shard,
      min_masterchain_block_id, std::move(prev), std::move(candidate), std::move(validator_set), std::move(manager),
      timeout, std::move(promise), mode)
      .release();
}
void run_collate_query(ShardIdFull shard, const BlockIdExt& min_masterchain_block_id, std::vector prev,
                       Ed25519_PublicKey collator_id, td::Ref validator_set,
                       td::actor::ActorId manager, td::Timestamp timeout,
                       td::Promise promise) {
  BlockSeqno seqno = 0;
  for (auto& p : prev) {
    if (p.seqno() > seqno) {
      seqno = p.seqno();
    }
  }
  td::actor::create_actor(PSTRING() << "collate" << shard.to_str() << ":" << (seqno + 1), shard, false,
                                    min_masterchain_block_id, std::move(prev), std::move(validator_set), collator_id,
                                    std::move(manager), timeout, std::move(promise))
      .release();
}
void run_collate_hardfork(ShardIdFull shard, const BlockIdExt& min_masterchain_block_id, std::vector prev,
                          td::actor::ActorId manager, td::Timestamp timeout,
                          td::Promise promise) {
  BlockSeqno seqno = 0;
  for (auto& p : prev) {
    if (p.seqno() > seqno) {
      seqno = p.seqno();
    }
  }
  td::actor::create_actor(PSTRING() << "collate" << shard.to_str() << ":" << (seqno + 1), shard, true,
                                    min_masterchain_block_id, std::move(prev), td::Ref{},
                                    Ed25519_PublicKey{Bits256::zero()}, std::move(manager), timeout, std::move(promise))
      .release();
}
void run_liteserver_query(td::BufferSlice data, td::actor::ActorId manager,
                          td::actor::ActorId cache, td::Promise promise) {
  LiteQuery::run_query(std::move(data), std::move(manager), std::move(promise));
}
void run_fetch_account_state(WorkchainId wc, StdSmcAddress  addr, td::actor::ActorId manager,
                             td::Promise,UnixTime,LogicalTime,std::unique_ptr>> promise) {
  LiteQuery::fetch_account_state(wc, addr, std::move(manager), std::move(promise));
}
void run_validate_shard_block_description(td::BufferSlice data, BlockHandle masterchain_block,
                                          td::Ref masterchain_state,
                                          td::actor::ActorId manager, td::Timestamp timeout,
                                          td::Promise> promise, bool is_fake) {
  auto id = masterchain_block->id();
  td::actor::create_actor("topshardfetch", std::move(data), id,
                                                      std::move(masterchain_block), std::move(masterchain_state),
                                                      manager, timeout, is_fake, std::move(promise))
      .release();
}
}  // namespace validator
}  // namespace ton