Skip to content

Commit c0cd658

Browse files
fix: player prefab null exception when using prefab hash via connection approval (#3042)
* fix fixing the issue where setting the prefab hash but not assigning a player prefab would cause an exception. * update change log entry * test Updating the ConnectionApproval test to include spawning from player prefab, prefab hash, spawning no player, and failing validation. * update updating changelog with PR number.
1 parent aee018c commit c0cd658

File tree

4 files changed

+125
-65
lines changed

4 files changed

+125
-65
lines changed

com.unity.netcode.gameobjects/CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ Additional documentation and release notes are available at [Multiplayer Documen
1919

2020
### Fixed
2121

22+
- Fixed issue where setting a prefab hash value during connection approval but not having a player prefab assigned could cause an exception when spawning a player. (#3042)
2223
- Fixed issue where the `NetworkSpawnManager.HandleNetworkObjectShow` could throw an exception if one of the `NetworkObject` components to show was destroyed during the same frame. (#3030)
2324
- Fixed issue where the `NetworkManagerHelper` was continuing to check for hierarchy changes when in play mode. (#3026)
2425
- Fixed issue with newly/late joined clients and `NetworkTransform` synchronization of parented `NetworkObject` instances. (#3013)

com.unity.netcode.gameobjects/Runtime/Connection/NetworkConnectionManager.cs

+8-26
Original file line numberDiff line numberDiff line change
@@ -735,42 +735,23 @@ internal void HandleConnectionApproval(ulong ownerClientId, NetworkManager.Conne
735735
RemovePendingClient(ownerClientId);
736736

737737
var client = AddClient(ownerClientId);
738-
if (!NetworkManager.DistributedAuthorityMode && response.CreatePlayerObject && NetworkManager.NetworkConfig.PlayerPrefab != null)
739-
{
740-
var prefabNetworkObject = NetworkManager.NetworkConfig.PlayerPrefab.GetComponent<NetworkObject>();
741-
var playerPrefabHash = response.PlayerPrefabHash ?? prefabNetworkObject.GlobalObjectIdHash;
742-
743-
// Generate a SceneObject for the player object to spawn
744-
// Note: This is only to create the local NetworkObject, many of the serialized properties of the player prefab will be set when instantiated.
745-
var sceneObject = new NetworkObject.SceneObject
746-
{
747-
OwnerClientId = ownerClientId,
748-
IsPlayerObject = true,
749-
IsSceneObject = false,
750-
HasTransform = prefabNetworkObject.SynchronizeTransform,
751-
Hash = playerPrefabHash,
752-
TargetClientId = ownerClientId,
753-
DontDestroyWithOwner = prefabNetworkObject.DontDestroyWithOwner,
754-
Transform = new NetworkObject.SceneObject.TransformData
755-
{
756-
Position = response.Position.GetValueOrDefault(),
757-
Rotation = response.Rotation.GetValueOrDefault()
758-
}
759-
};
760738

761-
// Create the player NetworkObject locally
762-
var networkObject = NetworkManager.SpawnManager.CreateLocalNetworkObject(sceneObject);
739+
// Server-side spawning (only if there is a prefab hash or player prefab provided)
740+
if (!NetworkManager.DistributedAuthorityMode && response.CreatePlayerObject && (response.PlayerPrefabHash.HasValue || NetworkManager.NetworkConfig.PlayerPrefab != null))
741+
{
742+
var playerObject = response.PlayerPrefabHash.HasValue ? NetworkManager.SpawnManager.GetNetworkObjectToSpawn(response.PlayerPrefabHash.Value, ownerClientId, response.Position.GetValueOrDefault(), response.Rotation.GetValueOrDefault())
743+
: NetworkManager.SpawnManager.GetNetworkObjectToSpawn(NetworkManager.NetworkConfig.PlayerPrefab.GetComponent<NetworkObject>().GlobalObjectIdHash, ownerClientId, response.Position.GetValueOrDefault(), response.Rotation.GetValueOrDefault());
763744

764745
// Spawn the player NetworkObject locally
765746
NetworkManager.SpawnManager.SpawnNetworkObjectLocally(
766-
networkObject,
747+
playerObject,
767748
NetworkManager.SpawnManager.GetNetworkObjectId(),
768749
sceneObject: false,
769750
playerObject: true,
770751
ownerClientId,
771752
destroyWithScene: false);
772753

773-
client.AssignPlayerObject(ref networkObject);
754+
client.AssignPlayerObject(ref playerObject);
774755
}
775756

776757
// Server doesn't send itself the connection approved message
@@ -871,6 +852,7 @@ internal void HandleConnectionApproval(ulong ownerClientId, NetworkManager.Conne
871852
}
872853
}
873854

855+
// Exit early if no player object was spawned
874856
if (!response.CreatePlayerObject || (response.PlayerPrefabHash == null && NetworkManager.NetworkConfig.PlayerPrefab == null))
875857
{
876858
return;

com.unity.netcode.gameobjects/TestHelpers/Runtime/NetcodeIntegrationTest.cs

+16-2
Original file line numberDiff line numberDiff line change
@@ -850,6 +850,12 @@ protected void ClientNetworkManagerPostStartInit()
850850

851851
protected virtual bool LogAllMessages => false;
852852

853+
protected virtual bool ShouldCheckForSpawnedPlayers()
854+
{
855+
return true;
856+
}
857+
858+
853859
/// <summary>
854860
/// This starts the server and clients as long as <see cref="CanStartServerAndClients"/>
855861
/// returns true.
@@ -938,7 +944,12 @@ protected IEnumerator StartServerAndClients()
938944
AssertOnTimeout($"{nameof(CreateAndStartNewClient)} timed out waiting for all sessions to spawn Client-{m_ServerNetworkManager.LocalClientId}'s player object!\n {m_InternalErrorLog}");
939945
}
940946
}
941-
ClientNetworkManagerPostStartInit();
947+
948+
if (ShouldCheckForSpawnedPlayers())
949+
{
950+
ClientNetworkManagerPostStartInit();
951+
}
952+
942953
// Notification that at this time the server and client(s) are instantiated,
943954
// started, and connected on both sides.
944955
yield return OnServerAndClientsConnected();
@@ -1030,7 +1041,10 @@ protected void StartServerAndClientsWithTimeTravel()
10301041
}
10311042
}
10321043

1033-
ClientNetworkManagerPostStartInit();
1044+
if (ShouldCheckForSpawnedPlayers())
1045+
{
1046+
ClientNetworkManagerPostStartInit();
1047+
}
10341048

10351049
// Notification that at this time the server and client(s) are instantiated,
10361050
// started, and connected on both sides.
Original file line numberDiff line numberDiff line change
@@ -1,65 +1,135 @@
11
using System;
22
using System.Collections;
3+
using System.Collections.Generic;
34
using System.Text;
45
using NUnit.Framework;
56
using Unity.Netcode.TestHelpers.Runtime;
6-
using UnityEngine;
77
using UnityEngine.TestTools;
88

99
namespace Unity.Netcode.RuntimeTests
1010
{
11-
internal class ConnectionApprovalTests
11+
[TestFixture(PlayerCreation.Prefab)]
12+
[TestFixture(PlayerCreation.PrefabHash)]
13+
[TestFixture(PlayerCreation.NoPlayer)]
14+
[TestFixture(PlayerCreation.FailValidation)]
15+
internal class ConnectionApprovalTests : NetcodeIntegrationTest
1216
{
17+
private const string k_InvalidToken = "Invalid validation token!";
18+
public enum PlayerCreation
19+
{
20+
Prefab,
21+
PrefabHash,
22+
NoPlayer,
23+
FailValidation
24+
}
25+
private PlayerCreation m_PlayerCreation;
26+
private bool m_ClientDisconnectReasonValidated;
27+
28+
private Dictionary<ulong, bool> m_Validated = new Dictionary<ulong, bool>();
29+
30+
public ConnectionApprovalTests(PlayerCreation playerCreation)
31+
{
32+
m_PlayerCreation = playerCreation;
33+
}
34+
35+
protected override int NumberOfClients => 1;
36+
1337
private Guid m_ValidationToken;
14-
private bool m_IsValidated;
1538

16-
[SetUp]
17-
public void Setup()
39+
protected override bool ShouldCheckForSpawnedPlayers()
40+
{
41+
return m_PlayerCreation != PlayerCreation.NoPlayer;
42+
}
43+
44+
protected override void OnServerAndClientsCreated()
1845
{
19-
// Create, instantiate, and host
20-
Assert.IsTrue(NetworkManagerHelper.StartNetworkManager(out _, NetworkManagerHelper.NetworkManagerOperatingMode.None));
46+
m_ClientDisconnectReasonValidated = false;
47+
m_BypassConnectionTimeout = m_PlayerCreation == PlayerCreation.FailValidation;
48+
m_Validated.Clear();
2149
m_ValidationToken = Guid.NewGuid();
50+
var validationToken = Encoding.UTF8.GetBytes(m_ValidationToken.ToString());
51+
m_ServerNetworkManager.ConnectionApprovalCallback = NetworkManagerObject_ConnectionApprovalCallback;
52+
m_ServerNetworkManager.NetworkConfig.PlayerPrefab = m_PlayerCreation == PlayerCreation.Prefab ? m_PlayerPrefab : null;
53+
if (m_PlayerCreation == PlayerCreation.PrefabHash)
54+
{
55+
m_ServerNetworkManager.NetworkConfig.Prefabs.Add(new NetworkPrefab() { Prefab = m_PlayerPrefab });
56+
}
57+
m_ServerNetworkManager.NetworkConfig.ConnectionApproval = true;
58+
m_ServerNetworkManager.NetworkConfig.ConnectionData = validationToken;
59+
60+
foreach (var client in m_ClientNetworkManagers)
61+
{
62+
client.NetworkConfig.PlayerPrefab = m_PlayerCreation == PlayerCreation.Prefab ? m_PlayerPrefab : null;
63+
if (m_PlayerCreation == PlayerCreation.PrefabHash)
64+
{
65+
client.NetworkConfig.Prefabs.Add(new NetworkPrefab() { Prefab = m_PlayerPrefab });
66+
}
67+
client.NetworkConfig.ConnectionApproval = true;
68+
client.NetworkConfig.ConnectionData = m_PlayerCreation == PlayerCreation.FailValidation ? Encoding.UTF8.GetBytes(Guid.NewGuid().ToString()) : validationToken;
69+
if (m_PlayerCreation == PlayerCreation.FailValidation)
70+
{
71+
client.OnClientDisconnectCallback += Client_OnClientDisconnectCallback;
72+
}
73+
}
74+
75+
base.OnServerAndClientsCreated();
2276
}
2377

24-
[UnityTest]
25-
public IEnumerator ConnectionApproval()
78+
private void Client_OnClientDisconnectCallback(ulong clientId)
79+
{
80+
m_ClientNetworkManagers[0].OnClientDisconnectCallback -= Client_OnClientDisconnectCallback;
81+
m_ClientDisconnectReasonValidated = m_ClientNetworkManagers[0].LocalClientId == clientId && m_ClientNetworkManagers[0].DisconnectReason == k_InvalidToken;
82+
}
83+
84+
private bool ClientAndHostValidated()
2685
{
27-
NetworkManagerHelper.NetworkManagerObject.ConnectionApprovalCallback = NetworkManagerObject_ConnectionApprovalCallback;
28-
NetworkManagerHelper.NetworkManagerObject.NetworkConfig.ConnectionApproval = true;
29-
NetworkManagerHelper.NetworkManagerObject.NetworkConfig.PlayerPrefab = null;
30-
NetworkManagerHelper.NetworkManagerObject.NetworkConfig.ConnectionData = Encoding.UTF8.GetBytes(m_ValidationToken.ToString());
31-
m_IsValidated = false;
32-
NetworkManagerHelper.NetworkManagerObject.StartHost();
33-
34-
var timeOut = Time.realtimeSinceStartup + 3.0f;
35-
var timedOut = false;
36-
while (!m_IsValidated)
86+
if (!m_Validated.ContainsKey(m_ServerNetworkManager.LocalClientId) || !m_Validated[m_ServerNetworkManager.LocalClientId])
3787
{
38-
yield return new WaitForSeconds(0.01f);
39-
if (timeOut < Time.realtimeSinceStartup)
88+
return false;
89+
}
90+
if (m_PlayerCreation == PlayerCreation.FailValidation)
91+
{
92+
return m_ClientDisconnectReasonValidated;
93+
}
94+
else
95+
{
96+
foreach (var client in m_ClientNetworkManagers)
4097
{
41-
timedOut = true;
98+
if (!m_Validated.ContainsKey(client.LocalClientId) || !m_Validated[client.LocalClientId])
99+
{
100+
return false;
101+
}
42102
}
43103
}
104+
return true;
105+
}
44106

45-
//Make sure we didn't time out
46-
Assert.False(timedOut);
47-
Assert.True(m_IsValidated);
107+
[UnityTest]
108+
public IEnumerator ConnectionApproval()
109+
{
110+
yield return WaitForConditionOrTimeOut(ClientAndHostValidated);
111+
AssertOnTimeout("Timed out waiting for all clients to be approved!");
48112
}
49113

50114
private void NetworkManagerObject_ConnectionApprovalCallback(NetworkManager.ConnectionApprovalRequest request, NetworkManager.ConnectionApprovalResponse response)
51115
{
52116
var stringGuid = Encoding.UTF8.GetString(request.Payload);
117+
53118
if (m_ValidationToken.ToString() == stringGuid)
54119
{
55-
m_IsValidated = true;
120+
m_Validated.Add(request.ClientNetworkId, true);
121+
response.Approved = true;
122+
}
123+
else
124+
{
125+
response.Approved = false;
126+
response.Reason = "Invalid validation token!";
56127
}
57128

58-
response.Approved = m_IsValidated;
59-
response.CreatePlayerObject = false;
129+
response.CreatePlayerObject = ShouldCheckForSpawnedPlayers();
60130
response.Position = null;
61131
response.Rotation = null;
62-
response.PlayerPrefabHash = null;
132+
response.PlayerPrefabHash = m_PlayerCreation == PlayerCreation.PrefabHash ? m_PlayerPrefab.GetComponent<NetworkObject>().GlobalObjectIdHash : null;
63133
}
64134

65135

@@ -78,13 +148,6 @@ public void VerifyUniqueNetworkConfigPerRequest()
78148

79149
Assert.True(currentHash != newHash, $"Hashed {nameof(NetworkConfig)} values {currentHash} and {newHash} should not be the same!");
80150
}
81-
82-
[TearDown]
83-
public void TearDown()
84-
{
85-
// Stop, shutdown, and destroy
86-
NetworkManagerHelper.ShutdownNetworkManager();
87-
}
88-
89151
}
90152
}
153+

0 commit comments

Comments
 (0)