Ghost snapshots and synchronization
A ghost is a networked object that the server simulates. During every frame, the server sends a snapshot of the current state of all ghosts to the client. The client presents them, but cannot directly control or affect them because the server owns them (it has authority).
Under the hood: The ghost snapshot system synchronizes entities, which exist on the server, to all clients. To improve performance, the server processes ghosts per entity chunk, rather than per entity. However, on the receiving client side, processing is done per entity. It's not possible to process per chunk on both sides because one set of entities in one chunk on the server may not correspond with the same set of entities in one chunk on the client. There are also multiple clients, each with their own entity-in-chunk layout.
Should you use ghosts or RPCs?
You can use both ghosts and RPCs in your game. Each one has specific use cases where it excels compared to the other.
Ghost snapshot synchronization (eventual consistency) use cases
Use ghosts to:
- Replicate spatially local, ephemeral, and relevant per-entity data.
- Enable client prediction of ghost entities, which is the most effective latency-hiding multiplayer technique.
RPC use cases
Use RPCs to:
- Communicate high-level game flow events. For example, making every client do a certain thing, like load a specific level.
- Send one-off, non-predicted commands from the client to the server. For example: Join this squad. Send a chat message. Unmute this player. Request to exit this realm.
Key differences
- RPCs are one-shot events, and are therefore not automatically persisted.
- For example, if you send an RPC when a treasure chest is opened, if a player disconnects and reconnects, the chest will appear closed.
- Ghost data persists for the lifetime of its ghost entity (and the lifetime of the ghost entity is itself replicated). Therefore, long-lived user-interactable entities should have their persistent state stored in ghost components.
- For example, a chest's finite-state machine (FSM) can be stored as an enum on a component. If a player opens the chest, disconnects, then reconnects, they will re-receive the chest, as well as its open state.
- RPCs are sent as reliable packets, while ghosts snapshots are unreliable (with eventual consistency).
- RPC data is sent and received as it is, while ghost data goes through optimizations like diff and delta-compression, and can go through value smoothing when received.
- RPCs are not tied to any particular tick, or other snapshot timing data. They are simply processed on the frame that they are received.
- Ghost snapshot data can work with interpolation and prediction (with snapshot history), and thus history, rollback, and resimulation.
- Ghost snapshot data can be bandwidth optimized via relevancy and importance. RPCs are either broadcast, or sent to a single client.
Authoring ghosts
Ghost can be authored in the Editor by creating a prefab with a GhostAuthoringComponent.
The GhostAuthoringComponent has a small editor that you can use to configure how Netcode for Entities synchronizes the prefab.
You must set the Name, Importance, Supported Ghost Mode, Default Ghost Mode and Optimization Mode property on each ghost.
Netcode for Entities uses the Importance property to control which entities are sent when there is not enough bandwidth to send all instantiated ghosts. A higher value makes it more likely that the ghost will be sent.
The (optional) MaxSendRate property denotes the absolute maximum send frequency (in Hz) for ghost chunks of this ghost prefab type (excluding a few nuanced exceptions).
Important Note: MaxSendRate
only denotes the maximum possible replication frequency, and cannot be enforced in all cases.
I.e. Other factors (like ClientServerTickRate.NetworkTickRate
, ghost instance count, Importance,
Importance-Scaling, GhostSendSystemData.DefaultSnapshotPacketSize
, and structural changes etc.) will determine the final send rate.
Examples:
- A ghost with a
MaxSendRate
of 100Hz will still be rate limited by theNetworkTickRate
itself, which is 60Hz by default. - Similarly, a ghost with a
MaxSendRate
of 60Hz instantiated in a project with aNetworkTickRate
of 30Hz will be sent at a maximum of 30Hz. - As this calculation can only be performed on integer/whole
ticksSinceLastSent
ticks, a ghost with aMaxSendRate
in-between multiples of theNetworkTickRate
will be rounded down to the next multiple. E.g.NetworkTickRate:30Hz
,MaxSendRate:45
means 30Hz is the actual maximum send rate.
You can select from three different Supported Ghost Mode types:
- All - this ghost supports both being interpolated and predicted.
- Interpolated - this ghost only supports being interpolated, it cannot be spawned as a predicted ghost.
- Predicted - this ghost only supports being predicted, it cannot be spawned as a interpolated ghost.
You can select from three different Default Ghost Mode types:
- Interpolated - all ghosts Unity receives from the server are treated as interpolated.
- Predicted - all ghosts Unity receives from the server are treated as predicted.
- Owner predicted - the ghost is predicted for the client that owns it, and interpolated for all other clients. When you select this property, you must also add a GhostOwner and set its NetworkId field in your code. Unity compares this field to each clients' network ID to find the correct owner.
You can select from two different Optimization Mode types:
- Dynamic - the ghost will be optimized for having small snapshot size both when changing and when not changing.
- Static - the ghost will not be optimized for having small snapshot size when changing, but it will not be sent at all when it is not changing.
Replicating components and buffers
Netcode for Entities uses C# attributes to configure which components and fields are synchronized as part of a ghost. There are three fundamental attributes you can use:
NetCode Attribute | Usage |
---|---|
GhostFieldAttribute | The GhostFieldAttribute should be used to mark which component (or buffer) fields should be serialized. The attribute can be added to struct fields and properties. Once a component has at least one field marked with [GhostField] , it becomes replicated, and will be transmitted as part of the ghost data. |
GhostEnabledBitAttribute | Similarly, the GhostEnabledBitAttribute should be used on an IEnableableComponent struct definition, to denote that the enabled bit for this component should be serialized. Once a component is flagged with [GhostEnabledBit] , its enabled-bit will be replicated, and thus transmitted as part of the ghost data. |
GhostComponentAttribute | The GhostComponentAttribute should be used on a ComponentType struct definition to:- Declare for which version of the prefab the component should be present. - Declare if the component should be serialized also for child entities. - Declare to which subset of clients a component should be replicated. IMPORTANT NOTE: Adding a GhostComponent attribute won't make your component fields replicate. You must mark each field with a GhostField attribute individually. |
Authoring component serialization
To signal the Netcode for Entities that a component should be serialized, you need to add a [GhostField]
attribute to the values you want to send.
public struct MySerialisedComponent : IComponentData
{
[GhostField]public int MyIntField;
[GhostField(Quantization=1000)]public float MyFloatField;
[GhostField(Quantization=1000, Smoothing=SmoothingAction.Interpolate)]public float2 Position;
public float2 NonSerialisedField;
...
}
For a component to support serialization, the following conditions must be met:
- The component itself must be declared as
public
.- Only
public
members are considered. Adding a[GhostField]
to a private member has no effect.
- Only
- The
GhostField
can specify quantization for floating point numbers (and other supported types, see Ghost Type Templates). The floating point number is multiplied by the quantization value and converted to an integer to save bandwidth. Quantization is disabled by default on float and unavailable on integers. - The
GhostField
Composite
flag controls how the delta compression computes the change field's bitmask for non-primitive fields (such as structs). When set totrue
, Netcode's delta compression templating generates only 1 bit to indicate if the entire struct contains any changes. IfComposite
is false, however, each field will have its own change-bit. Thus, useComposite=true
if all fields are typically modified together (example:GUID
). - The
GhostField
SendData
flag (which defaults totrue
) can be used to instruct code-generation to not include this field in the serialization data. This is particularly useful for non primitive members (like structs), which will have all fields serialized by default. - The
GhostField
also has aSmoothing
property, which controls how the field will be updated, when the ghost is inGhostMode.Interpolated
. I.e. When a given client is not predicting said ghost. Possible values are:SmoothingAction.Clamp
- Every time a snapshot is received, jump (i.e. clamp) the client value to the latest snapshot value.SmoothingAction.Interpolate
- Every frame, interpolate the field between the last two snapshot values. If no data is available for the next tick, clamp to the latest value.SmoothingAction.InterpolateAndExtrapolate
- Every frame, interpolate the field between the last two snapshot values. If no data is available for the next tick, the next value is linearly extrapolated using the previous two snapshot values. Extrapolation is limited (i.e. clamped) viaClientTickRate.MaxExtrapolationTimeSimTicks
.
GhostField
MaxSmoothingDistance
allows you to disable interpolation when the values change more than the specified limit between two snapshots. This is useful for dealing with teleportation, for example.- Finally the
GhostField
has aSubType
property which can be set to an integer value to use special serialization rules supplied for that specific field.
[!NOTE] Speaking of teleportation: To support short range teleportation, you'd need some other replicated bit to distinguish a teleport from a move (lerp).
Authoring dynamic buffer serialization
Dynamic buffers serialization is natively supported. Unlike components, to replicate a buffer, all public fields must be marked with at [GhostField]
attribute.
[!NOTE] This restriction has been added to guarantee that in the case where an element is added to the buffer, when it is replicated to the client, all fields on said element will have meaningful values. This restriction may be removed in the future (e.g. by instead, defaulting this undefined behavior to
default(T)
).
public struct SerialisedBuffer : IBufferElementData
{
[GhostField]public int Field0;
[GhostField(Quantization=1000)]public float Field1;
[GhostField(Quantization=1000)]public float2 Position;
public float2 NonSerialisedField; // This is an explicit error!
private float2 NonSerialisedField; // We allow this. Ensure you set this on the client, before reading from it.
[GhostField(SendData=false)]public int NotSentAndUninitialised; // We allow this. Ensure you set this on the client, before reading from it.
...
}
Furthermore, in line with the IComponentData
:
- The buffer must be declared as
public
. - Only
public
members are considered. Adding a[GhostField]
to a private member has no effect. - By using the
GhostField.SendData
you can instrument the serialization code to skip certain field. In such a case:- The value of the fields that aren't replicated are never altered.
- For new buffer elements, their content is not set to default and the content is undefined (can be any value).
Dynamic buffers fields don't support SmoothingAction
s. Thus, the GhostFieldAttribute.Smoothing
and GhostFieldAttribute.MaxSmoothingDistance
properties will be ignored when used on buffers.
ICommandData
and IInputComponentData
serialization
You can annotate your input's fields with [GhostField]
to replicate them from server to client. This can be useful, for example, to enable client-side prediction of other players' character controllers on your local machine.
When using automated input synchronization with IInputComponentData
:
public struct MyCommand : IInputComponentData
{
[GhostField] public int Value;
}
ICommandData
is a subclass of IBufferElementData
and so can be serialized from the server to clients. As such, the same rules for buffers apply: if the command buffer must be serialized, then all fields must be annotated.
When using ICommandData
:
[GhostComponent()]
public struct MyCommand : ICommandData
{
[GhostField] public NetworkTick Tick {get; set;}
[GhostField] public int Value;
}
The command data serialization is particularly useful for implementing RemotePlayerPrediction.
GhostField
inheritance
If a [GhostField]
is specified for a non primitive field type, the attribute (and some of its properties) are automatically inherited by all the sub-fields which do not themselves implement a [GhostField]
attribute. For example:
public struct Vector2
{
public float x;
[GhostField(Quantization=100)] public float y;
}
public struct MyComponent : IComponentData
{
//Value.x will inherit the quantization value specified by the parent definition (1000).
//Value.y will maintain its original quantization value (100).
[GhostField(Quantized=1000)] public Vector2 Value;
}
The following properties are not inherited:
- SubType - The subtype is always reset to the default
Using the GhostComponentAttribute
The GhostComponentAttribue
does not indicate or signal that a component is replicated (that's what the other two attributes are for). Instead, it should be used to instruct the runtime how to handle the component when it comes to:
SendDataForChildEntity
denotes whether or not to replicate this component when it is attached to a child of a ghost entity. Replicating child of ghost entities is significantly slower than replicating those ghost root entities. Thus, we default tofalse
. This flag also applies to the[GhostEnabledBit]
.PrefabType
allows you to remove the component from the specific version of the ghost prefab. Ghost prefabs have three versions:Interpolated Client (IC)
,Predicted Client (PC)
, andServer (S)
. Example: Removing rendering related components from the server world's version of this ghost.GhostSendType
denotes whether or not the component should be sent when the ghost isGhostMode.Predicted
vsGhostMode.Interpolated
. Example: Only sendingPhysicsVelocity
when you're actually predicting the physics of a ghost.SendToOwnerType
denotes whether or not to replicate this data when the ghost is owned, vs not owned, vs either. Example: Replicating input commands only to other players (as you already know your own).
[GhostComponent(PrefabType=GhostPrefabType.All, SendTypeOptimization=GhostSendType.OnlyInterpolatedClients, SendDataForChildEntity=false)]
public struct MyComponent : IComponentData
{
[GhostField(Quantized=1000)] public float3 Value;
}
PrefabType
details
To change which versions of a ghost prefab a component is available on, use PrefabType
in a GhostComponentAttribute
on the component. PrefabType
can be on of the these types:
InterpolatedClient
- the component is only available on clients, and only when the ghost is interpolated.PredictedClient
- the component is only available on clients, and only when the ghost is predicted.Client
- the component is only available on clients, regardless of theGhostMode
(e.g. either predicted or interpolated).Server
- the component is only available on the server.AllPredicted
- the component is only available on the server, and on clients, only when the ghost is predicted.All
- the component is available on the server and all clients.
For example, if you add [GhostComponent(PrefabType=GhostPrefabType.Client)]
to RenderMesh
, the ghost won’t have a RenderMesh
when it is instantiated on the server world, but it will have it when instantiated on the client world.
[!NOTE] Runtime Prediction Switching therefore has the potential to add and remove components on a ghost, live.
SendTypeOptimization
details
A component can set SendTypeOptimization
in the GhostComponentAttribute
to control which clients the component is sent to, whenever a ghost type is known at compile time. The available modes are:
None
- the component is never sent to any clients. Netcode will not modify the component on the clients which do not receive it.Interpolated
- the component is only sent to clients which are interpolating the ghost.Predicted
- the component is only sent to clients which are predicting the ghost.All
- the component is sent to all clients.
A component can also set SendDataForChildEntity to true in order to change the default (of not serializing children), allowing this component to be serialized when on a child.
A component can also set SendToOwner in the GhostComponentAttribute to specify if the component should be sent to client who owns the entity. The available values are:
- SendToOwner - the component is only sent to the client who own the ghost
- SendToNonOwner - the component is sent to all clients except the one who owns the ghost
- All - the component is sent to all clients.
[!NOTE] By setting either the
SendTypeOptimisation
and/orSendToOwner
(to specify to which types of client(s) the component should be replicated to), will not affect the presence of the component on the prefab, nor modify the component on the clients which did not receive it.
How to add serialization support for custom types
The types you can serialize via GhostFieldAttribute
are specified via templates. You can see the default supported types here.
In addition to the default out-of-the-box types you can also:
- add your own templates for new types.
- provide a custom serialization templates for a types and target by using the SubTypes property of the
GhostFieldAttribute
.
Please check how to use and write templates for more information on the topic.
[!NOTE] Writing templates is non-trivial. If it is possible to replicate the type simply by adding GhostFields, it's often easier to just do so. If you do not have access to a type, create a Variant instead (see section below).
Ghost component variants
The GhostComponentVariationAttribute is special attribute that can be used to declare at compile time
a "replication schema" for a type, without the need to markup the fields in the original type, or the original type itself.
[!NOTE]This new declared type act as proxy from a code-generation perspective. Instead of using the original type, the code-generation system use the declared "variant" to generate a specific version of the serialization code.
[!NOTE] Ghost components variants for
IBufferElementData
are not fully supported.
The GhostComponentVariationAttribute
has some specific use-cases in mind:
- Variants allow user-code (you) to declare serialization rules for a component that you don't have direct write access too (i.e. components in a package or external assembly). Example: Making
Unity.Entities.LocalTransform
replicated. - Generate multiple serialization strategies for a single type (i.e: provide multiple ways to replicate a single type, allowing individual ghosts to select their version). Example: Replicate only the Yaw value of
Unity.Entities.LocalRotation
, or the fullquaternion
. - Strip components (i.e:
RenderMesh
) from certain prefab types (e.g. from the Server) by overriding or adding aGhostComponentAttribute
to the type, without changing the original declaration.
[GhostComponentVariation(typeof(LocalTransform), "Transform - 2D")]
[GhostComponent(PrefabType=GhostPrefabType.All, SendTypeOptimization=GhostSendType.AllClients)]
public struct PositionRotation2d
{
[GhostField(Quantization=1000, Smoothing=SmoothingAction.InterpolateAndExtrapolate, SubType=GhostFieldSubType.Translation2D)]
public float3 Position;
[GhostField(Quantization=1000, Smoothing=SmoothingAction.InterpolateAndExtrapolate, SubType=GhostFieldSubType.Rotation2D)]
public quaternion Rotation;
}
In the example above, the PositionRotation2d
Variation will generate serialization code for LocalTransform
, using the properties and the attribute present in the variant declaration.
The attribute constructor takes a few arguments:
- The
Type type
of theComponentType
you want to specify the variant for (ex:LocalTransform
). - The user-friendly
string variantName
, which will allow you to better interpretGhostAuthoringInspectionComponent
UI.
Then, for each field in the original struct (in this case, LocalTransform
) that you wish to replicate, you should add a GhostField
attribute like you usually do, and define the field identically to that of the base struct.
[!NOTE] Only members that are present in the component type are allowed. Validation occurs at compile time, and exceptions are thrown in case this rule is not respected.
An optional GhostComponentAttribute
attribute can be added to the variant to further specify the component serialization properties.
It is possible to declare multiple serialization variant for a component (example: 2D and 3D variations for LocalRotation
).
[!NOTE] If you only define only one Variant for a
ComponentType
, it becomes the default serialization strategy for that type automatically.
Preventing a component from supporting variations
There are cases where you'd like to prevent a component from having its serialization modified via variants.
Example: From the Netcode package itself, we must always replicate the GhostComponent
for netcode systems to work properly, so we don't let user-code (you) modify serialization rules for it).
Thus, to prevent a component from supporting variation, use the DontSupportPrefabOverridesAttribute attribute.
An error will be reported at compile time, if a GhostComponentVariation
is defined for that type.
Specify which variant to use on a ghost prefab
Using the GhostAuthoringInspectionComponent
MonoBehaviour
in conjunction with the GhostAuthoringComponent
MonoBehaviour
, it's possible to select what serialization variants to use on a per-prefab basis.
You can choose a variant for each individual component (including the ability to set the special-case variant: DontSerializeVariant
).
All variants for that specific component type present in the project will be shown in a dropbox selection.
To modify how children of ghost prefabs are replicated, add a GhostAuthoringInspectionComponent
to each individual child.
[!NOTE] The
GhostAuthoringInspectionComponent
is also an incredibly valuable debugging tool. Add it to a ghost prefab (or one of its children) to view all replicated types on said ghost, and to diagnose why a specific type is not replicating in the way you'd expect.
Assigning a default variant to use for a type
In cases where multiple variants are present for a type, Netcode for Entities may be unable to infer which variant should be used. If the "Default Serializer" for the Type is replicated, it'll default to it. If not, it's considered a conflict, and you'll get runtime exceptions when creating any Netcode world (including Baking worlds). We use a built-in, deterministic, fallback method to guess which variant you likely want, but, in general, it is the user's responsibility to indicate what variant should be the default here.
To setup which variant to use as the default
for a given type, you need to create a system that inherits from
DefaultVariantSystemBase class, and implements the RegisterDefaultVariants
method.
using System.Collections.Generic;
using Unity.Entities;
using Unity.Transforms;
namespace Unity.NetCode.Samples
{
sealed partial class DefaultVariantSystem : DefaultVariantSystemBase
{
protected override void RegisterDefaultVariants(Dictionary<ComponentType, Rule> defaultVariants)
{
defaultVariants.Add(typeof(LocalTransform), Rule.OnlyParents(typeof(TransformDefaultVariant)));
}
}
}
This example code would make sure the default LocalTransform
variant to us as default is the TransformDefaultVariant
. For more information, please refer to the DefaultVariantSystemBase documentation.
[!NOTE] This is the recommended approach to setup the default variant for a ghost "project-wide". Prefer
DefaultVariantSystemBase
over direct variant manipulation (via theGhostAuthoringInspectionComponent
overrides).
Special variant types
Special built-in variant | Details |
---|---|
ClientOnlyVariant |
Use this to specify that a given ComponentType should only appear on client worlds. |
ServerOnlyVariant |
The inverse. |
DontSerializeVariant |
Use this to disable serialization of a Type entirely. I.e. Use it to ignore replication attributes ([GhostField] and [GhostEnabledBit] ). |
using System.Collections.Generic;
using Unity.Entities;
using Unity.Transforms;
namespace Unity.NetCode.Samples
{
sealed class DefaultVariantSystem : DefaultVariantSystemBase
{
protected override void RegisterDefaultVariants(Dictionary<ComponentType, Rule> defaultVariants)
{
defaultVariants.Add(typeof(SomeClientOnlyThing), Rule.ForAll(typeof(ClientOnlyVariant)));
defaultVariants.Add(typeof(SomeServerOnlyThing), Rule.ForAll(typeof(ServerOnlyVariant)));
defaultVariants.Add(typeof(NoNeedToSyncThis), Rule.ForAll(typeof(DontSerializeVariant)));
}
}
}
You can also manually pick the DontSerializeVariant
in the ghost component on ghost prefabs (via the GhostAuthoringInspectionComponent
).
Assign variants and override GhostComponentAttribute
settings on ghost prefabs
It's possible to override the following meta-data on per-prefab basis by using the GhostAuthoringInspectionComponent editor.
The GhostAuthoringInspectionComponent
should be added to the GameObject
you would like to customize. Once added, the editor will show which components present in the runtime entity are replicated.
The editor allows you to change the following properties:
- Change the
PrefabType
in which the component should be present/replicated. - Change the
SendToOptimization
for this component (if applicable) - Assign the serialization Variant to use for that component.
It's possible to prevent a component from supporting per-prefab overrides by using the DontSupportPrefabOverrides
attribute.
When present, the component can't be customized in the inspector, nor can a programmer add custom or default variants for this type (as that will trigger errors during ghost validation).
For example: The Netcode for Entities package requires the GhostOwner
to be added to all ghost types, sent for all ghost types, and serialized using the default variant. Thus, we add the [DontSupportPrefabOverride]
attribute to it.
[!NOTE] Components on child entities are not serialised by default, thus by default when you look to
GhostAuthoringInspectionComponent
on a child GameObject you will see that the selected variant for the type is theDontSerializeVariant
.
Snapshot visualization tool
To understand what is being put on the wire in the Netcode, you can use the snapshot visualization tool, Network Debugger tool.
To open the tool, go to menu: Multiplayer > Open NetDbg, and the tool opens in a browser window. It displays a vertical bar for each received snapshot, with a breakdown of the snapshot's ghost types, size etc.
To see more detailed information about the snapshot, click on one of the bars.
[!NOTE] This tool is a prototype. In future versions of the package, it will integrate with the Unity Profiler so you can easily correlate network traffic with memory usage and CPU performance.
Partial snapshots
When you start replicating a lot of ghosts (or ghost data), you'll notice that the per-tick snapshot data size will be clamped to the upper MTU
(Max Transmission Unit) limit. As a result, it's common and expected for the snapshot to only contain a subset of all ghosts. Internally these kinds of snapshots are referred to as 'partial snapshots'. Ghosts in chunks with the highest importance values are added first. In other words, Netcode streams your large world, a few ghost chunks at a time, instead of sending all of them at once in a huge packet.
You can change the maximum size of the snapshot. Reducing it saves bandwidth at the cost of a higher relative header overhead and less usable data, while increasing it may cause multiple UDP packets to have to be sent per snapshot, which can increase the probability of packet loss.
See Importance Scaling for more details on how to configure this.
NetworkTickRate
Netcode sends one snapshot to each connection, once per NetworkTickRate
tick (not SimulationTickRate
tick). Refer to Network Tick Rate config. If NetworkTickRate
is below SimulationTickRate
, Netcode collects a subset of all connections and sends one snapshot to each connection in this subset. This is referred to as a 'round robin' approach, as it distributes the GhostSendSystem
load across multiple SimulationTickRate
ticks.