mirror of
				https://github.com/ton-blockchain/ton
				synced 2025-03-09 15:40:10 +00:00 
			
		
		
		
	* Rename chunk to piece in MerkleTree for consistency * Refactor PeerManager * Make PeerState thread-safe * Download torrent by hash * First version of storage daemon * Download torrents partially * Improve storing and loading torrent state in DB * Rewrite MerkleTree * "Remove torrent" in storage daemon * Process errors, fix bugs in storage * Move TonlibClientWrapper from rldp-http-proxy to tonlib * Initial version of storage provider * Move interaction with contracts to smc-util * Improve TonlibClientWrapper interface * Various improvements in storage provider * Fix TorrentCreator.cpp * Improve interface for partial download * Client mode in storage-daemon * Improve interface of storage-daemon-cli * Fix calculating speed, show peers in storage-daemon * Use permanent adnl id in storage daemon * Fix sending large "storage.addUpdate" messages * Improve printing torrents in cli * Update tlo * Fix RldpSender::on_ack * Update storage provider * Add "address" parameter to get-provider-params * Allow client to close storage contract * Limit torrent description * Add more logs to storage provider * smc.forget tonlib method * Use smc.forget in storage daemon * Optimize sending messages in smc-util.cpp * Fix verbosity, remove excessive logs * Json output in storage-daemon-cli * Update storage provider contracts * Fix rldp2 acks * Change verbosity of logs in rldp2 * Update help and output of commands and in storage-daemon-cli Co-authored-by: SpyCheese <mikle98@yandex.ru>
		
			
				
	
	
		
			298 lines
		
	
	
	
		
			8.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			298 lines
		
	
	
	
		
			8.3 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
 | 
						|
*/
 | 
						|
 | 
						|
#include "PeerState.h"
 | 
						|
#include "Bitset.h"
 | 
						|
 | 
						|
#include "td/utils/Random.h"
 | 
						|
#include "td/utils/Status.h"
 | 
						|
 | 
						|
