quantum-space-buddies/QuantumUNET/QSBNetworkServerSimple.cs
Mister_Nebula 77c92887ad cleanup
2020-12-14 21:58:00 +00:00

456 lines
12 KiB
C#

using QuantumUNET.Messages;
using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using UnityEngine;
using UnityEngine.Networking;
using UnityEngine.Networking.Types;
namespace QuantumUNET
{
public class QSBNetworkServerSimple
{
public QSBNetworkServerSimple()
{
connections = new ReadOnlyCollection<QSBNetworkConnection>(m_Connections);
}
public int listenPort { get; set; }
public int serverHostId { get; set; } = -1;
public HostTopology hostTopology { get; private set; }
public bool useWebSockets { get; set; }
public ReadOnlyCollection<QSBNetworkConnection> connections { get; }
public Dictionary<short, QSBNetworkMessageDelegate> handlers => m_MessageHandlers.GetHandlers();
public byte[] messageBuffer { get; private set; } = null;
public NetworkReader messageReader { get; private set; } = null;
public Type networkConnectionClass { get; private set; } = typeof(QSBNetworkConnection);
public void SetNetworkConnectionClass<T>() where T : QSBNetworkConnection => networkConnectionClass = typeof(T);
public virtual void Initialize()
{
if (!m_Initialized)
{
m_Initialized = true;
NetworkTransport.Init();
messageBuffer = new byte[65535];
messageReader = new NetworkReader(messageBuffer);
if (hostTopology == null)
{
var connectionConfig = new ConnectionConfig();
connectionConfig.AddChannel(QosType.ReliableSequenced);
connectionConfig.AddChannel(QosType.Unreliable);
hostTopology = new HostTopology(connectionConfig, 8);
}
if (LogFilter.logDebug)
{
Debug.Log("NetworkServerSimple initialize.");
}
}
}
public bool Configure(ConnectionConfig config, int maxConnections)
{
var topology = new HostTopology(config, maxConnections);
return Configure(topology);
}
public bool Configure(HostTopology topology)
{
hostTopology = topology;
return true;
}
public bool Listen(string ipAddress, int serverListenPort)
{
Initialize();
listenPort = serverListenPort;
if (useWebSockets)
{
serverHostId = NetworkTransport.AddWebsocketHost(hostTopology, serverListenPort, ipAddress);
}
else
{
serverHostId = NetworkTransport.AddHost(hostTopology, serverListenPort, ipAddress);
}
bool result;
if (serverHostId == -1)
{
result = false;
}
else
{
if (LogFilter.logDebug)
{
Debug.Log(string.Concat(new object[]
{
"NetworkServerSimple listen: ",
ipAddress,
":",
listenPort
}));
}
result = true;
}
return result;
}
public bool Listen(int serverListenPort) => Listen(serverListenPort, hostTopology);
public bool Listen(int serverListenPort, HostTopology topology)
{
hostTopology = topology;
Initialize();
listenPort = serverListenPort;
if (useWebSockets)
{
serverHostId = NetworkTransport.AddWebsocketHost(hostTopology, serverListenPort);
}
else
{
serverHostId = NetworkTransport.AddHost(hostTopology, serverListenPort);
}
bool result;
if (serverHostId == -1)
{
result = false;
}
else
{
if (LogFilter.logDebug)
{
Debug.Log("NetworkServerSimple listen " + listenPort);
}
result = true;
}
return result;
}
public void ListenRelay(string relayIp, int relayPort, NetworkID netGuid, SourceID sourceId, NodeID nodeId)
{
Initialize();
serverHostId = NetworkTransport.AddHost(hostTopology, listenPort);
if (LogFilter.logDebug)
{
Debug.Log("Server Host Slot Id: " + serverHostId);
}
Update();
NetworkTransport.ConnectAsNetworkHost(serverHostId, relayIp, relayPort, netGuid, sourceId, nodeId, out var b);
m_RelaySlotId = 0;
if (LogFilter.logDebug)
{
Debug.Log("Relay Slot Id: " + m_RelaySlotId);
}
}
public void Stop()
{
if (LogFilter.logDebug)
{
Debug.Log("NetworkServerSimple stop ");
}
NetworkTransport.RemoveHost(serverHostId);
serverHostId = -1;
}
internal void RegisterHandlerSafe(short msgType, QSBNetworkMessageDelegate handler) => m_MessageHandlers.RegisterHandlerSafe(msgType, handler);
public void RegisterHandler(short msgType, QSBNetworkMessageDelegate handler) => m_MessageHandlers.RegisterHandler(msgType, handler);
public void UnregisterHandler(short msgType) => m_MessageHandlers.UnregisterHandler(msgType);
public void ClearHandlers() => m_MessageHandlers.ClearMessageHandlers();
public void UpdateConnections()
{
for (var i = 0; i < m_Connections.Count; i++)
{
var networkConnection = m_Connections[i];
if (networkConnection != null)
{
networkConnection.FlushChannels();
}
}
}
public void Update()
{
if (serverHostId != -1)
{
NetworkEventType networkEventType;
if (m_RelaySlotId != -1)
{
networkEventType = NetworkTransport.ReceiveRelayEventFromHost(serverHostId, out var b);
if (networkEventType != NetworkEventType.Nothing)
{
if (LogFilter.logDebug)
{
Debug.Log("NetGroup event:" + networkEventType);
}
}
if (networkEventType == NetworkEventType.ConnectEvent)
{
if (LogFilter.logDebug)
{
Debug.Log("NetGroup server connected");
}
}
if (networkEventType == NetworkEventType.DisconnectEvent)
{
if (LogFilter.logDebug)
{
Debug.Log("NetGroup server disconnected");
}
}
}
do
{
networkEventType = NetworkTransport.ReceiveFromHost(serverHostId, out var connectionId, out var channelId, messageBuffer, messageBuffer.Length, out var receivedSize, out var b);
if (networkEventType != NetworkEventType.Nothing)
{
Debug.Log(string.Concat(new object[]
{
"Server event: host=",
serverHostId,
" event=",
networkEventType,
" error=",
b
}));
}
switch (networkEventType)
{
case NetworkEventType.DataEvent:
HandleData(connectionId, channelId, receivedSize, b);
break;
case NetworkEventType.ConnectEvent:
HandleConnect(connectionId, b);
break;
case NetworkEventType.DisconnectEvent:
HandleDisconnect(connectionId, b);
break;
case NetworkEventType.Nothing:
break;
default:
if (LogFilter.logError)
{
Debug.LogError("Unknown network message type received: " + networkEventType);
}
break;
}
}
while (networkEventType != NetworkEventType.Nothing);
UpdateConnections();
}
}
public QSBNetworkConnection FindConnection(int connectionId)
{
QSBNetworkConnection result;
if (connectionId < 0 || connectionId >= m_Connections.Count)
{
result = null;
}
else
{
result = m_Connections[connectionId];
}
return result;
}
public bool SetConnectionAtIndex(QSBNetworkConnection conn)
{
while (m_Connections.Count <= conn.connectionId)
{
m_Connections.Add(null);
}
bool result;
if (m_Connections[conn.connectionId] != null)
{
result = false;
}
else
{
m_Connections[conn.connectionId] = conn;
conn.SetHandlers(m_MessageHandlers);
result = true;
}
return result;
}
public bool RemoveConnectionAtIndex(int connectionId)
{
bool result;
if (connectionId < 0 || connectionId >= m_Connections.Count)
{
result = false;
}
else
{
m_Connections[connectionId] = null;
result = true;
}
return result;
}
private void HandleConnect(int connectionId, byte error)
{
if (LogFilter.logDebug)
{
Debug.Log("NetworkServerSimple accepted client:" + connectionId);
}
if (error != 0)
{
OnConnectError(connectionId, error);
}
else
{
NetworkTransport.GetConnectionInfo(serverHostId, connectionId, out var networkAddress, out var num, out var networkID, out var nodeID, out var lastError);
var networkConnection = (QSBNetworkConnection)Activator.CreateInstance(networkConnectionClass);
networkConnection.SetHandlers(m_MessageHandlers);
networkConnection.Initialize(networkAddress, serverHostId, connectionId, hostTopology);
networkConnection.LastError = (NetworkError)lastError;
while (m_Connections.Count <= connectionId)
{
m_Connections.Add(null);
}
m_Connections[connectionId] = networkConnection;
OnConnected(networkConnection);
}
}
private void HandleDisconnect(int connectionId, byte error)
{
if (LogFilter.logDebug)
{
Debug.Log("NetworkServerSimple disconnect client:" + connectionId);
}
var networkConnection = FindConnection(connectionId);
if (networkConnection != null)
{
networkConnection.LastError = (NetworkError)error;
if (error != 0)
{
if (error != 6)
{
m_Connections[connectionId] = null;
if (LogFilter.logError)
{
Debug.LogError(string.Concat(new object[]
{
"Server client disconnect error, connectionId: ",
connectionId,
" error: ",
(NetworkError)error
}));
}
OnDisconnectError(networkConnection, error);
return;
}
}
networkConnection.Disconnect();
m_Connections[connectionId] = null;
if (LogFilter.logDebug)
{
Debug.Log("Server lost client:" + connectionId);
}
OnDisconnected(networkConnection);
}
}
private void HandleData(int connectionId, int channelId, int receivedSize, byte error)
{
var networkConnection = FindConnection(connectionId);
if (networkConnection == null)
{
if (LogFilter.logError)
{
Debug.LogError("HandleData Unknown connectionId:" + connectionId);
}
}
else
{
networkConnection.LastError = (NetworkError)error;
if (error != 0)
{
OnDataError(networkConnection, error);
}
else
{
messageReader.SeekZero();
OnData(networkConnection, receivedSize, channelId);
}
}
}
public void SendBytesTo(int connectionId, byte[] bytes, int numBytes, int channelId)
{
var networkConnection = FindConnection(connectionId);
if (networkConnection != null)
{
networkConnection.SendBytes(bytes, numBytes, channelId);
}
}
public void SendWriterTo(int connectionId, QSBNetworkWriter writer, int channelId)
{
var networkConnection = FindConnection(connectionId);
if (networkConnection != null)
{
networkConnection.SendWriter(writer, channelId);
}
}
public void Disconnect(int connectionId)
{
var networkConnection = FindConnection(connectionId);
if (networkConnection != null)
{
networkConnection.Disconnect();
m_Connections[connectionId] = null;
}
}
public void DisconnectAllConnections()
{
for (var i = 0; i < m_Connections.Count; i++)
{
var networkConnection = m_Connections[i];
if (networkConnection != null)
{
networkConnection.Disconnect();
networkConnection.Dispose();
}
}
}
public virtual void OnConnectError(int connectionId, byte error) => Debug.LogError("OnConnectError error:" + error);
public virtual void OnDataError(QSBNetworkConnection conn, byte error) => Debug.LogError("OnDataError error:" + error);
public virtual void OnDisconnectError(QSBNetworkConnection conn, byte error) => Debug.LogError("OnDisconnectError error:" + error);
public virtual void OnConnected(QSBNetworkConnection conn) => conn.InvokeHandlerNoData(32);
public virtual void OnDisconnected(QSBNetworkConnection conn) => conn.InvokeHandlerNoData(33);
public virtual void OnData(QSBNetworkConnection conn, int receivedSize, int channelId) => conn.TransportReceive(messageBuffer, receivedSize, channelId);
private bool m_Initialized = false;
private int m_RelaySlotId = -1;
private List<QSBNetworkConnection> m_Connections = new List<QSBNetworkConnection>();
private QSBNetworkMessageHandlers m_MessageHandlers = new QSBNetworkMessageHandlers();
}
}