mirror of
				https://github.com/DarkflameUniverse/DarkflameServer.git
				synced 2025-10-25 00:38:08 +00:00 
			
		
		
		
	Merge branch 'main' into guild_temp
This commit is contained in:
		| @@ -214,7 +214,12 @@ set(INCLUDED_DIRECTORIES | |||||||
| 	"dNavigation/dTerrain" | 	"dNavigation/dTerrain" | ||||||
| 	"dZoneManager" | 	"dZoneManager" | ||||||
| 	"dDatabase" | 	"dDatabase" | ||||||
| 	"dDatabase/Tables" | 	"dDatabase/CDClientDatabase" | ||||||
|  | 	"dDatabase/CDClientDatabase/CDClientTables" | ||||||
|  | 	"dDatabase/GameDatabase" | ||||||
|  | 	"dDatabase/GameDatabase/ITables" | ||||||
|  | 	"dDatabase/GameDatabase/MySQL" | ||||||
|  | 	"dDatabase/GameDatabase/MySQL/Tables" | ||||||
| 	"dNet" | 	"dNet" | ||||||
| 	"dScripts" | 	"dScripts" | ||||||
| 	"dScripts/02_server" | 	"dScripts/02_server" | ||||||
| @@ -329,8 +334,9 @@ add_subdirectory(thirdparty) | |||||||
| file( | file( | ||||||
| 	GLOB HEADERS_DDATABASE | 	GLOB HEADERS_DDATABASE | ||||||
| 	LIST_DIRECTORIES false | 	LIST_DIRECTORIES false | ||||||
| 	${PROJECT_SOURCE_DIR}/dDatabase/*.h | 	${PROJECT_SOURCE_DIR}/dDatabase/CDClientDatabase/*.h | ||||||
| 	${PROJECT_SOURCE_DIR}/dDatabase/Tables/*.h | 	${PROJECT_SOURCE_DIR}/dDatabase/CDClientDatabase/CDClientTables/*.h | ||||||
|  | 	${PROJECT_SOURCE_DIR}/dDatabase/GameDatabase/ITables/*.h | ||||||
| 	${PROJECT_SOURCE_DIR}/thirdparty/SQLite/*.h | 	${PROJECT_SOURCE_DIR}/thirdparty/SQLite/*.h | ||||||
| ) | ) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -55,14 +55,8 @@ int main(int argc, char** argv) { | |||||||
| 	LOG("Version: %i.%i", PROJECT_VERSION_MAJOR, PROJECT_VERSION_MINOR); | 	LOG("Version: %i.%i", PROJECT_VERSION_MAJOR, PROJECT_VERSION_MINOR); | ||||||
| 	LOG("Compiled on: %s", __TIMESTAMP__); | 	LOG("Compiled on: %s", __TIMESTAMP__); | ||||||
|  |  | ||||||
| 	//Connect to the MySQL Database |  | ||||||
| 	std::string mysql_host = Game::config->GetValue("mysql_host"); |  | ||||||
| 	std::string mysql_database = Game::config->GetValue("mysql_database"); |  | ||||||
| 	std::string mysql_username = Game::config->GetValue("mysql_username"); |  | ||||||
| 	std::string mysql_password = Game::config->GetValue("mysql_password"); |  | ||||||
|  |  | ||||||
| 	try { | 	try { | ||||||
| 		Database::Connect(mysql_host, mysql_database, mysql_username, mysql_password); | 		Database::Connect(); | ||||||
| 	} catch (sql::SQLException& ex) { | 	} catch (sql::SQLException& ex) { | ||||||
| 		LOG("Got an error while connecting to the database: %s", ex.what()); | 		LOG("Got an error while connecting to the database: %s", ex.what()); | ||||||
| 		Database::Destroy("AuthServer"); | 		Database::Destroy("AuthServer"); | ||||||
| @@ -74,15 +68,12 @@ int main(int argc, char** argv) { | |||||||
| 	//Find out the master's IP: | 	//Find out the master's IP: | ||||||
| 	std::string masterIP; | 	std::string masterIP; | ||||||
| 	uint32_t masterPort = 1500; | 	uint32_t masterPort = 1500; | ||||||
| 	sql::PreparedStatement* stmt = Database::CreatePreppedStmt("SELECT ip, port FROM servers WHERE name='master';"); |  | ||||||
| 	auto res = stmt->executeQuery(); |  | ||||||
| 	while (res->next()) { |  | ||||||
| 		masterIP = res->getString(1).c_str(); |  | ||||||
| 		masterPort = res->getInt(2); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	delete res; | 	auto masterInfo = Database::Get()->GetMasterInfo(); | ||||||
| 	delete stmt; | 	if (masterInfo) { | ||||||
|  | 		masterIP = masterInfo->ip; | ||||||
|  | 		masterPort = masterInfo->port; | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	Game::randomEngine = std::mt19937(time(0)); | 	Game::randomEngine = std::mt19937(time(0)); | ||||||
|  |  | ||||||
| @@ -134,16 +125,12 @@ int main(int argc, char** argv) { | |||||||
| 			//Find out the master's IP for absolutely no reason: | 			//Find out the master's IP for absolutely no reason: | ||||||
| 			std::string masterIP; | 			std::string masterIP; | ||||||
| 			uint32_t masterPort; | 			uint32_t masterPort; | ||||||
| 			sql::PreparedStatement* stmt = Database::CreatePreppedStmt("SELECT ip, port FROM servers WHERE name='master';"); | 			auto masterInfo = Database::Get()->GetMasterInfo(); | ||||||
| 			auto res = stmt->executeQuery(); | 			if (masterInfo) { | ||||||
| 			while (res->next()) { | 				masterIP = masterInfo->ip; | ||||||
| 				masterIP = res->getString(1).c_str(); | 				masterPort = masterInfo->port; | ||||||
| 				masterPort = res->getInt(2); |  | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			delete res; |  | ||||||
| 			delete stmt; |  | ||||||
|  |  | ||||||
| 			framesSinceLastSQLPing = 0; | 			framesSinceLastSQLPing = 0; | ||||||
| 		} else framesSinceLastSQLPing++; | 		} else framesSinceLastSQLPing++; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -32,15 +32,11 @@ dChatFilter::dChatFilter(const std::string& filepath, bool dontGenerateDCF) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	//Read player names that are ok as well: | 	//Read player names that are ok as well: | ||||||
| 	auto stmt = Database::CreatePreppedStmt("select name from charinfo;"); | 	auto approvedNames = Database::Get()->GetApprovedCharacterNames(); | ||||||
| 	auto res = stmt->executeQuery(); | 	for (auto& name : approvedNames) { | ||||||
| 	while (res->next()) { | 		std::transform(name.begin(), name.end(), name.begin(), ::tolower); //Transform to lowercase | ||||||
| 		std::string line = res->getString(1).c_str(); | 		m_ApprovedWords.push_back(CalculateHash(name)); | ||||||
| 		std::transform(line.begin(), line.end(), line.begin(), ::tolower); //Transform to lowercase |  | ||||||
| 		m_ApprovedWords.push_back(CalculateHash(line)); |  | ||||||
| 	} | 	} | ||||||
| 	delete res; |  | ||||||
| 	delete stmt; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| dChatFilter::~dChatFilter() { | dChatFilter::~dChatFilter() { | ||||||
|   | |||||||
| @@ -30,32 +30,17 @@ void ChatPacketHandler::HandleFriendlistRequest(Packet* packet) { | |||||||
| 	auto player = playerContainer.GetPlayerData(playerID); | 	auto player = playerContainer.GetPlayerData(playerID); | ||||||
| 	if (!player) return; | 	if (!player) return; | ||||||
|  |  | ||||||
| 	//Get our friends list from the Db.  Using a derived table since the friend of a player can be in either column. | 	auto friendsList = Database::Get()->GetFriendsList(playerID); | ||||||
| 	std::unique_ptr<sql::PreparedStatement> stmt(Database::CreatePreppedStmt( | 	for (const auto& friendData : friendsList) { | ||||||
| 		"SELECT fr.requested_player, best_friend, ci.name FROM " |  | ||||||
| 		"(SELECT CASE " |  | ||||||
| 		"WHEN player_id = ? THEN friend_id " |  | ||||||
| 		"WHEN friend_id = ? THEN player_id " |  | ||||||
| 		"END AS requested_player, best_friend FROM friends) AS fr " |  | ||||||
| 		"JOIN charinfo AS ci ON ci.id = fr.requested_player " |  | ||||||
| 		"WHERE fr.requested_player IS NOT NULL AND fr.requested_player != ?;")); |  | ||||||
| 	stmt->setUInt(1, static_cast<uint32_t>(playerID)); |  | ||||||
| 	stmt->setUInt(2, static_cast<uint32_t>(playerID)); |  | ||||||
| 	stmt->setUInt(3, static_cast<uint32_t>(playerID)); |  | ||||||
|  |  | ||||||
| 	std::vector<FriendData> friends; |  | ||||||
|  |  | ||||||
| 	std::unique_ptr<sql::ResultSet> res(stmt->executeQuery()); |  | ||||||
| 	while (res->next()) { |  | ||||||
| 		FriendData fd; | 		FriendData fd; | ||||||
| 		fd.isFTP = false; // not a thing in DLU | 		fd.isFTP = false; // not a thing in DLU | ||||||
| 		fd.friendID = res->getUInt(1); | 		fd.friendID = friendData.friendID; | ||||||
| 		GeneralUtils::SetBit(fd.friendID, eObjectBits::PERSISTENT); | 		GeneralUtils::SetBit(fd.friendID, eObjectBits::PERSISTENT); | ||||||
| 		GeneralUtils::SetBit(fd.friendID, eObjectBits::CHARACTER); | 		GeneralUtils::SetBit(fd.friendID, eObjectBits::CHARACTER); | ||||||
|  |  | ||||||
| 		fd.isBestFriend = res->getInt(2) == 3; //0 = friends, 1 = left_requested, 2 = right_requested, 3 = both_accepted - are now bffs | 		fd.isBestFriend = friendData.isBestFriend; //0 = friends, 1 = left_requested, 2 = right_requested, 3 = both_accepted - are now bffs | ||||||
| 		if (fd.isBestFriend) player->countOfBestFriends += 1; | 		if (fd.isBestFriend) player->countOfBestFriends += 1; | ||||||
| 		fd.friendName = res->getString(3); | 		fd.friendName = friendData.friendName; | ||||||
|  |  | ||||||
| 		//Now check if they're online: | 		//Now check if they're online: | ||||||
| 		auto fr = playerContainer.GetPlayerData(fd.friendID); | 		auto fr = playerContainer.GetPlayerData(fd.friendID); | ||||||
| @@ -71,7 +56,7 @@ void ChatPacketHandler::HandleFriendlistRequest(Packet* packet) { | |||||||
| 			fd.zoneID = LWOZONEID(); | 			fd.zoneID = LWOZONEID(); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		friends.push_back(fd); | 		player->friends.push_back(fd); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	//Now, we need to send the friendlist to the server they came from: | 	//Now, we need to send the friendlist to the server they came from: | ||||||
| @@ -83,22 +68,17 @@ void ChatPacketHandler::HandleFriendlistRequest(Packet* packet) { | |||||||
| 	BitStreamUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::GET_FRIENDS_LIST_RESPONSE); | 	BitStreamUtils::WriteHeader(bitStream, eConnectionType::CLIENT, eClientMessageType::GET_FRIENDS_LIST_RESPONSE); | ||||||
| 	bitStream.Write<uint8_t>(0); | 	bitStream.Write<uint8_t>(0); | ||||||
| 	bitStream.Write<uint16_t>(1); //Length of packet -- just writing one as it doesn't matter, client skips it. | 	bitStream.Write<uint16_t>(1); //Length of packet -- just writing one as it doesn't matter, client skips it. | ||||||
| 	bitStream.Write((uint16_t)friends.size()); | 	bitStream.Write((uint16_t)player->friends.size()); | ||||||
|  |  | ||||||
| 	for (auto& data : friends) { | 	for (auto& data : player->friends) { | ||||||
| 		data.Serialize(bitStream); | 		data.Serialize(bitStream); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	player->friends = friends; |  | ||||||
|  |  | ||||||
| 	SystemAddress sysAddr = player->sysAddr; | 	SystemAddress sysAddr = player->sysAddr; | ||||||
| 	SEND_PACKET; | 	SEND_PACKET; | ||||||
| } | } | ||||||
|  |  | ||||||
| void ChatPacketHandler::HandleFriendRequest(Packet* packet) { | void ChatPacketHandler::HandleFriendRequest(Packet* packet) { | ||||||
| 	auto maxNumberOfBestFriendsAsString = Game::config->GetValue("max_number_of_best_friends"); |  | ||||||
| 	// If this config option doesn't exist, default to 5 which is what live used. |  | ||||||
| 	auto maxNumberOfBestFriends = maxNumberOfBestFriendsAsString != "" ? std::stoi(maxNumberOfBestFriendsAsString) : 5U; |  | ||||||
| 	CINSTREAM_SKIP_HEADER; | 	CINSTREAM_SKIP_HEADER; | ||||||
| 	LWOOBJID requestorPlayerID; | 	LWOOBJID requestorPlayerID; | ||||||
| 	inStream.Read(requestorPlayerID); | 	inStream.Read(requestorPlayerID); | ||||||
| @@ -155,35 +135,26 @@ void ChatPacketHandler::HandleFriendRequest(Packet* packet) { | |||||||
| 	// If at this point we dont have a target, then they arent online and we cant send the request. | 	// If at this point we dont have a target, then they arent online and we cant send the request. | ||||||
| 	// Send the response code that corresponds to what the error is. | 	// Send the response code that corresponds to what the error is. | ||||||
| 	if (!requestee) { | 	if (!requestee) { | ||||||
| 		std::unique_ptr<sql::PreparedStatement> nameQuery(Database::CreatePreppedStmt("SELECT name from charinfo where name = ?;")); |  | ||||||
| 		nameQuery->setString(1, playerName); |  | ||||||
| 		std::unique_ptr<sql::ResultSet> result(nameQuery->executeQuery()); |  | ||||||
|  |  | ||||||
| 		requestee.reset(new PlayerData()); | 		requestee.reset(new PlayerData()); | ||||||
| 		requestee->playerName = playerName; | 		requestee->playerName = playerName; | ||||||
|  | 		auto responseType = Database::Get()->GetCharacterInfo(playerName) | ||||||
|  | 			? eAddFriendResponseType::NOTONLINE | ||||||
|  | 			: eAddFriendResponseType::INVALIDCHARACTER; | ||||||
|  |  | ||||||
| 		SendFriendResponse(requestor, requestee.get(), result->next() ? eAddFriendResponseType::NOTONLINE : eAddFriendResponseType::INVALIDCHARACTER); | 		SendFriendResponse(requestor, requestee.get(), responseType); | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (isBestFriendRequest) { | 	if (isBestFriendRequest) { | ||||||
| 		std::unique_ptr<sql::PreparedStatement> friendUpdate(Database::CreatePreppedStmt("SELECT * FROM friends WHERE (player_id = ? AND friend_id = ?) OR (player_id = ? AND friend_id = ?) LIMIT 1;")); |  | ||||||
| 		friendUpdate->setUInt(1, static_cast<uint32_t>(requestorPlayerID)); |  | ||||||
| 		friendUpdate->setUInt(2, static_cast<uint32_t>(requestee->playerID)); |  | ||||||
| 		friendUpdate->setUInt(3, static_cast<uint32_t>(requestee->playerID)); |  | ||||||
| 		friendUpdate->setUInt(4, static_cast<uint32_t>(requestorPlayerID)); |  | ||||||
| 		std::unique_ptr<sql::ResultSet> result(friendUpdate->executeQuery()); |  | ||||||
|  |  | ||||||
| 		LWOOBJID queryPlayerID = LWOOBJID_EMPTY; |  | ||||||
| 		LWOOBJID queryFriendID = LWOOBJID_EMPTY; |  | ||||||
| 		uint8_t oldBestFriendStatus{}; | 		uint8_t oldBestFriendStatus{}; | ||||||
| 		uint8_t bestFriendStatus{}; | 		uint8_t bestFriendStatus{}; | ||||||
|  | 		auto bestFriendInfo = Database::Get()->GetBestFriendStatus(requestorPlayerID, requestee->playerID); | ||||||
| 		if (result->next()) { | 		if (bestFriendInfo) { | ||||||
| 			// Get the IDs | 			// Get the IDs | ||||||
| 			queryPlayerID = result->getInt(1); | 			LWOOBJID queryPlayerID = bestFriendInfo->playerCharacterId; | ||||||
| 			queryFriendID = result->getInt(2); | 			LWOOBJID queryFriendID = bestFriendInfo->friendCharacterId; | ||||||
| 			oldBestFriendStatus = result->getInt(3); | 			oldBestFriendStatus = bestFriendInfo->bestFriendStatus; | ||||||
| 			bestFriendStatus = oldBestFriendStatus; | 			bestFriendStatus = oldBestFriendStatus; | ||||||
|  |  | ||||||
| 			// Set the bits | 			// Set the bits | ||||||
| @@ -204,22 +175,17 @@ void ChatPacketHandler::HandleFriendRequest(Packet* packet) { | |||||||
|  |  | ||||||
| 		// Only do updates if there was a change in the bff status. | 		// Only do updates if there was a change in the bff status. | ||||||
| 		if (oldBestFriendStatus != bestFriendStatus) { | 		if (oldBestFriendStatus != bestFriendStatus) { | ||||||
| 			if (requestee->countOfBestFriends >= maxNumberOfBestFriends || requestor->countOfBestFriends >= maxNumberOfBestFriends) { | 			auto maxBestFriends = playerContainer.GetMaxNumberOfBestFriends(); | ||||||
| 				if (requestee->countOfBestFriends >= maxNumberOfBestFriends) { | 			if (requestee->countOfBestFriends >= maxBestFriends || requestor->countOfBestFriends >= maxBestFriends) { | ||||||
|  | 				if (requestee->countOfBestFriends >= maxBestFriends) { | ||||||
| 					SendFriendResponse(requestor, requestee.get(), eAddFriendResponseType::THEIRFRIENDLISTFULL, false); | 					SendFriendResponse(requestor, requestee.get(), eAddFriendResponseType::THEIRFRIENDLISTFULL, false); | ||||||
| 				} | 				} | ||||||
| 				if (requestor->countOfBestFriends >= maxNumberOfBestFriends) { | 				if (requestor->countOfBestFriends >= maxBestFriends) { | ||||||
| 					SendFriendResponse(requestor, requestee.get(), eAddFriendResponseType::YOURFRIENDSLISTFULL, false); | 					SendFriendResponse(requestor, requestee.get(), eAddFriendResponseType::YOURFRIENDSLISTFULL, false); | ||||||
| 				} | 				} | ||||||
| 			} else { | 			} else { | ||||||
| 				// Then update the database with this new info. | 				// Then update the database with this new info. | ||||||
| 				std::unique_ptr<sql::PreparedStatement> updateQuery(Database::CreatePreppedStmt("UPDATE friends SET best_friend = ? WHERE (player_id = ? AND friend_id = ?) OR (player_id = ? AND friend_id = ?) LIMIT 1;")); | 				Database::Get()->SetBestFriendStatus(requestorPlayerID, requestee->playerID, bestFriendStatus); | ||||||
| 				updateQuery->setUInt(1, bestFriendStatus); |  | ||||||
| 				updateQuery->setUInt(2, static_cast<uint32_t>(requestorPlayerID)); |  | ||||||
| 				updateQuery->setUInt(3, static_cast<uint32_t>(requestee->playerID)); |  | ||||||
| 				updateQuery->setUInt(4, static_cast<uint32_t>(requestee->playerID)); |  | ||||||
| 				updateQuery->setUInt(5, static_cast<uint32_t>(requestorPlayerID)); |  | ||||||
| 				updateQuery->executeUpdate(); |  | ||||||
| 				// Sent the best friend update here if the value is 3 | 				// Sent the best friend update here if the value is 3 | ||||||
| 				if (bestFriendStatus == 3U) { | 				if (bestFriendStatus == 3U) { | ||||||
| 					requestee->countOfBestFriends += 1; | 					requestee->countOfBestFriends += 1; | ||||||
| @@ -242,8 +208,15 @@ void ChatPacketHandler::HandleFriendRequest(Packet* packet) { | |||||||
| 			if (requestor->sysAddr != UNASSIGNED_SYSTEM_ADDRESS) SendFriendResponse(requestor, requestee.get(), eAddFriendResponseType::WAITINGAPPROVAL, true, true); | 			if (requestor->sysAddr != UNASSIGNED_SYSTEM_ADDRESS) SendFriendResponse(requestor, requestee.get(), eAddFriendResponseType::WAITINGAPPROVAL, true, true); | ||||||
| 		} | 		} | ||||||
| 	} else { | 	} else { | ||||||
| 		// Do not send this if we are requesting to be a best friend. | 		auto maxFriends = playerContainer.GetMaxNumberOfFriends(); | ||||||
| 		SendFriendRequest(requestee.get(), requestor); | 		if (requestee->friends.size() >= maxFriends) { | ||||||
|  | 			SendFriendResponse(requestor, requestee.get(), eAddFriendResponseType::THEIRFRIENDLISTFULL, false); | ||||||
|  | 		} else if (requestor->friends.size() >= maxFriends) { | ||||||
|  | 			SendFriendResponse(requestor, requestee.get(), eAddFriendResponseType::YOURFRIENDSLISTFULL, false); | ||||||
|  | 		} else { | ||||||
|  | 			// Do not send this if we are requesting to be a best friend. | ||||||
|  | 			SendFriendRequest(requestee.get(), requestor); | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// If the player is actually a player and not a ghost one defined above, release it from being deleted. | 	// If the player is actually a player and not a ghost one defined above, release it from being deleted. | ||||||
| @@ -314,11 +287,7 @@ void ChatPacketHandler::HandleFriendResponse(Packet* packet) { | |||||||
| 		requesteeData.isOnline = true; | 		requesteeData.isOnline = true; | ||||||
| 		requestor->friends.push_back(requesteeData); | 		requestor->friends.push_back(requesteeData); | ||||||
|  |  | ||||||
| 		std::unique_ptr<sql::PreparedStatement> statement(Database::CreatePreppedStmt("INSERT IGNORE INTO `friends` (`player_id`, `friend_id`, `best_friend`) VALUES (?,?,?);")); | 		Database::Get()->AddFriend(requestor->playerID, requestee->playerID); | ||||||
| 		statement->setUInt(1, static_cast<uint32_t>(requestor->playerID)); |  | ||||||
| 		statement->setUInt(2, static_cast<uint32_t>(requestee->playerID)); |  | ||||||
| 		statement->setInt(3, 0); |  | ||||||
| 		statement->execute(); |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if (serverResponseCode != eAddFriendResponseType::DECLINED) SendFriendResponse(requestor, requestee, serverResponseCode, isAlreadyBestFriends); | 	if (serverResponseCode != eAddFriendResponseType::DECLINED) SendFriendResponse(requestor, requestee, serverResponseCode, isAlreadyBestFriends); | ||||||
| @@ -333,25 +302,17 @@ void ChatPacketHandler::HandleRemoveFriend(Packet* packet) { | |||||||
|  |  | ||||||
| 	//we'll have to query the db here to find the user, since you can delete them while they're offline. | 	//we'll have to query the db here to find the user, since you can delete them while they're offline. | ||||||
| 	//First, we need to find their ID: | 	//First, we need to find their ID: | ||||||
| 	std::unique_ptr<sql::PreparedStatement> stmt(Database::CreatePreppedStmt("SELECT id FROM charinfo WHERE name=? LIMIT 1;")); |  | ||||||
| 	stmt->setString(1, friendName.c_str()); |  | ||||||
|  |  | ||||||
| 	LWOOBJID friendID = 0; | 	LWOOBJID friendID = 0; | ||||||
| 	std::unique_ptr<sql::ResultSet> res(stmt->executeQuery()); | 	auto friendIdResult = Database::Get()->GetCharacterInfo(friendName); | ||||||
| 	while (res->next()) { | 	if (friendIdResult) { | ||||||
| 		friendID = res->getUInt(1); | 		friendID = friendIdResult->id; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Convert friendID to LWOOBJID | 	// Convert friendID to LWOOBJID | ||||||
| 	GeneralUtils::SetBit(friendID, eObjectBits::PERSISTENT); | 	GeneralUtils::SetBit(friendID, eObjectBits::PERSISTENT); | ||||||
| 	GeneralUtils::SetBit(friendID, eObjectBits::CHARACTER); | 	GeneralUtils::SetBit(friendID, eObjectBits::CHARACTER); | ||||||
|  |  | ||||||
| 	std::unique_ptr<sql::PreparedStatement> deletestmt(Database::CreatePreppedStmt("DELETE FROM friends WHERE (player_id = ? AND friend_id = ?) OR (player_id = ? AND friend_id = ?) LIMIT 1;")); | 	Database::Get()->RemoveFriend(playerID, friendID); | ||||||
| 	deletestmt->setUInt(1, static_cast<uint32_t>(playerID)); |  | ||||||
| 	deletestmt->setUInt(2, static_cast<uint32_t>(friendID)); |  | ||||||
| 	deletestmt->setUInt(3, static_cast<uint32_t>(friendID)); |  | ||||||
| 	deletestmt->setUInt(4, static_cast<uint32_t>(playerID)); |  | ||||||
| 	deletestmt->execute(); |  | ||||||
|  |  | ||||||
| 	//Now, we need to send an update to notify the sender (and possibly, receiver) that their friendship has been ended: | 	//Now, we need to send an update to notify the sender (and possibly, receiver) that their friendship has been ended: | ||||||
| 	auto goonA = playerContainer.GetPlayerData(playerID); | 	auto goonA = playerContainer.GetPlayerData(playerID); | ||||||
|   | |||||||
| @@ -78,13 +78,8 @@ int main(int argc, char** argv) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	//Connect to the MySQL Database | 	//Connect to the MySQL Database | ||||||
| 	std::string mysql_host = Game::config->GetValue("mysql_host"); |  | ||||||
| 	std::string mysql_database = Game::config->GetValue("mysql_database"); |  | ||||||
| 	std::string mysql_username = Game::config->GetValue("mysql_username"); |  | ||||||
| 	std::string mysql_password = Game::config->GetValue("mysql_password"); |  | ||||||
|  |  | ||||||
| 	try { | 	try { | ||||||
| 		Database::Connect(mysql_host, mysql_database, mysql_username, mysql_password); | 		Database::Connect(); | ||||||
| 	} catch (sql::SQLException& ex) { | 	} catch (sql::SQLException& ex) { | ||||||
| 		LOG("Got an error while connecting to the database: %s", ex.what()); | 		LOG("Got an error while connecting to the database: %s", ex.what()); | ||||||
| 		Database::Destroy("ChatServer"); | 		Database::Destroy("ChatServer"); | ||||||
| @@ -96,16 +91,11 @@ int main(int argc, char** argv) { | |||||||
| 	//Find out the master's IP: | 	//Find out the master's IP: | ||||||
| 	std::string masterIP; | 	std::string masterIP; | ||||||
| 	uint32_t masterPort = 1000; | 	uint32_t masterPort = 1000; | ||||||
| 	sql::PreparedStatement* stmt = Database::CreatePreppedStmt("SELECT ip, port FROM servers WHERE name='master';"); | 	auto masterInfo = Database::Get()->GetMasterInfo(); | ||||||
| 	auto res = stmt->executeQuery(); | 	if (masterInfo) { | ||||||
| 	while (res->next()) { | 		masterIP = masterInfo->ip; | ||||||
| 		masterIP = res->getString(1).c_str(); | 		masterPort = masterInfo->port; | ||||||
| 		masterPort = res->getInt(2); |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	delete res; |  | ||||||
| 	delete stmt; |  | ||||||
|  |  | ||||||
| 	//It's safe to pass 'localhost' here, as the IP is only used as the external IP. | 	//It's safe to pass 'localhost' here, as the IP is only used as the external IP. | ||||||
| 	uint32_t maxClients = 50; | 	uint32_t maxClients = 50; | ||||||
| 	uint32_t ourPort = 1501; | 	uint32_t ourPort = 1501; | ||||||
| @@ -118,6 +108,8 @@ int main(int argc, char** argv) { | |||||||
| 	 | 	 | ||||||
| 	Game::randomEngine = std::mt19937(time(0)); | 	Game::randomEngine = std::mt19937(time(0)); | ||||||
|  |  | ||||||
|  | 	playerContainer.Initialize(); | ||||||
|  |  | ||||||
| 	//Run it until server gets a kill message from Master: | 	//Run it until server gets a kill message from Master: | ||||||
| 	auto t = std::chrono::high_resolution_clock::now(); | 	auto t = std::chrono::high_resolution_clock::now(); | ||||||
| 	Packet* packet = nullptr; | 	Packet* packet = nullptr; | ||||||
| @@ -158,15 +150,12 @@ int main(int argc, char** argv) { | |||||||
| 			//Find out the master's IP for absolutely no reason: | 			//Find out the master's IP for absolutely no reason: | ||||||
| 			std::string masterIP; | 			std::string masterIP; | ||||||
| 			uint32_t masterPort; | 			uint32_t masterPort; | ||||||
| 			sql::PreparedStatement* stmt = Database::CreatePreppedStmt("SELECT ip, port FROM servers WHERE name='master';"); |  | ||||||
| 			auto res = stmt->executeQuery(); |  | ||||||
| 			while (res->next()) { |  | ||||||
| 				masterIP = res->getString(1).c_str(); |  | ||||||
| 				masterPort = res->getInt(2); |  | ||||||
| 			} |  | ||||||
|  |  | ||||||
| 			delete res; | 			auto masterInfo = Database::Get()->GetMasterInfo(); | ||||||
| 			delete stmt; | 			if (masterInfo) { | ||||||
|  | 				masterIP = masterInfo->ip; | ||||||
|  | 				masterPort = masterInfo->port; | ||||||
|  | 			} | ||||||
|  |  | ||||||
| 			framesSinceLastSQLPing = 0; | 			framesSinceLastSQLPing = 0; | ||||||
| 		} else framesSinceLastSQLPing++; | 		} else framesSinceLastSQLPing++; | ||||||
|   | |||||||
| @@ -13,11 +13,13 @@ | |||||||
| #include "ChatPackets.h" | #include "ChatPackets.h" | ||||||
| #include "dConfig.h" | #include "dConfig.h" | ||||||
|  |  | ||||||
| PlayerContainer::PlayerContainer() { | void PlayerContainer::Initialize() { | ||||||
|  | 	GeneralUtils::TryParse<uint32_t>(Game::config->GetValue("max_number_of_best_friends"), m_MaxNumberOfBestFriends); | ||||||
|  | 	GeneralUtils::TryParse<uint32_t>(Game::config->GetValue("max_number_of_friends"), m_MaxNumberOfFriends); | ||||||
| } | } | ||||||
|  |  | ||||||
| PlayerContainer::~PlayerContainer() { | PlayerContainer::~PlayerContainer() { | ||||||
| 	mPlayers.clear(); | 	m_Players.clear(); | ||||||
| } | } | ||||||
|  |  | ||||||
| TeamData::TeamData() { | TeamData::TeamData() { | ||||||
| @@ -41,19 +43,12 @@ void PlayerContainer::InsertPlayer(Packet* packet) { | |||||||
| 	inStream.Read(data->muteExpire); | 	inStream.Read(data->muteExpire); | ||||||
| 	data->sysAddr = packet->systemAddress; | 	data->sysAddr = packet->systemAddress; | ||||||
|  |  | ||||||
| 	mNames[data->playerID] = GeneralUtils::UTF8ToUTF16(data->playerName); | 	m_Names[data->playerID] = GeneralUtils::UTF8ToUTF16(data->playerName); | ||||||
|  |  | ||||||
| 	mPlayers.insert(std::make_pair(data->playerID, data)); | 	m_Players.insert(std::make_pair(data->playerID, data)); | ||||||
| 	LOG("Added user: %s (%llu), zone: %i", data->playerName.c_str(), data->playerID, data->zoneID.GetMapID()); | 	LOG("Added user: %s (%llu), zone: %i", data->playerName.c_str(), data->playerID, data->zoneID.GetMapID()); | ||||||
|  |  | ||||||
| 	auto* insertLog = Database::CreatePreppedStmt("INSERT INTO activity_log (character_id, activity, time, map_id) VALUES (?, ?, ?, ?);"); | 	Database::Get()->UpdateActivityLog(data->playerID, eActivityType::PlayerLoggedIn, data->zoneID.GetMapID()); | ||||||
|  |  | ||||||
| 	insertLog->setInt(1, data->playerID); |  | ||||||
| 	insertLog->setInt(2, 0); |  | ||||||
| 	insertLog->setUInt64(3, time(nullptr)); |  | ||||||
| 	insertLog->setInt(4, data->zoneID.GetMapID()); |  | ||||||
|  |  | ||||||
| 	insertLog->executeUpdate(); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| void PlayerContainer::RemovePlayer(Packet* packet) { | void PlayerContainer::RemovePlayer(Packet* packet) { | ||||||
| @@ -88,16 +83,9 @@ void PlayerContainer::RemovePlayer(Packet* packet) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	LOG("Removed user: %llu", playerID); | 	LOG("Removed user: %llu", playerID); | ||||||
| 	mPlayers.erase(playerID); | 	m_Players.erase(playerID); | ||||||
|  |  | ||||||
| 	auto* insertLog = Database::CreatePreppedStmt("INSERT INTO activity_log (character_id, activity, time, map_id) VALUES (?, ?, ?, ?);"); | 	Database::Get()->UpdateActivityLog(playerID, eActivityType::PlayerLoggedOut, player->zoneID.GetMapID()); | ||||||
|  |  | ||||||
| 	insertLog->setInt(1, playerID); |  | ||||||
| 	insertLog->setInt(2, 1); |  | ||||||
| 	insertLog->setUInt64(3, time(nullptr)); |  | ||||||
| 	insertLog->setInt(4, player->zoneID.GetMapID()); |  | ||||||
|  |  | ||||||
| 	insertLog->executeUpdate(); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| void PlayerContainer::MuteUpdate(Packet* packet) { | void PlayerContainer::MuteUpdate(Packet* packet) { | ||||||
| @@ -191,7 +179,7 @@ TeamData* PlayerContainer::CreateLocalTeam(std::vector<LWOOBJID> members) { | |||||||
| TeamData* PlayerContainer::CreateTeam(LWOOBJID leader, bool local) { | TeamData* PlayerContainer::CreateTeam(LWOOBJID leader, bool local) { | ||||||
| 	auto* team = new TeamData(); | 	auto* team = new TeamData(); | ||||||
|  |  | ||||||
| 	team->teamID = ++mTeamIDCounter; | 	team->teamID = ++m_TeamIDCounter; | ||||||
| 	team->leaderID = leader; | 	team->leaderID = leader; | ||||||
| 	team->local = local; | 	team->local = local; | ||||||
|  |  | ||||||
| @@ -376,15 +364,15 @@ void PlayerContainer::UpdateTeamsOnWorld(TeamData* team, bool deleteTeam) { | |||||||
| } | } | ||||||
|  |  | ||||||
| std::u16string PlayerContainer::GetName(LWOOBJID playerID) { | std::u16string PlayerContainer::GetName(LWOOBJID playerID) { | ||||||
| 	const auto& pair = mNames.find(playerID); | 	const auto& pair = m_Names.find(playerID); | ||||||
|  |  | ||||||
| 	if (pair == mNames.end()) return u""; | 	if (pair == m_Names.end()) return u""; | ||||||
|  |  | ||||||
| 	return pair->second; | 	return pair->second; | ||||||
| } | } | ||||||
|  |  | ||||||
| LWOOBJID PlayerContainer::GetId(const std::u16string& playerName) { | LWOOBJID PlayerContainer::GetId(const std::u16string& playerName) { | ||||||
| 	for (const auto& pair : mNames) { | 	for (const auto& pair : m_Names) { | ||||||
| 		if (pair.second == playerName) { | 		if (pair.second == playerName) { | ||||||
| 			return pair.first; | 			return pair.first; | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -29,9 +29,9 @@ struct TeamData { | |||||||
|  |  | ||||||
| class PlayerContainer { | class PlayerContainer { | ||||||
| public: | public: | ||||||
| 	PlayerContainer(); |  | ||||||
| 	~PlayerContainer(); | 	~PlayerContainer(); | ||||||
|  |  | ||||||
|  | 	void Initialize(); | ||||||
| 	void InsertPlayer(Packet* packet); | 	void InsertPlayer(Packet* packet); | ||||||
| 	void RemovePlayer(Packet* packet); | 	void RemovePlayer(Packet* packet); | ||||||
| 	void MuteUpdate(Packet* packet); | 	void MuteUpdate(Packet* packet); | ||||||
| @@ -39,13 +39,13 @@ public: | |||||||
| 	void BroadcastMuteUpdate(LWOOBJID player, time_t time); | 	void BroadcastMuteUpdate(LWOOBJID player, time_t time); | ||||||
|  |  | ||||||
| 	PlayerData* GetPlayerData(const LWOOBJID& playerID) { | 	PlayerData* GetPlayerData(const LWOOBJID& playerID) { | ||||||
| 		auto it = mPlayers.find(playerID); | 		auto it = m_Players.find(playerID); | ||||||
| 		if (it != mPlayers.end()) return it->second; | 		if (it != m_Players.end()) return it->second; | ||||||
| 		return nullptr; | 		return nullptr; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	PlayerData* GetPlayerData(const std::string& playerName) { | 	PlayerData* GetPlayerData(const std::string& playerName) { | ||||||
| 		for (auto player : mPlayers) { | 		for (auto player : m_Players) { | ||||||
| 			if (player.second) { | 			if (player.second) { | ||||||
| 				std::string pn = player.second->playerName.c_str(); | 				std::string pn = player.second->playerName.c_str(); | ||||||
| 				if (pn == playerName) return player.second; | 				if (pn == playerName) return player.second; | ||||||
| @@ -67,13 +67,17 @@ public: | |||||||
| 	std::u16string GetName(LWOOBJID playerID); | 	std::u16string GetName(LWOOBJID playerID); | ||||||
| 	LWOOBJID GetId(const std::u16string& playerName); | 	LWOOBJID GetId(const std::u16string& playerName); | ||||||
| 	bool GetIsMuted(PlayerData* data); | 	bool GetIsMuted(PlayerData* data); | ||||||
|  | 	uint32_t GetMaxNumberOfBestFriends() { return m_MaxNumberOfBestFriends; } | ||||||
|  | 	uint32_t GetMaxNumberOfFriends() { return m_MaxNumberOfFriends; } | ||||||
|  |  | ||||||
| 	std::map<LWOOBJID, PlayerData*>& GetAllPlayerData() { return mPlayers; } | 	std::map<LWOOBJID, PlayerData*>& GetAllPlayerData() { return m_Players; } | ||||||
|  |  | ||||||
| private: | private: | ||||||
| 	LWOOBJID mTeamIDCounter = 0; | 	LWOOBJID m_TeamIDCounter = 0; | ||||||
| 	std::map<LWOOBJID, PlayerData*> mPlayers; | 	std::map<LWOOBJID, PlayerData*> m_Players; | ||||||
| 	std::vector<TeamData*> mTeams; | 	std::vector<TeamData*> mTeams; | ||||||
| 	std::unordered_map<LWOOBJID, std::u16string> mNames; | 	std::unordered_map<LWOOBJID, std::u16string> m_Names; | ||||||
|  | 	uint32_t m_MaxNumberOfBestFriends = 5; | ||||||
|  | 	uint32_t m_MaxNumberOfFriends = 50; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -13,9 +13,8 @@ | |||||||
|  |  | ||||||
| //! Forward declarations | //! Forward declarations | ||||||
|  |  | ||||||
| std::unique_ptr<sql::ResultSet> GetModelsFromDatabase(); |  | ||||||
| void WriteSd0Magic(char* input, uint32_t chunkSize); | void WriteSd0Magic(char* input, uint32_t chunkSize); | ||||||
| bool CheckSd0Magic(sql::Blob* streamToCheck); | bool CheckSd0Magic(std::istream& streamToCheck); | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * @brief Truncates all models with broken data from the database. |  * @brief Truncates all models with broken data from the database. | ||||||
| @@ -24,28 +23,24 @@ bool CheckSd0Magic(sql::Blob* streamToCheck); | |||||||
|  */ |  */ | ||||||
| uint32_t BrickByBrickFix::TruncateBrokenBrickByBrickXml() { | uint32_t BrickByBrickFix::TruncateBrokenBrickByBrickXml() { | ||||||
| 	uint32_t modelsTruncated{}; | 	uint32_t modelsTruncated{}; | ||||||
| 	auto modelsToTruncate = GetModelsFromDatabase(); | 	auto modelsToTruncate = Database::Get()->GetAllUgcModels(); | ||||||
| 	bool previousCommitValue = Database::GetAutoCommit(); | 	bool previousCommitValue = Database::Get()->GetAutoCommit(); | ||||||
| 	Database::SetAutoCommit(false); | 	Database::Get()->SetAutoCommit(false); | ||||||
| 	while (modelsToTruncate->next()) { | 	for (auto& model : modelsToTruncate) { | ||||||
| 		std::unique_ptr<sql::PreparedStatement> ugcModelToDelete(Database::CreatePreppedStmt("DELETE FROM ugc WHERE ugc.id = ?;")); |  | ||||||
| 		std::unique_ptr<sql::PreparedStatement> pcModelToDelete(Database::CreatePreppedStmt("DELETE FROM properties_contents WHERE ugc_id = ?;")); |  | ||||||
| 		std::string completeUncompressedModel{}; | 		std::string completeUncompressedModel{}; | ||||||
| 		uint32_t chunkCount{}; | 		uint32_t chunkCount{}; | ||||||
| 		uint64_t modelId = modelsToTruncate->getInt(1); |  | ||||||
| 		std::unique_ptr<sql::Blob> modelAsSd0(modelsToTruncate->getBlob(2)); |  | ||||||
| 		// Check that header is sd0 by checking for the sd0 magic. | 		// Check that header is sd0 by checking for the sd0 magic. | ||||||
| 		if (CheckSd0Magic(modelAsSd0.get())) { | 		if (CheckSd0Magic(model.lxfmlData)) { | ||||||
| 			while (true) { | 			while (true) { | ||||||
| 				uint32_t chunkSize{}; | 				uint32_t chunkSize{}; | ||||||
| 				modelAsSd0->read(reinterpret_cast<char*>(&chunkSize), sizeof(uint32_t)); // Extract chunk size from istream | 				model.lxfmlData.read(reinterpret_cast<char*>(&chunkSize), sizeof(uint32_t)); // Extract chunk size from istream | ||||||
|  |  | ||||||
| 				// Check if good here since if at the end of an sd0 file, this will have eof flagged. | 				// Check if good here since if at the end of an sd0 file, this will have eof flagged. | ||||||
| 				if (!modelAsSd0->good()) break; | 				if (!model.lxfmlData.good()) break; | ||||||
|  |  | ||||||
| 				std::unique_ptr<uint8_t[]> compressedChunk(new uint8_t[chunkSize]); | 				std::unique_ptr<uint8_t[]> compressedChunk(new uint8_t[chunkSize]); | ||||||
| 				for (uint32_t i = 0; i < chunkSize; i++) { | 				for (uint32_t i = 0; i < chunkSize; i++) { | ||||||
| 					compressedChunk[i] = modelAsSd0->get(); | 					compressedChunk[i] = model.lxfmlData.get(); | ||||||
| 				} | 				} | ||||||
|  |  | ||||||
| 				// Ignore the valgrind warning about uninitialized values.  These are discarded later when we know the actual uncompressed size. | 				// Ignore the valgrind warning about uninitialized values.  These are discarded later when we know the actual uncompressed size. | ||||||
| @@ -59,7 +54,7 @@ uint32_t BrickByBrickFix::TruncateBrokenBrickByBrickXml() { | |||||||
| 					completeUncompressedModel.append((char*)uncompressedChunk.get()); | 					completeUncompressedModel.append((char*)uncompressedChunk.get()); | ||||||
| 					completeUncompressedModel.resize(previousSize + actualUncompressedSize); | 					completeUncompressedModel.resize(previousSize + actualUncompressedSize); | ||||||
| 				} else { | 				} else { | ||||||
| 					LOG("Failed to inflate chunk %i for model %llu.  Error: %i", chunkCount, modelId, err); | 					LOG("Failed to inflate chunk %i for model %llu.  Error: %i", chunkCount, model.id, err); | ||||||
| 					break; | 					break; | ||||||
| 				} | 				} | ||||||
| 				chunkCount++; | 				chunkCount++; | ||||||
| @@ -75,26 +70,20 @@ uint32_t BrickByBrickFix::TruncateBrokenBrickByBrickXml() { | |||||||
| 					"</LXFML>", | 					"</LXFML>", | ||||||
| 					completeUncompressedModel.length() >= 15 ? completeUncompressedModel.length() - 15 : 0) == std::string::npos | 					completeUncompressedModel.length() >= 15 ? completeUncompressedModel.length() - 15 : 0) == std::string::npos | ||||||
| 					) { | 					) { | ||||||
| 					LOG("Brick-by-brick model %llu will be deleted!", modelId); | 					LOG("Brick-by-brick model %llu will be deleted!", model.id); | ||||||
| 					ugcModelToDelete->setInt64(1, modelsToTruncate->getInt64(1)); | 					Database::Get()->DeleteUgcModelData(model.id); | ||||||
| 					pcModelToDelete->setInt64(1, modelsToTruncate->getInt64(1)); |  | ||||||
| 					ugcModelToDelete->execute(); |  | ||||||
| 					pcModelToDelete->execute(); |  | ||||||
| 					modelsTruncated++; | 					modelsTruncated++; | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			LOG("Brick-by-brick model %llu will be deleted!", modelId); | 			LOG("Brick-by-brick model %llu will be deleted!", model.id); | ||||||
| 			ugcModelToDelete->setInt64(1, modelsToTruncate->getInt64(1)); | 			Database::Get()->DeleteUgcModelData(model.id); | ||||||
| 			pcModelToDelete->setInt64(1, modelsToTruncate->getInt64(1)); |  | ||||||
| 			ugcModelToDelete->execute(); |  | ||||||
| 			pcModelToDelete->execute(); |  | ||||||
| 			modelsTruncated++; | 			modelsTruncated++; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	Database::Commit(); | 	Database::Get()->Commit(); | ||||||
| 	Database::SetAutoCommit(previousCommitValue); | 	Database::Get()->SetAutoCommit(previousCommitValue); | ||||||
| 	return modelsTruncated; | 	return modelsTruncated; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -106,21 +95,17 @@ uint32_t BrickByBrickFix::TruncateBrokenBrickByBrickXml() { | |||||||
|  */ |  */ | ||||||
| uint32_t BrickByBrickFix::UpdateBrickByBrickModelsToSd0() { | uint32_t BrickByBrickFix::UpdateBrickByBrickModelsToSd0() { | ||||||
| 	uint32_t updatedModels = 0; | 	uint32_t updatedModels = 0; | ||||||
| 	auto modelsToUpdate = GetModelsFromDatabase(); | 	auto modelsToUpdate = Database::Get()->GetAllUgcModels(); | ||||||
| 	auto previousAutoCommitState = Database::GetAutoCommit(); | 	auto previousAutoCommitState = Database::Get()->GetAutoCommit(); | ||||||
| 	Database::SetAutoCommit(false); | 	Database::Get()->SetAutoCommit(false); | ||||||
| 	std::unique_ptr<sql::PreparedStatement> insertionStatement(Database::CreatePreppedStmt("UPDATE ugc SET lxfml = ? WHERE id = ?;")); | 	for (auto& model : modelsToUpdate) { | ||||||
| 	while (modelsToUpdate->next()) { |  | ||||||
| 		int64_t modelId = modelsToUpdate->getInt64(1); |  | ||||||
| 		std::unique_ptr<sql::Blob> oldLxfml(modelsToUpdate->getBlob(2)); |  | ||||||
| 		// Check if the stored blob starts with zlib magic (0x78 0xDA - best compression of zlib) | 		// Check if the stored blob starts with zlib magic (0x78 0xDA - best compression of zlib) | ||||||
| 		// If it does, convert it to sd0. | 		// If it does, convert it to sd0. | ||||||
| 		if (oldLxfml->get() == 0x78 && oldLxfml->get() == 0xDA) { | 		if (model.lxfmlData.get() == 0x78 && model.lxfmlData.get() == 0xDA) { | ||||||
|  |  | ||||||
| 			// Get and save size of zlib compressed chunk. | 			// Get and save size of zlib compressed chunk. | ||||||
| 			oldLxfml->seekg(0, std::ios::end); | 			model.lxfmlData.seekg(0, std::ios::end); | ||||||
| 			uint32_t oldLxfmlSize = static_cast<uint32_t>(oldLxfml->tellg()); | 			uint32_t oldLxfmlSize = static_cast<uint32_t>(model.lxfmlData.tellg()); | ||||||
| 			oldLxfml->seekg(0); | 			model.lxfmlData.seekg(0); | ||||||
|  |  | ||||||
| 			// Allocate 9 extra bytes.  5 for sd0 magic, 4 for the only zlib compressed size. | 			// Allocate 9 extra bytes.  5 for sd0 magic, 4 for the only zlib compressed size. | ||||||
| 			uint32_t oldLxfmlSizeWithHeader = oldLxfmlSize + 9; | 			uint32_t oldLxfmlSizeWithHeader = oldLxfmlSize + 9; | ||||||
| @@ -128,34 +113,27 @@ uint32_t BrickByBrickFix::UpdateBrickByBrickModelsToSd0() { | |||||||
|  |  | ||||||
| 			WriteSd0Magic(sd0ConvertedModel.get(), oldLxfmlSize); | 			WriteSd0Magic(sd0ConvertedModel.get(), oldLxfmlSize); | ||||||
| 			for (uint32_t i = 9; i < oldLxfmlSizeWithHeader; i++) { | 			for (uint32_t i = 9; i < oldLxfmlSizeWithHeader; i++) { | ||||||
| 				sd0ConvertedModel.get()[i] = oldLxfml->get(); | 				sd0ConvertedModel.get()[i] = model.lxfmlData.get(); | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			std::string outputString(sd0ConvertedModel.get(), oldLxfmlSizeWithHeader); | 			std::string outputString(sd0ConvertedModel.get(), oldLxfmlSizeWithHeader); | ||||||
| 			std::istringstream outputStringStream(outputString); | 			std::istringstream outputStringStream(outputString); | ||||||
|  |  | ||||||
| 			insertionStatement->setBlob(1, static_cast<std::istream*>(&outputStringStream)); |  | ||||||
| 			insertionStatement->setInt64(2, modelId); |  | ||||||
| 			try { | 			try { | ||||||
| 				insertionStatement->executeUpdate(); | 				Database::Get()->UpdateUgcModelData(model.id, outputStringStream); | ||||||
| 				LOG("Updated model %i to sd0", modelId); | 				LOG("Updated model %i to sd0", model.id); | ||||||
| 				updatedModels++; | 				updatedModels++; | ||||||
| 			} catch (sql::SQLException exception) { | 			} catch (sql::SQLException exception) { | ||||||
| 				LOG("Failed to update model %i.  This model should be inspected manually to see why." | 				LOG("Failed to update model %i.  This model should be inspected manually to see why." | ||||||
| 					"The database error is %s", modelId, exception.what()); | 					"The database error is %s", model.id, exception.what()); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	Database::Commit(); | 	Database::Get()->Commit(); | ||||||
| 	Database::SetAutoCommit(previousAutoCommitState); | 	Database::Get()->SetAutoCommit(previousAutoCommitState); | ||||||
| 	return updatedModels; | 	return updatedModels; | ||||||
| } | } | ||||||
|  |  | ||||||
| std::unique_ptr<sql::ResultSet> GetModelsFromDatabase() { |  | ||||||
| 	std::unique_ptr<sql::PreparedStatement> modelsRawDataQuery(Database::CreatePreppedStmt("SELECT id, lxfml FROM ugc;")); |  | ||||||
| 	return std::unique_ptr<sql::ResultSet>(modelsRawDataQuery->executeQuery()); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * @brief Writes sd0 magic at the front of a char* |  * @brief Writes sd0 magic at the front of a char* | ||||||
|  * |  * | ||||||
| @@ -171,6 +149,6 @@ void WriteSd0Magic(char* input, uint32_t chunkSize) { | |||||||
| 	*reinterpret_cast<uint32_t*>(input + 5) = chunkSize; // Write the integer to the character array | 	*reinterpret_cast<uint32_t*>(input + 5) = chunkSize; // Write the integer to the character array | ||||||
| } | } | ||||||
|  |  | ||||||
| bool CheckSd0Magic(sql::Blob* streamToCheck) { | bool CheckSd0Magic(std::istream& streamToCheck) { | ||||||
| 	return streamToCheck->get() == 's' && streamToCheck->get() == 'd' && streamToCheck->get() == '0' && streamToCheck->get() == 0x01 && streamToCheck->get() == 0xFF; | 	return streamToCheck.get() == 's' && streamToCheck.get() == 'd' && streamToCheck.get() == '0' && streamToCheck.get() == 0x01 && streamToCheck.get() == 0xFF; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -151,6 +151,11 @@ namespace GeneralUtils { | |||||||
| 		return std::stod(value); | 		return std::stod(value); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	template <> | ||||||
|  | 	inline uint16_t Parse(const char* value) { | ||||||
|  | 		return std::stoul(value); | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	template <> | 	template <> | ||||||
| 	inline uint32_t Parse(const char* value) { | 	inline uint32_t Parse(const char* value) { | ||||||
| 		return std::stoul(value); | 		return std::stoul(value); | ||||||
|   | |||||||
| @@ -3,6 +3,8 @@ | |||||||
| // Custom Classes
 | // Custom Classes
 | ||||||
| #include "CDTable.h" | #include "CDTable.h" | ||||||
| 
 | 
 | ||||||
|  | #include <unordered_map> | ||||||
|  | 
 | ||||||
| enum class eReplicaComponentType : uint32_t; | enum class eReplicaComponentType : uint32_t; | ||||||
| struct CDComponentsRegistry { | struct CDComponentsRegistry { | ||||||
| 	unsigned int id;                    //!< The LOT is used as the ID
 | 	unsigned int id;                    //!< The LOT is used as the ID
 | ||||||
| @@ -42,9 +42,9 @@ std::vector<CDFeatureGating> CDFeatureGatingTable::Query(std::function<bool(CDFe | |||||||
| 	return data; | 	return data; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool CDFeatureGatingTable::FeatureUnlocked(const std::string& feature) const { | bool CDFeatureGatingTable::FeatureUnlocked(const CDFeatureGating& feature) const { | ||||||
| 	for (const auto& entry : entries) { | 	for (const auto& entry : entries) { | ||||||
| 		if (entry.featureName == feature) { | 		if (entry.featureName == feature.featureName && entry >= feature) { | ||||||
| 			return true; | 			return true; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -9,6 +9,12 @@ struct CDFeatureGating { | |||||||
| 	int32_t current; | 	int32_t current; | ||||||
| 	int32_t minor; | 	int32_t minor; | ||||||
| 	std::string description; | 	std::string description; | ||||||
|  | 
 | ||||||
|  | 	bool operator>=(const CDFeatureGating& b) const { | ||||||
|  | 		return 	(this->major > b.major) || | ||||||
|  | 				(this->major == b.major && this->current > b.current) || | ||||||
|  | 				(this->major == b.major && this->current == b.current && this->minor >= b.minor); | ||||||
|  | 	} | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| class CDFeatureGatingTable : public CDTable<CDFeatureGatingTable> { | class CDFeatureGatingTable : public CDTable<CDFeatureGatingTable> { | ||||||
| @@ -21,7 +27,7 @@ public: | |||||||
| 	// Queries the table with a custom "where" clause
 | 	// Queries the table with a custom "where" clause
 | ||||||
| 	std::vector<CDFeatureGating> Query(std::function<bool(CDFeatureGating)> predicate); | 	std::vector<CDFeatureGating> Query(std::function<bool(CDFeatureGating)> predicate); | ||||||
| 
 | 
 | ||||||
| 	bool FeatureUnlocked(const std::string& feature) const; | 	bool FeatureUnlocked(const CDFeatureGating& feature) const; | ||||||
| 
 | 
 | ||||||
| 	const std::vector<CDFeatureGating>& GetEntries(void) const; | 	const std::vector<CDFeatureGating>& GetEntries(void) const; | ||||||
| }; | }; | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| set(DDATABASE_TABLES_SOURCES "CDActivitiesTable.cpp" | set(DDATABASE_CDCLIENTDATABASE_CDCLIENTTABLES_SOURCES "CDActivitiesTable.cpp" | ||||||
| 	"CDActivityRewardsTable.cpp" | 	"CDActivityRewardsTable.cpp" | ||||||
| 	"CDAnimationsTable.cpp" | 	"CDAnimationsTable.cpp" | ||||||
| 	"CDBehaviorParameterTable.cpp" | 	"CDBehaviorParameterTable.cpp" | ||||||
							
								
								
									
										12
									
								
								dDatabase/CDClientDatabase/CMakeLists.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								dDatabase/CDClientDatabase/CMakeLists.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | set(DDATABASE_CDCLIENTDATABASE_SOURCES | ||||||
|  | 	"CDClientDatabase.cpp" | ||||||
|  | 	"CDClientManager.cpp" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | add_subdirectory(CDClientTables) | ||||||
|  |  | ||||||
|  | foreach(file ${DDATABASE_CDCLIENTDATABASE_CDCLIENTTABLES_SOURCES}) | ||||||
|  | 	set(DDATABASE_CDCLIENTDATABASE_SOURCES ${DDATABASE_CDCLIENTDATABASE_SOURCES} "CDClientTables/${file}") | ||||||
|  | endforeach() | ||||||
|  |  | ||||||
|  | set(DDATABASE_CDCLIENTDATABASE_SOURCES ${DDATABASE_CDCLIENTDATABASE_SOURCES} PARENT_SCOPE) | ||||||
| @@ -1,12 +1,15 @@ | |||||||
| set(DDATABASE_SOURCES "CDClientDatabase.cpp" | set(DDATABASE_SOURCES) | ||||||
| 		"CDClientManager.cpp" |  | ||||||
| 		"Database.cpp" |  | ||||||
| 		"MigrationRunner.cpp") |  | ||||||
|  |  | ||||||
| add_subdirectory(Tables) | add_subdirectory(CDClientDatabase) | ||||||
|  |  | ||||||
| foreach(file ${DDATABASE_TABLES_SOURCES}) | foreach(file ${DDATABASE_CDCLIENTDATABASE_SOURCES}) | ||||||
| 	set(DDATABASE_SOURCES ${DDATABASE_SOURCES} "Tables/${file}") | 	set(DDATABASE_SOURCES ${DDATABASE_SOURCES} "CDClientDatabase/${file}") | ||||||
|  | endforeach() | ||||||
|  |  | ||||||
|  | add_subdirectory(GameDatabase) | ||||||
|  |  | ||||||
|  | foreach(file ${DDATABASE_GAMEDATABASE_SOURCES}) | ||||||
|  | 	set(DDATABASE_SOURCES ${DDATABASE_SOURCES} "GameDatabase/${file}") | ||||||
| endforeach() | endforeach() | ||||||
|  |  | ||||||
| add_library(dDatabase STATIC ${DDATABASE_SOURCES}) | add_library(dDatabase STATIC ${DDATABASE_SOURCES}) | ||||||
|   | |||||||
| @@ -1,118 +0,0 @@ | |||||||
| #include "Database.h" |  | ||||||
| #include "Game.h" |  | ||||||
| #include "dConfig.h" |  | ||||||
| #include "Logger.h" |  | ||||||
| using namespace std; |  | ||||||
|  |  | ||||||
| #pragma warning (disable:4251) //Disables SQL warnings |  | ||||||
|  |  | ||||||
| sql::Driver* Database::driver; |  | ||||||
| sql::Connection* Database::con; |  | ||||||
| sql::Properties Database::props; |  | ||||||
| std::string Database::database; |  | ||||||
|  |  | ||||||
| void Database::Connect(const string& host, const string& database, const string& username, const string& password) { |  | ||||||
|  |  | ||||||
| 	//To bypass debug issues: |  | ||||||
| 	const char* szDatabase = database.c_str(); |  | ||||||
| 	const char* szUsername = username.c_str(); |  | ||||||
| 	const char* szPassword = password.c_str(); |  | ||||||
|  |  | ||||||
| 	driver = sql::mariadb::get_driver_instance(); |  | ||||||
|  |  | ||||||
| 	sql::Properties properties; |  | ||||||
| 	// The mariadb connector is *supposed* to handle unix:// and pipe:// prefixes to hostName, but there are bugs where |  | ||||||
| 	// 1) it tries to parse a database from the connection string (like in tcp://localhost:3001/darkflame) based on the |  | ||||||
| 	//    presence of a / |  | ||||||
| 	// 2) even avoiding that, the connector still assumes you're connecting with a tcp socket |  | ||||||
| 	// So, what we do in the presence of a unix socket or pipe is to set the hostname to the protocol and localhost, |  | ||||||
| 	// which avoids parsing errors while still ensuring the correct connection type is used, and then setting the appropriate |  | ||||||
| 	// property manually (which the URL parsing fails to do) |  | ||||||
| 	const std::string UNIX_PROTO = "unix://"; |  | ||||||
| 	const std::string PIPE_PROTO = "pipe://"; |  | ||||||
|     if (host.find(UNIX_PROTO) == 0) { |  | ||||||
| 		properties["hostName"] = "unix://localhost"; |  | ||||||
| 		properties["localSocket"] = host.substr(UNIX_PROTO.length()).c_str(); |  | ||||||
|     } else if (host.find(PIPE_PROTO) == 0) { |  | ||||||
| 		properties["hostName"] = "pipe://localhost"; |  | ||||||
| 		properties["pipe"] = host.substr(PIPE_PROTO.length()).c_str(); |  | ||||||
|     } else { |  | ||||||
| 		properties["hostName"] = host.c_str(); |  | ||||||
|     } |  | ||||||
| 	properties["user"] = szUsername; |  | ||||||
| 	properties["password"] = szPassword; |  | ||||||
| 	properties["autoReconnect"] = "true"; |  | ||||||
|  |  | ||||||
| 	Database::props = properties; |  | ||||||
| 	Database::database = database; |  | ||||||
|  |  | ||||||
| 	Database::Connect(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Database::Connect() { |  | ||||||
| 	// `connect(const Properties& props)` segfaults in windows debug, but |  | ||||||
| 	// `connect(const SQLString& host, const SQLString& user, const SQLString& pwd)` doesn't handle pipes/unix sockets correctly |  | ||||||
| 	if (Database::props.find("localSocket") != Database::props.end() || Database::props.find("pipe") != Database::props.end()) { |  | ||||||
| 		con = driver->connect(Database::props); |  | ||||||
| 	} else { |  | ||||||
| 		con = driver->connect(Database::props["hostName"].c_str(), Database::props["user"].c_str(), Database::props["password"].c_str()); |  | ||||||
| 	} |  | ||||||
| 	con->setSchema(Database::database.c_str()); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Database::Destroy(std::string source, bool log) { |  | ||||||
| 	if (!con) return; |  | ||||||
|  |  | ||||||
| 	if (log) { |  | ||||||
| 		if (source != "") LOG("Destroying MySQL connection from %s!", source.c_str()); |  | ||||||
| 		else LOG("Destroying MySQL connection!"); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	con->close(); |  | ||||||
| 	delete con; |  | ||||||
| } //Destroy |  | ||||||
|  |  | ||||||
| sql::Statement* Database::CreateStmt() { |  | ||||||
| 	sql::Statement* toReturn = con->createStatement(); |  | ||||||
| 	return toReturn; |  | ||||||
| } //CreateStmt |  | ||||||
|  |  | ||||||
| sql::PreparedStatement* Database::CreatePreppedStmt(const std::string& query) { |  | ||||||
| 	const char* test = query.c_str(); |  | ||||||
| 	size_t size = query.length(); |  | ||||||
| 	sql::SQLString str(test, size); |  | ||||||
|  |  | ||||||
| 	if (!con) { |  | ||||||
| 		Connect(); |  | ||||||
| 		LOG("Trying to reconnect to MySQL"); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	if (!con->isValid() || con->isClosed()) { |  | ||||||
| 		delete con; |  | ||||||
|  |  | ||||||
| 		con = nullptr; |  | ||||||
|  |  | ||||||
| 		Connect(); |  | ||||||
| 		LOG("Trying to reconnect to MySQL from invalid or closed connection"); |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	auto* stmt = con->prepareStatement(str); |  | ||||||
|  |  | ||||||
| 	return stmt; |  | ||||||
| } //CreatePreppedStmt |  | ||||||
|  |  | ||||||
| void Database::Commit() { |  | ||||||
| 	Database::con->commit(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| bool Database::GetAutoCommit() { |  | ||||||
| 	// TODO This should not just access a pointer.  A future PR should update this |  | ||||||
| 	// to check for null and throw an error if the connection is not valid. |  | ||||||
| 	return con->getAutoCommit(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void Database::SetAutoCommit(bool value) { |  | ||||||
| 	// TODO This should not just access a pointer.  A future PR should update this |  | ||||||
| 	// to check for null and throw an error if the connection is not valid. |  | ||||||
| 	Database::con->setAutoCommit(value); |  | ||||||
| } |  | ||||||
| @@ -1,31 +0,0 @@ | |||||||
| #pragma once |  | ||||||
|  |  | ||||||
| #include <string> |  | ||||||
| #include <conncpp.hpp> |  | ||||||
|  |  | ||||||
| class MySqlException : public std::runtime_error { |  | ||||||
| public: |  | ||||||
| 	MySqlException() : std::runtime_error("MySQL error!") {} |  | ||||||
| 	MySqlException(const std::string& msg) : std::runtime_error(msg.c_str()) {} |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| class Database { |  | ||||||
| private: |  | ||||||
| 	static sql::Driver* driver; |  | ||||||
| 	static sql::Connection* con; |  | ||||||
| 	static sql::Properties props; |  | ||||||
| 	static std::string database; |  | ||||||
| public: |  | ||||||
| 	static void Connect(const std::string& host, const std::string& database, const std::string& username, const std::string& password); |  | ||||||
| 	static void Connect(); |  | ||||||
| 	static void Destroy(std::string source = "", bool log = true); |  | ||||||
|  |  | ||||||
| 	static sql::Statement* CreateStmt(); |  | ||||||
| 	static sql::PreparedStatement* CreatePreppedStmt(const std::string& query); |  | ||||||
| 	static void Commit(); |  | ||||||
| 	static bool GetAutoCommit(); |  | ||||||
| 	static void SetAutoCommit(bool value); |  | ||||||
|  |  | ||||||
| 	static std::string GetDatabase() { return database; } |  | ||||||
| 	static sql::Properties GetProperties() { return props; } |  | ||||||
| }; |  | ||||||
							
								
								
									
										12
									
								
								dDatabase/GameDatabase/CMakeLists.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								dDatabase/GameDatabase/CMakeLists.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | set(DDATABASE_GAMEDATABASE_SOURCES | ||||||
|  | 	"Database.cpp" | ||||||
|  | 	"MigrationRunner.cpp" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | add_subdirectory(MySQL) | ||||||
|  |  | ||||||
|  | foreach(file ${DDATABSE_DATABSES_MYSQL_SOURCES}) | ||||||
|  | 	set(DDATABASE_GAMEDATABASE_SOURCES ${DDATABASE_GAMEDATABASE_SOURCES} "MySQL/${file}") | ||||||
|  | endforeach() | ||||||
|  |  | ||||||
|  | set(DDATABASE_GAMEDATABASE_SOURCES ${DDATABASE_GAMEDATABASE_SOURCES} PARENT_SCOPE) | ||||||
							
								
								
									
										40
									
								
								dDatabase/GameDatabase/Database.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								dDatabase/GameDatabase/Database.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | #include "Database.h" | ||||||
|  | #include "Game.h" | ||||||
|  | #include "dConfig.h" | ||||||
|  | #include "Logger.h" | ||||||
|  | #include "MySQLDatabase.h" | ||||||
|  | #include "DluAssert.h" | ||||||
|  |  | ||||||
|  | #pragma warning (disable:4251) //Disables SQL warnings | ||||||
|  |  | ||||||
|  | namespace { | ||||||
|  | 	GameDatabase* database = nullptr; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void Database::Connect() { | ||||||
|  | 	if (database) { | ||||||
|  | 		LOG("Tried to connect to database when it's already connected!"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	database = new MySQLDatabase(); | ||||||
|  | 	database->Connect(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | GameDatabase* Database::Get() { | ||||||
|  | 	if (!database) { | ||||||
|  | 		LOG("Tried to get database when it's not connected!"); | ||||||
|  | 		Connect(); | ||||||
|  | 	} | ||||||
|  | 	return database; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void Database::Destroy(std::string source) { | ||||||
|  | 	if (database) { | ||||||
|  | 		database->Destroy(source); | ||||||
|  | 		delete database; | ||||||
|  | 		database = nullptr; | ||||||
|  | 	} else { | ||||||
|  | 		LOG("Trying to destroy database when it's not connected!"); | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										12
									
								
								dDatabase/GameDatabase/Database.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								dDatabase/GameDatabase/Database.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | #pragma once | ||||||
|  |  | ||||||
|  | #include <string> | ||||||
|  | #include <conncpp.hpp> | ||||||
|  |  | ||||||
|  | #include "GameDatabase.h" | ||||||
|  |  | ||||||
|  | namespace Database { | ||||||
|  | 	void Connect(); | ||||||
|  | 	GameDatabase* Get(); | ||||||
|  | 	void Destroy(std::string source = ""); | ||||||
|  | }; | ||||||
							
								
								
									
										55
									
								
								dDatabase/GameDatabase/GameDatabase.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								dDatabase/GameDatabase/GameDatabase.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | |||||||
|  | #ifndef __GAMEDATABASE__H__ | ||||||
|  | #define __GAMEDATABASE__H__ | ||||||
|  |  | ||||||
|  | #include <optional> | ||||||
|  |  | ||||||
|  | #include "ILeaderboard.h" | ||||||
|  | #include "IPlayerCheatDetections.h" | ||||||
|  | #include "ICommandLog.h" | ||||||
|  | #include "IMail.h" | ||||||
|  | #include "IObjectIdTracker.h" | ||||||
|  | #include "IPlayKeys.h" | ||||||
|  | #include "IServers.h" | ||||||
|  | #include "IBugReports.h" | ||||||
|  | #include "IPropertyContents.h" | ||||||
|  | #include "IProperty.h" | ||||||
|  | #include "IPetNames.h" | ||||||
|  | #include "ICharXml.h" | ||||||
|  | #include "IMigrationHistory.h" | ||||||
|  | #include "IUgc.h" | ||||||
|  | #include "IFriends.h" | ||||||
|  | #include "ICharInfo.h" | ||||||
|  | #include "IAccounts.h" | ||||||
|  | #include "IActivityLog.h" | ||||||
|  |  | ||||||
|  | namespace sql { | ||||||
|  | 	class Statement; | ||||||
|  | 	class PreparedStatement; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | #ifdef _DEBUG | ||||||
|  | #  define DLU_SQL_TRY_CATCH_RETHROW(x) do { try { x; } catch (sql::SQLException& ex) { LOG("SQL Error: %s", ex.what()); throw; } } while(0) | ||||||
|  | #else | ||||||
|  | #  define DLU_SQL_TRY_CATCH_RETHROW(x) x | ||||||
|  | #endif // _DEBUG | ||||||
|  |  | ||||||
|  | class GameDatabase : | ||||||
|  | 	public IPlayKeys, public ILeaderboard, public IObjectIdTracker, public IServers, | ||||||
|  | 	public IMail, public ICommandLog, public IPlayerCheatDetections, public IBugReports, | ||||||
|  | 	public IPropertyContents, public IProperty, public IPetNames, public ICharXml, | ||||||
|  | 	public IMigrationHistory, public IUgc, public IFriends, public ICharInfo, | ||||||
|  | 	public IAccounts, public IActivityLog { | ||||||
|  | public: | ||||||
|  | 	virtual ~GameDatabase() = default; | ||||||
|  | 	// TODO: These should be made private. | ||||||
|  | 	virtual void Connect() = 0; | ||||||
|  | 	virtual void Destroy(std::string source = "") = 0; | ||||||
|  | 	virtual void ExecuteCustomQuery(const std::string_view query) = 0; | ||||||
|  | 	virtual sql::PreparedStatement* CreatePreppedStmt(const std::string& query) = 0; | ||||||
|  | 	virtual void Commit() = 0; | ||||||
|  | 	virtual bool GetAutoCommit() = 0; | ||||||
|  | 	virtual void SetAutoCommit(bool value) = 0; | ||||||
|  | 	virtual void DeleteCharacter(const uint32_t characterId) = 0; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | #endif  //!__GAMEDATABASE__H__ | ||||||
							
								
								
									
										37
									
								
								dDatabase/GameDatabase/ITables/IAccounts.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								dDatabase/GameDatabase/ITables/IAccounts.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | #ifndef __IACCOUNTS__H__ | ||||||
|  | #define __IACCOUNTS__H__ | ||||||
|  |  | ||||||
|  | #include <cstdint> | ||||||
|  | #include <optional> | ||||||
|  | #include <string_view> | ||||||
|  |  | ||||||
|  | enum class eGameMasterLevel : uint8_t; | ||||||
|  |  | ||||||
|  | class IAccounts { | ||||||
|  | public: | ||||||
|  | 	struct Info { | ||||||
|  | 		std::string bcryptPassword; | ||||||
|  | 		uint32_t id{}; | ||||||
|  | 		uint32_t playKeyId{}; | ||||||
|  | 		bool banned{}; | ||||||
|  | 		bool locked{}; | ||||||
|  | 		eGameMasterLevel maxGmLevel{}; | ||||||
|  | 	}; | ||||||
|  |  | ||||||
|  | 	// Get the account info for the given username. | ||||||
|  | 	virtual std::optional<IAccounts::Info> GetAccountInfo(const std::string_view username) = 0; | ||||||
|  |  | ||||||
|  | 	// Update the account's unmute time. | ||||||
|  | 	virtual void UpdateAccountUnmuteTime(const uint32_t accountId, const uint64_t timeToUnmute) = 0; | ||||||
|  |  | ||||||
|  | 	// Update the account's ban status. | ||||||
|  | 	virtual void UpdateAccountBan(const uint32_t accountId, const bool banned) = 0; | ||||||
|  |  | ||||||
|  | 	// Update the account's password. | ||||||
|  | 	virtual void UpdateAccountPassword(const uint32_t accountId, const std::string_view bcryptpassword) = 0; | ||||||
|  |  | ||||||
|  | 	// Add a new account to the database. | ||||||
|  | 	virtual void InsertNewAccount(const std::string_view username, const std::string_view bcryptpassword) = 0; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | #endif  //!__IACCOUNTS__H__ | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	 Aaron Kimbre
					Aaron Kimbre