namespace ton {
 | 
						|
struct PartsHelper {
 | 
						|
 public:
 | 
						|
  explicit PartsHelper(size_t parts_count = 0) : parts_(parts_count), peers_(64) {
 | 
						|
    peers_[0].is_valid = true;
 | 
						|
  }
 | 
						|
  using PartId = size_t;
 | 
						|
  using PeerToken = size_t;
 | 
						|
 | 
						|
  void init_parts_count(size_t parts_count) {
 | 
						|
    CHECK(parts_.empty());
 | 
						|
    parts_.resize(parts_count);
 | 
						|
  }
 | 
						|
  PeerToken register_self() {
 | 
						|
    return self_token_;
 | 
						|
  }
 | 
						|
  PeerToken register_peer(PeerId peer_id) {
 | 
						|
    PeerToken new_peer_token{free_peer_tokens_.empty() ? next_peer_token_ : free_peer_tokens_.back()};
 | 
						|
    auto it = peer_id_to_token_.emplace(peer_id, new_peer_token);
 | 
						|
    if (it.second) {
 | 
						|
      if (free_peer_tokens_.empty()) {
 | 
						|
        next_peer_token_++;
 | 
						|
        peers_.resize(next_peer_token_);
 | 
						|
      } else {
 | 
						|
        free_peer_tokens_.pop_back();
 | 
						|
      }
 | 
						|
      auto peer = get_peer(new_peer_token, true);
 | 
						|
      peer->is_valid = true;
 | 
						|
      peer->peer_id = peer_id;
 | 
						|
      peer->want_download_count = 0;
 | 
						|
    }
 | 
						|
    return it.first->second;
 | 
						|
  }
 | 
						|
  void forget_peer(PeerToken peer_token) {
 | 
						|
    CHECK(peer_token != self_token_);
 | 
						|
    free_peer_tokens_.push_back(peer_token);
 | 
						|
    auto *peer = get_peer(peer_token);
 | 
						|
    peer_id_to_token_.erase(get_peer(peer_token)->peer_id);
 | 
						|
    *peer = Peer{};
 | 
						|
    peer->rnd = td::Random::fast_uint32();
 | 
						|
  }
 | 
						|
 | 
						|
  void set_peer_limit(PeerToken peer, td::uint32 limit) {
 | 
						|
    get_peer(peer)->limit = limit;
 | 
						|
  }
 | 
						|
 | 
						|
  void on_peer_part_ready(PeerToken peer_token, PartId part_id) {
 | 
						|
    auto peer = get_peer(peer_token);
 | 
						|
    if (!peer->ready_parts.set_one(part_id)) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    auto part = get_part(part_id);
 | 
						|
    if (part->is_ready) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    peer->want_download_count++;
 | 
						|
    if (!part->rnd) {
 | 
						|
      part->rnd = td::Random::fast_uint32();
 | 
						|
    }
 | 
						|
    change_key(part_id, part->rnd, part->peers_count, part->peers_count + 1, part->priority, part->priority);
 | 
						|
    part->peers_count++;
 | 
						|
  }
 | 
						|
 | 
						|
  void lock_part(PartId part_id) {
 | 
						|
    auto *part = get_part(part_id);
 | 
						|
    CHECK(!part->is_locked);
 | 
						|
    part->is_locked = true;
 | 
						|
  }
 | 
						|
  void unlock_part(PartId part_id) {
 | 
						|
    auto *part = get_part(part_id);
 | 
						|
    CHECK(part->is_locked);
 | 
						|
    part->is_locked = false;
 | 
						|
  }
 | 
						|
 | 
						|
  void set_part_priority(PartId part_id, td::uint8 priority) {
 | 
						|
    auto *part = get_part(part_id);
 | 
						|
    if (part->is_ready) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    change_key(part_id, part->rnd, part->peers_count, part->peers_count, part->priority, priority);
 | 
						|
    part->priority = priority;
 | 
						|
  }
 | 
						|
 | 
						|
  td::uint8 get_part_priority(PartId part_id) {
 | 
						|
    auto *part = get_part(part_id);
 | 
						|
    return part->priority;
 | 
						|
  }
 | 
						|
 | 
						|
  void on_self_part_ready(PartId part_id) {
 | 
						|
    auto peer = get_peer(self_token_);
 | 
						|
    if (!peer->ready_parts.set_one(part_id)) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    auto part = get_part(part_id);
 | 
						|
    CHECK(!part->is_ready);
 | 
						|
    part->is_ready = true;
 | 
						|
    for (auto &peer : peers_) {
 | 
						|
      if (peer.ready_parts.get(part_id)) {
 | 
						|
        peer.want_download_count--;
 | 
						|
      }
 | 
						|
    }
 | 
						|
    change_key(part_id, part->rnd, part->peers_count, 0, part->priority, part->priority);
 | 
						|
  }
 | 
						|
 | 
						|
  void on_self_part_not_ready(PartId part_id) {
 | 
						|
    auto peer = get_peer(self_token_);
 | 
						|
    if (!peer->ready_parts.set_zero(part_id)) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    auto part = get_part(part_id);
 | 
						|
    CHECK(part->is_ready);
 | 
						|
    part->is_ready = false;
 | 
						|
    for (auto &peer : peers_) {
 | 
						|
      if (peer.ready_parts.get(part_id)) {
 | 
						|
        peer.want_download_count++;
 | 
						|
      }
 | 
						|
    }
 | 
						|
    change_key(part_id, part->rnd, 0, part->peers_count, part->priority, part->priority);
 | 
						|
  }
 | 
						|
 | 
						|
  struct RarePart {
 | 
						|
    PartId part_id;
 | 
						|
    PeerId peer_id;
 | 
						|
  };
 | 
						|
  std::vector<RarePart> get_rarest_parts(size_t max_count) {
 | 
						|
    struct It {
 | 
						|
      std::set<Peer::Key>::iterator begin, end;
 | 
						|
      PeerId peer_id;
 | 
						|
      td::uint32 limit{0};
 | 
						|
      bool empty() const {
 | 
						|
        return begin == end || limit == 0;
 | 
						|
      }
 | 
						|
      auto key() const {
 | 
						|
        return std::tie(*begin, peer_id);
 | 
						|
      }
 | 
						|
      bool operator<(const It &other) const {
 | 
						|
        return key() < other.key();
 | 
						|
      }
 | 
						|
    };
 | 
						|
    std::set<It> its;
 | 
						|
 | 
						|
    for (auto &peer : peers_) {
 | 
						|
      if (!peer.is_valid || peer.limit == 0) {
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
      It it;
 | 
						|
      it.begin = peer.rarest_parts.begin();
 | 
						|
      it.end = peer.rarest_parts.end();
 | 
						|
      it.peer_id = peer.peer_id;
 | 
						|
      it.limit = peer.limit;
 | 
						|
      if (it.empty()) {
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
      its.insert(it);
 | 
						|
    }
 | 
						|
 | 
						|
    std::vector<RarePart> res;
 | 
						|
    while (res.size() < max_count && !its.empty()) {
 | 
						|
      auto it = *its.begin();
 | 
						|
      its.erase(its.begin());
 | 
						|
      auto part_id = it.begin->part_id;
 | 
						|
      if ((res.empty() || res.back().part_id != part_id) && !get_part(part_id)->is_locked) {
 | 
						|
        res.push_back({part_id, it.peer_id});
 | 
						|
        CHECK(get_peer(register_peer(it.peer_id))->ready_parts.get(part_id));
 | 
						|
        it.limit--;
 | 
						|
      }
 | 
						|
      it.begin++;
 | 
						|
      if (it.empty()) {
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
      its.insert(it);
 | 
						|
    }
 | 
						|
    return res;
 | 
						|
  }
 | 
						|
 | 
						|
  td::uint32 get_want_download_count(PeerToken peer_token) {
 | 
						|
    return get_peer(peer_token, false)->want_download_count;
 | 
						|
  }
 | 
						|
  const td::Bitset &get_ready_parts(PeerToken peer_token) {
 | 
						|
    return get_peer(peer_token, false)->ready_parts;
 | 
						|
  }
 | 
						|
 | 
						|
 private:
 | 
						|
  PeerToken self_token_{0};
 | 
						|
  size_t parts_count_;
 | 
						|
  struct Part {
 | 
						|
    bool is_locked{false};
 | 
						|
    bool is_ready{false};
 | 
						|
    td::uint8 priority{1};
 | 
						|
    td::uint32 rnd{0};
 | 
						|
    td::uint32 peers_count{0};  // invalid for ready parts
 | 
						|
  };
 | 
						|
  struct Peer {
 | 
						|
    PeerId peer_id{0};
 | 
						|
    bool is_valid{false};
 | 
						|
    td::uint32 rnd{0};
 | 
						|
    td::uint32 limit{0};
 | 
						|
    td::Bitset ready_parts;
 | 
						|
    // sum_i (peer.ready_parts[i] && !node.ready_parts[i])
 | 
						|
    td::uint32 want_download_count{0};
 | 
						|
 | 
						|
    // peers_count - count of peers which has this part
 | 
						|
    // key_count = !is_ready * peers_count;
 | 
						|
    struct Key {
 | 
						|
      td::uint8 priority{0};
 | 
						|
      td::uint32 count{0};
 | 
						|
      PartId part_id{0};
 | 
						|
      td::uint32 rnd{0};
 | 
						|
      auto key() const {
 | 
						|
        return std::make_tuple(255 - priority, count, rnd, part_id);
 | 
						|
      }
 | 
						|
      bool operator<(const Key &other) const {
 | 
						|
        return key() < other.key();
 | 
						|
      }
 | 
						|
    };
 | 
						|
    std::set<Key> rarest_parts;  // TODO: use vector instead of set
 | 
						|
  };
 | 
						|
 | 
						|
  std::vector<Part> parts_;
 | 
						|
  std::vector<Peer> peers_;
 | 
						|
  td::uint32 next_peer_token_{1};
 | 
						|
  std::map<PeerId, PeerToken> peer_id_to_token_;
 | 
						|
  std::vector<PeerToken> free_peer_tokens_;
 | 
						|
 | 
						|
  Part *get_part(PartId part_id) {
 | 
						|
    CHECK(part_id < parts_.size());
 | 
						|
    return &parts_[part_id];
 | 
						|
  }
 | 
						|
  Peer *get_peer(PeerToken peer_token, bool can_be_uninited = false) {
 | 
						|
    CHECK(peer_token < peers_.size());
 | 
						|
    auto res = &peers_[peer_token];
 | 
						|
    CHECK(res->is_valid || can_be_uninited);
 | 
						|
    return res;
 | 
						|
  }
 | 
						|
 | 
						|
  void change_key(PartId part_id, td::uint32 rnd, td::uint32 from_count, td::uint32 to_count, td::uint8 from_priority,
 | 
						|
                  td::uint8 to_priority) {
 | 
						|
    if (from_count == 0 && to_count == 0) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    if (from_count == to_count && from_priority == to_priority) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    for (auto &peer : peers_) {
 | 
						|
      if (!peer.is_valid) {
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
      if (peer.peer_id == 0) {
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
      if (!peer.ready_parts.get(part_id)) {
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
      //TODO: xor is not a perfect solution as it keeps a lot order between part_ids
 | 
						|
      Peer::Key key;
 | 
						|
      key.part_id = part_id;
 | 
						|
      key.rnd = rnd ^ peer.rnd;
 | 
						|
 | 
						|
      if (from_count != 0 && from_priority != 0) {
 | 
						|
        key.count = from_count;
 | 
						|
        key.priority = from_priority;
 | 
						|
        peer.rarest_parts.erase(key);
 | 
						|
      }
 | 
						|
 | 
						|
      if (to_count != 0 && to_priority != 0) {
 | 
						|
        key.count = to_count;
 | 
						|
        key.priority = to_priority;
 | 
						|
        peer.rarest_parts.insert(key);
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
};
 | 
						|
}  // namespace ton
 |