mirror of
https://github.com/misternebula/quantum-space-buddies.git
synced 2025-01-07 13:05:41 +00:00
336 lines
10 KiB
C#
336 lines
10 KiB
C#
using Epic.OnlineServices;
|
|
using Epic.OnlineServices.P2P;
|
|
using System;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using UnityEngine;
|
|
|
|
namespace EpicTransport;
|
|
|
|
public abstract class Common
|
|
{
|
|
private PacketReliability[] channels;
|
|
private int internal_ch => channels.Length;
|
|
|
|
protected enum InternalMessages : byte
|
|
{
|
|
CONNECT,
|
|
ACCEPT_CONNECT,
|
|
DISCONNECT
|
|
}
|
|
|
|
protected struct PacketKey
|
|
{
|
|
public ProductUserId productUserId;
|
|
public byte channel;
|
|
}
|
|
|
|
private OnIncomingConnectionRequestCallback OnIncomingConnectionRequest;
|
|
private ulong incomingNotificationId = 0;
|
|
private OnRemoteConnectionClosedCallback OnRemoteConnectionClosed;
|
|
private ulong outgoingNotificationId = 0;
|
|
|
|
protected readonly EosTransport transport;
|
|
|
|
protected List<string> deadSockets;
|
|
public bool ignoreAllMessages = false;
|
|
|
|
// Mapping from PacketKey to a List of Packet Lists
|
|
protected Dictionary<PacketKey, List<List<Packet>>> incomingPackets = new();
|
|
|
|
protected Common(EosTransport transport)
|
|
{
|
|
channels = transport.Channels;
|
|
|
|
deadSockets = new List<string>();
|
|
|
|
var addNotifyPeerConnectionRequestOptions = new AddNotifyPeerConnectionRequestOptions();
|
|
addNotifyPeerConnectionRequestOptions.LocalUserId = EOSSDKComponent.LocalUserProductId;
|
|
addNotifyPeerConnectionRequestOptions.SocketId = null;
|
|
|
|
OnIncomingConnectionRequest += OnNewConnection;
|
|
OnRemoteConnectionClosed += OnConnectFail;
|
|
|
|
incomingNotificationId = EOSSDKComponent.GetP2PInterface().AddNotifyPeerConnectionRequest(addNotifyPeerConnectionRequestOptions,
|
|
null, OnIncomingConnectionRequest);
|
|
|
|
var addNotifyPeerConnectionClosedOptions = new AddNotifyPeerConnectionClosedOptions();
|
|
addNotifyPeerConnectionClosedOptions.LocalUserId = EOSSDKComponent.LocalUserProductId;
|
|
addNotifyPeerConnectionClosedOptions.SocketId = null;
|
|
|
|
outgoingNotificationId = EOSSDKComponent.GetP2PInterface().AddNotifyPeerConnectionClosed(addNotifyPeerConnectionClosedOptions,
|
|
null, OnRemoteConnectionClosed);
|
|
|
|
if (outgoingNotificationId == 0 || incomingNotificationId == 0)
|
|
{
|
|
Debug.LogError("Couldn't bind notifications with P2P interface");
|
|
}
|
|
|
|
incomingPackets = new Dictionary<PacketKey, List<List<Packet>>>();
|
|
|
|
this.transport = transport;
|
|
}
|
|
|
|
protected void Dispose()
|
|
{
|
|
EOSSDKComponent.GetP2PInterface().RemoveNotifyPeerConnectionRequest(incomingNotificationId);
|
|
EOSSDKComponent.GetP2PInterface().RemoveNotifyPeerConnectionClosed(outgoingNotificationId);
|
|
|
|
transport.ResetIgnoreMessagesAtStartUpTimer();
|
|
}
|
|
|
|
protected abstract void OnNewConnection(OnIncomingConnectionRequestInfo result);
|
|
|
|
private void OnConnectFail(OnRemoteConnectionClosedInfo result)
|
|
{
|
|
if (ignoreAllMessages)
|
|
{
|
|
return;
|
|
}
|
|
|
|
OnConnectionFailed(result.RemoteUserId);
|
|
|
|
switch (result.Reason)
|
|
{
|
|
case ConnectionClosedReason.ClosedByLocalUser:
|
|
throw new Exception("Connection cLosed: The Connection was gracecfully closed by the local user.");
|
|
case ConnectionClosedReason.ClosedByPeer:
|
|
throw new Exception("Connection closed: The connection was gracefully closed by remote user.");
|
|
case ConnectionClosedReason.ConnectionClosed:
|
|
throw new Exception("Connection closed: The connection was unexpectedly closed.");
|
|
case ConnectionClosedReason.ConnectionFailed:
|
|
throw new Exception("Connection failed: Failled to establish connection.");
|
|
case ConnectionClosedReason.InvalidData:
|
|
throw new Exception("Connection failed: The remote user sent us invalid data..");
|
|
case ConnectionClosedReason.InvalidMessage:
|
|
throw new Exception("Connection failed: The remote user sent us an invalid message.");
|
|
case ConnectionClosedReason.NegotiationFailed:
|
|
throw new Exception("Connection failed: Negotiation failed.");
|
|
case ConnectionClosedReason.TimedOut:
|
|
throw new Exception("Connection failed: Timeout.");
|
|
case ConnectionClosedReason.TooManyConnections:
|
|
throw new Exception("Connection failed: Too many connections.");
|
|
case ConnectionClosedReason.UnexpectedError:
|
|
throw new Exception("Unexpected Error, connection will be closed");
|
|
case ConnectionClosedReason.Unknown:
|
|
default:
|
|
throw new Exception("Unknown Error, connection has been closed.");
|
|
}
|
|
}
|
|
|
|
protected void SendInternal(ProductUserId target, SocketId socketId, InternalMessages type)
|
|
{
|
|
Debug.LogError($"SendInternal type:{type} to {target}");
|
|
|
|
EOSSDKComponent.GetP2PInterface().SendPacket(new SendPacketOptions()
|
|
{
|
|
AllowDelayedDelivery = true,
|
|
Channel = (byte)internal_ch,
|
|
Data = new byte[] { (byte)type },
|
|
LocalUserId = EOSSDKComponent.LocalUserProductId,
|
|
Reliability = PacketReliability.ReliableOrdered,
|
|
RemoteUserId = target,
|
|
SocketId = socketId
|
|
});
|
|
}
|
|
|
|
protected void Send(ProductUserId host, SocketId socketId, byte[] msgBuffer, byte channel)
|
|
{
|
|
var result = EOSSDKComponent.GetP2PInterface().SendPacket(new SendPacketOptions()
|
|
{
|
|
AllowDelayedDelivery = true,
|
|
Channel = channel,
|
|
Data = msgBuffer,
|
|
LocalUserId = EOSSDKComponent.LocalUserProductId,
|
|
Reliability = channels[channel],
|
|
RemoteUserId = host,
|
|
SocketId = socketId
|
|
});
|
|
|
|
if (result != Result.Success)
|
|
{
|
|
Debug.LogError("Send failed " + result);
|
|
}
|
|
}
|
|
|
|
private bool Receive(out ProductUserId clientProductUserId, out SocketId socketId, out byte[] receiveBuffer, byte channel)
|
|
{
|
|
var result = EOSSDKComponent.GetP2PInterface().ReceivePacket(new ReceivePacketOptions()
|
|
{
|
|
LocalUserId = EOSSDKComponent.LocalUserProductId,
|
|
MaxDataSizeBytes = P2PInterface.MaxPacketSize,
|
|
RequestedChannel = channel
|
|
}, out clientProductUserId, out socketId, out channel, out receiveBuffer);
|
|
|
|
if (result == Result.Success)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
receiveBuffer = null;
|
|
clientProductUserId = null;
|
|
return false;
|
|
}
|
|
|
|
protected virtual void CloseP2PSessionWithUser(ProductUserId clientUserID, SocketId socketId)
|
|
{
|
|
if (socketId == null)
|
|
{
|
|
Debug.LogWarning("Socket ID == null | " + ignoreAllMessages);
|
|
return;
|
|
}
|
|
|
|
if (deadSockets == null)
|
|
{
|
|
Debug.LogWarning("DeadSockets == null");
|
|
return;
|
|
}
|
|
|
|
if (deadSockets.Contains(socketId.SocketName))
|
|
{
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
deadSockets.Add(socketId.SocketName);
|
|
}
|
|
}
|
|
|
|
protected void WaitForClose(ProductUserId clientUserID, SocketId socketId) => transport.StartCoroutine(DelayedClose(clientUserID, socketId));
|
|
|
|
private IEnumerator DelayedClose(ProductUserId clientUserID, SocketId socketId)
|
|
{
|
|
yield return null;
|
|
CloseP2PSessionWithUser(clientUserID, socketId);
|
|
}
|
|
|
|
public void ReceiveData()
|
|
{
|
|
try
|
|
{
|
|
// Internal Channel, no fragmentation here
|
|
var socketId = new SocketId();
|
|
while (transport.enabled && Receive(out var clientUserID, out socketId, out var internalMessage, (byte)internal_ch))
|
|
{
|
|
if (internalMessage.Length == 1)
|
|
{
|
|
OnReceiveInternalData((InternalMessages)internalMessage[0], clientUserID, socketId);
|
|
return; // Wait one frame
|
|
}
|
|
else
|
|
{
|
|
Debug.Log("Incorrect package length on internal channel.");
|
|
}
|
|
}
|
|
|
|
// Insert new packet at the correct location in the incoming queue
|
|
for (var chNum = 0; chNum < channels.Length; chNum++)
|
|
{
|
|
while (transport.enabled && Receive(out var clientUserID, out socketId, out var receiveBuffer, (byte)chNum))
|
|
{
|
|
var incomingPacketKey = new PacketKey();
|
|
incomingPacketKey.productUserId = clientUserID;
|
|
incomingPacketKey.channel = (byte)chNum;
|
|
|
|
var packet = new Packet();
|
|
packet.FromBytes(receiveBuffer);
|
|
|
|
if (!incomingPackets.ContainsKey(incomingPacketKey))
|
|
{
|
|
incomingPackets.Add(incomingPacketKey, new List<List<Packet>>());
|
|
}
|
|
|
|
var packetListIndex = incomingPackets[incomingPacketKey].Count;
|
|
for (var i = 0; i < incomingPackets[incomingPacketKey].Count; i++)
|
|
{
|
|
if (incomingPackets[incomingPacketKey][i][0].id == packet.id)
|
|
{
|
|
packetListIndex = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (packetListIndex == incomingPackets[incomingPacketKey].Count)
|
|
{
|
|
incomingPackets[incomingPacketKey].Add(new List<Packet>());
|
|
}
|
|
|
|
var insertionIndex = -1;
|
|
|
|
for (var i = 0; i < incomingPackets[incomingPacketKey][packetListIndex].Count; i++)
|
|
{
|
|
if (incomingPackets[incomingPacketKey][packetListIndex][i].fragment > packet.fragment)
|
|
{
|
|
insertionIndex = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (insertionIndex >= 0)
|
|
{
|
|
incomingPackets[incomingPacketKey][packetListIndex].Insert(insertionIndex, packet);
|
|
}
|
|
else
|
|
{
|
|
incomingPackets[incomingPacketKey][packetListIndex].Add(packet);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Find fully received packets
|
|
var emptyPacketLists = new List<List<Packet>>();
|
|
foreach (var keyValuePair in incomingPackets)
|
|
{
|
|
for (var packetList = 0; packetList < keyValuePair.Value.Count; packetList++)
|
|
{
|
|
var packetReady = true;
|
|
var packetLength = 0;
|
|
for (var packet = 0; packet < keyValuePair.Value[packetList].Count; packet++)
|
|
{
|
|
var tempPacket = keyValuePair.Value[packetList][packet];
|
|
if (tempPacket.fragment != packet || packet == keyValuePair.Value[packetList].Count - 1 && tempPacket.moreFragments)
|
|
{
|
|
packetReady = false;
|
|
}
|
|
else
|
|
{
|
|
packetLength += tempPacket.data.Length;
|
|
}
|
|
}
|
|
|
|
if (packetReady)
|
|
{
|
|
var data = new byte[packetLength];
|
|
var dataIndex = 0;
|
|
|
|
for (var packet = 0; packet < keyValuePair.Value[packetList].Count; packet++)
|
|
{
|
|
Array.Copy(keyValuePair.Value[packetList][packet].data, 0, data, dataIndex, keyValuePair.Value[packetList][packet].data.Length);
|
|
dataIndex += keyValuePair.Value[packetList][packet].data.Length;
|
|
}
|
|
|
|
OnReceiveData(data, keyValuePair.Key.productUserId, keyValuePair.Key.channel);
|
|
|
|
//keyValuePair.Value[packetList].Clear();
|
|
emptyPacketLists.Add(keyValuePair.Value[packetList]);
|
|
}
|
|
}
|
|
|
|
for (var i = 0; i < emptyPacketLists.Count; i++)
|
|
{
|
|
keyValuePair.Value.Remove(emptyPacketLists[i]);
|
|
}
|
|
|
|
emptyPacketLists.Clear();
|
|
}
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
Debug.LogException(e);
|
|
}
|
|
}
|
|
|
|
protected abstract void OnReceiveInternalData(InternalMessages type, ProductUserId clientUserID, SocketId socketId);
|
|
protected abstract void OnReceiveData(byte[] data, ProductUserId clientUserID, int channel);
|
|
protected abstract void OnConnectionFailed(ProductUserId remoteId);
|
|
} |