2019-01-06 21:01:33 +01:00
|
|
|
/*
|
|
|
|
* IXSocketServer.h
|
|
|
|
* Author: Benjamin Sergeant
|
|
|
|
* Copyright (c) 2018 Machine Zone, Inc. All rights reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2019-03-21 02:34:24 +01:00
|
|
|
#include "IXConnectionState.h"
|
2020-12-18 07:42:14 +01:00
|
|
|
#include "IXNetSystem.h"
|
2020-08-16 00:28:15 +02:00
|
|
|
#include "IXSelectInterrupt.h"
|
2019-10-01 03:21:20 +02:00
|
|
|
#include "IXSocketTLSOptions.h"
|
2019-01-06 21:12:39 +01:00
|
|
|
#include <atomic>
|
2019-01-06 21:01:33 +01:00
|
|
|
#include <condition_variable>
|
2019-05-30 17:46:50 +02:00
|
|
|
#include <functional>
|
|
|
|
#include <list>
|
|
|
|
#include <memory>
|
|
|
|
#include <mutex>
|
|
|
|
#include <set>
|
|
|
|
#include <string>
|
|
|
|
#include <thread>
|
|
|
|
#include <utility> // pair
|
2019-01-06 21:01:33 +01:00
|
|
|
|
2019-02-21 03:59:07 +01:00
|
|
|
namespace ix
|
2019-01-06 21:01:33 +01:00
|
|
|
{
|
2019-10-01 06:48:55 +02:00
|
|
|
class Socket;
|
|
|
|
|
2019-05-30 17:46:50 +02:00
|
|
|
class SocketServer
|
|
|
|
{
|
2019-01-06 21:01:33 +01:00
|
|
|
public:
|
2019-03-21 02:34:24 +01:00
|
|
|
using ConnectionStateFactory = std::function<std::shared_ptr<ConnectionState>()>;
|
|
|
|
|
2019-04-18 05:31:34 +02:00
|
|
|
// Each connection is handled by its own worker thread.
|
2019-04-18 01:23:24 +02:00
|
|
|
// We use a list as we only care about remove and append operations.
|
2019-05-30 17:46:50 +02:00
|
|
|
using ConnectionThreads =
|
|
|
|
std::list<std::pair<std::shared_ptr<ConnectionState>, std::thread>>;
|
2019-04-18 01:23:24 +02:00
|
|
|
|
2019-01-06 21:01:33 +01:00
|
|
|
SocketServer(int port = SocketServer::kDefaultPort,
|
|
|
|
const std::string& host = SocketServer::kDefaultHost,
|
|
|
|
int backlog = SocketServer::kDefaultTcpBacklog,
|
2020-01-27 01:17:26 +01:00
|
|
|
size_t maxConnections = SocketServer::kDefaultMaxConnections,
|
|
|
|
int addressFamily = SocketServer::kDefaultAddressFamily);
|
2019-01-06 21:01:33 +01:00
|
|
|
virtual ~SocketServer();
|
|
|
|
virtual void stop();
|
|
|
|
|
2019-04-18 05:31:34 +02:00
|
|
|
// It is possible to override ConnectionState through inheritance
|
|
|
|
// this method allows user to change the factory by returning an object
|
|
|
|
// that inherits from ConnectionState but has its own methods.
|
2019-03-21 02:34:24 +01:00
|
|
|
void setConnectionStateFactory(const ConnectionStateFactory& connectionStateFactory);
|
|
|
|
|
2022-04-12 17:55:43 +02:00
|
|
|
const static int kDefaultPort;
|
|
|
|
const static std::string kDefaultHost;
|
|
|
|
const static int kDefaultTcpBacklog;
|
|
|
|
const static size_t kDefaultMaxConnections;
|
|
|
|
const static int kDefaultAddressFamily;
|
2019-01-06 21:01:33 +01:00
|
|
|
|
|
|
|
void start();
|
|
|
|
std::pair<bool, std::string> listen();
|
|
|
|
void wait();
|
|
|
|
|
2019-10-01 03:21:20 +02:00
|
|
|
void setTLSOptions(const SocketTLSOptions& socketTLSOptions);
|
|
|
|
|
2021-11-24 17:28:25 +01:00
|
|
|
int getPort();
|
|
|
|
std::string getHost();
|
|
|
|
int getBacklog();
|
|
|
|
std::size_t getMaxConnections();
|
|
|
|
int getAddressFamily();
|
2019-01-06 21:01:33 +01:00
|
|
|
protected:
|
|
|
|
// Logging
|
|
|
|
void logError(const std::string& str);
|
|
|
|
void logInfo(const std::string& str);
|
|
|
|
|
2019-04-24 18:45:53 +02:00
|
|
|
void stopAcceptingConnections();
|
|
|
|
|
2019-01-06 21:01:33 +01:00
|
|
|
private:
|
|
|
|
// Member variables
|
|
|
|
int _port;
|
|
|
|
std::string _host;
|
|
|
|
int _backlog;
|
|
|
|
size_t _maxConnections;
|
2020-01-27 01:17:26 +01:00
|
|
|
int _addressFamily;
|
2019-01-06 21:01:33 +01:00
|
|
|
|
|
|
|
// socket for accepting connections
|
2020-12-12 20:00:55 +01:00
|
|
|
socket_t _serverFd;
|
2019-01-06 21:01:33 +01:00
|
|
|
|
2019-09-24 06:14:20 +02:00
|
|
|
std::atomic<bool> _stop;
|
|
|
|
|
2019-01-06 21:01:33 +01:00
|
|
|
std::mutex _logMutex;
|
|
|
|
|
2019-04-18 05:31:34 +02:00
|
|
|
// background thread to wait for incoming connections
|
2019-01-06 21:01:33 +01:00
|
|
|
std::thread _thread;
|
2019-05-13 21:20:03 +02:00
|
|
|
void run();
|
2020-08-16 01:03:40 +02:00
|
|
|
void onSetTerminatedCallback();
|
2019-05-13 21:20:03 +02:00
|
|
|
|
|
|
|
// background thread to cleanup (join) terminated threads
|
|
|
|
std::atomic<bool> _stopGc;
|
|
|
|
std::thread _gcThread;
|
|
|
|
void runGC();
|
2019-01-06 21:01:33 +01:00
|
|
|
|
2019-04-18 05:31:34 +02:00
|
|
|
// the list of (connectionState, threads) for each connections
|
2019-04-18 01:23:24 +02:00
|
|
|
ConnectionThreads _connectionsThreads;
|
2019-04-25 05:50:10 +02:00
|
|
|
std::mutex _connectionsThreadsMutex;
|
2019-04-17 07:19:44 +02:00
|
|
|
|
2019-04-18 05:31:34 +02:00
|
|
|
// used to have the main control thread for a server
|
|
|
|
// wait for a 'terminate' notification without busy polling
|
2019-01-06 21:01:33 +01:00
|
|
|
std::condition_variable _conditionVariable;
|
|
|
|
std::mutex _conditionVariableMutex;
|
|
|
|
|
2019-04-18 05:31:34 +02:00
|
|
|
// the factory to create ConnectionState objects
|
2019-03-21 02:34:24 +01:00
|
|
|
ConnectionStateFactory _connectionStateFactory;
|
|
|
|
|
2020-03-24 20:40:58 +01:00
|
|
|
virtual void handleConnection(std::unique_ptr<Socket>,
|
2020-08-28 23:55:40 +02:00
|
|
|
std::shared_ptr<ConnectionState> connectionState) = 0;
|
2019-01-06 21:01:33 +01:00
|
|
|
virtual size_t getConnectedClientsCount() = 0;
|
2019-04-18 01:23:24 +02:00
|
|
|
|
2019-04-25 05:50:10 +02:00
|
|
|
// Returns true if all connection threads are joined
|
2019-05-13 21:20:03 +02:00
|
|
|
void closeTerminatedThreads();
|
|
|
|
size_t getConnectionsThreadsCount();
|
2019-10-01 03:21:20 +02:00
|
|
|
|
|
|
|
SocketTLSOptions _socketTLSOptions;
|
2020-08-16 00:28:15 +02:00
|
|
|
|
|
|
|
// to wake up from select
|
|
|
|
SelectInterruptPtr _acceptSelectInterrupt;
|
2020-08-16 01:24:35 +02:00
|
|
|
|
|
|
|
// used by the gc thread, to know that a thread needs to be garbage collected
|
|
|
|
// as a connection
|
|
|
|
std::condition_variable _conditionVariableGC;
|
|
|
|
std::mutex _conditionVariableMutexGC;
|
2023-02-25 23:41:05 +01:00
|
|
|
bool _canContinueGC{ false };
|
2019-01-06 21:01:33 +01:00
|
|
|
};
|
2019-05-30 17:46:50 +02:00
|
|
|
} // namespace ix
|