The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/) and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). Additional documentation and release notes are available at [Multiplayer Documentation](https://docs-multiplayer.unity3d.com). ## [1.9.1] - 2024-04-18 ### Added - Added AnticipatedNetworkVariable<T>, which adds support for client anticipation of NetworkVariable values, allowing for more responsive gameplay (#2820) - Added AnticipatedNetworkTransform, which adds support for client anticipation of NetworkTransforms (#2820) - Added NetworkVariableBase.ExceedsDirtinessThreshold to allow network variables to throttle updates by only sending updates when the difference between the current and previous values exceeds a threshold. (This is exposed in NetworkVariable<T> with the callback NetworkVariable<T>.CheckExceedsDirtinessThreshold) (#2820) - Added NetworkVariableUpdateTraits, which add additional throttling support: MinSecondsBetweenUpdates will prevent the NetworkVariable from sending updates more often than the specified time period (even if it exceeds the dirtiness threshold), while MaxSecondsBetweenUpdates will force a dirty NetworkVariable to send an update after the specified time period even if it has not yet exceeded the dirtiness threshold. (#2820) - Added virtual method NetworkVariableBase.OnInitialize() which can be used by NetworkVariable subclasses to add initialization code (#2820) - Added virtual method NetworkVariableBase.Update(), which is called once per frame to support behaviors such as interpolation between an anticipated value and an authoritative one. (#2820) - Added NetworkTime.TickWithPartial, which represents the current tick as a double that includes the fractional/partial tick value. (#2820) - Added NetworkTickSystem.AnticipationTick, which can be helpful with implementation of client anticipation. This value represents the tick the current local client was at at the beginning of the most recent network round trip, which enables it to correlate server update ticks with the client tick that may have triggered them. (#2820) - `NetworkVariable` now includes built-in support for `NativeHashSet`, `NativeHashMap`, `List`, `HashSet`, and `Dictionary` (#2813) - `NetworkVariable` now includes delta compression for collection values (`NativeList`, `NativeArray`, `NativeHashSet`, `NativeHashMap`, `List`, `HashSet`, `Dictionary`, and `FixedString` types) to save bandwidth by only sending the values that changed. (Note: For `NativeList`, `NativeArray`, and `List`, this algorithm works differently than that used in `NetworkList`. This algorithm will use less bandwidth for "set" and "add" operations, but `NetworkList` is more bandwidth-efficient if you are performing frequent "insert" operations.) (#2813) - `UserNetworkVariableSerialization` now has optional callbacks for `WriteDelta` and `ReadDelta`. If both are provided, they will be used for all serialization operations on NetworkVariables of that type except for the first one for each client. If either is missing, the existing `Write` and `Read` will always be used. (#2813) - Network variables wrapping `INetworkSerializable` types can perform delta serialization by setting `UserNetworkVariableSerialization<T>.WriteDelta` and `UserNetworkVariableSerialization<T>.ReadDelta` for those types. The built-in `INetworkSerializable` serializer will continue to be used for all other serialization operations, but if those callbacks are set, it will call into them on all but the initial serialization to perform delta serialization. (This could be useful if you have a large struct where most values do not change regularly and you want to send only the fields that did change.) (#2813) ### Fixed - Fixed issue where NetworkTransformEditor would throw and exception if you excluded the physics package. (#2871) - Fixed issue where `NetworkTransform` could not properly synchronize its base position when using half float precision. (#2845) - Fixed issue where the host was not invoking `OnClientDisconnectCallback` for its own local client when internally shutting down. (#2822) - Fixed issue where NetworkTransform could potentially attempt to "unregister" a named message prior to it being registered. (#2807) - Fixed issue where in-scene placed `NetworkObject`s with complex nested children `NetworkObject`s (more than one child in depth) would not synchronize properly if WorldPositionStays was set to true. (#2796) ### Changed - Changed `NetworkObjectReference` and `NetworkBehaviourReference` to allow null references when constructing and serializing. (#2874) - Changed `NetworkAnimator` no longer requires the `Animator` component to exist on the same `GameObject`. (#2872) - Changed `NetworkTransform` to now use `NetworkTransformMessage` as opposed to named messages for NetworkTransformState updates. (#2810) - Changed `CustomMessageManager` so it no longer attempts to register or "unregister" a null or empty string and will log an error if this condition occurs. (#2807)
220 lines
11 KiB
C#
220 lines
11 KiB
C#
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using NUnit.Framework;
|
|
using Unity.Netcode.TestHelpers.Runtime;
|
|
using UnityEngine.TestTools;
|
|
|
|
namespace Unity.Netcode.RuntimeTests
|
|
{
|
|
/// <summary>
|
|
/// Validates the client disconnection process.
|
|
/// This assures that:
|
|
/// - When a client disconnects from the server that the server:
|
|
/// -- Detects the client disconnected.
|
|
/// -- Cleans up the transport to NGO client (and vice versa) mappings.
|
|
/// - When a server disconnects a client that:
|
|
/// -- The client detects this disconnection.
|
|
/// -- The server cleans up the transport to NGO client (and vice versa) mappings.
|
|
/// - When <see cref="OwnerPersistence.DestroyWithOwner"/> the server-side player object is destroyed
|
|
/// - When <see cref="OwnerPersistence.DontDestroyWithOwner"/> the server-side player object ownership is transferred back to the server
|
|
/// </summary>
|
|
[TestFixture(OwnerPersistence.DestroyWithOwner)]
|
|
[TestFixture(OwnerPersistence.DontDestroyWithOwner)]
|
|
public class DisconnectTests : NetcodeIntegrationTest
|
|
{
|
|
public enum OwnerPersistence
|
|
{
|
|
DestroyWithOwner,
|
|
DontDestroyWithOwner
|
|
}
|
|
|
|
public enum ClientDisconnectType
|
|
{
|
|
ServerDisconnectsClient,
|
|
ClientDisconnectsFromServer
|
|
}
|
|
|
|
protected override int NumberOfClients => 1;
|
|
|
|
private OwnerPersistence m_OwnerPersistence;
|
|
private ClientDisconnectType m_ClientDisconnectType;
|
|
private bool m_ClientDisconnected;
|
|
private Dictionary<NetworkManager, ConnectionEventData> m_DisconnectedEvent = new Dictionary<NetworkManager, ConnectionEventData>();
|
|
private ulong m_DisconnectEventClientId;
|
|
private ulong m_TransportClientId;
|
|
private ulong m_ClientId;
|
|
|
|
|
|
public DisconnectTests(OwnerPersistence ownerPersistence)
|
|
{
|
|
m_OwnerPersistence = ownerPersistence;
|
|
}
|
|
|
|
protected override void OnCreatePlayerPrefab()
|
|
{
|
|
m_PlayerPrefab.GetComponent<NetworkObject>().DontDestroyWithOwner = m_OwnerPersistence == OwnerPersistence.DontDestroyWithOwner;
|
|
base.OnCreatePlayerPrefab();
|
|
}
|
|
|
|
protected override void OnServerAndClientsCreated()
|
|
{
|
|
// Adjusting client and server timeout periods to reduce test time
|
|
// Get the tick frequency in milliseconds and triple it for the heartbeat timeout
|
|
var heartBeatTimeout = (int)(300 * (1.0f / m_ServerNetworkManager.NetworkConfig.TickRate));
|
|
var unityTransport = m_ServerNetworkManager.NetworkConfig.NetworkTransport as Transports.UTP.UnityTransport;
|
|
if (unityTransport != null)
|
|
{
|
|
unityTransport.HeartbeatTimeoutMS = heartBeatTimeout;
|
|
}
|
|
|
|
unityTransport = m_ClientNetworkManagers[0].NetworkConfig.NetworkTransport as Transports.UTP.UnityTransport;
|
|
if (unityTransport != null)
|
|
{
|
|
unityTransport.HeartbeatTimeoutMS = heartBeatTimeout;
|
|
}
|
|
|
|
base.OnServerAndClientsCreated();
|
|
}
|
|
|
|
protected override IEnumerator OnSetup()
|
|
{
|
|
m_ClientDisconnected = false;
|
|
m_ClientId = 0;
|
|
m_TransportClientId = 0;
|
|
return base.OnSetup();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Used to detect the client disconnected on the server side
|
|
/// </summary>
|
|
private void OnClientDisconnectCallback(ulong obj)
|
|
{
|
|
m_ClientDisconnected = true;
|
|
}
|
|
|
|
private void OnConnectionEvent(NetworkManager networkManager, ConnectionEventData connectionEventData)
|
|
{
|
|
if (connectionEventData.EventType != ConnectionEvent.ClientDisconnected)
|
|
{
|
|
return;
|
|
}
|
|
|
|
m_DisconnectedEvent.Add(networkManager, connectionEventData);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Conditional check to assure the transport to client (and vice versa) mappings are cleaned up
|
|
/// </summary>
|
|
private bool TransportIdCleanedUp()
|
|
{
|
|
if (m_ServerNetworkManager.ConnectionManager.TransportIdToClientId(m_TransportClientId) == m_ClientId)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (m_ServerNetworkManager.ConnectionManager.ClientIdToTransportId(m_ClientId) == m_TransportClientId)
|
|
{
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Conditional check to make sure the client player object no longer exists on the server side
|
|
/// </summary>
|
|
private bool DoesServerStillHaveSpawnedPlayerObject()
|
|
{
|
|
if (m_PlayerNetworkObjects[m_ServerNetworkManager.LocalClientId].ContainsKey(m_ClientId))
|
|
{
|
|
var playerObject = m_PlayerNetworkObjects[m_ServerNetworkManager.LocalClientId][m_ClientId];
|
|
if (playerObject != null && playerObject.IsSpawned)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
return !m_ServerNetworkManager.SpawnManager.SpawnedObjects.Any(x => x.Value.IsPlayerObject && x.Value.OwnerClientId == m_ClientId);
|
|
}
|
|
|
|
[UnityTest]
|
|
public IEnumerator ClientPlayerDisconnected([Values] ClientDisconnectType clientDisconnectType)
|
|
{
|
|
m_ClientId = m_ClientNetworkManagers[0].LocalClientId;
|
|
m_ClientDisconnectType = clientDisconnectType;
|
|
|
|
var serverSideClientPlayer = m_ServerNetworkManager.ConnectionManager.ConnectedClients[m_ClientId].PlayerObject;
|
|
|
|
m_TransportClientId = m_ServerNetworkManager.ConnectionManager.ClientIdToTransportId(m_ClientId);
|
|
|
|
var clientManager = m_ClientNetworkManagers[0];
|
|
|
|
if (clientDisconnectType == ClientDisconnectType.ServerDisconnectsClient)
|
|
{
|
|
m_ClientNetworkManagers[0].OnClientDisconnectCallback += OnClientDisconnectCallback;
|
|
m_ClientNetworkManagers[0].OnConnectionEvent += OnConnectionEvent;
|
|
m_ServerNetworkManager.OnConnectionEvent += OnConnectionEvent;
|
|
m_ServerNetworkManager.DisconnectClient(m_ClientId);
|
|
}
|
|
else
|
|
{
|
|
m_ServerNetworkManager.OnClientDisconnectCallback += OnClientDisconnectCallback;
|
|
m_ServerNetworkManager.OnConnectionEvent += OnConnectionEvent;
|
|
m_ClientNetworkManagers[0].OnConnectionEvent += OnConnectionEvent;
|
|
|
|
yield return StopOneClient(m_ClientNetworkManagers[0]);
|
|
}
|
|
|
|
yield return WaitForConditionOrTimeOut(() => m_ClientDisconnected);
|
|
AssertOnTimeout("Timed out waiting for client to disconnect!");
|
|
|
|
if (clientDisconnectType == ClientDisconnectType.ServerDisconnectsClient)
|
|
{
|
|
Assert.IsTrue(m_DisconnectedEvent.ContainsKey(m_ServerNetworkManager), $"Could not find the server {nameof(NetworkManager)} disconnect event entry!");
|
|
Assert.IsTrue(m_DisconnectedEvent[m_ServerNetworkManager].ClientId == m_ClientId, $"Expected ClientID {m_ClientId} but found ClientID {m_DisconnectedEvent[m_ServerNetworkManager].ClientId} for the server {nameof(NetworkManager)} disconnect event entry!");
|
|
Assert.IsTrue(m_DisconnectedEvent.ContainsKey(clientManager), $"Could not find the client {nameof(NetworkManager)} disconnect event entry!");
|
|
Assert.IsTrue(m_DisconnectedEvent[clientManager].ClientId == m_ClientId, $"Expected ClientID {m_ClientId} but found ClientID {m_DisconnectedEvent[m_ServerNetworkManager].ClientId} for the client {nameof(NetworkManager)} disconnect event entry!");
|
|
// Unregister for this event otherwise it will be invoked during teardown
|
|
m_ServerNetworkManager.OnConnectionEvent -= OnConnectionEvent;
|
|
}
|
|
else
|
|
{
|
|
Assert.IsTrue(m_DisconnectedEvent.ContainsKey(m_ServerNetworkManager), $"Could not find the server {nameof(NetworkManager)} disconnect event entry!");
|
|
Assert.IsTrue(m_DisconnectedEvent[m_ServerNetworkManager].ClientId == m_ClientId, $"Expected ClientID {m_ClientId} but found ClientID {m_DisconnectedEvent[m_ServerNetworkManager].ClientId} for the server {nameof(NetworkManager)} disconnect event entry!");
|
|
Assert.IsTrue(m_DisconnectedEvent.ContainsKey(clientManager), $"Could not find the client {nameof(NetworkManager)} disconnect event entry!");
|
|
Assert.IsTrue(m_DisconnectedEvent[clientManager].ClientId == m_ClientId, $"Expected ClientID {m_ClientId} but found ClientID {m_DisconnectedEvent[m_ServerNetworkManager].ClientId} for the client {nameof(NetworkManager)} disconnect event entry!");
|
|
}
|
|
|
|
if (m_OwnerPersistence == OwnerPersistence.DestroyWithOwner)
|
|
{
|
|
// When we are destroying with the owner, validate the player object is destroyed on the server side
|
|
yield return WaitForConditionOrTimeOut(DoesServerStillHaveSpawnedPlayerObject);
|
|
AssertOnTimeout("Timed out waiting for client's player object to be destroyed!");
|
|
}
|
|
else
|
|
{
|
|
// When we are not destroying with the owner, ensure the player object's ownership was transferred back to the server
|
|
yield return WaitForConditionOrTimeOut(() => serverSideClientPlayer.IsOwnedByServer);
|
|
AssertOnTimeout("The client's player object's ownership was not transferred back to the server!");
|
|
}
|
|
|
|
yield return WaitForConditionOrTimeOut(TransportIdCleanedUp);
|
|
AssertOnTimeout("Timed out waiting for transport and client id mappings to be cleaned up!");
|
|
|
|
// Validate the host-client generates a OnClientDisconnected event when it shutsdown.
|
|
// Only test when the test run is the client disconnecting from the server (otherwise the server will be shutdown already)
|
|
if (clientDisconnectType == ClientDisconnectType.ClientDisconnectsFromServer)
|
|
{
|
|
m_DisconnectedEvent.Clear();
|
|
m_ClientDisconnected = false;
|
|
m_ServerNetworkManager.Shutdown();
|
|
|
|
yield return WaitForConditionOrTimeOut(() => m_ClientDisconnected);
|
|
AssertOnTimeout("Timed out waiting for host-client to generate disconnect message!");
|
|
|
|
Assert.IsTrue(m_DisconnectedEvent.ContainsKey(m_ServerNetworkManager), $"Could not find the server {nameof(NetworkManager)} disconnect event entry!");
|
|
Assert.IsTrue(m_DisconnectedEvent[m_ServerNetworkManager].ClientId == NetworkManager.ServerClientId, $"Expected ClientID {m_ClientId} but found ClientID {m_DisconnectedEvent[m_ServerNetworkManager].ClientId} for the server {nameof(NetworkManager)} disconnect event entry!");
|
|
}
|
|
}
|
|
}
|
|
}
|