999 lines
24 KiB
C#
Raw Normal View History

2020-12-04 22:15:41 +00:00
using OWML.Logging;
2020-12-07 21:19:16 +00:00
using QuantumUNET.Messages;
2020-12-15 16:37:52 +00:00
using QuantumUNET.Transport;
2020-12-02 21:23:01 +00:00
using System;
2020-12-02 09:51:53 +00:00
using System.Collections.Generic;
using System.Collections.ObjectModel;
using UnityEngine;
2020-12-07 21:19:16 +00:00
namespace QuantumUNET.Components
2020-12-02 09:51:53 +00:00
{
public sealed class QSBNetworkIdentity : MonoBehaviour
{
public bool IsClient { get; private set; }
2020-12-02 18:40:38 +00:00
public bool IsServer => m_IsServer && QSBNetworkServer.active && m_IsServer;
2020-12-02 09:51:53 +00:00
public bool HasAuthority { get; private set; }
2020-12-15 09:56:03 +00:00
public QSBNetworkInstanceId NetId { get; private set; }
public QSBNetworkSceneId SceneId => m_SceneId;
2020-12-02 12:42:26 +00:00
public QSBNetworkConnection ClientAuthorityOwner { get; private set; }
2020-12-15 09:56:03 +00:00
public QSBNetworkHash128 AssetId => m_AssetId;
2020-12-02 09:51:53 +00:00
public bool IsLocalPlayer { get; private set; }
public short PlayerControllerId { get; private set; } = -1;
2020-12-02 12:42:26 +00:00
public QSBNetworkConnection ConnectionToServer { get; private set; }
public QSBNetworkConnection ConnectionToClient { get; private set; }
2020-12-12 18:14:04 +00:00
public QSBNetworkIdentity RootIdentity { get; private set; }
public List<QSBNetworkIdentity> SubIdentities { get; private set; } = new List<QSBNetworkIdentity>();
2020-12-02 09:51:53 +00:00
public bool ServerOnly
{
get
{
return m_ServerOnly;
}
set
{
m_ServerOnly = value;
}
}
public bool LocalPlayerAuthority
{
get
{
return m_LocalPlayerAuthority;
}
set
{
m_LocalPlayerAuthority = value;
}
}
2020-12-12 18:14:04 +00:00
public void SetRootIdentity(QSBNetworkIdentity newRoot)
{
if (RootIdentity != null)
{
RootIdentity.RemoveSubIdentity(this);
}
RootIdentity = newRoot;
RootIdentity.AddSubIndentity(this);
}
internal void AddSubIndentity(QSBNetworkIdentity identityToAdd)
=> SubIdentities.Add(identityToAdd);
internal void RemoveSubIdentity(QSBNetworkIdentity identityToRemove)
=> SubIdentities.Remove(identityToRemove);
2020-12-15 09:56:03 +00:00
internal void SetDynamicAssetId(QSBNetworkHash128 newAssetId)
2020-12-02 09:51:53 +00:00
{
if (!m_AssetId.IsValid() || m_AssetId.Equals(newAssetId))
{
m_AssetId = newAssetId;
}
else
{
Debug.LogWarning("SetDynamicAssetId object already has an assetId <" + m_AssetId + ">");
}
}
2020-12-02 12:42:26 +00:00
internal void SetClientOwner(QSBNetworkConnection conn)
2020-12-02 09:51:53 +00:00
{
if (ClientAuthorityOwner != null)
{
Debug.LogError("SetClientOwner m_ClientAuthorityOwner already set!");
}
ClientAuthorityOwner = conn;
2020-12-02 18:40:38 +00:00
ClientAuthorityOwner.AddOwnedObject(this);
2020-12-02 09:51:53 +00:00
}
internal void ClearClientOwner() => ClientAuthorityOwner = null;
internal void ForceAuthority(bool authority)
{
if (HasAuthority != authority)
{
HasAuthority = authority;
if (authority)
{
OnStartAuthority();
}
else
{
OnStopAuthority();
}
}
}
2020-12-02 12:42:26 +00:00
public ReadOnlyCollection<QSBNetworkConnection> Observers
2020-12-02 09:51:53 +00:00
{
get
{
2020-12-02 12:42:26 +00:00
ReadOnlyCollection<QSBNetworkConnection> result;
2020-12-02 09:51:53 +00:00
if (m_Observers == null)
{
result = null;
}
else
{
2020-12-02 12:42:26 +00:00
result = new ReadOnlyCollection<QSBNetworkConnection>(m_Observers);
2020-12-02 09:51:53 +00:00
}
return result;
}
}
2020-12-15 09:56:03 +00:00
internal static QSBNetworkInstanceId GetNextNetworkId()
2020-12-02 09:51:53 +00:00
{
var value = s_NextNetworkId;
s_NextNetworkId += 1U;
2020-12-15 09:56:03 +00:00
return new QSBNetworkInstanceId(value);
2020-12-02 09:51:53 +00:00
}
private void CacheBehaviours()
{
if (m_NetworkBehaviours == null)
{
2020-12-14 21:58:00 +00:00
m_NetworkBehaviours = GetComponents<QSBNetworkBehaviour>();
2020-12-02 09:51:53 +00:00
}
}
internal static void AddNetworkId(uint id)
{
if (id >= s_NextNetworkId)
{
s_NextNetworkId = id + 1U;
}
}
2020-12-15 09:56:03 +00:00
internal void SetNetworkInstanceId(QSBNetworkInstanceId newNetId)
2020-12-02 09:51:53 +00:00
{
NetId = newNetId;
if (newNetId.Value == 0U)
{
m_IsServer = false;
}
}
2020-12-15 09:56:03 +00:00
public void ForceSceneId(int newSceneId) => m_SceneId = new QSBNetworkSceneId((uint)newSceneId);
2020-12-02 09:51:53 +00:00
internal void UpdateClientServer(bool isClientFlag, bool isServerFlag)
{
IsClient = (IsClient || isClientFlag);
m_IsServer = (m_IsServer || isServerFlag);
}
internal void SetNotLocalPlayer()
{
IsLocalPlayer = false;
2020-12-02 18:40:38 +00:00
if (!QSBNetworkServer.active || !QSBNetworkServer.localClientActive)
2020-12-02 09:51:53 +00:00
{
HasAuthority = false;
}
}
2020-12-02 12:42:26 +00:00
internal void RemoveObserverInternal(QSBNetworkConnection conn)
2020-12-02 09:51:53 +00:00
{
if (m_Observers != null)
{
m_Observers.Remove(conn);
m_ObserverConnections.Remove(conn.connectionId);
}
}
2020-12-14 21:58:00 +00:00
public void OnDestroy()
2020-12-02 09:51:53 +00:00
{
2020-12-02 18:40:38 +00:00
if (m_IsServer && QSBNetworkServer.active)
2020-12-02 09:51:53 +00:00
{
2020-12-14 21:58:00 +00:00
QSBNetworkServer.Destroy(gameObject);
2020-12-02 09:51:53 +00:00
}
}
internal void OnStartServer(bool allowNonZeroNetId)
{
if (!m_IsServer)
{
m_IsServer = true;
if (m_LocalPlayerAuthority)
{
HasAuthority = false;
}
else
{
HasAuthority = true;
}
2020-12-02 12:42:26 +00:00
m_Observers = new List<QSBNetworkConnection>();
2020-12-02 09:51:53 +00:00
m_ObserverConnections = new HashSet<int>();
CacheBehaviours();
if (NetId.IsEmpty())
{
NetId = GetNextNetworkId();
}
else if (!allowNonZeroNetId)
{
2020-12-04 22:14:53 +00:00
ModConsole.OwmlConsole.WriteLine(string.Concat(new object[]
2020-12-02 09:51:53 +00:00
{
"Object has non-zero netId ",
NetId,
" for ",
2020-12-02 18:40:38 +00:00
gameObject
2020-12-02 09:51:53 +00:00
}));
return;
}
2020-12-02 18:40:38 +00:00
QSBNetworkServer.instance.SetLocalObjectOnServer(NetId, gameObject);
2020-12-02 09:51:53 +00:00
for (var i = 0; i < m_NetworkBehaviours.Length; i++)
{
var networkBehaviour = m_NetworkBehaviours[i];
try
{
networkBehaviour.OnStartServer();
}
catch (Exception ex)
{
Debug.LogError("Exception in OnStartServer:" + ex.Message + " " + ex.StackTrace);
}
}
2020-12-02 18:40:38 +00:00
if (QSBNetworkClient.active && QSBNetworkServer.localClientActive)
2020-12-02 09:51:53 +00:00
{
2020-12-14 21:58:00 +00:00
QSBClientScene.SetLocalObject(NetId, gameObject);
2020-12-02 09:51:53 +00:00
OnStartClient();
}
if (HasAuthority)
{
OnStartAuthority();
}
}
}
internal void OnStartClient()
{
if (!IsClient)
{
IsClient = true;
}
CacheBehaviours();
Debug.Log(string.Concat(new object[]
{
"OnStartClient ",
2020-12-14 21:58:00 +00:00
gameObject,
2020-12-02 09:51:53 +00:00
" GUID:",
NetId,
" localPlayerAuthority:",
LocalPlayerAuthority
}));
for (var i = 0; i < m_NetworkBehaviours.Length; i++)
{
var networkBehaviour = m_NetworkBehaviours[i];
try
{
networkBehaviour.PreStartClient();
networkBehaviour.OnStartClient();
}
catch (Exception ex)
{
Debug.LogError("Exception in OnStartClient:" + ex.Message + " " + ex.StackTrace);
}
}
}
internal void OnStartAuthority()
{
for (var i = 0; i < m_NetworkBehaviours.Length; i++)
{
var networkBehaviour = m_NetworkBehaviours[i];
try
{
networkBehaviour.OnStartAuthority();
}
catch (Exception ex)
{
Debug.LogError("Exception in OnStartAuthority:" + ex.Message + " " + ex.StackTrace);
}
}
}
internal void OnStopAuthority()
{
for (var i = 0; i < m_NetworkBehaviours.Length; i++)
{
var networkBehaviour = m_NetworkBehaviours[i];
try
{
networkBehaviour.OnStopAuthority();
}
catch (Exception ex)
{
Debug.LogError("Exception in OnStopAuthority:" + ex.Message + " " + ex.StackTrace);
}
}
}
internal void OnSetLocalVisibility(bool vis)
{
for (var i = 0; i < m_NetworkBehaviours.Length; i++)
{
var networkBehaviour = m_NetworkBehaviours[i];
try
{
networkBehaviour.OnSetLocalVisibility(vis);
}
catch (Exception ex)
{
Debug.LogError("Exception in OnSetLocalVisibility:" + ex.Message + " " + ex.StackTrace);
}
}
}
2020-12-02 18:40:38 +00:00
internal bool OnCheckObserver(QSBNetworkConnection conn)
2020-12-02 09:51:53 +00:00
{
for (var i = 0; i < m_NetworkBehaviours.Length; i++)
{
var networkBehaviour = m_NetworkBehaviours[i];
try
{
if (!networkBehaviour.OnCheckObserver(conn))
{
return false;
}
}
catch (Exception ex)
{
Debug.LogError("Exception in OnCheckObserver:" + ex.Message + " " + ex.StackTrace);
}
}
return true;
}
2020-12-04 09:23:27 +00:00
internal void UNetSerializeAllVars(QSBNetworkWriter writer)
2020-12-02 09:51:53 +00:00
{
for (var i = 0; i < m_NetworkBehaviours.Length; i++)
{
var networkBehaviour = m_NetworkBehaviours[i];
networkBehaviour.OnSerialize(writer, true);
}
}
internal void HandleClientAuthority(bool authority)
{
if (!LocalPlayerAuthority)
{
2020-12-14 21:58:00 +00:00
Debug.LogError("HandleClientAuthority " + gameObject + " does not have localPlayerAuthority");
2020-12-02 09:51:53 +00:00
}
else
{
ForceAuthority(authority);
}
}
private bool GetInvokeComponent(int cmdHash, Type invokeClass, out QSBNetworkBehaviour invokeComponent)
{
QSBNetworkBehaviour networkBehaviour = null;
for (var i = 0; i < m_NetworkBehaviours.Length; i++)
{
var networkBehaviour2 = m_NetworkBehaviours[i];
if (networkBehaviour2.GetType() == invokeClass || networkBehaviour2.GetType().IsSubclassOf(invokeClass))
{
networkBehaviour = networkBehaviour2;
break;
}
}
bool result;
if (networkBehaviour == null)
{
var cmdHashHandlerName = QSBNetworkBehaviour.GetCmdHashHandlerName(cmdHash);
Debug.LogError(string.Concat(new object[]
{
"Found no behaviour for incoming [",
cmdHashHandlerName,
"] on ",
2020-12-14 21:58:00 +00:00
gameObject,
2020-12-02 09:51:53 +00:00
", the server and client should have the same NetworkBehaviour instances [netId=",
NetId,
"]."
}));
invokeComponent = null;
result = false;
}
else
{
invokeComponent = networkBehaviour;
result = true;
}
return result;
}
2020-12-04 09:23:27 +00:00
internal void HandleSyncEvent(int cmdHash, QSBNetworkReader reader)
2020-12-02 09:51:53 +00:00
{
2020-12-14 21:58:00 +00:00
if (gameObject == null)
2020-12-02 09:51:53 +00:00
{
var cmdHashHandlerName = QSBNetworkBehaviour.GetCmdHashHandlerName(cmdHash);
Debug.LogWarning(string.Concat(new object[]
{
"SyncEvent [",
cmdHashHandlerName,
"] received for deleted object [netId=",
NetId,
"]"
}));
}
else if (!QSBNetworkBehaviour.GetInvokerForHashSyncEvent(cmdHash, out var invokeClass, out var cmdDelegate))
{
var cmdHashHandlerName2 = QSBNetworkBehaviour.GetCmdHashHandlerName(cmdHash);
Debug.LogError(string.Concat(new object[]
{
"Found no receiver for incoming [",
cmdHashHandlerName2,
"] on ",
2020-12-14 21:58:00 +00:00
gameObject,
2020-12-02 09:51:53 +00:00
", the server and client should have the same NetworkBehaviour instances [netId=",
NetId,
"]."
}));
}
else if (!GetInvokeComponent(cmdHash, invokeClass, out var obj))
{
var cmdHashHandlerName3 = QSBNetworkBehaviour.GetCmdHashHandlerName(cmdHash);
Debug.LogWarning(string.Concat(new object[]
{
"SyncEvent [",
cmdHashHandlerName3,
"] handler not found [netId=",
NetId,
"]"
}));
}
else
{
cmdDelegate(obj, reader);
}
}
2020-12-04 09:23:27 +00:00
internal void HandleSyncList(int cmdHash, QSBNetworkReader reader)
2020-12-02 09:51:53 +00:00
{
2020-12-14 21:58:00 +00:00
if (gameObject == null)
2020-12-02 09:51:53 +00:00
{
var cmdHashHandlerName = QSBNetworkBehaviour.GetCmdHashHandlerName(cmdHash);
Debug.LogWarning(string.Concat(new object[]
{
"SyncList [",
cmdHashHandlerName,
"] received for deleted object [netId=",
NetId,
"]"
}));
}
else if (!QSBNetworkBehaviour.GetInvokerForHashSyncList(cmdHash, out var invokeClass, out var cmdDelegate))
{
var cmdHashHandlerName2 = QSBNetworkBehaviour.GetCmdHashHandlerName(cmdHash);
Debug.LogError(string.Concat(new object[]
{
"Found no receiver for incoming [",
cmdHashHandlerName2,
"] on ",
2020-12-14 21:58:00 +00:00
gameObject,
2020-12-02 09:51:53 +00:00
", the server and client should have the same NetworkBehaviour instances [netId=",
NetId,
"]."
}));
}
else if (!GetInvokeComponent(cmdHash, invokeClass, out var obj))
{
var cmdHashHandlerName3 = QSBNetworkBehaviour.GetCmdHashHandlerName(cmdHash);
Debug.LogWarning(string.Concat(new object[]
{
"SyncList [",
cmdHashHandlerName3,
"] handler not found [netId=",
NetId,
"]"
}));
}
else
{
cmdDelegate(obj, reader);
}
}
2020-12-04 09:23:27 +00:00
internal void HandleCommand(int cmdHash, QSBNetworkReader reader)
2020-12-02 09:51:53 +00:00
{
2020-12-14 21:58:00 +00:00
if (gameObject == null)
2020-12-02 09:51:53 +00:00
{
var cmdHashHandlerName = QSBNetworkBehaviour.GetCmdHashHandlerName(cmdHash);
Debug.LogWarning(string.Concat(new object[]
{
"Command [",
cmdHashHandlerName,
"] received for deleted object [netId=",
NetId,
"]"
}));
}
else if (!QSBNetworkBehaviour.GetInvokerForHashCommand(cmdHash, out var invokeClass, out var cmdDelegate))
{
var cmdHashHandlerName2 = QSBNetworkBehaviour.GetCmdHashHandlerName(cmdHash);
Debug.LogError(string.Concat(new object[]
{
"Found no receiver for incoming [",
cmdHashHandlerName2,
"] on ",
2020-12-14 21:58:00 +00:00
gameObject,
2020-12-02 09:51:53 +00:00
", the server and client should have the same NetworkBehaviour instances [netId=",
NetId,
"]."
}));
}
else if (!GetInvokeComponent(cmdHash, invokeClass, out var obj))
{
var cmdHashHandlerName3 = QSBNetworkBehaviour.GetCmdHashHandlerName(cmdHash);
Debug.LogWarning(string.Concat(new object[]
{
"Command [",
cmdHashHandlerName3,
"] handler not found [netId=",
NetId,
"]"
}));
}
else
{
cmdDelegate(obj, reader);
}
}
2020-12-04 09:23:27 +00:00
internal void HandleRPC(int cmdHash, QSBNetworkReader reader)
2020-12-02 09:51:53 +00:00
{
2020-12-14 21:58:00 +00:00
if (gameObject == null)
2020-12-02 09:51:53 +00:00
{
var cmdHashHandlerName = QSBNetworkBehaviour.GetCmdHashHandlerName(cmdHash);
Debug.LogWarning(string.Concat(new object[]
{
"ClientRpc [",
cmdHashHandlerName,
"] received for deleted object [netId=",
NetId,
"]"
}));
}
else if (!QSBNetworkBehaviour.GetInvokerForHashClientRpc(cmdHash, out var invokeClass, out var cmdDelegate))
{
var cmdHashHandlerName2 = QSBNetworkBehaviour.GetCmdHashHandlerName(cmdHash);
Debug.LogError(string.Concat(new object[]
{
"Found no receiver for incoming [",
cmdHashHandlerName2,
"] on ",
2020-12-14 21:58:00 +00:00
gameObject,
2020-12-02 09:51:53 +00:00
", the server and client should have the same NetworkBehaviour instances [netId=",
NetId,
"]."
}));
}
else if (!GetInvokeComponent(cmdHash, invokeClass, out var obj))
{
var cmdHashHandlerName3 = QSBNetworkBehaviour.GetCmdHashHandlerName(cmdHash);
Debug.LogWarning(string.Concat(new object[]
{
"ClientRpc [",
cmdHashHandlerName3,
"] handler not found [netId=",
NetId,
"]"
}));
}
else
{
cmdDelegate(obj, reader);
}
}
internal void UNetUpdate()
{
var num = 0U;
for (var i = 0; i < m_NetworkBehaviours.Length; i++)
{
var networkBehaviour = m_NetworkBehaviours[i];
var dirtyChannel = networkBehaviour.GetDirtyChannel();
if (dirtyChannel != -1)
{
num |= 1U << dirtyChannel;
}
}
if (num != 0U)
{
var j = 0;
2020-12-02 18:40:38 +00:00
while (j < QSBNetworkServer.numChannels)
2020-12-02 09:51:53 +00:00
{
if ((num & (1U << j)) != 0U)
{
s_UpdateWriter.StartMessage(8);
s_UpdateWriter.Write(NetId);
var flag = false;
for (var k = 0; k < m_NetworkBehaviours.Length; k++)
{
var position = s_UpdateWriter.Position;
var networkBehaviour2 = m_NetworkBehaviours[k];
if (networkBehaviour2.GetDirtyChannel() != j)
{
networkBehaviour2.OnSerialize(s_UpdateWriter, false);
}
else
{
if (networkBehaviour2.OnSerialize(s_UpdateWriter, false))
{
networkBehaviour2.ClearAllDirtyBits();
flag = true;
}
2020-12-02 18:40:38 +00:00
var maxPacketSize = QSBNetworkServer.maxPacketSize;
2020-12-02 09:51:53 +00:00
if (s_UpdateWriter.Position - position > maxPacketSize)
{
Debug.LogWarning(string.Concat(new object[]
{
"Large state update of ",
2020-12-14 21:58:00 +00:00
s_UpdateWriter.Position - position,
2020-12-02 09:51:53 +00:00
" bytes for netId:",
NetId,
" from script:",
networkBehaviour2
}));
}
}
}
if (flag)
{
s_UpdateWriter.FinishMessage();
2020-12-14 21:58:00 +00:00
QSBNetworkServer.SendWriterToReady(gameObject, s_UpdateWriter, j);
2020-12-02 09:51:53 +00:00
}
}
IL_197:
j++;
continue;
goto IL_197;
}
}
}
2020-12-04 09:23:27 +00:00
internal void OnUpdateVars(QSBNetworkReader reader, bool initialState)
2020-12-02 09:51:53 +00:00
{
if (initialState && m_NetworkBehaviours == null)
{
2020-12-14 21:58:00 +00:00
m_NetworkBehaviours = GetComponents<QSBNetworkBehaviour>();
2020-12-02 09:51:53 +00:00
}
for (var i = 0; i < m_NetworkBehaviours.Length; i++)
{
var networkBehaviour = m_NetworkBehaviours[i];
networkBehaviour.OnDeserialize(reader, initialState);
}
}
internal void SetLocalPlayer(short localPlayerControllerId)
{
2020-12-04 22:14:53 +00:00
ModConsole.OwmlConsole.WriteLine($"SetLocalPlayer {localPlayerControllerId}");
2020-12-02 09:51:53 +00:00
IsLocalPlayer = true;
PlayerControllerId = localPlayerControllerId;
2020-12-14 21:58:00 +00:00
var hasAuthority = HasAuthority;
2020-12-02 09:51:53 +00:00
if (LocalPlayerAuthority)
{
HasAuthority = true;
}
for (var i = 0; i < m_NetworkBehaviours.Length; i++)
{
var networkBehaviour = m_NetworkBehaviours[i];
networkBehaviour.OnStartLocalPlayer();
if (LocalPlayerAuthority && !hasAuthority)
{
networkBehaviour.OnStartAuthority();
}
}
}
2020-12-02 12:42:26 +00:00
internal void SetConnectionToServer(QSBNetworkConnection conn) => ConnectionToServer = conn;
2020-12-02 09:51:53 +00:00
2020-12-02 12:42:26 +00:00
internal void SetConnectionToClient(QSBNetworkConnection conn, short newPlayerControllerId)
2020-12-02 09:51:53 +00:00
{
PlayerControllerId = newPlayerControllerId;
ConnectionToClient = conn;
}
internal void OnNetworkDestroy()
{
var num = 0;
while (m_NetworkBehaviours != null && num < m_NetworkBehaviours.Length)
{
var networkBehaviour = m_NetworkBehaviours[num];
networkBehaviour.OnNetworkDestroy();
num++;
}
m_IsServer = false;
}
internal void ClearObservers()
{
if (m_Observers != null)
{
var count = m_Observers.Count;
for (var i = 0; i < count; i++)
{
var networkConnection = m_Observers[i];
2020-12-02 18:40:38 +00:00
networkConnection.RemoveFromVisList(this, true);
2020-12-02 09:51:53 +00:00
}
m_Observers.Clear();
m_ObserverConnections.Clear();
}
}
2020-12-02 12:42:26 +00:00
internal void AddObserver(QSBNetworkConnection conn)
2020-12-02 09:51:53 +00:00
{
if (m_Observers == null)
{
2020-12-14 21:58:00 +00:00
Debug.LogError("AddObserver for " + gameObject + " observer list is null");
2020-12-02 09:51:53 +00:00
}
else if (m_ObserverConnections.Contains(conn.connectionId))
{
Debug.Log(string.Concat(new object[]
{
"Duplicate observer ",
conn.address,
" added for ",
2020-12-14 21:58:00 +00:00
gameObject
2020-12-02 09:51:53 +00:00
}));
}
else
{
Debug.Log(string.Concat(new object[]
{
"Added observer ",
conn.address,
" added for ",
2020-12-14 21:58:00 +00:00
gameObject
2020-12-02 09:51:53 +00:00
}));
m_Observers.Add(conn);
m_ObserverConnections.Add(conn.connectionId);
2020-12-02 18:40:38 +00:00
conn.AddToVisList(this);
2020-12-02 09:51:53 +00:00
}
}
2020-12-02 12:42:26 +00:00
internal void RemoveObserver(QSBNetworkConnection conn)
2020-12-02 09:51:53 +00:00
{
if (m_Observers != null)
{
m_Observers.Remove(conn);
m_ObserverConnections.Remove(conn.connectionId);
2020-12-02 18:40:38 +00:00
conn.RemoveFromVisList(this, true);
2020-12-02 09:51:53 +00:00
}
}
public void RebuildObservers(bool initialize)
{
if (m_Observers != null)
{
var flag = false;
var flag2 = false;
2020-12-02 12:42:26 +00:00
var hashSet = new HashSet<QSBNetworkConnection>();
var hashSet2 = new HashSet<QSBNetworkConnection>(m_Observers);
2020-12-02 09:51:53 +00:00
for (var i = 0; i < m_NetworkBehaviours.Length; i++)
{
var networkBehaviour = m_NetworkBehaviours[i];
flag2 |= networkBehaviour.OnRebuildObservers(hashSet, initialize);
}
if (!flag2)
{
if (initialize)
{
2020-12-02 12:42:26 +00:00
for (var j = 0; j < QSBNetworkServer.connections.Count; j++)
2020-12-02 09:51:53 +00:00
{
2020-12-02 12:42:26 +00:00
var networkConnection = QSBNetworkServer.connections[j];
2020-12-02 09:51:53 +00:00
if (networkConnection != null)
{
if (networkConnection.isReady)
{
AddObserver(networkConnection);
}
}
}
2020-12-02 12:42:26 +00:00
for (var k = 0; k < QSBNetworkServer.localConnections.Count; k++)
2020-12-02 09:51:53 +00:00
{
2020-12-02 12:42:26 +00:00
var networkConnection2 = QSBNetworkServer.localConnections[k];
2020-12-02 09:51:53 +00:00
if (networkConnection2 != null)
{
if (networkConnection2.isReady)
{
AddObserver(networkConnection2);
}
}
}
}
}
else
{
foreach (var networkConnection3 in hashSet)
{
if (networkConnection3 != null)
{
if (!networkConnection3.isReady)
{
Debug.LogWarning(string.Concat(new object[]
{
"Observer is not ready for ",
2020-12-14 21:58:00 +00:00
gameObject,
2020-12-02 09:51:53 +00:00
" ",
networkConnection3
}));
}
else if (initialize || !hashSet2.Contains(networkConnection3))
{
2020-12-02 18:40:38 +00:00
networkConnection3.AddToVisList(this);
2020-12-02 09:51:53 +00:00
Debug.Log(string.Concat(new object[]
{
"New Observer for ",
2020-12-14 21:58:00 +00:00
gameObject,
2020-12-02 09:51:53 +00:00
" ",
networkConnection3
}));
flag = true;
}
}
}
foreach (var networkConnection4 in hashSet2)
{
if (!hashSet.Contains(networkConnection4))
{
2020-12-02 18:40:38 +00:00
networkConnection4.RemoveFromVisList(this, true);
2020-12-02 09:51:53 +00:00
Debug.Log(string.Concat(new object[]
{
"Removed Observer for ",
2020-12-14 21:58:00 +00:00
gameObject,
2020-12-02 09:51:53 +00:00
" ",
networkConnection4
}));
flag = true;
}
}
if (initialize)
{
2020-12-02 12:42:26 +00:00
for (var l = 0; l < QSBNetworkServer.localConnections.Count; l++)
2020-12-02 09:51:53 +00:00
{
2020-12-02 12:42:26 +00:00
if (!hashSet.Contains(QSBNetworkServer.localConnections[l]))
2020-12-02 09:51:53 +00:00
{
OnSetLocalVisibility(false);
}
}
}
if (flag)
{
2020-12-02 12:42:26 +00:00
m_Observers = new List<QSBNetworkConnection>(hashSet);
2020-12-02 09:51:53 +00:00
m_ObserverConnections.Clear();
for (var m = 0; m < m_Observers.Count; m++)
{
m_ObserverConnections.Add(m_Observers[m].connectionId);
}
}
}
}
}
2020-12-02 12:42:26 +00:00
public bool RemoveClientAuthority(QSBNetworkConnection conn)
2020-12-02 09:51:53 +00:00
{
if (!IsServer)
{
2020-12-04 22:14:53 +00:00
ModConsole.OwmlConsole.WriteLine($"Warning - Cannot remove authority on client-side. (NetId:{NetId}, Gameobject:{gameObject.name})");
2020-12-04 09:23:27 +00:00
return false;
2020-12-02 09:51:53 +00:00
}
else if (ConnectionToClient != null)
{
Debug.LogError("RemoveClientAuthority cannot remove authority for a player object");
2020-12-04 09:23:27 +00:00
return false;
2020-12-02 09:51:53 +00:00
}
else if (ClientAuthorityOwner == null)
{
2020-12-14 21:58:00 +00:00
Debug.LogError("RemoveClientAuthority for " + gameObject + " has no clientAuthority owner.");
2020-12-04 09:23:27 +00:00
return false;
2020-12-02 09:51:53 +00:00
}
else if (ClientAuthorityOwner != conn)
{
2020-12-14 21:58:00 +00:00
Debug.LogError("RemoveClientAuthority for " + gameObject + " has different owner.");
2020-12-04 09:23:27 +00:00
return false;
2020-12-02 09:51:53 +00:00
}
2020-12-04 09:23:27 +00:00
ClientAuthorityOwner.RemoveOwnedObject(this);
ClientAuthorityOwner = null;
ForceAuthority(true);
conn.Send(15, new QSBClientAuthorityMessage
2020-12-02 09:51:53 +00:00
{
2020-12-04 09:23:27 +00:00
netId = NetId,
authority = false
});
clientAuthorityCallback?.Invoke(conn, this, false);
return true;
2020-12-02 09:51:53 +00:00
}
2020-12-02 12:42:26 +00:00
public bool AssignClientAuthority(QSBNetworkConnection conn)
2020-12-02 09:51:53 +00:00
{
if (!IsServer)
{
2020-12-04 22:14:53 +00:00
ModConsole.OwmlConsole.WriteLine($"Warning - Cannot assign authority on client-side. (NetId:{NetId}, Gameobject:{gameObject.name})");
2020-12-04 09:23:27 +00:00
return false;
2020-12-02 09:51:53 +00:00
}
else if (!LocalPlayerAuthority)
{
2020-12-04 22:14:53 +00:00
ModConsole.OwmlConsole.WriteLine($"Warning - Cannot assign authority on object without LocalPlayerAuthority. (NetId:{NetId}, Gameobject:{gameObject.name})");
2020-12-04 09:23:27 +00:00
return false;
2020-12-02 09:51:53 +00:00
}
else if (ClientAuthorityOwner != null && conn != ClientAuthorityOwner)
{
2020-12-14 21:58:00 +00:00
ModConsole.OwmlConsole.WriteLine("AssignClientAuthority for " + gameObject + " already has an owner. Use RemoveClientAuthority() first.");
2020-12-04 09:23:27 +00:00
return false;
2020-12-02 09:51:53 +00:00
}
else if (conn == null)
{
2020-12-14 21:58:00 +00:00
ModConsole.OwmlConsole.WriteLine("AssignClientAuthority for " + gameObject + " owner cannot be null. Use RemoveClientAuthority() instead.");
2020-12-04 09:23:27 +00:00
return false;
2020-12-02 09:51:53 +00:00
}
2020-12-04 09:23:27 +00:00
ClientAuthorityOwner = conn;
ClientAuthorityOwner.AddOwnedObject(this);
2020-12-02 09:51:53 +00:00
2020-12-04 09:23:27 +00:00
ForceAuthority(false);
conn.Send(15, new QSBClientAuthorityMessage
{
netId = NetId,
authority = true
});
clientAuthorityCallback?.Invoke(conn, this, true);
return true;
2020-12-02 09:51:53 +00:00
}
internal void MarkForReset() => m_Reset = true;
internal void Reset()
{
if (m_Reset)
{
m_Reset = false;
m_IsServer = false;
IsClient = false;
HasAuthority = false;
2020-12-15 09:56:03 +00:00
NetId = (QSBNetworkInstanceId)typeof(QSBNetworkInstanceId).GetField("Zero", System.Reflection.BindingFlags.Static).GetValue(null);
2020-12-02 09:51:53 +00:00
IsLocalPlayer = false;
ConnectionToServer = null;
ConnectionToClient = null;
PlayerControllerId = -1;
m_NetworkBehaviours = null;
ClearObservers();
ClientAuthorityOwner = null;
}
}
2020-12-04 22:14:53 +00:00
public static void UNetStaticUpdate()
2020-12-02 09:51:53 +00:00
{
2020-12-02 18:40:38 +00:00
QSBNetworkServer.Update();
QSBNetworkClient.UpdateClients();
QSBNetworkManagerUNET.UpdateScene();
2020-12-02 09:51:53 +00:00
}
[SerializeField]
2020-12-15 09:56:03 +00:00
private QSBNetworkSceneId m_SceneId;
2020-12-02 09:51:53 +00:00
[SerializeField]
2020-12-15 09:56:03 +00:00
private QSBNetworkHash128 m_AssetId;
2020-12-02 09:51:53 +00:00
[SerializeField]
private bool m_ServerOnly;
[SerializeField]
private bool m_LocalPlayerAuthority;
2020-12-03 08:28:05 +00:00
2020-12-02 09:51:53 +00:00
private bool m_IsServer;
2020-12-12 18:14:04 +00:00
2020-12-02 09:51:53 +00:00
private QSBNetworkBehaviour[] m_NetworkBehaviours;
private HashSet<int> m_ObserverConnections;
2020-12-02 12:42:26 +00:00
private List<QSBNetworkConnection> m_Observers;
2020-12-12 18:14:04 +00:00
2020-12-02 09:51:53 +00:00
private bool m_Reset = false;
private static uint s_NextNetworkId = 1U;
2020-12-04 09:23:27 +00:00
private static readonly QSBNetworkWriter s_UpdateWriter = new QSBNetworkWriter();
2020-12-02 09:51:53 +00:00
public static ClientAuthorityCallback clientAuthorityCallback;
2020-12-02 12:42:26 +00:00
public delegate void ClientAuthorityCallback(QSBNetworkConnection conn, QSBNetworkIdentity uv, bool authorityState);
2020-12-02 09:51:53 +00:00
}
}