2021-10-26 14:53:56 +08:00
|
|
|
// Copyright 2021 Memgraph Ltd.
|
|
|
|
//
|
|
|
|
// Use of this software is governed by the Business Source License
|
|
|
|
// included in the file licenses/BSL.txt; by using this file, you agree to be bound by the terms of the Business Source
|
|
|
|
// License, and you may not use this file except in compliance with the Business Source License.
|
|
|
|
//
|
|
|
|
// As of the Change Date specified in that file, in accordance with
|
|
|
|
// the Business Source License, use of this software will be governed
|
|
|
|
// by the Apache License, Version 2.0, included in the file
|
|
|
|
// licenses/APL.txt.
|
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
#include <chrono>
|
|
|
|
#include <iostream>
|
2018-09-21 22:53:27 +08:00
|
|
|
#include <thread>
|
2017-10-17 20:05:08 +08:00
|
|
|
|
|
|
|
#include <gflags/gflags.h>
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
|
|
#include "communication/server.hpp"
|
|
|
|
#include "io/network/socket.hpp"
|
|
|
|
|
|
|
|
using namespace std::chrono_literals;
|
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
class TestData {};
|
|
|
|
|
|
|
|
class TestSession {
|
2017-10-17 20:05:08 +08:00
|
|
|
public:
|
2021-02-18 22:32:43 +08:00
|
|
|
TestSession(TestData *, const io::network::Endpoint &, communication::InputStream *input_stream,
|
2018-09-03 21:29:06 +08:00
|
|
|
communication::OutputStream *output_stream)
|
2018-03-23 23:32:17 +08:00
|
|
|
: input_stream_(input_stream), output_stream_(output_stream) {}
|
|
|
|
|
|
|
|
void Execute() {
|
2021-02-18 22:32:43 +08:00
|
|
|
spdlog::info("Received data: '{}'",
|
|
|
|
std::string(reinterpret_cast<const char *>(input_stream_->data()), input_stream_->size()));
|
2018-09-21 22:53:27 +08:00
|
|
|
if (input_stream_->data()[0] == 'e') {
|
|
|
|
std::this_thread::sleep_for(std::chrono::seconds(5));
|
|
|
|
}
|
2018-09-03 21:29:06 +08:00
|
|
|
output_stream_->Write(input_stream_->data(), input_stream_->size());
|
|
|
|
input_stream_->Shift(input_stream_->size());
|
2018-03-23 23:32:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2018-09-03 21:29:06 +08:00
|
|
|
communication::InputStream *input_stream_;
|
|
|
|
communication::OutputStream *output_stream_;
|
2017-10-17 20:05:08 +08:00
|
|
|
};
|
|
|
|
|
2018-09-21 22:53:27 +08:00
|
|
|
const std::string safe_query("tttt");
|
|
|
|
const std::string expensive_query("eeee");
|
2018-03-23 23:32:17 +08:00
|
|
|
|
2018-09-21 22:53:27 +08:00
|
|
|
bool QueryServer(io::network::Socket &socket, const std::string &query) {
|
2018-03-23 23:32:17 +08:00
|
|
|
if (!socket.Write(query)) return false;
|
|
|
|
char response[105];
|
|
|
|
int len = 0;
|
|
|
|
while (len < query.size()) {
|
|
|
|
int got = socket.Read(response + len, query.size() - len);
|
|
|
|
if (got <= 0) return false;
|
2018-03-28 21:15:59 +08:00
|
|
|
response[got] = 0;
|
2018-03-23 23:32:17 +08:00
|
|
|
len += got;
|
|
|
|
}
|
|
|
|
if (std::string(response, strlen(response)) != query) return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-10-17 20:05:08 +08:00
|
|
|
TEST(NetworkTimeouts, InactiveSession) {
|
2018-03-23 23:32:17 +08:00
|
|
|
// Instantiate the server and set the session timeout to 2 seconds.
|
|
|
|
TestData test_data;
|
2018-06-20 23:44:47 +08:00
|
|
|
communication::ServerContext context;
|
2021-02-18 22:32:43 +08:00
|
|
|
communication::Server<TestSession, TestData> server{{"127.0.0.1", 0}, &test_data, &context, 2, "Test", 1};
|
2018-10-16 16:58:41 +08:00
|
|
|
ASSERT_TRUE(server.Start());
|
2017-10-17 20:05:08 +08:00
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
// Create the client and connect to the server.
|
|
|
|
io::network::Socket client;
|
|
|
|
ASSERT_TRUE(client.Connect(server.endpoint()));
|
2017-10-17 20:05:08 +08:00
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
// Send some data to the server.
|
2018-09-21 22:53:27 +08:00
|
|
|
ASSERT_TRUE(QueryServer(client, safe_query));
|
2017-10-17 20:05:08 +08:00
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
// After this sleep the session should still be alive.
|
|
|
|
std::this_thread::sleep_for(500ms);
|
2017-10-17 20:05:08 +08:00
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
// Send some data to the server.
|
2018-09-21 22:53:27 +08:00
|
|
|
ASSERT_TRUE(QueryServer(client, safe_query));
|
2018-03-23 23:32:17 +08:00
|
|
|
}
|
2017-10-17 20:05:08 +08:00
|
|
|
|
2018-03-23 23:32:17 +08:00
|
|
|
// After this sleep the session should have timed out.
|
2018-02-22 23:17:45 +08:00
|
|
|
std::this_thread::sleep_for(3500ms);
|
2018-09-21 22:53:27 +08:00
|
|
|
ASSERT_FALSE(QueryServer(client, safe_query));
|
2018-09-27 21:07:46 +08:00
|
|
|
|
|
|
|
// Shutdown the server.
|
|
|
|
server.Shutdown();
|
|
|
|
server.AwaitShutdown();
|
2018-09-21 22:53:27 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(NetworkTimeouts, ActiveSession) {
|
|
|
|
// Instantiate the server and set the session timeout to 2 seconds.
|
|
|
|
TestData test_data;
|
|
|
|
communication::ServerContext context;
|
2021-02-18 22:32:43 +08:00
|
|
|
communication::Server<TestSession, TestData> server{{"127.0.0.1", 0}, &test_data, &context, 2, "Test", 1};
|
2018-10-16 16:58:41 +08:00
|
|
|
ASSERT_TRUE(server.Start());
|
2018-09-21 22:53:27 +08:00
|
|
|
|
|
|
|
// Create the client and connect to the server.
|
|
|
|
io::network::Socket client;
|
|
|
|
ASSERT_TRUE(client.Connect(server.endpoint()));
|
|
|
|
|
|
|
|
// Send some data to the server.
|
|
|
|
ASSERT_TRUE(QueryServer(client, expensive_query));
|
|
|
|
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
// After this sleep the session should still be alive.
|
|
|
|
std::this_thread::sleep_for(500ms);
|
|
|
|
|
|
|
|
// Send some data to the server.
|
|
|
|
ASSERT_TRUE(QueryServer(client, safe_query));
|
|
|
|
}
|
|
|
|
|
|
|
|
// After this sleep the session should have timed out.
|
|
|
|
std::this_thread::sleep_for(3500ms);
|
|
|
|
ASSERT_FALSE(QueryServer(client, safe_query));
|
|
|
|
|
2018-09-27 21:07:46 +08:00
|
|
|
// Shutdown the server.
|
|
|
|
server.Shutdown();
|
|
|
|
server.AwaitShutdown();
|
2017-10-17 20:05:08 +08:00
|
|
|
}
|