Support channel packet format.

This commit is contained in:
Ray Koopa 2019-01-04 03:21:28 +01:00
parent 1d8c090957
commit 7e801057ac
15 changed files with 201 additions and 60 deletions

View File

@ -74,7 +74,7 @@ namespace Syroot.Worms.OnlineWorms.Server
EndPoint = new IPEndPoint(IPAddress.Loopback, 1),
Type = ChannelType.Normal,
Color = Color.LightGreen,
UserLoad = 5
Load = ChannelLoad.Highest
},
new ChannelInfo
{
@ -87,7 +87,7 @@ namespace Syroot.Worms.OnlineWorms.Server
Name = "Boredom Time",
EndPoint = new IPEndPoint(IPAddress.Loopback, 3),
Type = ChannelType.Normal,
UserLoad = 3
Load = ChannelLoad.Medium
},
new ChannelInfo
{
@ -96,7 +96,7 @@ namespace Syroot.Worms.OnlineWorms.Server
Type = ChannelType.Roping,
Color = Color.Orange,
Coins = 2,
UserLoad = 6
Load = ChannelLoad.Closed
},
new ChannelInfo
{
@ -105,7 +105,7 @@ namespace Syroot.Worms.OnlineWorms.Server
Type = ChannelType.Roping,
Color = Color.Orange,
Coins = 1,
UserLoad = 5
Load = ChannelLoad.Full
},
new ChannelInfo
{
@ -126,8 +126,12 @@ namespace Syroot.Worms.OnlineWorms.Server
});
}
public void HandleChannelConnect(ChannelConnectQuery packet)
{
}
#if DEBUG
public void HandleRaw(RawQuery packet) { }
public void HandleRaw(RawPacket packet) { }
#endif
// ---- METHODS (PROTECTED) ------------------------------------------------------------------------------------

View File

@ -16,6 +16,9 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
// ---- CONSTANTS ----------------------------------------------------------------------------------------------
private const int _maxDataSize = 2048;
private const ushort _channelPacketStartTag = 0xFFFE;
private const byte _channelPacketStartTag2 = 1;
private const ushort _channelPacketEndTag = 0xFEFF;
// ---- FIELDS -------------------------------------------------------------------------------------------------
@ -123,19 +126,39 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
private Packet ReceivePacket()
{
// Receive the raw packet data.
ushort id;
PacketType type;
int id;
ushort dataSize;
try
{
id = _receiveStream.ReadUInt16();
dataSize = _receiveStream.ReadUInt16();
_receiveStream.ReadAll(_receiveBuffer, 0, dataSize);
// Handle both server and channel specific packet formats.
ushort tag = _receiveStream.ReadUInt16();
if (tag == _channelPacketStartTag)
{
type = PacketType.Channel;
if (_receiveStream.Read1Byte() != 1)
throw new IOException("Invalid channel packet start tag.");
dataSize = _receiveStream.ReadUInt16();
id = _receiveStream.Read1Byte();
_receiveStream.ReadAll(_receiveBuffer, 0, dataSize - sizeof(byte));
_ = _receiveStream.Read1Byte(); // ?
ushort endTag = _receiveStream.ReadUInt16();
if (endTag != _channelPacketEndTag)
throw new IOException("Invalid channel packet end tag.");
}
else
{
type = PacketType.Server;
id = tag;
dataSize = _receiveStream.ReadUInt16();
_receiveStream.ReadAll(_receiveBuffer, 0, dataSize);
}
}
catch (IOException) { return null; } // The underlying socket closed.
catch (ObjectDisposedException) { return null; } // The underlying stream closed.
// Deserialize and return the packet.
Packet packet = PacketFactory.Create(id);
Packet packet = PacketFactory.Create(type, id);
using (PacketStream readStream = new PacketStream(new MemoryStream(_receiveBuffer, 0, dataSize, false)))
packet.Deserialize(readStream);
return packet;
@ -153,9 +176,25 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
// Send the data and return success.
try
{
_receiveStream.WriteUInt16(PacketFactory.GetID(packet));
_receiveStream.WriteUInt16(dataSize);
_receiveStream.Write(_sendDataBuffer, 0, dataSize);
PacketAttribute attribute = PacketFactory.GetAttribute(packet);
switch (attribute.PacketType)
{
case PacketType.Channel:
_receiveStream.WriteUInt16(_channelPacketStartTag);
_receiveStream.WriteByte(1);
_receiveStream.WriteUInt16(dataSize);
_receiveStream.Write(_sendDataBuffer, 0, dataSize);
_receiveStream.WriteByte(0); // ?
_receiveStream.Write(_channelPacketEndTag);
break;
case PacketType.Server:
_receiveStream.WriteUInt16((ushort)attribute.PacketID);
_receiveStream.WriteUInt16(dataSize);
_receiveStream.Write(_sendDataBuffer, 0, dataSize);
break;
default:
throw new InvalidOperationException("Cannot send unknown packet type.");
}
return true;
}
catch (IOException) { return false; } // A network error appeared, and communication should end.

View File

@ -15,9 +15,10 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
/// packet it represents.
/// </summary>
/// <param name="id">The ID of the packet which the decorated class represents.</param>
public PacketAttribute(ushort id)
public PacketAttribute(PacketType type, int id)
{
ID = id;
PacketType = type;
PacketID = id;
}
// ---- PROPERTIES ---------------------------------------------------------------------------------------------
@ -25,6 +26,14 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
/// <summary>
/// Gets the ID of the packet the decorated class represents.
/// </summary>
public ushort ID { get; }
public int PacketID { get; }
public PacketType PacketType { get; }
}
internal enum PacketType
{
Server,
Channel
}
}

