/*
    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-2019 Telegram Systems LLP
*/
#pragma once
#include "interfaces/validator-manager.h"
namespace ton {
namespace validator {
class WaitBlockState : public td::actor::Actor {
 public:
  WaitBlockState(BlockHandle handle, td::uint32 priority, td::actor::ActorId manager,
                 td::Timestamp timeout, td::Promise> promise)
      : handle_(std::move(handle))
      , priority_(priority)
      , manager_(manager)
      , timeout_(timeout)
      , promise_(std::move(promise)) {
  }
  void abort_query(td::Status reason);
  void finish_query();
  void alarm() override;
  void force_read_from_db();
  void start_up() override;
  void got_block_handle(BlockHandle handle);
  void start();
  void got_state_from_db(td::Ref data);
  void got_state_from_static_file(td::Ref state, td::BufferSlice data);
  void failed_to_get_state_from_db(td::Status reason);
  void got_prev_state(td::Ref state);
  void failed_to_get_prev_state(td::Status reason);
  void got_block_data(td::Ref data);
  void failed_to_get_block_data(td::Status reason);
  void got_state_from_net(td::BufferSlice data);
  void failed_to_get_zero_state();
  void failed_to_get_state_from_net(td::Status reason);
  void apply();
  void written_state(td::Ref upd_state);
  void written_state_file();
  void update_timeout(td::Timestamp timeout, td::uint32 priority) {
    timeout_ = timeout;
    alarm_timestamp() = timeout_;
    priority_ = priority;
  }
 private:
  BlockHandle handle_;
  td::uint32 priority_;
  td::actor::ActorId manager_;
  td::Timestamp timeout_;
  td::Promise> promise_;
  td::Ref prev_state_;
  td::Ref block_;
  bool reading_from_db_ = false;
  td::Timestamp next_static_file_attempt_;
  //td::PerfWarningTimer perf_timer_{"waitstate", 1.0};
};
}  // namespace validator
}  // namespace ton