Parenting a host NetworkObject causing soft synchronization errors on clients

When spawning the same network objects using the same ServerRpc on both hosts and clients, the host NetworkObject does not display on any clients, even though the clients can see each other and the host can see the clients. Also, this only occurs if the NetworkObject is parented to another NetworkObject. The image show the issue and the error message that arises. Spawning code is below as well.

What I have checked for so far:
The GlobalObjectIdHash is identical for both host and client spawned NetworkObjects.
The GlobalObjectIdHash matches with the prefab NetworkObject trying to be spawned.
Disabling the parenting code remedies the error, but the spawned NetworkObjects are left as root objects.
The issue arises regardless of using a ServerRpc on the host or spawning natively.

using UnityEngine;
using Unity.Netcode;

public class CarSpawner : NetworkBehaviour
{
    [SerializeField] private GameObject m_CarPrefab;
    [SerializeField] private NetworkObject m_SpawnedNetworkObject;
    public NetworkObject LocalCarObject => m_SpawnedNetworkObject;

    public override void OnNetworkSpawn()
    {
        //only the server spawns
        /*if (IsHost) //diabled to test if the using the same code on host and client would remedy the error
        {
            //instantiate the game object
            GameObject instance = Instantiate(m_CarPrefab);

            //get the instances NetworkObject and Spawn
            m_SpawnedNetworkObject = instance.GetComponent<NetworkObject>();
            m_SpawnedNetworkObject.Spawn();
            //m_SpawnedNetworkObject.TrySetParent(gameObject);
        }
        else if (IsClient)*/
        {
            //tell the server to spawn a car for us
            SpawnClientCarServerRpc(NetworkManager.Singleton.LocalClientId);
        }
    }

    [ServerRpc(RequireOwnership = false)]
    public void SpawnClientCarServerRpc(ulong clientID, ServerRpcParams rpcParams = default)
    {
        //instantiate the game object
        GameObject instance = Instantiate(m_CarPrefab);

        //get the instances NetworkObject and Spawn
        NetworkObject spawned = instance.GetComponent<NetworkObject>();
        spawned.SpawnWithOwnership(clientID);
        spawned.TrySetParent(gameObject);

        //send back the ID of the car to the client
        OnSpawnedNetworkObjectClientRpc(spawned.NetworkObjectId);
    }

    [ClientRpc]
    public void OnSpawnedNetworkObjectClientRpc(ulong spawnedObjectID, ClientRpcParams rpcParams = default)
    {
        if (IsClient && m_SpawnedNetworkObject == null)
        {
            //find the spawned object
            m_SpawnedNetworkObject = GetNetworkObject(spawnedObjectID);
        }
    }
}

I am not following exactly but the idea of spawning an object on both server and client sounds wrong. You are supposed to spawn on the server only, and the server will have the clients replicate the spawned object. Of course a client can trigger a spawn by calling a ServerRPC but the client itself cannot spawn network objects.

One thought: try reversing setting the parent and assigning ownerships. Also, something tells me that you cannot have diverging ownership in a network object tree. So if the parent is host owned the children may not be set to be owned by clients. Respectively ownership may have to be set on the root network object. Please check if there is any mention of that in the docs.

For the first point, everything is being spawned on the server side via ServerRpc, then the client finds the spawned object using the NetworkObjectId.

For the second point, that is a good thought. I’ll dig through the docs to see if there is any mention of parenting and ownership limitations.

Thanks!

After thinking it through, this seems like a bug. The server is instantiating and syncing everything and retains ownership of the NetworkObject that is having issues soft synching on clients. From what it sounds like, the netcode is mistaking the hosts NetworkObject as an ‘In-Scene placed NetworkObject’, when in fact it was instantiated. The error code backs up this theory exactly:

[Netcode] NetworkPrefab hash was not found! In-Scene placed NetworkObject soft synchronization failure for Hash: 257822384!

The Hash here is referring to the instantiated prefab, not an object pre-existing in the scene. I’m going to do some additional testing for a workaround, now that I have a better idea of what the problem is.

And bingo. Waiting to spawn anything on the host/server does the trick. Here is the new code:

using UnityEngine;
using Unity.Netcode;
using System.Collections;

public class CarSpawner : NetworkBehaviour
{
    [SerializeField] private GameObject m_CarPrefab;
    [SerializeField] private NetworkObject m_SpawnedNetworkObject;
    public NetworkObject LocalCarObject => m_SpawnedNetworkObject;

    public override void OnNetworkSpawn()
    {
        //tell the server to spawn a car for us
        //SpawnClientCarServerRpc(NetworkManager.Singleton.LocalClientId); //this causes a soft-sync error on the host
        StartCoroutine(WaitSpawn()); //wait until the server has had a chance to 'register' all the in-scene placed network objects before spawning anything
    }

    private IEnumerator WaitSpawn()
    {
        yield return new WaitForSeconds(1);
        SpawnClientCarServerRpc(NetworkManager.Singleton.LocalClientId);
    }

    [ServerRpc(RequireOwnership = false)]
    public void SpawnClientCarServerRpc(ulong clientID, ServerRpcParams rpcParams = default)
    {
        //instantiate the game object
        GameObject instance = Instantiate(m_CarPrefab);

        //get the instances NetworkObject and Spawn
        NetworkObject spawned = instance.GetComponent<NetworkObject>();
        spawned.SpawnWithOwnership(clientID);
        spawned.TrySetParent(gameObject);

        //send back the ID of the car to the client
        OnSpawnedNetworkObjectClientRpc(spawned.NetworkObjectId);
    }

    [ClientRpc]
    public void OnSpawnedNetworkObjectClientRpc(ulong spawnedObjectID, ClientRpcParams rpcParams = default)
    {
        if (IsClient && m_SpawnedNetworkObject == null)
        {
            //find the spawned object
            m_SpawnedNetworkObject = GetNetworkObject(spawnedObjectID);
        }
    }
}

I can guess all day as to why this is, but the Unity engineers would definitely have a better understanding than I.

However, my best guess is that when a scene is loaded on the host/server, the check for all ‘In-Scene placed NetworkObjects’ is happening after the OnNetworkSpawn() is being called. In this case, it causes anything that gets spawned using OnNetworkSpawn() on the host/server to be mistaken for an ‘In-Scene placed NetworkObject’.

Then, when a client connects, an attempt it made to soft sync the host/server NetworkObjects as ‘In-Scene placed’ instead of instantiated, hence the error. Meanwhile, any objects spawned for clients happened well after the scene was loaded on the host/server, so those NetworkObjects get synced correctly as instantiated prefabs.

The workaround is to have the host/server wait to spawn anything gives it a chance to check the scene for NetworkObjects that are indeed ‘In-Scene placed’. Those instantiated NetworkObjects are then no longer mistaken as placed ones, meaning they soft sync correctly on the clients.

1 Like