View File

@ -12,7 +12,7 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
{
// ---- FIELDS -------------------------------------------------------------------------------------------------
private static readonly Dictionary<ushort, Type> _packetTypes = new Dictionary<ushort, Type>();
private static readonly Dictionary<PacketAttribute, Type> _packetMetas = new Dictionary<PacketAttribute, Type>();
// ---- CONSTRUCTORS & DESTRUCTOR ------------------------------------------------------------------------------
@ -21,11 +21,14 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
// Cache the ID-to-packet-class map.
foreach (Type type in Assembly.GetExecutingAssembly().GetTypes())
{
foreach (PacketAttribute packetAttribute in type.GetCustomAttributes<PacketAttribute>())
foreach (PacketAttribute attrib in type.GetCustomAttributes<PacketAttribute>())
{
if (_packetTypes.ContainsKey(packetAttribute.ID))
throw new InvalidOperationException($"Packet {packetAttribute.ID} mapped to multiple classes.");
_packetTypes.Add(packetAttribute.ID, type);
if (_packetMetas.ContainsKey(attrib))
{
throw new InvalidOperationException(
$"{attrib.PacketType} packet with ID {attrib.PacketID} mapped to multiple classes.");
}
_packetMetas.Add(attrib, type);
}
}
}
@ -38,30 +41,33 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
/// <param name="id">The ID of the packet class to instantiate.</param>
/// <returns>The created <see cref="Packet"/> instance.</returns>
/// <exception cref="KeyNotFoundException">No class was mapped to the given packet ID.</exception>
internal static Packet Create(ushort id)
internal static Packet Create(PacketType type, int id)
{
foreach (KeyValuePair<PacketAttribute, Type> packetMeta in _packetMetas)
{
if (packetMeta.Key.PacketType == type && packetMeta.Key.PacketID == id)
return (Packet)Activator.CreateInstance(packetMeta.Value, true);
}
// No packet metadata matching.
#if DEBUG
return _packetTypes.TryGetValue(id, out Type type)
? (Packet)Activator.CreateInstance(type, true)
: new RawQuery(id);
#else
return (Packet)Activator.CreateInstance(_packetTypes[id], true);
return new RawPacket(type, id);
#endif
throw new ArgumentException($"{type} packet with ID {id} could not be created.");
}
/// <summary>
/// Gets the ID for the class of the given <paramref name="packet"/>.
/// Gets the <see cref="PacketAttribute"/> metadata for the class of the given <paramref name="packet"/>.
/// </summary>
/// <param name="packet">The packet, whose class ID will be returned.</param>
/// <returns>The ID of the <see cref="Packet"/> class.</returns>
internal static ushort GetID(Packet packet)
/// <param name="packet">The <see cref="Packet"/> whose metadata will be returned.</param>
/// <returns>The metadata of the packet class.</returns>
internal static PacketAttribute GetAttribute(Packet packet)
{
#if DEBUG
if (packet is RawQuery rawPacket)
return rawPacket.ID;
if (packet is RawPacket rawPacket)
return new PacketAttribute(rawPacket.Type, rawPacket.ID);
else
#endif
return _packetTypes.Where(x => x.Value == packet.GetType()).First().Key;
return _packetMetas.Where(x => x.Value == packet.GetType()).First().Key;
}
}
}

