quantum-space-buddies/QuantumUNET/QNetworkConnection.cs

458 lines
12 KiB
C#
Raw Normal View History

2020-12-23 13:48:31 +00:00
using QuantumUNET.Components;
using QuantumUNET.Logging;
2020-12-07 21:19:16 +00:00
using QuantumUNET.Messages;
using QuantumUNET.Transport;
2020-12-04 09:23:27 +00:00
using System;
2020-12-02 12:42:26 +00:00
using System.Collections.Generic;
using System.Text;
using UnityEngine;
using UnityEngine.Networking;
2020-12-04 22:14:53 +00:00
namespace QuantumUNET
2020-12-02 12:42:26 +00:00
{
2020-12-23 12:58:45 +00:00
public class QNetworkConnection : IDisposable
2020-12-02 12:42:26 +00:00
{
2020-12-24 16:34:24 +00:00
public QNetworkConnection() => m_Writer = new QNetworkWriter();
2020-12-02 12:42:26 +00:00
2020-12-23 12:58:45 +00:00
internal HashSet<QNetworkIdentity> VisList { get; } = new HashSet<QNetworkIdentity>();
2020-12-02 12:42:26 +00:00
2020-12-23 12:58:45 +00:00
public List<QPlayerController> PlayerControllers { get; } = new List<QPlayerController>();
2020-12-02 12:42:26 +00:00
2020-12-16 08:57:15 +00:00
public HashSet<NetworkInstanceId> ClientOwnedObjects { get; private set; }
2020-12-02 12:42:26 +00:00
public bool isConnected => hostId != -1;
public NetworkError LastError { get; internal set; }
internal Dictionary<short, PacketStat> PacketStats { get; } = new Dictionary<short, PacketStat>();
public virtual void Initialize(string networkAddress, int networkHostId, int networkConnectionId, HostTopology hostTopology)
{
2020-12-23 12:58:45 +00:00
m_Writer = new QNetworkWriter();
2020-12-02 12:42:26 +00:00
address = networkAddress;
hostId = networkHostId;
connectionId = networkConnectionId;
var channelCount = hostTopology.DefaultConfig.ChannelCount;
var packetSize = (int)hostTopology.DefaultConfig.PacketSize;
if (hostTopology.DefaultConfig.UsePlatformSpecificProtocols && Application.platform != RuntimePlatform.PS4 && Application.platform != RuntimePlatform.PSP2)
{
throw new ArgumentOutOfRangeException("Platform specific protocols are not supported on this platform");
}
2020-12-23 12:58:45 +00:00
m_Channels = new QChannelBuffer[channelCount];
2020-12-02 12:42:26 +00:00
for (var i = 0; i < channelCount; i++)
{
var channelQOS = hostTopology.DefaultConfig.Channels[i];
var bufferSize = packetSize;
if (channelQOS.QOS == QosType.ReliableFragmented || channelQOS.QOS == QosType.UnreliableFragmented)
{
bufferSize = hostTopology.DefaultConfig.FragmentSize * 128;
2020-12-02 12:42:26 +00:00
}
2020-12-23 12:58:45 +00:00
m_Channels[i] = new QChannelBuffer(this, bufferSize, (byte)i, IsReliableQoS(channelQOS.QOS), IsSequencedQoS(channelQOS.QOS));
2020-12-02 12:42:26 +00:00
}
}
2020-12-23 12:58:45 +00:00
~QNetworkConnection()
2020-12-02 12:42:26 +00:00
{
Dispose(false);
}
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
protected virtual void Dispose(bool disposing)
{
if (!m_Disposed && m_Channels != null)
{
foreach (var channel in m_Channels)
2020-12-02 12:42:26 +00:00
{
channel.Dispose();
2020-12-02 12:42:26 +00:00
}
}
m_Channels = null;
if (ClientOwnedObjects != null)
{
foreach (var netId in ClientOwnedObjects)
{
2020-12-23 12:58:45 +00:00
var gameObject = QNetworkServer.FindLocalObject(netId);
2020-12-02 12:42:26 +00:00
if (gameObject != null)
{
2020-12-23 12:58:45 +00:00
gameObject.GetComponent<QNetworkIdentity>().ClearClientOwner();
2020-12-02 12:42:26 +00:00
}
}
}
ClientOwnedObjects = null;
m_Disposed = true;
}
private static bool IsSequencedQoS(QosType qos) => qos == QosType.ReliableSequenced || qos == QosType.UnreliableSequenced;
private static bool IsReliableQoS(QosType qos) => qos == QosType.Reliable || qos == QosType.ReliableFragmented || qos == QosType.ReliableSequenced || qos == QosType.ReliableStateUpdate;
public bool SetChannelOption(int channelId, ChannelOption option, int value) => m_Channels != null && channelId >= 0 && channelId < m_Channels.Length && m_Channels[channelId].SetOption(option, value);
public void Disconnect()
{
address = "";
isReady = false;
2020-12-23 12:58:45 +00:00
QClientScene.HandleClientDisconnect(this);
2020-12-02 12:42:26 +00:00
if (hostId != -1)
{
NetworkTransport.Disconnect(hostId, connectionId, out var b);
2020-12-02 12:42:26 +00:00
RemoveObservers();
}
}
2020-12-23 12:58:45 +00:00
internal void SetHandlers(QNetworkMessageHandlers handlers)
2020-12-02 12:42:26 +00:00
{
m_MessageHandlers = handlers;
m_MessageHandlersDict = handlers.GetHandlers();
}
public bool CheckHandler(short msgType) => m_MessageHandlersDict.ContainsKey(msgType);
public bool InvokeHandlerNoData(short msgType) => InvokeHandler(msgType, null, 0);
2020-12-23 12:58:45 +00:00
public bool InvokeHandler(short msgType, QNetworkReader reader, int channelId)
2020-12-02 12:42:26 +00:00
{
bool result;
if (m_MessageHandlersDict.ContainsKey(msgType))
{
2020-12-03 11:56:32 +00:00
m_MessageInfo.MsgType = msgType;
m_MessageInfo.Connection = this;
m_MessageInfo.Reader = reader;
m_MessageInfo.ChannelId = channelId;
2020-12-02 12:42:26 +00:00
var networkMessageDelegate = m_MessageHandlersDict[msgType];
if (networkMessageDelegate == null)
{
result = false;
}
else
{
networkMessageDelegate(m_MessageInfo);
result = true;
}
}
else
{
result = false;
}
return result;
}
2020-12-23 12:58:45 +00:00
public bool InvokeHandler(QNetworkMessage netMsg)
2020-12-02 12:42:26 +00:00
{
bool result;
2020-12-03 11:56:32 +00:00
if (m_MessageHandlersDict.ContainsKey(netMsg.MsgType))
2020-12-02 12:42:26 +00:00
{
2020-12-03 11:56:32 +00:00
var networkMessageDelegate = m_MessageHandlersDict[netMsg.MsgType];
2020-12-02 12:42:26 +00:00
networkMessageDelegate(netMsg);
result = true;
}
else
{
result = false;
}
return result;
}
2020-12-23 12:58:45 +00:00
internal void HandleFragment(QNetworkReader reader, int channelId)
2020-12-02 12:42:26 +00:00
{
if (channelId >= 0 && channelId < m_Channels.Length)
{
var channelBuffer = m_Channels[channelId];
if (channelBuffer.HandleFragment(reader))
{
2020-12-23 12:58:45 +00:00
var networkReader = new QNetworkReader(channelBuffer._fragmentBuffer.AsArraySegment().Array);
2020-12-02 12:42:26 +00:00
networkReader.ReadInt16();
var msgType = networkReader.ReadInt16();
InvokeHandler(msgType, networkReader, channelId);
}
}
}
2020-12-23 12:58:45 +00:00
public void RegisterHandler(short msgType, QNetworkMessageDelegate handler) => m_MessageHandlers.RegisterHandler(msgType, handler);
2020-12-02 12:42:26 +00:00
public void UnregisterHandler(short msgType) => m_MessageHandlers.UnregisterHandler(msgType);
2020-12-23 12:58:45 +00:00
internal void SetPlayerController(QPlayerController player)
2020-12-02 12:42:26 +00:00
{
while (player.PlayerControllerId >= PlayerControllers.Count)
2020-12-02 12:42:26 +00:00
{
2020-12-23 12:58:45 +00:00
PlayerControllers.Add(new QPlayerController());
2020-12-02 12:42:26 +00:00
}
PlayerControllers[player.PlayerControllerId] = player;
2020-12-02 12:42:26 +00:00
}
internal void RemovePlayerController(short playerControllerId)
{
for (var i = PlayerControllers.Count; i >= 0; i--)
{
if (playerControllerId == i && playerControllerId == PlayerControllers[i].PlayerControllerId)
2020-12-02 12:42:26 +00:00
{
2020-12-23 12:58:45 +00:00
PlayerControllers[i] = new QPlayerController();
2020-12-02 12:42:26 +00:00
return;
}
}
QLog.Error($"RemovePlayer player at playerControllerId {playerControllerId} not found");
2020-12-02 12:42:26 +00:00
}
2020-12-23 12:58:45 +00:00
internal bool GetPlayerController(short playerControllerId, out QPlayerController playerController)
2020-12-02 12:42:26 +00:00
{
playerController = null;
bool result;
if (PlayerControllers.Count > 0)
{
foreach (var controller in PlayerControllers)
2020-12-02 12:42:26 +00:00
{
if (controller.IsValid && controller.PlayerControllerId == playerControllerId)
2020-12-02 12:42:26 +00:00
{
playerController = controller;
2020-12-02 12:42:26 +00:00
return true;
}
}
2020-12-02 12:42:26 +00:00
result = false;
}
else
{
result = false;
}
return result;
}
public void FlushChannels()
{
if (m_Channels != null)
{
foreach (var channel in m_Channels)
2020-12-02 12:42:26 +00:00
{
channel.CheckInternalBuffer();
2020-12-02 12:42:26 +00:00
}
}
}
public void SetMaxDelay(float seconds)
{
if (m_Channels != null)
{
foreach (var channel in m_Channels)
2020-12-02 12:42:26 +00:00
{
channel.MaxDelay = seconds;
2020-12-02 12:42:26 +00:00
}
}
}
2020-12-23 12:58:45 +00:00
public virtual bool Send(short msgType, QMessageBase msg) => SendByChannel(msgType, msg, 0);
2020-12-02 12:42:26 +00:00
2020-12-23 12:58:45 +00:00
public virtual bool SendUnreliable(short msgType, QMessageBase msg) => SendByChannel(msgType, msg, 1);
2020-12-02 12:42:26 +00:00
2020-12-23 12:58:45 +00:00
public virtual bool SendByChannel(short msgType, QMessageBase msg, int channelId)
2020-12-02 12:42:26 +00:00
{
m_Writer.StartMessage(msgType);
msg.Serialize(m_Writer);
m_Writer.FinishMessage();
return SendWriter(m_Writer, channelId);
}
public virtual bool SendBytes(byte[] bytes, int numBytes, int channelId) => CheckChannel(channelId) && m_Channels[channelId].SendBytes(bytes, numBytes);
2020-12-02 12:42:26 +00:00
2020-12-23 12:58:45 +00:00
public virtual bool SendWriter(QNetworkWriter writer, int channelId) => CheckChannel(channelId) && m_Channels[channelId].SendWriter(writer);
2020-12-02 12:42:26 +00:00
private void LogSend(byte[] bytes)
{
2020-12-23 12:58:45 +00:00
var networkReader = new QNetworkReader(bytes);
2020-12-02 12:42:26 +00:00
var num = networkReader.ReadUInt16();
var num2 = networkReader.ReadUInt16();
var stringBuilder = new StringBuilder();
for (var i = 4; i < 4 + num; i++)
2020-12-02 12:42:26 +00:00
{
stringBuilder.AppendFormat("{0:X2}", bytes[i]);
if (i > 150)
{
break;
}
}
2020-12-23 13:48:31 +00:00
QLog.Log(
$"ConnectionSend con:{connectionId} bytes:{num} msgId:{num2} {stringBuilder}");
2020-12-02 12:42:26 +00:00
}
private bool CheckChannel(int channelId)
{
bool result;
if (m_Channels == null)
{
QLog.Warning($"Channels not initialized sending on id '{channelId}");
2020-12-02 12:42:26 +00:00
result = false;
}
else if (channelId < 0 || channelId >= m_Channels.Length)
{
QLog.Error(
$"Invalid channel when sending buffered data, '{channelId}'. Current channel count is {m_Channels.Length}");
2020-12-02 12:42:26 +00:00
result = false;
}
else
{
result = true;
}
return result;
}
public void ResetStats()
{
}
protected void HandleBytes(byte[] buffer, int receivedSize, int channelId)
{
2020-12-23 12:58:45 +00:00
var reader = new QNetworkReader(buffer);
2020-12-02 12:42:26 +00:00
HandleReader(reader, receivedSize, channelId);
}
2020-12-23 12:58:45 +00:00
protected void HandleReader(QNetworkReader reader, int receivedSize, int channelId)
2020-12-02 12:42:26 +00:00
{
2020-12-04 09:23:27 +00:00
while (reader.Position < receivedSize)
2020-12-02 12:42:26 +00:00
{
var num = reader.ReadUInt16();
var num2 = reader.ReadInt16();
var array = reader.ReadBytes(num);
2020-12-23 12:58:45 +00:00
var reader2 = new QNetworkReader(array);
QNetworkMessageDelegate networkMessageDelegate = null;
2020-12-02 12:42:26 +00:00
if (m_MessageHandlersDict.ContainsKey(num2))
{
networkMessageDelegate = m_MessageHandlersDict[num2];
}
if (networkMessageDelegate == null)
{
QLog.Error($"Unknown message ID {num2} connId:{connectionId}");
2020-12-02 12:42:26 +00:00
break;
}
2020-12-03 11:56:32 +00:00
m_NetMsg.MsgType = num2;
m_NetMsg.Reader = reader2;
m_NetMsg.Connection = this;
m_NetMsg.ChannelId = channelId;
2020-12-02 12:42:26 +00:00
networkMessageDelegate(m_NetMsg);
lastMessageTime = Time.time;
}
}
public virtual void GetStatsOut(out int numMsgs, out int numBufferedMsgs, out int numBytes, out int lastBufferedPerSecond)
{
numMsgs = 0;
numBufferedMsgs = 0;
numBytes = 0;
lastBufferedPerSecond = 0;
foreach (var channelBuffer in m_Channels)
2020-12-02 12:42:26 +00:00
{
2020-12-03 11:56:32 +00:00
numMsgs += channelBuffer.NumMsgsOut;
numBufferedMsgs += channelBuffer.NumBufferedMsgsOut;
numBytes += channelBuffer.NumBytesOut;
lastBufferedPerSecond += channelBuffer.LastBufferedPerSecond;
2020-12-02 12:42:26 +00:00
}
}
public virtual void GetStatsIn(out int numMsgs, out int numBytes)
{
numMsgs = 0;
numBytes = 0;
foreach (var channelBuffer in m_Channels)
2020-12-02 12:42:26 +00:00
{
2020-12-03 11:56:32 +00:00
numMsgs += channelBuffer.NumMsgsIn;
numBytes += channelBuffer.NumBytesIn;
2020-12-02 12:42:26 +00:00
}
}
public override string ToString() =>
$"hostId: {hostId} connectionId: {connectionId} isReady: {isReady} channel count: {m_Channels?.Length ?? 0}";
2020-12-02 12:42:26 +00:00
2020-12-23 12:58:45 +00:00
internal void AddToVisList(QNetworkIdentity uv)
2020-12-02 12:42:26 +00:00
{
VisList.Add(uv);
2020-12-23 12:58:45 +00:00
QNetworkServer.ShowForConnection(uv, this);
2020-12-02 12:42:26 +00:00
}
2020-12-23 12:58:45 +00:00
internal void RemoveFromVisList(QNetworkIdentity uv, bool isDestroyed)
2020-12-02 12:42:26 +00:00
{
VisList.Remove(uv);
if (!isDestroyed)
{
2020-12-23 12:58:45 +00:00
QNetworkServer.HideForConnection(uv, this);
2020-12-02 12:42:26 +00:00
}
}
internal void RemoveObservers()
{
foreach (var networkIdentity in VisList)
{
networkIdentity.RemoveObserverInternal(this);
}
VisList.Clear();
}
public virtual void TransportReceive(byte[] bytes, int numBytes, int channelId) => HandleBytes(bytes, numBytes, channelId);
public virtual bool TransportSend(byte[] bytes, int numBytes, int channelId, out byte error) => NetworkTransport.Send(hostId, connectionId, channelId, bytes, numBytes, out error);
2020-12-23 12:58:45 +00:00
internal void AddOwnedObject(QNetworkIdentity obj)
2020-12-02 12:42:26 +00:00
{
if (ClientOwnedObjects == null)
{
2020-12-16 08:57:15 +00:00
ClientOwnedObjects = new HashSet<NetworkInstanceId>();
2020-12-02 12:42:26 +00:00
}
ClientOwnedObjects.Add(obj.NetId);
}
2020-12-23 12:58:45 +00:00
internal void RemoveOwnedObject(QNetworkIdentity obj) => ClientOwnedObjects?.Remove(obj.NetId);
2020-12-02 12:42:26 +00:00
2020-12-23 12:58:45 +00:00
internal static void OnFragment(QNetworkMessage netMsg) => netMsg.Connection.HandleFragment(netMsg.Reader, netMsg.ChannelId);
2020-12-02 12:42:26 +00:00
2020-12-23 12:58:45 +00:00
private QChannelBuffer[] m_Channels;
private readonly QNetworkMessage m_NetMsg = new QNetworkMessage();
private QNetworkWriter m_Writer;
2020-12-02 12:42:26 +00:00
2020-12-23 12:58:45 +00:00
private Dictionary<short, QNetworkMessageDelegate> m_MessageHandlersDict;
2020-12-02 12:42:26 +00:00
2020-12-23 12:58:45 +00:00
private QNetworkMessageHandlers m_MessageHandlers;
private readonly QNetworkMessage m_MessageInfo = new QNetworkMessage();
2020-12-02 12:42:26 +00:00
private const int k_MaxMessageLogSize = 150;
public int hostId = -1;
public int connectionId = -1;
public bool isReady;
public string address;
public float lastMessageTime;
public bool logNetworkMessages = false;
private bool m_Disposed;
public class PacketStat
{
public PacketStat()
{
msgType = 0;
count = 0;
bytes = 0;
}
public PacketStat(PacketStat s)
{
msgType = s.msgType;
count = s.count;
bytes = s.bytes;
}
2020-12-23 12:58:45 +00:00
public override string ToString() => $"{QMsgType.MsgTypeToString(msgType)}: count={count} bytes={bytes}";
2020-12-02 12:42:26 +00:00
public short msgType;
public int count;
public int bytes;
}
}
2020-12-03 08:28:05 +00:00
}