Monero
net_load_tests.h
Go to the documentation of this file.
1 // Copyright (c) 2014-2022, The Monero Project
2 //
3 // All rights reserved.
4 //
5 // Redistribution and use in source and binary forms, with or without modification, are
6 // permitted provided that the following conditions are met:
7 //
8 // 1. Redistributions of source code must retain the above copyright notice, this list of
9 // conditions and the following disclaimer.
10 //
11 // 2. Redistributions in binary form must reproduce the above copyright notice, this list
12 // of conditions and the following disclaimer in the documentation and/or other
13 // materials provided with the distribution.
14 //
15 // 3. Neither the name of the copyright holder nor the names of its contributors may be
16 // used to endorse or promote products derived from this software without specific
17 // prior written permission.
18 //
19 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
20 // EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
21 // MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
22 // THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
24 // PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
25 // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
26 // STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
27 // THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 //
29 // Parts of this file are originally copyright (c) 2012-2013 The Cryptonote developers
30 
31 #pragma once
32 
33 #include <atomic>
34 
35 #include <boost/uuid/uuid_io.hpp>
36 
37 #include "include_base_utils.h"
38 #include "string_tools.h"
42 
43 #include "../unit_tests/unit_tests_utils.h"
44 
45 namespace net_load_tests
46 {
48  {
49  test_connection_context(): epee::net_utils::connection_context_base(boost::uuids::nil_uuid(), {}, false, false), m_closed(false) {}
50  static constexpr int handshake_command() noexcept { return 1001; }
51  static constexpr bool handshake_complete() noexcept { return true; }
52  size_t get_max_bytes(int command) const { return LEVIN_DEFAULT_MAX_PACKET_SIZE; }
53  volatile bool m_closed;
54  };
55 
60 
61  struct test_levin_commands_handler : public epee::levin::levin_commands_handler<test_connection_context>
62  {
64  //: m_return_code(LEVIN_OK)
65  //, m_last_command(-1)
66  {
67  }
68 
69  virtual int invoke(int command, const epee::span<const uint8_t> in_buff, epee::byte_stream& buff_out, test_connection_context& context)
70  {
71  //m_invoke_counter.inc();
72  //std::unique_lock<std::mutex> lock(m_mutex);
73  //m_last_command = command;
74  //m_last_in_buf = in_buff;
75  //buff_out = m_invoke_out_buf;
76  //return m_return_code;
77  return LEVIN_OK;
78  }
79 
80  virtual int notify(int command, const epee::span<const uint8_t> in_buff, test_connection_context& context)
81  {
82  //m_notify_counter.inc();
83  //std::unique_lock<std::mutex> lock(m_mutex);
84  //m_last_command = command;
85  //m_last_in_buf = in_buff;
86  //return m_return_code;
87  return LEVIN_OK;
88  }
89 
91  {
92  //m_callback_counter.inc();
93  //std::cout << "test_levin_commands_handler::callback()" << std::endl;
94  }
95 
97  {
99  //std::cout << "test_levin_commands_handler::on_connection_new()" << std::endl;
100  }
101 
103  {
105  //std::cout << "test_levin_commands_handler::on_connection_close()" << std::endl;
106  }
107 
108  //size_t invoke_counter() const { return m_invoke_counter.get(); }
109  //size_t notify_counter() const { return m_notify_counter.get(); }
110  //size_t callback_counter() const { return m_callback_counter.get(); }
113 
114  //int return_code() const { return m_return_code; }
115  //void return_code(int v) { m_return_code = v; }
116 
117  //const std::string& invoke_out_buf() const { return m_invoke_out_buf; }
118  //void invoke_out_buf(const std::string& v) { m_invoke_out_buf = v; }
119 
120  //int last_command() const { return m_last_command; }
121  //const std::string& last_in_buf() const { return m_last_in_buf; }
122 
123  protected:
124  //unit_test::call_counter m_invoke_counter;
125  //unit_test::call_counter m_notify_counter;
126  //unit_test::call_counter m_callback_counter;
129 
130  //std::mutex m_mutex;
131 
132  //int m_return_code;
133  //std::string m_invoke_out_buf;
134 
135  //int m_last_command;
136  //std::string m_last_in_buf;
137  };
138 
140  {
141  public:
142  open_close_test_helper(test_tcp_server& tcp_server, size_t open_request_target, size_t max_opened_connection_count)
143  : m_tcp_server(tcp_server)
144  , m_max_opened_connection_count(max_opened_connection_count)
148  , m_connections(open_request_target)
149  {
150  for (auto& conn_id : m_connections)
151  conn_id = boost::uuids::nil_uuid();
152  }
153 
154  bool handle_new_connection(const boost::uuids::uuid& connection_id, bool ignore_close_fails = false)
155  {
156  size_t idx = m_next_opened_conn_idx.fetch_add(1, std::memory_order_relaxed);
157  if (idx >= m_connections.size())
158  {
159  LOG_PRINT_L0("ERROR: connections overflow");
160  exit(1);
161  }
162  m_connections[idx] = connection_id;
163 
164  size_t prev_connection_count = m_opened_connection_count.fetch_add(1, std::memory_order_relaxed);
165  if (m_max_opened_connection_count <= prev_connection_count)
166  {
167  return close_next_connection(ignore_close_fails);
168  }
169 
170  return true;
171  }
172 
174  {
175  while (close_next_connection(false));
176  }
177 
178  bool close_next_connection(bool ignore_close_fails)
179  {
180  size_t idx = m_next_closed_conn_idx.fetch_add(1, std::memory_order_relaxed);
181  if (m_next_opened_conn_idx.load(std::memory_order_relaxed) <= idx)
182  {
183  LOG_PRINT_L0("Not enough opened connections");
184  return false;
185  }
186  if (m_connections[idx].is_nil())
187  {
188  LOG_PRINT_L0("Connection isn't opened");
189  return false;
190  }
191  if (!m_tcp_server.get_config_object().close(m_connections[idx]))
192  {
193  LOG_PRINT_L0("Close connection error: " << m_connections[idx]);
194  if (!ignore_close_fails)
195  {
196  return false;
197  }
198  }
199 
200  m_connections[idx] = boost::uuids::nil_uuid();
201  m_opened_connection_count.fetch_sub(1, std::memory_order_relaxed);
202  return true;
203  }
204 
205  size_t opened_connection_count() const { return m_opened_connection_count.load(std::memory_order_relaxed); }
206 
207  private:
210  std::atomic<size_t> m_opened_connection_count;
211  std::atomic<size_t> m_next_opened_conn_idx;
212  std::atomic<size_t> m_next_closed_conn_idx;
213  std::vector<boost::uuids::uuid> m_connections;
214  };
215 
216  const unsigned int min_thread_count = 2;
217  const std::string clt_port("36230");
218  const std::string srv_port("36231");
219 
221  {
229  };
230 
232  {
233  const static int ID = cmd_close_all_connections_id;
234 
235  struct request
236  {
239  };
240  };
241 
243  {
244  const static int ID = cmd_start_open_close_test_id;
245 
246  struct request
247  {
250 
255  };
256 
257  struct response
258  {
261  };
262  };
263 
265  {
266  const static int ID = cmd_get_statistics_id;
267 
268  struct request
269  {
272  };
273 
274  struct response
275  {
279 
281  KV_SERIALIZE(opened_connections_count)
282  KV_SERIALIZE(new_connection_counter)
283  KV_SERIALIZE(close_connection_counter)
285 
286  std::string to_string() const
287  {
288  std::stringstream ss;
289  ss << "opened_connections_count = " << opened_connections_count <<
290  ", new_connection_counter = " << new_connection_counter <<
291  ", close_connection_counter = " << close_connection_counter;
292  return ss.str();
293  }
294  };
295  };
296 
298  {
299  const static int ID = cmd_reset_statistics_id;
300 
301  struct request
302  {
305  };
306 
307  struct response
308  {
311  };
312  };
313 
315  {
316  const static int ID = cmd_shutdown_id;
317 
318  struct request
319  {
322  };
323  };
324 
326  {
327  const static int ID = cmd_send_data_requests_id;
328 
329  struct request
330  {
332 
334  KV_SERIALIZE(request_size)
336  };
337  };
338 
340  {
341  const static int ID = cmd_data_request_id;
342 
343  struct request
344  {
347 
351  };
352 
353  struct response
354  {
356 
360  };
361  };
362 }
epee::net_utils::connection< test_levin_protocol_handler > test_connection
Definition: net_load_tests.h:58
Definition: net_load_tests.h:61
bool close_next_connection(bool ignore_close_fails)
Definition: net_load_tests.h:178
virtual int notify(int command, const epee::span< const uint8_t > in_buff, test_connection_context &context)
Definition: net_load_tests.h:80
std::vector< boost::uuids::uuid > m_connections
Definition: net_load_tests.h:213
Definition: abstract_tcp_server2.h:342
Definition: net_load_tests.h:231
void inc() volatile
Definition: unit_tests_utils.h:48
Definition: net_load_tests.h:329
Definition: portable_binary_archive.hpp:29
Definition: net_load_tests.h:225
size_t get_max_bytes(int command) const
Definition: net_load_tests.h:52
uint64_t close_connection_counter
Definition: net_load_tests.h:278
::std::string string
Definition: gtest-port.h:1097
Definition: net_load_tests.h:227
Definition: net_load_tests.h:223
A partial drop-in replacement for std::ostream.
Definition: byte_stream.h:57
const std::string clt_port("36230")
epee::net_utils::boosted_tcp_server< test_levin_protocol_handler > test_tcp_server
Definition: net_load_tests.h:59
Definition: net_load_tests.h:226
std::string data
Definition: net_load_tests.h:345
Definition: net_utils_base.h:366
std::string data
Definition: base58.cpp:37
static const int ID
Definition: net_load_tests.h:244
Definition: net_load_tests.h:318
Definition: enums.h:67
Non-owning sequence of data. Does not deep copy.
Definition: span.h:54
std::atomic< size_t > m_opened_connection_count
Definition: net_load_tests.h:210
#define KV_SERIALIZE(varialble)
Definition: keyvalue_serialization.h:120
command_ids
Definition: net_load_tests.h:220
volatile bool m_closed
Definition: net_load_tests.h:53
Definition: net_load_tests.h:297
#define LEVIN_DEFAULT_MAX_PACKET_SIZE
Definition: levin_base.h:77
Definition: net_load_tests.h:339
Definition: net_load_tests.h:47
void close_remaining_connections()
Definition: net_load_tests.h:173
unit_test::call_counter m_close_connection_counter
Definition: net_load_tests.h:128
uint64_t response_size
Definition: net_load_tests.h:346
Definition: net_load_tests.h:343
uint64_t open_request_target
Definition: net_load_tests.h:248
const char * uuid
Definition: minissdp.c:598
Definition: net_load_tests.h:45
Definition: levin_base.h:90
the connection templated-class for one peer connection
Represents a single connection from a client.
Definition: abstract_tcp_server2.h:95
virtual int invoke(int command, const epee::span< const uint8_t > in_buff, epee::byte_stream &buff_out, test_connection_context &context)
Definition: net_load_tests.h:69
epee::levin::async_protocol_handler_config< test_connection_context > test_levin_protocol_handler_config
Definition: net_load_tests.h:57
unit_test::call_counter m_new_connection_counter
Definition: net_load_tests.h:127
unsigned __int64 uint64_t
Definition: stdint.h:136
connection_context_base()
Definition: net_utils_base.h:401
size_t close_connection_counter() const
Definition: net_load_tests.h:112
std::unique_ptr< void, terminate > context
Unique ZMQ context handle, calls zmq_term on destruction.
Definition: zmq.h:105
static constexpr int handshake_command() noexcept
Definition: net_load_tests.h:50
virtual void callback(test_connection_context &context)
Definition: net_load_tests.h:90
uint64_t opened_connections_count
Definition: net_load_tests.h:276
std::atomic< size_t > m_next_closed_conn_idx
Definition: net_load_tests.h:212
virtual void on_connection_new(test_connection_context &context)
Definition: net_load_tests.h:96
epee::levin::async_protocol_handler< test_connection_context > test_levin_protocol_handler
Definition: net_load_tests.h:56
test_levin_commands_handler()
Definition: net_load_tests.h:63
bool handle_new_connection(const boost::uuids::uuid &connection_id, bool ignore_close_fails=false)
Definition: net_load_tests.h:154
std::string data
Definition: net_load_tests.h:355
TODO: (mj-xmr) This will be reduced in an another PR.
Definition: byte_slice.h:39
virtual void on_connection_close(test_connection_context &context)
Definition: net_load_tests.h:102
const std::string srv_port("36231")
Definition: net_load_tests.h:224
Definition: net_load_tests.h:301
open_close_test_helper(test_tcp_server &tcp_server, size_t open_request_target, size_t max_opened_connection_count)
Definition: net_load_tests.h:142
uint64_t request_size
Definition: net_load_tests.h:331
Definition: net_load_tests.h:228
Definition: levin_protocol_handler_async.h:77
Definition: levin_protocol_handler_async.h:80
Definition: net_load_tests.h:264
size_t opened_connection_count() const
Definition: net_load_tests.h:205
const unsigned int min_thread_count
Definition: net_load_tests.h:216
size_t m_max_opened_connection_count
Definition: net_load_tests.h:209
std::atomic< size_t > m_next_opened_conn_idx
Definition: net_load_tests.h:211
Definition: net_load_tests.h:325
uint64_t new_connection_counter
Definition: net_load_tests.h:277
Definition: net_load_tests.h:307
Definition: net_load_tests.h:274
std::string to_string(t_connection_type type)
Definition: connection_basic.cpp:70
Definition: net_load_tests.h:314
test_tcp_server & m_tcp_server
Definition: net_load_tests.h:208
#define END_KV_SERIALIZE_MAP()
Definition: keyvalue_serialization.h:118
Definition: net_load_tests.h:139
uint64_t max_opened_conn_count
Definition: net_load_tests.h:249
static const int ID
Definition: net_load_tests.h:233
test_connection_context()
Definition: net_load_tests.h:49
Definition: unit_tests_utils.h:40
#define LEVIN_OK
Definition: levin_base.h:102
#define BEGIN_KV_SERIALIZE_MAP()
Definition: keyvalue_serialization.h:43
Definition: net_load_tests.h:242
Definition: net_load_tests.h:353
static constexpr bool handshake_complete() noexcept
Definition: net_load_tests.h:51
size_t get() volatile const
Definition: unit_tests_utils.h:54
size_t new_connection_counter() const
Definition: net_load_tests.h:111
Definition: net_load_tests.h:222
Definition: net_load_tests.h:268
t_protocol_handler::config_type & get_config_object()
Definition: abstract_tcp_server2.h:412