View File

@ -91,6 +91,15 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
}
}
internal string ReadString(int bufferSize)
{
// Ensure to not try to decode any bytes after the 0 termination.
byte[] bytes = _baseStream.ReadBytes(bufferSize);
int length = bufferSize;
while (bytes[--length] == 0 && length > 0) { }
return _win949Encoding.GetString(bytes, 0, length + 1);
}
/// <summary>
/// Reads a 2-byte length-prefixed, Windows-949 enoded string.
/// </summary>
@ -121,6 +130,13 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
_baseStream.WriteByte(0);
}
internal void WriteString(string value, int bufferSize)
{
byte[] bytes = new byte[bufferSize];
_win949Encoding.GetBytes(value, 0, value.Length, bytes, 0);
_baseStream.WriteBytes(bytes);
}
/// <summary>
/// Writes a 2-byte length-prefixed, Windows-949 encoded string.
/// </summary>

View File

@ -0,0 +1,59 @@
using System;
using System.Collections.Generic;
using System.Net;
using Syroot.BinaryData;
namespace Syroot.Worms.OnlineWorms.Server.Net
{
/// <summary>
/// Represents the client request for a <see cref="?"/>.
/// </summary>
[Packet(PacketType.Channel, 0x10)]
internal class ChannelConnectQuery : Packet
{
// ---- PROPERTIES ---------------------------------------------------------------------------------------------
public IList<ChannelConnectPlayer> Players { get; set; }
public IPAddress ClientIP { get; set; }
public ushort ClientVersion { get; set; }
public ushort UnknownA { get; set; } // Always 1?
// ---- METHODS (INTERNAL) -------------------------------------------------------------------------------------
internal override void Deserialize(PacketStream stream)
{
Players = new List<ChannelConnectPlayer>
{
new ChannelConnectPlayer
{
ID = stream.ReadString(12),
Password = stream.ReadString(12)
}
};
ClientIP = IPAddress.Parse(stream.ReadString(15));
ClientVersion = stream.ReadUInt16();
UnknownA = stream.ReadUInt16();
ushort additionalPlayers = stream.ReadUInt16();
for (int i = 0; i < additionalPlayers; i++)
{
Players.Add(new ChannelConnectPlayer
{
ID = stream.ReadString(12),
Password = stream.ReadString(12)
});
}
}
internal override void Serialize(PacketStream stream) => throw new NotImplementedException();
}
internal class ChannelConnectPlayer
{
public string ID { get; set; } // Max. 12 characters.
public string Password { get; set; } // Max. 12 characters.
}
}

View File

@ -7,7 +7,7 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
/// <summary>
/// Represents the client request for a <see cref="ChannelEnterReply"/>.
/// </summary>
[Packet(0x8034)]
[Packet(PacketType.Server, 0x8034)]
internal class ChannelEnterQuery : Packet
{
// ---- PROPERTIES ---------------------------------------------------------------------------------------------

View File

@ -7,7 +7,7 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
/// <summary>
/// Represents the server response to a <see cref="ChannelEnterQuery"/>.
/// </summary>
[Packet(0x8035)]
[Packet(PacketType.Server, 0x8035)]
internal class ChannelEnterReply : Packet
{
// ---- PROPERTIES ---------------------------------------------------------------------------------------------

View File

@ -9,7 +9,7 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
/// <summary>
/// Represents an additional server response to a <see cref="LoginQuery"/>, providing available server channels.
/// </summary>
[Packet(0x80C9)]
[Packet(PacketType.Server, 0x80C9)]
internal class ChannelInfosReply : Packet
{
// ---- PROPERTIES ---------------------------------------------------------------------------------------------
@ -35,7 +35,7 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
stream.WriteBytes(channel.EndPoint.Address.GetAddressBytes());
stream.WriteUInt16((ushort)channel.EndPoint.Port);
stream.WriteByte(channel.UserLoad);
stream.WriteEnum(channel.Load);
}
}
}
@ -68,10 +68,9 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
internal Color Color { get; set; } = Color.White;
/// <summary>
/// Gets or sets an indicator of the number of players in this channel. 0 hides the channel, and 6 represents a
/// full channel.
/// Gets or sets an indicator or state visualizing the number of players in this channel.
/// </summary>
internal byte UserLoad { get; set; } = 1;
internal ChannelLoad Load { get; set; } = ChannelLoad.Lowest;
}
internal enum ChannelType : byte
@ -80,4 +79,16 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
Roping = 1,
Special = 2
}
internal enum ChannelLoad : byte
{
Hidden,
Lowest,
Low,
Medium,
High,
Highest,
Full,
Closed = 99
}
}

