| 
									
										
										
										
											2019-07-17 19:37:18 +02:00
										 |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <deque>
 | 
					
						
							|  |  |  | #include <EventLoop.h>
 | 
					
						
							|  |  |  | #include "client/voice/PrecomputedPuzzles.h"
 | 
					
						
							|  |  |  | #include "server/VoiceIOManager.h"
 | 
					
						
							| 
									
										
										
										
											2020-01-26 18:04:38 +01:00
										 |  |  | #include "VirtualServer.h"
 | 
					
						
							| 
									
										
										
										
											2019-07-17 19:37:18 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace ts { | 
					
						
							|  |  |  |     namespace server { | 
					
						
							|  |  |  |         class InstanceHandler; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         struct ServerReport { | 
					
						
							|  |  |  |             size_t avariable; | 
					
						
							|  |  |  |             size_t online; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             size_t slots; | 
					
						
							|  |  |  |             size_t onlineClients; | 
					
						
							|  |  |  |             size_t onlineChannels; | 
					
						
							|  |  |  |         }; | 
					
						
							| 
									
										
										
										
											2020-01-27 02:21:39 +01:00
										 |  |  |         class VirtualServerManager { | 
					
						
							| 
									
										
										
										
											2019-07-17 19:37:18 +02:00
										 |  |  |             public: | 
					
						
							|  |  |  |                 enum State { | 
					
						
							|  |  |  |                     STOPPED, | 
					
						
							|  |  |  |                     STARTING, | 
					
						
							|  |  |  |                     STARTED, | 
					
						
							|  |  |  |                     STOPPING | 
					
						
							|  |  |  |                 }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-27 02:21:39 +01:00
										 |  |  |                 explicit VirtualServerManager(InstanceHandler*); | 
					
						
							|  |  |  |                 ~VirtualServerManager(); | 
					
						
							| 
									
										
										
										
											2019-07-17 19:37:18 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 bool initialize(bool execute_autostart = true); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-15 15:37:19 +01:00
										 |  |  |                 std::shared_ptr<VirtualServer> create_server(std::string hosts, uint16_t port); | 
					
						
							| 
									
										
										
										
											2020-01-26 18:04:38 +01:00
										 |  |  |                 bool deleteServer(std::shared_ptr<VirtualServer>); | 
					
						
							| 
									
										
										
										
											2019-07-17 19:37:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-26 18:04:38 +01:00
										 |  |  |                 std::shared_ptr<VirtualServer> findServerById(ServerId); | 
					
						
							|  |  |  |                 std::shared_ptr<VirtualServer> findServerByPort(uint16_t); | 
					
						
							| 
									
										
										
										
											2019-07-17 19:37:18 +02:00
										 |  |  |                 uint16_t next_available_port(); | 
					
						
							|  |  |  |                 ServerId next_available_server_id(bool& /* success */); | 
					
						
							|  |  |  |                  | 
					
						
							| 
									
										
										
										
											2020-01-26 18:04:38 +01:00
										 |  |  |                 std::deque<std::shared_ptr<VirtualServer>> serverInstances(){ | 
					
						
							| 
									
										
										
										
											2019-07-17 19:37:18 +02:00
										 |  |  |                     threads::MutexLock l(this->instanceLock); | 
					
						
							|  |  |  |                     return instances; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 ServerReport report(); | 
					
						
							|  |  |  |                 OnlineClientReport clientReport(); | 
					
						
							|  |  |  |                 size_t runningServers(); | 
					
						
							|  |  |  |                 size_t usedSlots(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 void executeAutostart(); | 
					
						
							|  |  |  |                 void shutdownAll(const std::string&); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 //Dotn use shared_ptr references to keep sure that they be hold in memory
 | 
					
						
							| 
									
										
										
										
											2020-01-26 18:04:38 +01:00
										 |  |  |                 bool createServerSnapshot(Command &cmd, std::shared_ptr<VirtualServer> server, int version, std::string &error); | 
					
						
							|  |  |  |                 std::shared_ptr<VirtualServer> createServerFromSnapshot(std::shared_ptr<VirtualServer> old, std::string, uint16_t, const ts::Command &, std::string &); | 
					
						
							| 
									
										
										
										
											2019-07-17 19:37:18 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 protocol::PuzzleManager* rsaPuzzles() { return this->puzzles; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-24 02:57:58 +01:00
										 |  |  |                 event::EventExecutor* get_join_loop() { return this->join_loop; } | 
					
						
							| 
									
										
										
										
											2019-07-17 19:37:18 +02:00
										 |  |  |                 event::EventExecutor* get_executor_loop() { return this->execute_loop; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 inline void adjust_executor_threads() { | 
					
						
							|  |  |  |                     std::unique_lock instance_lock(this->instanceLock); | 
					
						
							|  |  |  |                     auto instance_count = this->instances.size(); | 
					
						
							|  |  |  |                     instance_lock.unlock(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     auto threads = std::min(config::threads::voice::execute_per_server * instance_count, config::threads::voice::execute_limit); | 
					
						
							|  |  |  |                     this->execute_loop->threads(threads); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 io::VoiceIOManager* ioManager(){ return this->_ioManager; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 threads::Mutex server_create_lock; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 State getState() { return this->state; } | 
					
						
							|  |  |  |             private: | 
					
						
							|  |  |  |                 State state = State::STOPPED; | 
					
						
							|  |  |  |                 InstanceHandler* handle; | 
					
						
							|  |  |  |                 threads::Mutex instanceLock; | 
					
						
							| 
									
										
										
										
											2020-01-26 18:04:38 +01:00
										 |  |  |                 std::deque<std::shared_ptr<VirtualServer>> instances; | 
					
						
							| 
									
										
										
										
											2019-07-17 19:37:18 +02:00
										 |  |  |                 protocol::PuzzleManager* puzzles = nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 event::EventExecutor* execute_loop = nullptr; | 
					
						
							| 
									
										
										
										
											2020-01-24 02:57:58 +01:00
										 |  |  |                 event::EventExecutor* join_loop = nullptr; | 
					
						
							|  |  |  |                 threads::Scheduler* handshakeTickers = nullptr; | 
					
						
							| 
									
										
										
										
											2019-07-17 19:37:18 +02:00
										 |  |  |                 io::VoiceIOManager* _ioManager = nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 struct { | 
					
						
							| 
									
										
										
										
											2020-01-27 02:21:39 +01:00
										 |  |  |                     std::thread executor{}; | 
					
						
							| 
									
										
										
										
											2019-07-17 19:37:18 +02:00
										 |  |  |                     std::condition_variable condition; | 
					
						
							|  |  |  |                     std::mutex lock; | 
					
						
							|  |  |  |                 } acknowledge; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 void tickHandshakeClients(); | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |