-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhole_punching.cpp
119 lines (100 loc) · 2.96 KB
/
hole_punching.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
#include <array>
#include <atomic>
#include <cstdlib>
#include <functional>
#include <iostream>
#include <memory>
#include <optional>
#include <random>
#include <string>
#include <utility>
#include <boost/asio.hpp>
#include <boost/asio/write.hpp>
#include <cereal/types/string.hpp>
#include <cppless/dispatcher/aws-lambda.hpp>
#include <cppless/dispatcher/common.hpp>
#include <cppless/utils/tcpunch.hpp>
namespace asio = boost::asio;
auto exchange_number(std::string host_name, std::string connection_key) -> int
{
asio::io_service ioc;
cppless::tcpunch::client tcpunch(ioc);
asio::ip::tcp::resolver resolver(ioc);
long long seed = std::chrono::system_clock::now().time_since_epoch().count();
std::mt19937 rng(seed);
std::uniform_int_distribution<> dist(0, 256);
auto random_number = dist(rng);
auto other_number = 0;
int x = 0;
int sum = 0;
auto handle_update = [&](asio::ip::tcp::socket& socket)
{
x++;
if (x >= 2) {
sum = other_number + random_number;
socket.close();
}
};
auto handle_connect =
[&](const boost::system::error_code& ec, asio::ip::tcp::socket& socket)
{
auto handle_send =
[&](const boost::system::error_code& ec, unsigned int /*num_bytes*/)
{
if (ec.failed()) {
std::cout << ec.what() << std::endl;
socket.close();
return;
}
handle_update(socket);
};
auto handle_rcv =
[&](const boost::system::error_code& ec, unsigned int /*num_bytes*/)
{
if (ec.failed()) {
std::cout << ec.what() << std::endl;
socket.close();
return;
}
handle_update(socket);
};
asio::async_write(socket,
asio::buffer(&random_number, sizeof(random_number)),
handle_send);
asio::async_read(
socket, asio::buffer(&other_number, sizeof(other_number)), handle_rcv);
};
auto handle_resolve =
[&](const boost::system::error_code& ec,
const asio::ip::tcp::resolver::results_type& endpoint)
{
if (ec.failed()) {
std::cout << ec.what() << std::endl;
return;
}
tcpunch.async_connect(endpoint, connection_key, handle_connect);
};
resolver.async_resolve(host_name, "10000", handle_resolve);
ioc.run();
return sum;
}
using dispatcher = cppless::aws_lambda_beast_dispatcher<>::from_env;
using task = cppless::task<dispatcher>;
auto main() -> int
{
dispatcher aws;
auto instance = aws.create_instance();
cppless::shared_future<int> t0_result;
cppless::shared_future<int> t1_result;
std::string host_name =
"ec2-18-185-121-125.eu-central-1.compute.amazonaws.com";
std::string cnnectioon_key = "connection_key";
task::sendable t0 = [=]()
{ return exchange_number(host_name, cnnectioon_key); };
instance.dispatch(t0, t0_result, {});
instance.dispatch(t0, t1_result, {});
instance.wait_one();
instance.wait_one();
std::cout << t0_result.value() << std::endl;
std::cout << t1_result.value() << std::endl;
}