using System; using System.Collections.Generic; using System.Linq; using System.Reflection; using Syntriax.Engine.Core; namespace Syntriax.Engine.Network; public class NetworkManager : UniverseObject, INetworkManager { private INetworkCommunicator networkCommunicator = null!; private readonly List<(Type packetType, Delegate callback)> delegates = []; public INetworkCommunicator NetworkCommunicator { get => networkCommunicator; set { if (networkCommunicator == value) return; var previousCommunicator = networkCommunicator; networkCommunicator = value; if (previousCommunicator is not null) UnsubscribeDelegates(networkCommunicator); if (networkCommunicator is not null) SubscribeDelegates(networkCommunicator); } } private readonly Dictionary> clientPacketListeners = []; private readonly Dictionary> serverPacketListeners = []; private readonly Dictionary _networkEntities = []; public IReadOnlyDictionary NetworkEntities => _networkEntities; private readonly BehaviourCollector _networkEntityCollector = new(); public IBehaviourCollector NetworkEntityCollector => _networkEntityCollector; public NetworkManager() { CacheDelegates(); _networkEntityCollector.OnCollected.AddListener(OnCollected); _networkEntityCollector.OnRemoved.AddListener(OnRemoved); } private void CacheDelegates() { // Find network packets implementing INetworkPacket IEnumerable packetTypes = AppDomain.CurrentDomain.GetAssemblies().SelectMany(a => a.GetTypes()) .Where(t => typeof(INetworkPacket).IsAssignableFrom(t) && !t.IsInterface && !t.IsAbstract && !t.IsGenericType); MethodInfo onPacketArrivedMethod = GetType() .GetMethod(nameof(OnPacketReceived), BindingFlags.NonPublic | BindingFlags.Instance)!; foreach (Type packetType in packetTypes) { MethodInfo genericOnPacketArrivedMethod = onPacketArrivedMethod.MakeGenericMethod(packetType); Type genericDelegateType = typeof(Action<,>).MakeGenericType(packetType, typeof(string)); Delegate genericPacketReceivedDelegate = Delegate.CreateDelegate(genericDelegateType, this, genericOnPacketArrivedMethod); delegates.Add((packetType, genericPacketReceivedDelegate)); } } private void SubscribeDelegates(INetworkCommunicator networkCommunicator) { MethodInfo subscribeToPacketsMethod = typeof(INetworkCommunicator) .GetMethod(nameof(INetworkCommunicator.SubscribeToPackets), BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.Instance)!; foreach ((Type packetType, Delegate callback) in delegates) { MethodInfo genericSubscribeMethod = subscribeToPacketsMethod.MakeGenericMethod(packetType); genericSubscribeMethod.Invoke(networkCommunicator, [callback]); } } private void UnsubscribeDelegates(INetworkCommunicator networkCommunicator) { MethodInfo unsubscribeFromPacketsMethod = typeof(INetworkCommunicator) .GetMethod(nameof(INetworkCommunicator.UnsubscribeFromPackets), BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.Instance)!; foreach ((Type packetType, Delegate callback) in delegates) { MethodInfo genericUnsubscribeMethod = unsubscribeFromPacketsMethod.MakeGenericMethod(packetType); genericUnsubscribeMethod.Invoke(networkCommunicator, [callback]); } } private void OnPacketReceived(T entityDataPacket, string fromClientId) { Type packetType = typeof(T); if (entityDataPacket is IEntityNetworkPacket entityPacket) { if (networkCommunicator is INetworkCommunicatorClient) if (clientPacketListeners.TryGetValue(packetType, out Dictionary? clientListeners)) if (clientListeners.TryGetValue(entityPacket.EntityId, out PacketListenerData clientListenerData)) clientListenerData.ListenerMethod.Invoke(clientListenerData.Object, [entityDataPacket]); if (networkCommunicator is INetworkCommunicatorServer) if (serverPacketListeners.TryGetValue(packetType, out Dictionary? serverListeners)) if (serverListeners.TryGetValue(entityPacket.EntityId, out PacketListenerData serverListenerData)) serverListenerData.ListenerMethod.Invoke(serverListenerData.Object, [entityDataPacket, fromClientId]); return; } if (networkCommunicator is INetworkCommunicatorClient) if (clientPacketListeners.TryGetValue(packetType, out Dictionary? clientListeners)) foreach ((string id, PacketListenerData clientListenerData) in clientListeners) clientListenerData.ListenerMethod.Invoke(clientListenerData.Object, [entityDataPacket]); if (networkCommunicator is INetworkCommunicatorServer) if (serverPacketListeners.TryGetValue(packetType, out Dictionary? serverListeners)) foreach ((string id, PacketListenerData serverListenerData) in serverListeners) serverListenerData.ListenerMethod.Invoke(serverListenerData.Object, [entityDataPacket, fromClientId]); } private void OnCollected(IBehaviourCollector sender, IBehaviourCollector.BehaviourCollectedArguments args) { INetworkEntity collectedBehaviour = args.BehaviourCollected; if (!_networkEntities.TryAdd(collectedBehaviour.Id, collectedBehaviour)) throw new($"Unable to add {collectedBehaviour.Id} to {nameof(NetworkManager)}"); foreach (Type clientListenerType in collectedBehaviour.GetType().GetInterfaces().Where(i => i.IsGenericType && i.GetGenericTypeDefinition() == typeof(IPacketListenerClient<>))) { Type clientListenerParameterType = clientListenerType.GetGenericArguments().First(); MethodInfo clientListenerReceiveMethod = clientListenerType.GetMethods().First(m => m.Name == nameof(IPacketListenerClient.OnClientPacketArrived)); if (!clientPacketListeners.TryGetValue(clientListenerParameterType, out Dictionary? clientListeners)) { clientListeners = []; clientPacketListeners.Add(clientListenerParameterType, clientListeners); } clientListeners.Add(collectedBehaviour.Id, new(collectedBehaviour, clientListenerReceiveMethod)); } foreach (Type serverListenerType in collectedBehaviour.GetType().GetInterfaces().Where(i => i.IsGenericType && i.GetGenericTypeDefinition() == typeof(IPacketListenerServer<>))) { Type serverListenerParameterType = serverListenerType.GetGenericArguments().First(); MethodInfo serverListenerReceiveMethod = serverListenerType.GetMethods().First(m => m.Name == nameof(IPacketListenerServer.OnServerPacketArrived)); if (!serverPacketListeners.TryGetValue(serverListenerParameterType, out Dictionary? serverListeners)) { serverListeners = []; serverPacketListeners.Add(serverListenerParameterType, serverListeners); } serverListeners.Add(collectedBehaviour.Id, new(collectedBehaviour, serverListenerReceiveMethod)); } } private void OnRemoved(IBehaviourCollector sender, IBehaviourCollector.BehaviourRemovedArguments args) { _networkEntities.Remove(args.BehaviourRemoved.Id); } protected override void OnExitingUniverse(IUniverse universe) => _networkEntityCollector.Unassign(); protected override void OnEnteringUniverse(IUniverse universe) { _networkEntityCollector.Assign(universe); NetworkCommunicator = this.GetRequiredUniverseObjectInParent(); } private record struct PacketListenerData(object Object, MethodInfo ListenerMethod) { public static implicit operator (object @object, MethodInfo listenerMethod)(PacketListenerData value) => (value.Object, value.ListenerMethod); public static implicit operator PacketListenerData((object @object, MethodInfo listenerMethod) value) => new(value.@object, value.listenerMethod); } }