From c36eda289a2b4f9315d50901e53716be1303b959 Mon Sep 17 00:00:00 2001 From: Li Date: Mon, 5 Sep 2022 14:55:16 +1200 Subject: [PATCH] Remove all MemoryStreams from PacketBuilder, use Byte[] and List only. --- .../LibHISP/Server/PacketBuilder.cs | 834 ++++++++---------- 1 file changed, 374 insertions(+), 460 deletions(-) diff --git a/HorseIsleServer/LibHISP/Server/PacketBuilder.cs b/HorseIsleServer/LibHISP/Server/PacketBuilder.cs index 81be215..5d89314 100755 --- a/HorseIsleServer/LibHISP/Server/PacketBuilder.cs +++ b/HorseIsleServer/LibHISP/Server/PacketBuilder.cs @@ -1,33 +1,32 @@ using System; using System.Collections.Generic; using System.Globalization; -using System.IO; using System.Text; + using HISP.Game; using HISP.Game.SwfModules; - +using HISP.Util; + namespace HISP.Server { public class PacketBuilder { - public const byte PACKET_TERMINATOR = 0x00; public const byte PACKET_CLIENT_TERMINATOR = 0x0A; - public const byte PACKET_FLASH_XML_CROSSDOMAIN = 0x3C; public const byte PACKET_LOGIN = 0x7F; public const byte PACKET_CHAT = 0x14; public const byte PACKET_MOVE = 0x15; public const byte PACKET_CLICK = 0x77; - public const byte PACKET_USERINFO = 0x81; + public const byte PACKET_SEC_CODE = 0x81; public const byte PACKET_WORLD = 0x7A; public const byte PACKET_BASE_STATS = 0x7B; public const byte PACKET_RANCH = 0x23; public const byte PACKET_SWF_CUTSCENE = 0x29; public const byte PACKET_SWF_MODULE_FORCE = 0x28; public const byte PACKET_SWF_MODULE_GENTLE = 0x2A; - public const byte PACKET_PLACE_INFO = 0x1E; + public const byte PACKET_META = 0x1E; public const byte PACKET_BIRDMAP = 0x76; public const byte PACKET_HORSE = 0x19; public const byte PACKET_AREA_DEFS = 0x79; @@ -227,6 +226,27 @@ namespace HISP.Server public const byte DIRECTION_TELEPORT = 4; public const byte DIRECTION_NONE = 10; + // Helper function for packets that return map data, (eg CreateMovementPacket or CreateBirdMap) + // To encode tile data and add it to a given packet represented as a List. + private static void encodeTileDataAndAddToPacket(List packet, int tileId, int otileId) + { + if (tileId >= 190) + { + packet.Add((byte)190); + tileId -= 100; + } + packet.Add((byte)tileId); + + if (otileId >= 190) + { + packet.Add((byte)190); + otileId -= 100; + } + packet.Add((byte)otileId); + } + + // Creates a byte array of a packet to inform the client that Player 2 in the current 2Player game + // has actually left / quit the game. public static byte[] Create2PlayerClose() { byte[] packet = new byte[3]; @@ -236,105 +256,100 @@ namespace HISP.Server return packet; } + + // Creates a byte array of a packet to inform the client that a peice in a dressup room + // was moved to another location. public static byte[] CreateDressupRoomPeiceMove(int peiceId, double x, double y, bool active) { - MemoryStream ms = new MemoryStream(); - ms.WriteByte(PACKET_SWFMODULE); - string packetStr = ""; - packetStr += peiceId.ToString() + "|"; + string peiceMoveStr = ""; + peiceMoveStr += peiceId.ToString() + "|"; if (active) { - packetStr += x.ToString() + "|"; - packetStr += y.ToString() + "|"; + peiceMoveStr += x.ToString() + "|"; + peiceMoveStr += y.ToString() + "|"; } else { - packetStr += "D|D|"; + peiceMoveStr += "D|D|"; } - packetStr += "^"; + peiceMoveStr += "^"; + byte[] peiceMoveBytes = Encoding.UTF8.GetBytes(peiceMoveStr); + + byte[] packet = new byte[(1 * 2) + peiceMoveBytes.Length]; - byte[] packetBytes = Encoding.UTF8.GetBytes(packetStr); - ms.Write(packetBytes, 0x00, packetBytes.Length); - ms.WriteByte(PACKET_TERMINATOR); - ms.Seek(0x00, SeekOrigin.Begin); - byte[] response = ms.ToArray(); - ms.Dispose(); - return response; + packet[0] = PACKET_SWFMODULE; + Array.Copy(peiceMoveBytes, 0, packet, 1, peiceMoveBytes.Length); + packet[packet.Length - 1] = PACKET_TERMINATOR; + + return packet; } - public static byte[] CreateDressupRoomPeiceResponse(Dressup.DressupPeice[] dressupPeices) + // Creates a byte array of a packet to inform the client of all the peices + // in a given dressup room. + public static byte[] CreateDressupRoomPeiceLoad(Dressup.DressupPeice[] dressupPeices) { - MemoryStream ms = new MemoryStream(); - ms.WriteByte(PACKET_SWFMODULE); - string packetStr = ""; + string peiceLoadStr = ""; foreach(Dressup.DressupPeice peice in dressupPeices) { if (!peice.Active) continue; - packetStr += peice.PeiceId.ToString() + "|"; - packetStr += peice.X.ToString() + "|"; - packetStr += peice.Y.ToString() + "|"; - packetStr += "^"; + peiceLoadStr += peice.PeiceId.ToString() + "|"; + peiceLoadStr += peice.X.ToString() + "|"; + peiceLoadStr += peice.Y.ToString() + "|"; + peiceLoadStr += "^"; } - byte[] packetBytes = Encoding.UTF8.GetBytes(packetStr); - ms.Write(packetBytes, 0x00, packetBytes.Length); - ms.WriteByte(PACKET_TERMINATOR); - ms.Seek(0x00, SeekOrigin.Begin); - byte[] response = ms.ToArray(); - ms.Dispose(); - return response; + byte[] peiceLoadBytes = Encoding.UTF8.GetBytes(peiceLoadStr); + byte[] packet = new byte[(1 * 2) + peiceLoadBytes.Length]; + + packet[0] = PACKET_SWFMODULE; + Array.Copy(peiceLoadBytes, 0, packet, 1, peiceLoadBytes.Length); + packet[packet.Length - 1] = PACKET_TERMINATOR; + + return packet; } - + // Creates a byte array that contains the contents of a request byte array + // as the response, it basically just forwards it onwards + // this is used for *most* SwfModule public static byte[] CreateForwardedSwfRequest(byte[] request) { - MemoryStream ms = new MemoryStream(); - ms.WriteByte(PACKET_SWFMODULE); - ms.Write(request, 0x2, request.Length - 0x4); - ms.WriteByte(PACKET_TERMINATOR); - ms.Seek(0x00, SeekOrigin.Begin); - byte[] response = ms.ToArray(); - ms.Dispose(); - return response; + byte[] packet = new byte[(1 * 2) + request.Length]; + packet[0] = PACKET_SWFMODULE; + Array.Copy(request, 0, packet, 1, request.Length); + packet[packet.Length - 1] = PACKET_TERMINATOR; + return packet; } - - public static byte[] CreateBirdMap(int playerX, int playerY) + // Creates a byte array that contains "Bird Map" data + // From a given X/Y Position, this is primarily used to handle + // using the telescope item in game + public static byte[] CreateBirdMap(int X, int Y) { - MemoryStream ms = new MemoryStream(); + // The size is always fixed in this case, but i still have to use a List because + // encodeTileDataAndAddToPacket expects packet as a List. + List packet = new List(); - int xstart = playerX - 24; - int ystart = playerY - 15; + // Calculate top left corner of BirdMap viewport + // from given X/Y position. + int startX = X - 24; + int startY = Y - 15; - ms.WriteByte(PACKET_BIRDMAP); + packet.Add(PACKET_BIRDMAP); for (int rely = 0; rely <= 30; rely++) { for (int relx = 0; relx <= 48; relx++) { - int tileId = Map.GetTileId(xstart + relx, ystart + rely, false); - int otileId = Map.GetTileId(xstart + relx, ystart + rely, true); - - if (tileId >= 190) - { - ms.WriteByte((byte)190); - tileId -= 100; - } - ms.WriteByte((byte)tileId); - - if (otileId >= 190) - { - ms.WriteByte((byte)190); - otileId -= 100; - } - ms.WriteByte((byte)otileId); - + int tileId = Map.GetTileId(startX + relx, startY + rely, false); + int otileId = Map.GetTileId(startX + relx, startY + rely, true); + encodeTileDataAndAddToPacket(packet, tileId, otileId); } } - ms.WriteByte(PACKET_TERMINATOR); - ms.Seek(0x00, SeekOrigin.Begin); - return ms.ToArray(); + packet.Add(PACKET_TERMINATOR); + + return packet.ToArray(); } + // Creates a byte array for a packet to inform the client that the image in a drawing room has changed. public static byte[] CreateDrawingUpdatePacket(string Drawing) { byte[] drawingBytes = Encoding.UTF8.GetBytes(Drawing); @@ -346,15 +361,16 @@ namespace HISP.Server return packet; } + // Creates a byte array for a packet to inform the client that a poetry peice in a brick poet room has moved. public static byte[] CreateBrickPoetMovePacket(Brickpoet.PoetryPeice peice) { - string packetStr = "|"; - packetStr += peice.Id + "|"; - packetStr += peice.X + "|"; - packetStr += peice.Y + "|"; - packetStr += "^"; + string peiceUpdateStr = "|"; + peiceUpdateStr += peice.Id + "|"; + peiceUpdateStr += peice.X + "|"; + peiceUpdateStr += peice.Y + "|"; + peiceUpdateStr += "^"; - byte[] infoBytes = Encoding.UTF8.GetBytes(packetStr); + byte[] infoBytes = Encoding.UTF8.GetBytes(peiceUpdateStr); byte[] packet = new byte[(1 * 3) + infoBytes.Length]; packet[0] = PACKET_SWFMODULE; @@ -366,32 +382,36 @@ namespace HISP.Server return packet; } + // Creates a byte array for a packet to inform the client of all all Poetry Peices in a Brick Poet room public static byte[] CreateBrickPoetListPacket(Brickpoet.PoetryPeice[] room) { - MemoryStream ms = new MemoryStream(); - ms.WriteByte(PACKET_SWFMODULE); - string packetStr = ""; + string peicesStr = ""; foreach(Brickpoet.PoetryPeice peice in room) { - packetStr += "A"; - packetStr += "|"; - packetStr += peice.Id; - packetStr += "|"; - packetStr += peice.Word.ToUpper(); - packetStr += "|"; - packetStr += peice.X; - packetStr += "|"; - packetStr += peice.Y; - packetStr += "|"; - packetStr += "^"; + peicesStr += "A"; + peicesStr += "|"; + peicesStr += peice.Id; + peicesStr += "|"; + peicesStr += peice.Word.ToUpper(); + peicesStr += "|"; + peicesStr += peice.X; + peicesStr += "|"; + peicesStr += peice.Y; + peicesStr += "|"; + peicesStr += "^"; } - byte[] packetBytes = Encoding.UTF8.GetBytes(packetStr); - ms.Write(packetBytes, 0x00, packetBytes.Length); - ms.WriteByte(PACKET_TERMINATOR); + byte[] peicesBytes = Encoding.UTF8.GetBytes(peicesStr); + byte[] packet = new byte[(1 * 2) + peicesBytes.Length]; - ms.Seek(0x00, SeekOrigin.Begin); - return ms.ToArray(); + packet[0] = PACKET_SWFMODULE; + + Array.Copy(peicesBytes, 0, packet, 1, peicesBytes.Length); + + packet[packet.Length - 1] = PACKET_TERMINATOR; + + return packet; } + // Creates a byte array of a packet requesting the client to play a sound effect. public static byte[] CreatePlaysoundPacket(string sound) { byte[] soundBytes = Encoding.UTF8.GetBytes(sound); @@ -405,6 +425,8 @@ namespace HISP.Server return packet; } + // Creates a byte array of a packet informing the client that a given user has left the game + // So they can be removed from the chat list public static byte[] CreatePlayerLeavePacket(string username) { byte[] userBytes = Encoding.UTF8.GetBytes(username); @@ -419,6 +441,8 @@ namespace HISP.Server return packet; } + // Creates a byte array of a packet informing the client that a given player has changed position, + // changed direction, or changed character sprites public static byte[] CreatePlayerInfoUpdateOrCreate(int x, int y, int facing, int charId, string username) { byte[] userBytes = Encoding.UTF8.GetBytes(username); @@ -444,14 +468,15 @@ namespace HISP.Server return packet; } - - public static byte[] CreateLoginPacket(bool Success, string message="") + // Creates a byte array of a packet to inform the client + // if a given Login Attempt was successful or not + public static byte[] CreateLoginPacket(bool Success, string ErrorMessage="") { - byte[] loginFailMessage = Encoding.UTF8.GetBytes(message); + byte[] loginFailMessage = Encoding.UTF8.GetBytes(ErrorMessage); byte[] packet = new byte[(1 * 3) + loginFailMessage.Length]; packet[0] = PACKET_LOGIN; - if (message != "") + if (ErrorMessage != "") packet[1] = LOGIN_CUSTOM_MESSAGE; else if (Success) packet[1] = LOGIN_SUCCESS; @@ -464,8 +489,10 @@ namespace HISP.Server return packet; } - - public static byte[] CreateProfilePacket(string userProfile) + // Creates a byte array of a packet to inform the client of + // the users current profile page, or "about me" + // This is for the the "Profile" button + public static byte[] CreateProfilePage(string userProfile) { byte[] profileBytes = Encoding.UTF8.GetBytes(userProfile); byte[] packet = new byte[(1 * 2) + profileBytes.Length]; @@ -476,205 +503,138 @@ namespace HISP.Server return packet; } - + // Creates a byte array of a packet to inform the client of the players + // new X/Y position, there character id, facing direction, and Tile Data for their position in the map. public static byte[] CreateMovementPacket(int x, int y, int charId, int facing, int direction, bool walk) - { - // Header information + { + + /* Packet HEADER */ + + // Packet size varries too much and so using a dynamically sized list of bytes instead of a byte[] List packet = new List(); - packet.Add(PACKET_MOVE); + packet.Add(PACKET_MOVE); // 0x0 - packet.Add((byte)(((x-4) / 64) + 20)); //1 - packet.Add((byte)(((x-4) % 64) + 20)); //2 + packet.Add((byte)(((x-4) / 64) + 20)); //0x1 + packet.Add((byte)(((x-4) % 64) + 20)); //0x2 - packet.Add((byte)(((y-1) / 64) + 20)); //3 - packet.Add((byte)(((y-1) % 64) + 20)); //4 + packet.Add((byte)(((y-1) / 64) + 20)); //0x3 + packet.Add((byte)(((y-1) % 64) + 20)); //0x4 - packet.Add((byte)(facing + 20)); //5 + packet.Add((byte)(facing + 20)); //0x5 - packet.Add((byte)((charId / 64) + 20)); //6 - packet.Add((byte)((charId % 64) + 20)); //7 - packet.Add((byte)(direction + 20)); //8 - packet.Add((byte)(Convert.ToInt32(walk) + 20)); //9 + packet.Add((byte)((charId / 64) + 20)); //0x6 + packet.Add((byte)((charId % 64) + 20)); //0x7 + packet.Add((byte)(direction + 20)); //0x8 + packet.Add((byte)(Convert.ToInt32(walk) + 20)); //0x9 - - // Map Data + /* Packet PAYLOAD */ bool moveTwo = false; - if(direction >= 20) + if(direction >= 20) // is the player riding a horse? { direction -= 20; moveTwo = true; } - int ystart = y - 4; - int xstart = x - 6; - int xend = xstart + 12; - int yend = ystart + 9; + // Calculate start of the client's viewport start offset from top-left origin + int startY = y - 4; + int startX = x - 6; + int endX = startX + 12; + int endY = startY + 9; + + // This giant if case logic essentially + // Pulls the missing tile data portion from map file + // And encodes it into packet data if (direction == DIRECTION_UP) { int totalY = 0; if (moveTwo) { - ystart++; + startY++; totalY = 1; } - - for (int yy = ystart; yy >= ystart - totalY; yy--) + + for (int curY = startY; curY >= startY - totalY; curY--) { - for (int xx = xstart; xx <= xend; xx++) + for (int curX = startX; curX <= endX; curX++) { - int tileId = Map.GetTileId(xx, yy, false); - int otileId = Map.GetTileId(xx, yy, true); - - if (tileId >= 190) - { - packet.Add((byte)190); - tileId -= 100; - } - packet.Add((byte)tileId); - - if (otileId >= 190) - { - packet.Add((byte)190); - otileId -= 100; - } - packet.Add((byte)otileId); + int tileId = Map.GetTileId(curX, curY, false); + int otileId = Map.GetTileId(curX, curY, true); + encodeTileDataAndAddToPacket(packet, tileId, otileId); } } } - - if (direction == DIRECTION_LEFT) + else if (direction == DIRECTION_LEFT) { int totalX = 0; if (moveTwo) { - xstart++; + startX++; totalX = 1; } - for (int xx = xstart; xx >= xstart - totalX; xx--) + for (int curX = startX; curX >= startX - totalX; curX--) { - for (int yy = ystart; yy <= yend; yy++) + for (int curY = startY; curY <= endY; curY++) { - int tileId = Map.GetTileId(xx, yy, false); - int otileId = Map.GetTileId(xx, yy, true); - - - - if (tileId >= 190) - { - packet.Add((byte)190); - tileId -= 100; - } - packet.Add((byte)tileId); - - if (otileId >= 190) - { - packet.Add((byte)190); - otileId -= 100; - } - packet.Add((byte)otileId); + int tileId = Map.GetTileId(curX, curY, false); + int otileId = Map.GetTileId(curX, curY, true); + encodeTileDataAndAddToPacket(packet, tileId, otileId); } } } - - - if (direction == DIRECTION_RIGHT) + else if (direction == DIRECTION_RIGHT) { int totalX = 0; if (moveTwo) { - xend--; + endX--; totalX = 1; } - for (int xx = xend; xx <= xend + totalX; xx++) + for (int curX = endX; curX <= endX + totalX; curX++) { - for (int yy = ystart; yy <= yend; yy++) + for (int curY = startY; curY <= endY; curY++) { - int tileId = Map.GetTileId(xx, yy, false); - int otileId = Map.GetTileId(xx, yy, true); - - - if (tileId >= 190) - { - packet.Add((byte)190); - tileId -= 100; - } - packet.Add((byte)tileId); - - if (otileId >= 190) - { - packet.Add((byte)190); - otileId -= 100; - } - packet.Add((byte)otileId); + int tileId = Map.GetTileId(curX, curY, false); + int otileId = Map.GetTileId(curX, curY, true); + encodeTileDataAndAddToPacket(packet, tileId, otileId); } } } - - if (direction == DIRECTION_DOWN) + else if (direction == DIRECTION_DOWN) { int totalY = 0; if (moveTwo) { - yend--; + endY--; totalY = 1; } - for (int yy = yend; yy <= yend + totalY; yy++) + for (int curY = endY; curY <= endY + totalY; curY++) { - for (int xx = xstart; xx <= xend; xx++) + for (int curX = startX; curX <= endX; curX++) { - int tileId = Map.GetTileId(xx, yy, false); - int otileId = Map.GetTileId(xx, yy, true); - - - if (tileId >= 190) - { - packet.Add((byte)190); - tileId -= 100; - } - packet.Add((byte)tileId); - - if (otileId >= 190) - { - packet.Add((byte)190); - otileId -= 100; - } - packet.Add((byte)otileId); - + int tileId = Map.GetTileId(curX, curY, false); + int otileId = Map.GetTileId(curX, curY, true); + encodeTileDataAndAddToPacket(packet, tileId, otileId); } } } - if (direction == DIRECTION_TELEPORT) + else if (direction == DIRECTION_TELEPORT) { for(int rely = 0; rely <= 9; rely++) { for (int relx = 0; relx <= 12; relx++) { - int tileId = Map.GetTileId(xstart + relx, ystart + rely, false); - int otileId = Map.GetTileId(xstart + relx, ystart + rely, true); - - if(tileId >= 190) - { - packet.Add((byte)190); - tileId -= 100; - } - packet.Add((byte)tileId); - - if (otileId >= 190) - { - packet.Add((byte)190); - otileId -= 100; - } - packet.Add((byte)otileId); - + int tileId = Map.GetTileId(startX + relx, startY + rely, false); + int otileId = Map.GetTileId(startX + relx, startY + rely, true); + encodeTileDataAndAddToPacket(packet, tileId, otileId); } } @@ -683,322 +643,276 @@ namespace HISP.Server return packet.ToArray(); } - - public static byte[] CreateClickTileInfoPacket(string text) + // Creates a byte array of a packet containing Information about a specific tile + // used when you click on a tile in the client, it gives you some extra info about it. + public static byte[] CreateTileClickInfo(string text) { byte[] strBytes = Encoding.UTF8.GetBytes(text); - MemoryStream ms = new MemoryStream(); - ms.WriteByte(PACKET_CLICK); - ms.Write(strBytes, 0x00, strBytes.Length); - ms.WriteByte(PACKET_TERMINATOR); - - ms.Seek(0x00, SeekOrigin.Begin); - byte[] Packet = ms.ToArray(); - ms.Dispose(); - - return Packet; + byte[] packet = new byte[(1 * 2) + strBytes.Length]; + packet[0] = PACKET_CLICK; + Array.Copy(strBytes, 0, packet, 1, strBytes.Length); + packet[packet.Length - 1] = PACKET_TERMINATOR; + return packet; } - public static byte[] CreateMetaPacket(string formattedText) + // Creates a byte array of a packet containing information to be displayed in the "Meta" window + // (Thats the one on the top-right corner of the screent hat contains buttons and other widgets) + public static byte[] CreateMeta(string formattedText) { - byte[] strBytes = Encoding.UTF8.GetBytes(formattedText); + byte[] formattedBytes = Encoding.UTF8.GetBytes(formattedText); + byte[] packet = new byte[(1 * 2) + formattedBytes.Length]; - MemoryStream ms = new MemoryStream(); + packet[0] = PACKET_META; + Array.Copy(formattedBytes, 0, packet, 1, formattedBytes.Length); + packet[packet.Length - 1] = PACKET_TERMINATOR; - ms.WriteByte(PACKET_PLACE_INFO); - ms.Write(strBytes, 0x00, strBytes.Length); - ms.WriteByte(PACKET_TERMINATOR); - - ms.Seek(0x00, SeekOrigin.Begin); - byte[] Packet = ms.ToArray(); - ms.Dispose(); - - return Packet; + return packet; } + // Creates a byte array of a packet informing the client to display a chat message + // And which of the two chat windows to include it in. public static byte[] CreateChat(string formattedText, byte chatWindow) { - byte[] strBytes = Encoding.UTF8.GetBytes(formattedText); - - MemoryStream ms = new MemoryStream(); - - ms.WriteByte(PACKET_CHAT); - ms.WriteByte(chatWindow); - - ms.Write(strBytes, 0x00, strBytes.Length); - - ms.WriteByte(PACKET_TERMINATOR); - - ms.Seek(0x00, SeekOrigin.Begin); - byte[] Packet = ms.ToArray(); - ms.Dispose(); - - return Packet; + byte[] formattedBytes = Encoding.UTF8.GetBytes(formattedText); + byte[] packet = new byte[(1 * 3) + formattedBytes.Length]; + packet[0] = PACKET_CHAT; + packet[1] = chatWindow; + Array.Copy(formattedBytes, 0, packet, 2, formattedBytes.Length); + packet[packet.Length - 1] = PACKET_TERMINATOR; + return packet; } + // Creates a byte array of a packet informing the client to change the current Weather Effect. public static byte[] CreateWeatherUpdatePacket(string newWeather) { - byte[] strBytes = Encoding.UTF8.GetBytes(newWeather); - - MemoryStream ms = new MemoryStream(); - ms.WriteByte(PACKET_WORLD); - ms.WriteByte(WEATHER_UPDATE); - ms.Write(strBytes, 0x00, strBytes.Length); - ms.WriteByte(PACKET_TERMINATOR); - - ms.Seek(0x00, SeekOrigin.Begin); - byte[] Packet = ms.ToArray(); - ms.Dispose(); - - return Packet; + byte[] weatherBytes = Encoding.UTF8.GetBytes(newWeather); + byte[] packet = new byte[(1 * 3) + weatherBytes.Length]; + packet[0] = PACKET_WORLD; + packet[1] = WEATHER_UPDATE; + Array.Copy(weatherBytes, 0, packet, 2, weatherBytes.Length); + packet[packet.Length - 1] = PACKET_TERMINATOR; + return packet; } - - public static byte[] CreateWorldData(int gameTime, int gameDay, int gameYear, string weather) + // Creates a byte array of a packet informing the client of the current game time, and weather effect. + public static byte[] CreateTimeAndWeatherUpdate(int gameTime, int gameDay, int gameYear, string weather) { - byte[] strBytes = Encoding.UTF8.GetBytes(weather); + byte[] weatherBytes = Encoding.UTF8.GetBytes(weather); - MemoryStream ms = new MemoryStream(); - ms.WriteByte(PACKET_WORLD); + byte[] packet = new byte[(1 * 7) + weatherBytes.Length]; + + packet[0] = PACKET_WORLD; + // Encode current time + packet[1] = (byte)((gameTime / 64) + 20); + packet[2] = (byte)((gameTime % 64) + 20); + // Encode current day + packet[3] = (byte)((gameDay / 64) + 20); + packet[4] = (byte)((gameDay % 64) + 20); + // Encode current year + packet[5] = (byte)((gameYear / 64) + 20); + packet[6] = (byte)((gameYear % 64) + 20); + + // Copy weather information to packet + Array.Copy(weatherBytes, 0, packet, 7, weatherBytes.Length); - ms.WriteByte((byte)((gameTime / 64) + 20)); - ms.WriteByte((byte)((gameTime % 64) + 20)); + packet[packet.Length - 1] = PACKET_TERMINATOR; - ms.WriteByte((byte)((gameDay / 64) + 20)); - ms.WriteByte((byte)((gameDay % 64) + 20)); - - ms.WriteByte((byte)((gameYear / 64) + 20)); - ms.WriteByte((byte)((gameYear % 64) + 20)); - - ms.Write(strBytes,0x00, strBytes.Length); - ms.WriteByte(PACKET_TERMINATOR); - - ms.Seek(0x00, SeekOrigin.Begin); - byte[] Packet = ms.ToArray(); - ms.Dispose(); - - return Packet; + return packet; } - + // Creates a byte array of a "keep alive" packet, to check if the client is still connected + // and to inform the client the server is still here too and has not crashed / disconnected the client. public static byte[] CreateKeepAlive() { - MemoryStream ms = new MemoryStream(); - - ms.WriteByte(PACKET_KEEP_ALIVE); - ms.WriteByte(PACKET_TERMINATOR); - - ms.Seek(0x00, SeekOrigin.Begin); - byte[] Packet = ms.ToArray(); - ms.Dispose(); - - return Packet; + byte[] packet = new byte[2]; + packet[0] = PACKET_KEEP_ALIVE; + packet[1] = PACKET_TERMINATOR; + return packet; } + // Creates a byte array of a packet to inform the client of all "Places" that exist in the world + // (as defined by gamedata json) This is used in the map view when you hover over certain areas public static byte[] CreatePlaceData(World.Isle[] isles, World.Town[] towns, World.Area[] areas) { - MemoryStream ms = new MemoryStream(); - ms.WriteByte(PACKET_AREA_DEFS); + // As this information is defined by gamedata.json file + // the size of it can vary alot, so im using a List instead of a byte[] here. + List packet = new List(); + packet.Add(PACKET_AREA_DEFS); - // Write Towns + // Encode Towns foreach (World.Town town in towns) { - byte[] strBytes = Encoding.UTF8.GetBytes(town.Name); + byte[] townBytes = Encoding.UTF8.GetBytes(town.Name); - ms.WriteByte(AREA_SEPERATOR); - ms.WriteByte(AREA_TOWN); - - ms.WriteByte((byte)(((town.StartX - 4) / 64) + 20)); - ms.WriteByte((byte)(((town.StartX - 4) % 64) + 20)); - - ms.WriteByte((byte)(((town.EndX - 4) / 64) + 20)); - ms.WriteByte((byte)(((town.EndX - 4) % 64) + 20)); - - ms.WriteByte((byte)(((town.StartY - 1) / 64) + 20)); - ms.WriteByte((byte)(((town.StartY - 1) % 64) + 20)); - - ms.WriteByte((byte)(((town.EndY - 1) / 64) + 20)); - ms.WriteByte((byte)(((town.EndY - 1) % 64) + 20)); + packet.Add(AREA_SEPERATOR); + packet.Add(AREA_TOWN); + + packet.Add((byte)(((town.StartX - 4) / 64) + 20)); + packet.Add((byte)(((town.StartX - 4) % 64) + 20)); + + packet.Add((byte)(((town.EndX - 4) / 64) + 20)); + packet.Add((byte)(((town.EndX - 4) % 64) + 20)); + + packet.Add((byte)(((town.StartY - 1) / 64) + 20)); + packet.Add((byte)(((town.StartY - 1) % 64) + 20)); + + packet.Add((byte)(((town.EndY - 1) / 64) + 20)); + packet.Add((byte)(((town.EndY - 1) % 64) + 20)); - ms.Write(strBytes, 0x00, strBytes.Length); + Helper.ByteArrayToByteList(townBytes, packet); } - // Write Areas + // Encode Areas foreach (World.Area area in areas) { - byte[] strBytes = Encoding.UTF8.GetBytes(area.Name); + byte[] areaBytes = Encoding.UTF8.GetBytes(area.Name); - ms.WriteByte(AREA_SEPERATOR); - ms.WriteByte(AREA_AREA); + packet.Add(AREA_SEPERATOR); + packet.Add(AREA_AREA); - ms.WriteByte((byte)(((area.StartX - 4) / 64) + 20)); - ms.WriteByte((byte)(((area.StartX - 4) % 64) + 20)); + packet.Add((byte)(((area.StartX - 4) / 64) + 20)); + packet.Add((byte)(((area.StartX - 4) % 64) + 20)); - ms.WriteByte((byte)(((area.EndX - 4) / 64) + 20)); - ms.WriteByte((byte)(((area.EndX - 4) % 64) + 20)); - - ms.WriteByte((byte)(((area.StartY - 1) / 64) + 20)); - ms.WriteByte((byte)(((area.StartY - 1) % 64) + 20)); - - ms.WriteByte((byte)(((area.EndY - 1) / 64) + 20)); - ms.WriteByte((byte)(((area.EndY - 1) % 64) + 20)); + packet.Add((byte)(((area.EndX - 4) / 64) + 20)); + packet.Add((byte)(((area.EndX - 4) % 64) + 20)); + + packet.Add((byte)(((area.StartY - 1) / 64) + 20)); + packet.Add((byte)(((area.StartY - 1) % 64) + 20)); + + packet.Add((byte)(((area.EndY - 1) / 64) + 20)); + packet.Add((byte)(((area.EndY - 1) % 64) + 20)); - ms.Write(strBytes, 0x00, strBytes.Length); + Helper.ByteArrayToByteList(areaBytes, packet); } - // Write Isles + // Encode Isles foreach (World.Isle isle in isles) { - byte[] strBytes = Encoding.UTF8.GetBytes(isle.Name); + byte[] isleBytes = Encoding.UTF8.GetBytes(isle.Name); - ms.WriteByte(AREA_SEPERATOR); - ms.WriteByte(AREA_ISLE); + packet.Add(AREA_SEPERATOR); + packet.Add(AREA_ISLE); - ms.WriteByte((byte)(((isle.StartX - 4) / 64) + 20)); - ms.WriteByte((byte)(((isle.StartX - 4) % 64) + 20)); + packet.Add((byte)(((isle.StartX - 4) / 64) + 20)); + packet.Add((byte)(((isle.StartX - 4) % 64) + 20)); - ms.WriteByte((byte)(((isle.EndX - 4) / 64) + 20)); - ms.WriteByte((byte)(((isle.EndX - 4) % 64) + 20)); + packet.Add((byte)(((isle.EndX - 4) / 64) + 20)); + packet.Add((byte)(((isle.EndX - 4) % 64) + 20)); - ms.WriteByte((byte)(((isle.StartY - 1) / 64) + 20)); - ms.WriteByte((byte)(((isle.StartY - 1) % 64) + 20)); + packet.Add((byte)(((isle.StartY - 1) / 64) + 20)); + packet.Add((byte)(((isle.StartY - 1) % 64) + 20)); - ms.WriteByte((byte)(((isle.EndY - 1) / 64) + 20)); - ms.WriteByte((byte)(((isle.EndY - 1) % 64) + 20)); + packet.Add((byte)(((isle.EndY - 1) / 64) + 20)); + packet.Add((byte)(((isle.EndY - 1) % 64) + 20)); - ms.WriteByte((byte)isle.Tileset.ToString()[0]); - - ms.Write(strBytes, 0x00, strBytes.Length); + packet.Add((byte)isle.Tileset.ToString()[0]); + + Helper.ByteArrayToByteList(isleBytes, packet); } - ms.WriteByte(PACKET_TERMINATOR); + packet.Add(PACKET_TERMINATOR); - ms.Seek(0x00, SeekOrigin.Begin); - byte[] Packet = ms.ToArray(); - ms.Dispose(); - - return Packet; + return packet.ToArray(); } - - public static byte[] CreatePlayerData(int money, int playerCount, int mail) + // Creates a byte array of a packet informing the client of the players money, total player count and, + // how many mail messages they have. + public static byte[] CreateMoneyPlayerCountAndMail(int money, int playerCount, int mail) { - byte[] moneyStrBytes = Encoding.UTF8.GetBytes(money.ToString("N0", CultureInfo.InvariantCulture)); - byte[] playerStrBytes = Encoding.UTF8.GetBytes(playerCount.ToString("N0", CultureInfo.InvariantCulture)); - byte[] mailStrBytes = Encoding.UTF8.GetBytes(mail.ToString("N0", CultureInfo.InvariantCulture)); + byte[] playerDataBytes = Encoding.UTF8.GetBytes(money.ToString("N0", CultureInfo.InvariantCulture) + "|" + + playerCount.ToString("N0", CultureInfo.InvariantCulture) + "|" + + mail.ToString("N0", CultureInfo.InvariantCulture) + "|"); - MemoryStream ms = new MemoryStream(); - ms.WriteByte(PACKET_BASE_STATS); - ms.Write(moneyStrBytes, 0x00, moneyStrBytes.Length); - ms.WriteByte((byte)'|'); - ms.Write(playerStrBytes, 0x00, playerStrBytes.Length); - ms.WriteByte((byte)'|'); - ms.Write(mailStrBytes, 0x00, mailStrBytes.Length); - ms.WriteByte((byte)'|'); - ms.WriteByte(PACKET_TERMINATOR); - ms.Seek(0x00, SeekOrigin.Begin); - byte[] Packet = ms.ToArray(); - ms.Dispose(); - - return Packet; + byte[] packet = new byte[(1*2) + playerDataBytes.Length]; + packet[0] = PACKET_BASE_STATS; + Array.Copy(playerDataBytes, 0, packet, 1, playerDataBytes.Length); + packet[packet.Length - 1] = PACKET_TERMINATOR; + return packet; } - + // Creates a byte array of a packet informing the client of Tile Overlay flags + // these tell the client what tiles are and are not passable, which ones the player + // should appear ontop of or under, and stuff like that. public static byte[] CreateTileOverlayFlags(int[] tileDepthFlags) { - MemoryStream ms = new MemoryStream(); - ms.WriteByte(PACKET_TILE_FLAGS); + byte[] packet = new byte[(1 * 2) + tileDepthFlags.Length]; + packet[0] = PACKET_TILE_FLAGS; - foreach(int tileDepthFlag in tileDepthFlags) - { - ms.WriteByte((byte)tileDepthFlag.ToString()[0]); - } + for(int i = 0; i < tileDepthFlags.Length; i++) + packet[1 + i] = (byte)(tileDepthFlags[i].ToString()[0]); - ms.WriteByte(PACKET_TERMINATOR); - - ms.Seek(0x00, SeekOrigin.Begin); - byte[] Packet = ms.ToArray(); - ms.Dispose(); - - return Packet; + packet[packet.Length - 1] = PACKET_TERMINATOR; + return packet; } - + // Creates a byte array of a packet informing the client of its current Sec Code seed and Inc values, + // Some client packets (eg minigame rewards) require this special Message Authentication Code to validate them + // Its not at all secure, you can easily just forge these packets by just implementing sec codes, but i didnt make it -- public static byte[] CreateSecCode(byte[] SecCodeSeed, int SecCodeInc, bool Admin, bool Moderator) - { - MemoryStream ms = new MemoryStream(); - ms.WriteByte(PACKET_USERINFO); - - ms.WriteByte((byte)(SecCodeSeed[0] + 33)); - ms.WriteByte((byte)(SecCodeSeed[1] + 33)); - ms.WriteByte((byte)(SecCodeSeed[2] + 33)); - ms.WriteByte((byte)(SecCodeInc + 33)); - + { + char userType = 'N'; // Normal? + if (Moderator) - userType = 'M'; + userType = 'M'; // Moderator + if (Admin) - userType = 'A'; + userType = 'A'; // Admin - ms.WriteByte((byte)userType); - ms.WriteByte(PACKET_TERMINATOR); + byte[] packet = new byte[7]; - ms.Seek(0x00, SeekOrigin.Begin); - byte[] Packet = ms.ToArray(); - ms.Dispose(); + packet[0] = PACKET_SEC_CODE; - return Packet; + packet[1] = (byte)(SecCodeSeed[0] + 33); + packet[2] = (byte)(SecCodeSeed[1] + 33); + packet[3] = (byte)(SecCodeSeed[2] + 33); + packet[4] = (byte)(SecCodeInc + 33); + + + packet[5] = (byte)userType; + packet[6] = PACKET_TERMINATOR; + + return packet; } - public static byte[] CreateSwfModulePacket(string swf,byte type) + // Creates a byte array of a packet to tell the client to please GET + // a certain SWF in the mod/ directory on web, and then load it as a MovieClip + // into the actual game client. + public static byte[] CreateSwfModule(string swf,byte headerByte) { - MemoryStream ms = new MemoryStream(); - ms.WriteByte(type); - byte[] strBytes = Encoding.UTF8.GetBytes(swf); - ms.Write(strBytes, 0x00, strBytes.Length); - ms.WriteByte(PACKET_TERMINATOR); + byte[] swfBytes = Encoding.UTF8.GetBytes(swf); + byte[] packet = new byte[(1 * 2) + swfBytes.Length]; + + packet[0] = headerByte; + Array.Copy(swfBytes, 0, packet, 1, swfBytes.Length); + packet[packet.Length - 1] = PACKET_TERMINATOR; - ms.Seek(0x00, SeekOrigin.Begin); - byte[] Packet = ms.ToArray(); - ms.Dispose(); - return Packet; + return packet; } + // Creates a byte array of a packet to show the client an "Annoucement" message + // This has the exact same effect as CreateChat with CHAT_BOTTOM_RIGHT but for some reason + // the header byte is different, + // This is basically only used for MOTD. public static byte[] CreateAnnouncement(string announcement) { - MemoryStream ms = new MemoryStream(); - ms.WriteByte(PACKET_ANNOUNCEMENT); - byte[] strBytes = Encoding.UTF8.GetBytes(announcement); - ms.Write(strBytes, 0x00, strBytes.Length); - ms.WriteByte(PACKET_TERMINATOR); + byte[] annouceBytes = Encoding.UTF8.GetBytes(announcement); + byte[] packet = new byte[(1 * 2) + annouceBytes.Length]; - ms.Seek(0x00, SeekOrigin.Begin); - byte[] Packet = ms.ToArray(); - ms.Dispose(); + packet[0] = PACKET_ANNOUNCEMENT; + Array.Copy(annouceBytes, 0, packet, 1, annouceBytes.Length); + packet[packet.Length - 1] = PACKET_TERMINATOR; - return Packet; + return packet; } - + // Creates a byte array of a packet informing the clent that they have been kicked from the server + // and includes a reason for them being kicked, public static byte[] CreateKickMessage(string reason) { - MemoryStream ms = new MemoryStream(); - ms.WriteByte(PACKET_KICK); - byte[] strBytes = Encoding.UTF8.GetBytes(reason); - ms.Write(strBytes, 0x00, strBytes.Length); - ms.WriteByte(PACKET_TERMINATOR); + byte[] kickMsgBytes = Encoding.UTF8.GetBytes(reason); + byte[] packet = new byte[(1 * 2) + kickMsgBytes.Length]; - ms.Seek(0x00, SeekOrigin.Begin); - byte[] Packet = ms.ToArray(); - ms.Dispose(); + packet[0] = PACKET_KICK; + Array.Copy(kickMsgBytes, 0, packet, 1, kickMsgBytes.Length); + packet[packet.Length - 1] = PACKET_TERMINATOR; - return Packet; - } - - public static byte[] CreateMotd() - { - string formattedMotd = Messages.FormatMOTD(); - return CreateAnnouncement(formattedMotd); - } - public static byte[] CreateWelcomeMessage(string username) - { - string formattedStr = Messages.FormatWelcomeMessage(username); - return CreateChat(formattedStr, CHAT_BOTTOM_RIGHT); + return packet; } }