/* * Darkflame Universe * Copyright 2018 */ #include "ClientPackets.h" #include "dCommonVars.h" #include "LDFFormat.h" #include "ZCompression.h" namespace ClientPackets { void LoadStaticZone::Serialize(RakNet::BitStream &bitStream) const { bitStream.Write(zoneID.GetMapID()); bitStream.Write(zoneID.GetInstanceID()); bitStream.Write(zoneID.GetCloneID()); bitStream.Write(checksum); bitStream.Write(editorEnabled); bitStream.Write(editorLevel); bitStream.Write(position.x); bitStream.Write(position.y); bitStream.Write(position.z); bitStream.Write(instanceType); } void CharacterCreationResponse::Serialize(RakNet::BitStream &bitStream) const { bitStream.Write(response); } void CharacterRenameResponse::Serialize(RakNet::BitStream &bitStream) const { bitStream.Write(response); } void CharacterDeleteResponse::Serialize(RakNet::BitStream &bitStream) const { bitStream.Write(success); } void TransferToWorld::Serialize(RakNet::BitStream &bitStream) const { bitStream.Write(LUString(serverIP)); bitStream.Write(serverPort); bitStream.Write(mythranShift); } void ServerState::Serialize(RakNet::BitStream &bitStream) const { bitStream.Write(serverReady); } void CreateCharacter::Serialize(RakNet::BitStream &bitStream) const { RakNet::BitStream data; data.Write(7); //LDF key count LDFData(u"objid", objid).WriteToPacket(data); LDFData(u"template", templateID).WriteToPacket(data);; LDFData(u"name", name).WriteToPacket(data);; LDFData(u"gmlevel", static_cast(gmLevel)).WriteToPacket(data);; LDFData(u"chatmode", static_cast(chatMode)).WriteToPacket(data);; LDFData(u"xmlData", xmlData).WriteToPacket(data);; LDFData(u"reputation", reputation).WriteToPacket(data);; //Compress the data before sending: const uint32_t reservedSize = ZCompression::GetMaxCompressedLength(data.GetNumberOfBytesUsed()); uint8_t* compressedData = new uint8_t[reservedSize]; if (!compressedData) { throw std::runtime_error("Failed to allocate memory for compressed data"); } size_t size = ZCompression::Compress(data.GetData(), data.GetNumberOfBytesUsed(), compressedData, reservedSize); assert(size <= reservedSize); bitStream.Write(size + 9); //size of data + header bytes (8) bitStream.Write(1); //compressed boolean, true bitStream.Write(data.GetNumberOfBytesUsed()); bitStream.Write(size); /** * In practice, this warning serves no purpose for us. We allocate the max memory needed on the heap * and then compress the data. In the off chance that the compression actually increases the size, * an assertion is done to prevent bad data from being saved or sent. */ #pragma warning(disable:6385) // C6385 Reading invalid data from 'compressedData'. bitStream.WriteAlignedBytes(compressedData, size); #pragma warning(default:6385) delete[] compressedData; }; void ChatModerationString::Serialize(RakNet::BitStream &bitStream) const { bitStream.Write(rejectedWords.empty()); // Accepted bitStream.Write(0); // padding bitStream.Write(chatChannel); bitStream.Write(chatMode); bitStream.Write(LUWString(receiver, 42)); for (auto it : rejectedWords) { bitStream.Write(it.first); // start index bitStream.Write(it.second); // length } // Pad out the rest of the packet // The client expects 64 items, so we need to write 64 - rejectedWords.size() empty items for (int i = rejectedWords.size(); 64 > i; i++) { bitStream.Write(0); } } void GMLevelChange::Serialize(RakNet::BitStream &bitStream) const { bitStream.Write(success); bitStream.Write(static_cast(highestLevel)); bitStream.Write(static_cast(prevLevel)); bitStream.Write(static_cast(newLevel)); } void DebugOutput::Serialize(RakNet::BitStream &bitStream) const { bitStream.Write(data.size()); bitStream.Write(data); } void HTTPMonitorInfoResponse::Serialize(RakNet::BitStream &bitStream) const { bitStream.Write(port); bitStream.Write(openWeb); bitStream.Write(supportsSum); bitStream.Write(supportsDetail); bitStream.Write(supportsWho); bitStream.Write(supportsObjects); } }