View File

@ -5,7 +5,7 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
/// <summary>
/// Represents the client request for a <see cref="ConnectReply"/>.
/// </summary>
[Packet(0x800E)]
[Packet(PacketType.Server, 0x800E)]
internal class ConnectQuery : Packet
{
// ---- METHODS (INTERNAL) -------------------------------------------------------------------------------------

View File

@ -6,7 +6,7 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
/// <summary>
/// Represents the server response to a <see cref="ConnectQuery"/>.
/// </summary>
[Packet(0x800F)]
[Packet(PacketType.Server, 0x800F)]
internal class ConnectReply : Packet
{
// ---- PROPERTIES ---------------------------------------------------------------------------------------------

View File

@ -7,7 +7,7 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
/// <summary>
/// Represents the client request for a <see cref="LoginReply"/>.
/// </summary>
[Packet(0x8000)]
[Packet(PacketType.Server, 0x8000)]
internal class LoginQuery : Packet
{
// ---- PROPERTIES ---------------------------------------------------------------------------------------------

View File

@ -6,7 +6,7 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
/// <summary>
/// Represents the server response to a <see cref="LoginQuery"/>.
/// </summary>
[Packet(0x8001)]
[Packet(PacketType.Server, 0x8001)]
internal class LoginReply : Packet
{
// ---- PROPERTIES ---------------------------------------------------------------------------------------------

View File

@ -1,36 +1,33 @@
#if DEBUG
using Syroot.BinaryData;
namespace Syroot.Worms.OnlineWorms.Server.Net
{
/// <summary>
/// Represents a special fallback packet which simply stores any ID and the raw client packet data.
/// Represents a special fallback packet which simply stores any ID and the raw packet data.
/// </summary>
internal class RawQuery : Packet
internal class RawPacket : Packet
{
// ---- CONSTRUCTORS & DESTRUCTOR ------------------------------------------------------------------------------
internal RawQuery(ushort id)
internal RawPacket(PacketType type, int id)
{
Type = type;
ID = id;
}
// ---- PROPERTIES ---------------------------------------------------------------------------------------------
internal ushort ID { get; }
internal PacketType Type { get; }
internal int ID { get; }
internal byte[] Data { get; set; }
// ---- METHODS (INTERNAL) -------------------------------------------------------------------------------------
internal override void Deserialize(PacketStream stream)
{
Data = new byte[stream.Length];
stream.Read(Data, 0, Data.Length);
}
internal override void Serialize(PacketStream stream)
{
stream.Write(Data, 0, Data.Length);
}
internal override void Deserialize(PacketStream stream) => Data = stream.ReadBytes((int)stream.Length);
internal override void Serialize(PacketStream stream) => stream.WriteBytes(Data);
}
}
#endif

View File

@ -6,7 +6,7 @@ namespace Syroot.Worms.OnlineWorms.Server.Net
/// Represents an additional server response to a <see cref="LoginQuery"/>, providing informational server
/// screen text.
/// </summary>
[Packet(0x8033)]
[Packet(PacketType.Server, 0x8033)]
internal class ServerInfoReply : Packet
{
// ---- PROPERTIES ---------------------------------------------------------------------------------------------