mirror of
				https://github.com/DarkflameUniverse/DarkflameServer.git
				synced 2025-10-25 08:48:12 +00:00 
			
		
		
		
	Merge pull request #887 from EmosewaMC/ServerShutdown
Improve server shutdown process
This commit is contained in:
		| @@ -48,17 +48,18 @@ namespace Game { | ||||
| 	InstanceManager* im = nullptr; | ||||
| 	dConfig* config = nullptr; | ||||
| 	AssetManager* assetManager = nullptr; | ||||
| 	bool shouldShutdown = false; | ||||
| } //namespace Game | ||||
|  | ||||
| bool shutdownSequenceStarted = false; | ||||
| void ShutdownSequence(); | ||||
| int FinalizeShutdown(); | ||||
| void ShutdownSequence(int signal = -1); | ||||
| int FinalizeShutdown(int signal = -1); | ||||
| dLogger* SetupLogger(); | ||||
| void StartAuthServer(); | ||||
| void StartChatServer(); | ||||
| void HandlePacket(Packet* packet); | ||||
| std::map<uint32_t, std::string> activeSessions; | ||||
| bool shouldShutdown = false; | ||||
| SystemAddress authServerMasterPeerSysAddr; | ||||
| SystemAddress chatServerMasterPeerSysAddr; | ||||
|  | ||||
| int main(int argc, char** argv) { | ||||
| @@ -71,9 +72,9 @@ int main(int argc, char** argv) { | ||||
| #endif | ||||
|  | ||||
| 	//Triggers the shutdown sequence at application exit | ||||
| 	std::atexit(ShutdownSequence); | ||||
| 	signal(SIGINT, [](int) { ShutdownSequence(); }); | ||||
| 	signal(SIGTERM, [](int) { ShutdownSequence(); }); | ||||
| 	std::atexit([]() { ShutdownSequence(); }); | ||||
| 	signal(SIGINT, [](int signal) { ShutdownSequence(EXIT_FAILURE); }); | ||||
| 	signal(SIGTERM, [](int signal) { ShutdownSequence(EXIT_FAILURE); }); | ||||
|  | ||||
| 	//Create all the objects we need to run our service: | ||||
| 	Game::logger = SetupLogger(); | ||||
| @@ -241,7 +242,7 @@ int main(int argc, char** argv) { | ||||
| 	if (Game::config->GetValue("max_clients") != "") maxClients = std::stoi(Game::config->GetValue("max_clients")); | ||||
| 	if (Game::config->GetValue("port") != "") ourPort = std::stoi(Game::config->GetValue("port")); | ||||
|  | ||||
| 	Game::server = new dServer(Game::config->GetValue("external_ip"), ourPort, 0, maxClients, true, false, Game::logger, "", 0, ServerType::Master, Game::config); | ||||
| 	Game::server = new dServer(Game::config->GetValue("external_ip"), ourPort, 0, maxClients, true, false, Game::logger, "", 0, ServerType::Master, Game::config, &Game::shouldShutdown); | ||||
|  | ||||
| 	//Query for the database for a server labeled "master" | ||||
| 	auto* masterLookupStatement = Database::CreatePreppedStmt("SELECT id FROM `servers` WHERE `name` = 'master'"); | ||||
| @@ -278,8 +279,8 @@ int main(int argc, char** argv) { | ||||
| 	if (Game::config->GetValue("prestart_servers") != "" && Game::config->GetValue("prestart_servers") == "1") { | ||||
| 		StartChatServer(); | ||||
|  | ||||
| 		Game::im->GetInstance(0, false, 0)->SetIsReady(true); | ||||
| 		Game::im->GetInstance(1000, false, 0)->SetIsReady(true); | ||||
| 		Game::im->GetInstance(0, false, 0); | ||||
| 		Game::im->GetInstance(1000, false, 0); | ||||
|  | ||||
| 		StartAuthServer(); | ||||
| 	} | ||||
| @@ -328,7 +329,7 @@ int main(int argc, char** argv) { | ||||
| 			framesSinceLastSQLPing++; | ||||
|  | ||||
| 		//10m shutdown for universe kill command | ||||
| 		if (shouldShutdown) { | ||||
| 		if (Game::shouldShutdown) { | ||||
| 			if (framesSinceKillUniverseCommand >= 40000) { | ||||
| 				//Break main loop and exit | ||||
| 				break; | ||||
| @@ -375,9 +376,7 @@ int main(int argc, char** argv) { | ||||
| 		t += std::chrono::milliseconds(highFrameRate); | ||||
| 		std::this_thread::sleep_until(t); | ||||
| 	} | ||||
| 	FinalizeShutdown(); | ||||
| 	exit(EXIT_SUCCESS); | ||||
| 	return EXIT_SUCCESS; | ||||
| 	return FinalizeShutdown(EXIT_SUCCESS); | ||||
| } | ||||
|  | ||||
| dLogger* SetupLogger() { | ||||
| @@ -406,9 +405,15 @@ void HandlePacket(Packet* packet) { | ||||
| 			Game::im->RemoveInstance(instance); //Delete the old | ||||
| 		} | ||||
|  | ||||
| 		if (packet->systemAddress == chatServerMasterPeerSysAddr && !shouldShutdown) { | ||||
| 		if (packet->systemAddress == chatServerMasterPeerSysAddr) { | ||||
| 			chatServerMasterPeerSysAddr = UNASSIGNED_SYSTEM_ADDRESS; | ||||
| 			StartChatServer(); | ||||
| 		} | ||||
|  | ||||
| 		if (packet->systemAddress == authServerMasterPeerSysAddr) { | ||||
| 			authServerMasterPeerSysAddr = UNASSIGNED_SYSTEM_ADDRESS; | ||||
| 			StartAuthServer(); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if (packet->data[0] == ID_CONNECTION_LOST) { | ||||
| @@ -422,9 +427,15 @@ void HandlePacket(Packet* packet) { | ||||
| 			//Game::im->GetInstance(zoneID.GetMapID(), false, 0); //Create the new | ||||
| 		} | ||||
|  | ||||
| 		if (packet->systemAddress == chatServerMasterPeerSysAddr && !shouldShutdown) { | ||||
| 		if (packet->systemAddress == chatServerMasterPeerSysAddr) { | ||||
| 			chatServerMasterPeerSysAddr = UNASSIGNED_SYSTEM_ADDRESS; | ||||
| 			StartChatServer(); | ||||
| 		} | ||||
|  | ||||
| 		if (packet->systemAddress == authServerMasterPeerSysAddr) { | ||||
| 			authServerMasterPeerSysAddr = UNASSIGNED_SYSTEM_ADDRESS; | ||||
| 			StartAuthServer(); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if (packet->data[1] == MASTER) { | ||||
| @@ -520,6 +531,14 @@ void HandlePacket(Packet* packet) { | ||||
| 				chatServerMasterPeerSysAddr = copy; | ||||
| 			} | ||||
|  | ||||
| 			if (theirServerType == ServerType::Auth) { | ||||
| 				SystemAddress copy; | ||||
| 				copy.binaryAddress = packet->systemAddress.binaryAddress; | ||||
| 				copy.port = packet->systemAddress.port; | ||||
|  | ||||
| 				authServerMasterPeerSysAddr = copy; | ||||
| 			} | ||||
|  | ||||
| 			Game::logger->Log("MasterServer", "Received server info, instance: %i port: %i", theirInstanceID, theirPort); | ||||
|  | ||||
| 			break; | ||||
| @@ -744,17 +763,21 @@ void HandlePacket(Packet* packet) { | ||||
|  | ||||
| 		case MSG_MASTER_SHUTDOWN_UNIVERSE: { | ||||
| 			Game::logger->Log("MasterServer", "Received shutdown universe command, shutting down in 10 minutes."); | ||||
| 			shouldShutdown = true; | ||||
| 			Game::shouldShutdown = true; | ||||
| 			break; | ||||
| 		} | ||||
|  | ||||
| 		default: | ||||
| 			Game::logger->Log("MasterServer", "Unknown master packet ID from server: %i", packet->data[3]); | ||||
| 		} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| void StartChatServer() { | ||||
| 	if (Game::shouldShutdown) { | ||||
| 		Game::logger->Log("MasterServer", "Currently shutting down.  Chat will not be restarted."); | ||||
| 		return; | ||||
| 	} | ||||
| #ifdef __APPLE__ | ||||
| 	//macOS doesn't need sudo to run on ports < 1024 | ||||
| 	auto result = system(((BinaryPathFinder::GetBinaryDir() / "ChatServer").string() + "&").c_str()); | ||||
| @@ -767,9 +790,13 @@ void StartChatServer() { | ||||
| 		auto result = system(((BinaryPathFinder::GetBinaryDir() / "ChatServer").string() + "&").c_str()); | ||||
| 	} | ||||
| #endif | ||||
| } | ||||
| 	} | ||||
|  | ||||
| void StartAuthServer() { | ||||
| 	if (Game::shouldShutdown) { | ||||
| 		Game::logger->Log("MasterServer", "Currently shutting down.  Auth will not be restarted."); | ||||
| 		return; | ||||
| 	} | ||||
| #ifdef __APPLE__ | ||||
| 	auto result = system(((BinaryPathFinder::GetBinaryDir() / "AuthServer").string() + "&").c_str()); | ||||
| #elif _WIN32 | ||||
| @@ -783,21 +810,19 @@ void StartAuthServer() { | ||||
| #endif | ||||
| } | ||||
|  | ||||
| void ShutdownSequence() { | ||||
| void ShutdownSequence(int signal) { | ||||
| 	if (shutdownSequenceStarted) { | ||||
| 		return; | ||||
| 	} | ||||
|  | ||||
| 	shutdownSequenceStarted = true; | ||||
| 	Game::shouldShutdown = true; | ||||
|  | ||||
| 	if (Game::im) { | ||||
| 		for (auto* instance : Game::im->GetInstances()) { | ||||
| 			if (instance == nullptr) { | ||||
| 				continue; | ||||
| 			} | ||||
|  | ||||
| 			instance->Shutdown(); | ||||
| 		} | ||||
| 	{ | ||||
| 		CBITSTREAM; | ||||
| 		PacketUtils::WriteHeader(bitStream, MASTER, MSG_MASTER_SHUTDOWN); | ||||
| 		Game::server->Send(&bitStream, UNASSIGNED_SYSTEM_ADDRESS, true); | ||||
| 		Game::logger->Log("MasterServer", "Triggered master shutdown"); | ||||
| 	} | ||||
|  | ||||
| 	auto* objIdManager = ObjectIDManager::TryInstance(); | ||||
| @@ -810,13 +835,23 @@ void ShutdownSequence() { | ||||
| 	auto ticks = 0; | ||||
|  | ||||
| 	if (!Game::im) { | ||||
| 		exit(EXIT_SUCCESS); | ||||
| 		FinalizeShutdown(EXIT_FAILURE); | ||||
| 	} | ||||
|  | ||||
| 	// A server might not be finished spinning up yet, remove all of those here. | ||||
| 	for (auto instance : Game::im->GetInstances()) { | ||||
| 		if (!instance->GetIsReady()) { | ||||
| 			Game::im->RemoveInstance(instance); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	for (auto instance : Game::im->GetInstances()) { | ||||
| 		instance->SetIsShuttingDown(true); | ||||
| 	} | ||||
|  | ||||
| 	Game::logger->Log("MasterServer", "Attempting to shutdown instances, max 60 seconds..."); | ||||
|  | ||||
| 	while (true) { | ||||
|  | ||||
| 		auto packet = Game::server->Receive(); | ||||
| 		if (packet) { | ||||
| 			HandlePacket(packet); | ||||
| @@ -835,8 +870,8 @@ void ShutdownSequence() { | ||||
| 				done = false; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if (done) { | ||||
|   | ||||
| 		if (done && authServerMasterPeerSysAddr == UNASSIGNED_SYSTEM_ADDRESS && chatServerMasterPeerSysAddr == UNASSIGNED_SYSTEM_ADDRESS) { | ||||
| 			Game::logger->Log("MasterServer", "Finished shutting down MasterServer!"); | ||||
| 			break; | ||||
| 		} | ||||
| @@ -852,10 +887,10 @@ void ShutdownSequence() { | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	FinalizeShutdown(); | ||||
| 	FinalizeShutdown(signal); | ||||
| } | ||||
|  | ||||
| int FinalizeShutdown() { | ||||
| int FinalizeShutdown(int signal) { | ||||
| 	//Delete our objects here: | ||||
| 	Database::Destroy("MasterServer"); | ||||
| 	if (Game::config) delete Game::config; | ||||
| @@ -863,6 +898,6 @@ int FinalizeShutdown() { | ||||
| 	if (Game::server) delete Game::server; | ||||
| 	if (Game::logger) delete Game::logger; | ||||
|  | ||||
| 	exit(EXIT_SUCCESS); | ||||
| 	return EXIT_SUCCESS; | ||||
| 	if (signal != EXIT_SUCCESS) exit(signal); | ||||
| 	return signal; | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Gie "Max" Vanommeslaeghe
					Gie "Max" Vanommeslaeghe