mirror of
				https://github.com/ton-blockchain/ton
				synced 2025-03-09 15:40:10 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			147 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			147 lines
		
	
	
	
		
			5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/* 
 | 
						|
    This file is part of TON Blockchain source code.
 | 
						|
 | 
						|
    TON Blockchain is free software; you can redistribute it and/or
 | 
						|
    modify it under the terms of the GNU 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 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 General Public License for more details.
 | 
						|
 | 
						|
    You should have received a copy of the GNU General Public License
 | 
						|
    along with TON Blockchain.  If not, see <http://www.gnu.org/licenses/>.
 | 
						|
 | 
						|
    In addition, as a special exception, the copyright holders give permission 
 | 
						|
    to link the code of portions of this program with the OpenSSL library. 
 | 
						|
    You must obey the GNU General Public License in all respects for all 
 | 
						|
    of the code used other than OpenSSL. If you modify file(s) with this 
 | 
						|
    exception, you may extend this exception to your version of the file(s), 
 | 
						|
    but you are not obligated to do so. If you do not wish to do so, delete this 
 | 
						|
    exception statement from your version. If you delete this exception statement 
 | 
						|
    from all source files in the program, then also delete it here.
 | 
						|
 | 
						|
    Copyright 2017-2020 Telegram Systems LLP
 | 
						|
*/
 | 
						|
#include "td/actor/actor.h"
 | 
						|
 | 
						|
#include "td/utils/OptionParser.h"
 | 
						|
#include "td/utils/Observer.h"
 | 
						|
#include "td/utils/port/UdpSocketFd.h"
 | 
						|
 | 
						|
#include "td/net/UdpServer.h"
 | 
						|
 | 
						|
// PingPong
 | 
						|
class PingPong : public td::actor::Actor {
 | 
						|
 public:
 | 
						|
  PingPong(int port, td::IPAddress dest, bool use_tcp) : port_(port), dest_(std::move(dest)), use_tcp_(use_tcp) {
 | 
						|
  }
 | 
						|
 | 
						|
 private:
 | 
						|
  int port_;
 | 
						|
  td::actor::ActorOwn<td::UdpServer> udp_server_;
 | 
						|
  td::IPAddress dest_;
 | 
						|
  bool is_closing_{false};
 | 
						|
  bool use_tcp_{false};
 | 
						|
 | 
						|
  void start_up() override {
 | 
						|
    class Callback : public td::UdpServer::Callback {
 | 
						|
     public:
 | 
						|
      Callback(td::actor::ActorShared<PingPong> ping_pong) : ping_pong_(std::move(ping_pong)) {
 | 
						|
      }
 | 
						|
 | 
						|
     private:
 | 
						|
      td::actor::ActorShared<PingPong> ping_pong_;
 | 
						|
      void on_udp_message(td::UdpMessage udp_message) override {
 | 
						|
        send_closure(ping_pong_, &PingPong::on_udp_message, std::move(udp_message));
 | 
						|
      }
 | 
						|
    };
 | 
						|
 | 
						|
    if (use_tcp_) {
 | 
						|
      udp_server_ = td::UdpServer::create_via_tcp(PSLICE() << "UdpServer(via tcp) " << td::tag("port", port_), port_,
 | 
						|
                                                  std::make_unique<Callback>(actor_shared(this)))
 | 
						|
                        .move_as_ok();
 | 
						|
    } else {
 | 
						|
      udp_server_ = td::UdpServer::create(PSLICE() << "UdpServer " << td::tag("port", port_), port_,
 | 
						|
                                          std::make_unique<Callback>(actor_shared(this)))
 | 
						|
                        .move_as_ok();
 | 
						|
    }
 | 
						|
 | 
						|
    alarm_timestamp() = td::Timestamp::now();
 | 
						|
  }
 | 
						|
 | 
						|
  void on_udp_message(td::UdpMessage message) {
 | 
						|
    if (is_closing_) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    if (message.error.is_error()) {
 | 
						|
      LOG(ERROR) << "Got error " << message.error << " from " << message.address;
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    auto data_slice = message.data.as_slice();
 | 
						|
    LOG(INFO) << "Got query " << td::format::escaped(data_slice) << " from " << message.address;
 | 
						|
    if (data_slice[5] == 'i') {
 | 
						|
      send_closure(udp_server_, &td::UdpServer::send,
 | 
						|
                   td::UdpMessage{message.address, td::BufferSlice("magkpongpong"), {}});
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  int cnt_{0};
 | 
						|
  void alarm() override {
 | 
						|
    if (++cnt_ > 10) {
 | 
						|
      return close();
 | 
						|
    }
 | 
						|
    alarm_timestamp() = td::Timestamp::in(1);
 | 
						|
    LOG(INFO) << "Send ping";
 | 
						|
    send_closure(udp_server_, &td::UdpServer::send, td::UdpMessage{dest_, td::BufferSlice("magkpingping"), {}});
 | 
						|
  }
 | 
						|
 | 
						|
  void close() {
 | 
						|
    is_closing_ = true;
 | 
						|
    udp_server_.reset();
 | 
						|
  }
 | 
						|
  void hangup_shared() override {
 | 
						|
    // udp_server_ was_closed
 | 
						|
    stop();
 | 
						|
  }
 | 
						|
  void tear_down() override {
 | 
						|
    td::actor::SchedulerContext::get()->stop();
 | 
						|
  }
 | 
						|
};
 | 
						|
 | 
						|
int main(int argc, char *argv[]) {
 | 
						|
  td::OptionParser options_parser;
 | 
						|
  options_parser.set_description("Udp ping server/client (8083 <-> 8084) (based on td::actors2)");
 | 
						|
 | 
						|
  int from_port = 8083;
 | 
						|
  int to_port = 8084;
 | 
						|
  bool is_client = false;
 | 
						|
  bool use_tcp = false;
 | 
						|
  options_parser.add_option('c', "client", "Work as client (server by default)", [&]() { is_client = true; });
 | 
						|
  options_parser.add_option('t', "tcp", "Use tcp (udp by default)", [&]() { use_tcp = true; });
 | 
						|
  auto status = options_parser.run(argc, argv);
 | 
						|
  if (status.is_error()) {
 | 
						|
    LOG(ERROR) << status.error();
 | 
						|
    LOG(INFO) << options_parser;
 | 
						|
    return 1;
 | 
						|
  }
 | 
						|
  if (is_client) {
 | 
						|
    std::swap(from_port, to_port);
 | 
						|
  }
 | 
						|
  td::IPAddress to_ip;
 | 
						|
  to_ip.init_ipv4_port("127.0.0.1", to_port).ensure();
 | 
						|
 | 
						|
  // NB: Interface will be changed
 | 
						|
  td::actor::Scheduler scheduler({2});
 | 
						|
  LOG(INFO) << "Listen to " << from_port;
 | 
						|
  scheduler.run_in_context([&] {
 | 
						|
    td::actor::create_actor<PingPong>(td::actor::ActorOptions().with_name("TcpClient").with_poll(), from_port, to_ip,
 | 
						|
                                      use_tcp)
 | 
						|
        .release();
 | 
						|
  });
 | 
						|
  scheduler.run();
 | 
						|
  return 0;
 | 
						|
}
 |