Liberi
An exergame built for kids with CP!
Sync.Static.Internal.cs
1 using UnityEngine;
2 using System;
3 using System.Collections;
4 using System.Collections.Generic;
5 using System.Linq;
6 using System.Diagnostics;
7 using System.Reflection;
8 using Janus;
9 
10 public delegate void PropagateSpawnUpHandler (int requestIndex, SpawnInfo spawnInfo);
11 public delegate void PropagateSpawnDownHandler (int requesterIndex, int requestIndex, int objectIndex, SpawnInfo spawnInfo);
12 public delegate void PropagateDespawnUpHandler (int objectIndex);
13 public delegate void PropagateDespawnDownHandler (int requesterIndex, int objectIndex);
14 
15 public partial class Sync : MonoBehaviour
16 {
17  public static event PropagateSpawnUpHandler PropagateSpawnUp;
18  public static event PropagateSpawnDownHandler PropagateSpawnDown;
19  public static event PropagateDespawnUpHandler PropagateDespawnUp;
20  public static event PropagateDespawnDownHandler PropagateDespawnDown;
21 
22  static int _localPeerIndex = -1;
23  static bool _isClient;
24  static bool _isServer;
25  static bool _isMapSynced;
26  static bool _isMapUnsyncing;
27  static TimelineManager _timelineManager;
28  static TimelineSynchronizer _timelineSyncer;
29  static Dictionary<string, GameObject> _spawnablePrefabs;
30  static Dictionary<string, UJeli> _spawnDetailsTemplates;
31  static Dictionary<int, Sync> _syncedObjects;
32  static Dictionary<int, Sync> _syncingObjects;
33  static Dictionary<int, Sync> _requestedObjects;
34  static Dictionary<int, Sync> _syncingMapObjects;
35  static int _nextSpawnRequestIndex;
36  static int _nextObjectIndex;
37  static int _timelineObjectIndex;
38  static byte _timelineScriptIndex;
39  static byte _nextTimelineIndex;
40 
41  internal static void Initialize ()
42  {
43  _spawnablePrefabs = new Dictionary<string, GameObject>();
44  _spawnDetailsTemplates = new Dictionary<string, UJeli>();
45 
46  foreach (var prefab in Resources.LoadAll<GameObject>(""))
47  {
48  _spawnablePrefabs.Add(prefab.name, prefab);
49  }
50 
51  _syncedObjects = new Dictionary<int, Sync>();
52  _syncingObjects = new Dictionary<int, Sync>();
53  _requestedObjects = new Dictionary<int, Sync>();
54  _syncingMapObjects = new Dictionary<int, Sync>();
55  }
56 
57  internal static void SetSpawnDetailsTemplate (string prefabId, UJeli spawnDetailsTemplate)
58  {
59  _spawnDetailsTemplates[prefabId] = spawnDetailsTemplate;
60  }
61 
62  internal static void SyncServer ()
63  {
64  _isServer = true;
65  _localPeerIndex = 0;
66 
67  _timelineManager = new TimelineManager();
68  _timelineManager.ClockSyncCorrectionFactor = 50f;
69  _timelineManager.MinClockSyncCorrectionRate = 50f;
70 
71  _timelineSyncer = new TimelineSynchronizer();
72  _timelineSyncer.ClockSyncSamplingRule = ClockSyncSamplingRule.MostRecent;
73  _timelineSyncer.ConnectPeer(0, 0);
74 
75  AddMapObjects();
76 
77  _isMapSynced = true;
78  }
79 
80  internal static void BeginSyncClient (int peerIndex)
81  {
82  _isClient = true;
83  _localPeerIndex = peerIndex;
84 
85  _timelineManager = new TimelineManager();
86  _timelineManager.ClockSyncCorrectionFactor = 10f;
87  _timelineManager.MinClockSyncCorrectionRate = 10f;
88 
89  _nextSpawnRequestIndex = 0;
90 
91  AddMapObjects();
92  }
93 
94  internal static void FinishSyncClient ()
95  {
96  foreach (var sync in _syncingObjects.Values.ToArray())
97  {
98  if (sync._spawnInfo == null && !_syncingMapObjects.ContainsValue(sync))
99  {
100  _syncingObjects.Remove(sync._objectIndex);
101  Destroy(sync.gameObject);
102  }
103  }
104 
105  _syncingMapObjects.Clear();
106 
107  _isMapSynced = true;
108  }
109 
110  private static void AddMapObjects ()
111  {
112  var rootSyncsInScene = new List<Sync>(FindObjectsOfType<Sync>().Where(sync => sync.IsRoot));
113  rootSyncsInScene.Sort((a, b) => a.name.CompareTo(b.name));
114 
115  _nextObjectIndex = -1;
116 
117  foreach (var sync in rootSyncsInScene)
118  {
119  SetObjectOwner(sync, 0);
120  }
121 
122  foreach (var sync in rootSyncsInScene)
123  {
124  _nextObjectIndex++;
125  SyncObject(sync, _nextObjectIndex, false);
126  }
127  }
128 
129  private static void RemoveMapObjects ()
130  {
131  var rootSyncsInScene = new List<Sync>(FindObjectsOfType<Sync>().Where(sync => sync.IsRoot));
132 
133  foreach (var sync in rootSyncsInScene)
134  {
135  if (sync._spawnInfo != null)
136  continue;
137 
138  UnsyncObject(sync);
139  }
140  }
141 
142  internal static void RegisterClient (int peerIndex, float rtt)
143  {
144  _timelineSyncer.ConnectPeer((ushort)peerIndex, rtt);
145  }
146 
147  internal static void UnregisterClient (int peerIndex)
148  {
149  DespawnPeerObjects(peerIndex);
150  _timelineSyncer.DisconnectPeer((ushort)peerIndex);
151  }
152 
153  internal static void Step ()
154  {
155  if (_isServer)
156  {
157  foreach (var timelineMsg in _timelineManager.GetOutgoingMessages())
158  {
159  _timelineSyncer.ProcessIncomingMessage(0, timelineMsg);
160  }
161 
162  foreach (var timelineMsg in _timelineSyncer.GetOutgoingMessages(0))
163  {
164  _timelineManager.ProcessIncomingMessage(timelineMsg);
165  }
166 
167  if (_timelineSyncer != null)
168  _timelineSyncer.Step(Time.deltaTime);
169  }
170 
171  _timelineManager.Step(Time.deltaTime);
172 
173  foreach (var sync in _syncingObjects.Values.ToArray())
174  {
175  bool synced = true;
176 
177  foreach (var essentialTimeline in sync._essentialTimelines)
178  {
179  if (essentialTimeline.NumEntries == 0)
180  synced = false;
181  }
182 
183  if (synced)
184  FinalizeObject(sync, true);
185  }
186  }
187 
188  internal static void UnsyncClient ()
189  {
190  _isMapUnsyncing = true;
191 
192  foreach (var sync in _syncedObjects.Values.ToArray())
193  {
194  if (sync == null)
195  continue;
196 
197  if (sync._spawnInfo != null)
198  {
199  UnsyncObject(sync);
200  Destroy(sync.gameObject);
201  }
202  }
203 
204  RemoveMapObjects();
205 
206  _syncingObjects.Clear();
207  _syncingMapObjects.Clear();
208  _syncedObjects.Clear();
209  _requestedObjects.Clear();
210 
211  _localPeerIndex = -1;
212  _timelineManager = null;
213  _isMapSynced = false;
214  _isMapUnsyncing = false;
215  }
216 
217  internal static void ProcessPropagatedUpSpawn (int requesterIndex, int requestIndex, SpawnInfo spawnInfo)
218  {
219  var sync = SpawnFromInfo(spawnInfo);
220 
221  _nextObjectIndex++;
222  SyncObject(sync, _nextObjectIndex, true);
223 
224  if (PropagateSpawnDown != null)
225  PropagateSpawnDown(requesterIndex, requestIndex, sync._objectIndex, spawnInfo);
226  }
227 
228  internal static void ProcessPropagatedDownSpawn (SpawnInfo spawnInfo, int objectIndex, bool isNew)
229  {
230  Sync sync = null;
231 
232  if (spawnInfo != null)
233  {
234  sync = SpawnFromInfo(spawnInfo);
235  SyncObject(sync, objectIndex, isNew);
236  }
237  else
238  {
239  _syncingObjects.TryGetValue(objectIndex, out sync);
240 
241  if (sync != null)
242  _syncingMapObjects.Add(objectIndex, sync);
243  }
244  }
245 
246  internal static void ProcessSpawnRequestResponse (int requestIndex, int objectIndex)
247  {
248  Sync sync = null;
249 
250  if (!_requestedObjects.TryGetValue(requestIndex, out sync))
251  {
252  // This means that our spawn request was fulfilled, but we have already decided
253  // to despawn the object before it was fully propagated to the rest of our peers.
254  // Now it is our responsibility to tell the other peers that we have wasted their
255  // time and that we're very sorry.
256  if (PropagateDespawnUp != null)
257  PropagateDespawnUp(objectIndex);
258 
259  return;
260  }
261 
262  SyncObject(sync, objectIndex, true);
263  }
264 
265  internal static void ProcessPropagatedUpDespawn (int requesterIndex, int objectIndex)
266  {
267  Sync sync = null;
268 
269  if (!_syncedObjects.TryGetValue(objectIndex, out sync))
270  return;
271 
272  if (PropagateDespawnDown != null)
273  PropagateDespawnDown(requesterIndex, objectIndex);
274 
275  NotifyDespawn(sync.gameObject);
276  UnsyncObject(sync);
277  Destroy(sync.gameObject);
278  }
279 
280  internal static void ProcessPropagatedDownDespawn (int objectIndex)
281  {
282  Sync sync = null;
283 
284  if (!_syncedObjects.TryGetValue(objectIndex, out sync))
285  {
286  if (!_syncingObjects.TryGetValue(objectIndex, out sync))
287  return;
288  }
289 
290  NotifyDespawn(sync.gameObject);
291  UnsyncObject(sync);
292  Destroy(sync.gameObject);
293  }
294 
295  private static GameObject SpawnShell (GameObject prefab, Vector3 position)
296  {
297  var go = (GameObject)Instantiate(prefab, position, prefab.transform.rotation);
298  go.name = prefab.name;
299  return go;
300  }
301 
302  private static Sync SpawnFromInfo (SpawnInfo spawnInfo)
303  {
304  var go = SpawnShell(spawnInfo.Prefab, spawnInfo.Position);
305  var sync = go.GetComponent<Sync>();
306 
307  if (spawnInfo.Direction != Vector3.zero)
308  go.transform.SetDirection(spawnInfo.Direction);
309 
310  SetObjectOwner(sync, spawnInfo.OwnerPeerIndex);
311  sync._spawnInfo = spawnInfo;
312 
313  return sync;
314  }
315 
316  private static bool ShouldScriptExecute (Sync sync, ScriptRole scriptRole)
317  {
318  if (sync == null)
319  return true;
320 
321  var isOwner = sync._ownerPeerIndex == _localPeerIndex;
322 
323  if (scriptRole == ScriptRole.Logic
324  || scriptRole == ScriptRole.Owner)
325  {
326  return isOwner;
327  }
328  else if (scriptRole == ScriptRole.View
329  || scriptRole == ScriptRole.Client)
330  {
331  return _isClient;
332  }
333  else if (scriptRole == ScriptRole.Secure
334  || scriptRole == ScriptRole.Server)
335  {
336  return _isServer;
337  }
338  else if (scriptRole == ScriptRole.HumanBrain
339  || scriptRole == ScriptRole.ClientOwner
340  || scriptRole == ScriptRole.OwnerView)
341  {
342  return isOwner && _isClient;
343  }
344  else if (scriptRole == ScriptRole.BotBrain
345  || scriptRole == ScriptRole.ServerOwner)
346  {
347  return isOwner && _isServer;
348  }
349 
350  return false;
351  }
352 
353  private static void SetObjectOwner (Sync sync, int ownerPeerIndex)
354  {
355  foreach (var childSync in sync.GetComponentsInChildren<Sync>())
356  {
357  childSync._ownerPeerIndex = ownerPeerIndex;
358  }
359 
360  foreach (var mb in sync._originalBehaviours)
361  {
362  var mbType = mb.GetType();
363 
364  var scriptAttrs = mbType.GetCustomAttributes(typeof(ScriptAttribute), false);
365 
366  if (scriptAttrs.Length != 0)
367  {
368  var scriptAttr = (ScriptAttribute)scriptAttrs[0];
369  mb.enabled = ShouldScriptExecute(sync, scriptAttr.ScriptRole);
370  }
371  else mb.enabled = true;
372  }
373  }
374 
375  private static void SyncObject (Sync sync, int objectIndex, bool isNew)
376  {
377  if (_timelineManager == null)
378  return;
379 
380  foreach (var childSync in sync.GetComponentsInChildren<Sync>())
381  {
382  childSync._objectIndex = objectIndex;
383  }
384 
385  if (Sync.GetObject(objectIndex) != null)
386  {
387  print (Sync.GetObject(objectIndex).name);
388  }
389 
390  _syncingObjects.Add(objectIndex, sync);
391 
392  _timelineObjectIndex = objectIndex;
393  _timelineScriptIndex = 0;
394 
395  foreach (var mb in sync._originalBehaviours)
396  {
397  _nextTimelineIndex = 0;
398 
399  sync._timelines.AddRange(_timelineManager.AddTimelines(mb, GenerateTimelineID));
400 
401  sync._essentialTimelines.AddRange(
402  from timeline in sync._timelines
403  where timeline.Tags.Contains("Essential")
404  select timeline);
405 
406  _timelineScriptIndex++;
407  }
408 
409  if (sync._ownerPeerIndex == _localPeerIndex)
410  FinalizeObject(sync, isNew);
411  else sync.gameObject.SetActive(false);
412  }
413 
414  private static byte[] GenerateTimelineID (string s)
415  {
416  byte[] id = BuildTimelineID(_timelineObjectIndex, _timelineScriptIndex, _nextTimelineIndex);
417  _nextTimelineIndex++;
418  return id;
419  }
420 
421  private static byte[] BuildTimelineID (int objectIndex, byte timelineScriptIndex, byte timelineIndex)
422  {
423  byte[] id = new byte[6];
424 
425  Buffer.BlockCopy(BitConverter.GetBytes(objectIndex), 0, id, 0, 4);
426  id[4] = timelineScriptIndex;
427  id[5] = timelineIndex;
428 
429  return id;
430  }
431 
432  private static void FinalizeObject (Sync sync, bool isNew)
433  {
434  _syncingObjects.Remove(sync._objectIndex);
435 
436  try { _syncedObjects.Add(sync._objectIndex, sync); }
437 
438  catch (Exception e)
439  {
440  string errorMsg = e.Message;
441  errorMsg += "\nAn error occurred while trying to Finalize Object: " + sync.name + ", Object Index: " + sync._objectIndex + ".";
442  UnityEngine.Debug.LogError(errorMsg);
443 
444  if (_isClient)
446  }
447 
448  if (!sync.gameObject.activeSelf)
449  sync.gameObject.SetActive(true);
450 
451  if (sync._spawnInfo != null)
452  {
453  PreAssignValues(sync.gameObject, sync._spawnInfo.Details);
454  NotifySpawn(sync.gameObject, sync._spawnInfo.Details, isNew);
455 
456  if (isNew
457  && sync._ownerPeerIndex == _localPeerIndex
458  && sync.rigidbody != null
459  && !sync.rigidbody.isKinematic
460  && sync._spawnInfo.Details.HasChild("Velocity"))
461  {
462  sync.rigidbody.velocity = sync._spawnInfo.Details["Velocity"].Vector3Value;
463  }
464  }
465  else
466  {
467  PreAssignValues(sync.gameObject, new UJeli());
468  NotifySpawn(sync.gameObject, new UJeli(), isNew);
469  }
470  }
471 
472  private static void PreAssignValues (GameObject go, UJeli details)
473  {
474  foreach (var mb in go.GetComponentsInChildren<MonoBehaviour>())
475  {
476  if (!mb.enabled)
477  continue;
478 
479  var mbType = mb.GetType();
480 
481  foreach (var field in mbType.GetFields(
482  BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance))
483  {
484  var spawnDetailAttrs = field.GetCustomAttributes(typeof(SpawnDetailAttribute), false);
485 
486  if (spawnDetailAttrs.Length != 0)
487  {
488  var spawnDetailAttr = spawnDetailAttrs[0] as SpawnDetailAttribute;
489  string detailKey = spawnDetailAttr.DetailKey ?? field.Name;
490 
491  if (details.HasChild(detailKey))
492  {
493  var detail = details[detailKey];
494  SetFieldValueFromJeli(field, mb, detail);
495  }
496  }
497 
498  var mapSettingAttrs = field.GetCustomAttributes(typeof(MapSettingAttribute), false);
499 
500  if (mapSettingAttrs.Length != 0)
501  {
502  var mapSettingAttr = mapSettingAttrs[0] as MapSettingAttribute;
503  string settingKey = mapSettingAttr.SettingKey ?? field.Name;
504 
505  if (Game.MapSettings.HasChild(settingKey))
506  {
507  var setting = Game.MapSettings[settingKey];
508  SetFieldValueFromJeli(field, mb, setting);
509  }
510  }
511  }
512  }
513  }
514 
515  private static void SetFieldValueFromJeli (FieldInfo field, object obj, UJeli jeli)
516  {
517  var fieldType = field.FieldType;
518 
519  try
520  {
521  if (fieldType == typeof(int))
522  field.SetValue(obj, jeli.IntValue);
523  else if (fieldType == typeof(uint))
524  field.SetValue(obj, jeli.UIntValue);
525  else if (fieldType == typeof(long))
526  field.SetValue(obj, jeli.LongValue);
527  else if (fieldType == typeof(ulong))
528  field.SetValue(obj, jeli.ULongValue);
529  else if (fieldType == typeof(short))
530  field.SetValue(obj, jeli.ShortValue);
531  else if (fieldType == typeof(ushort))
532  field.SetValue(obj, jeli.UShortValue);
533  else if (fieldType == typeof(float))
534  field.SetValue(obj, jeli.FloatValue);
535  else if (fieldType == typeof(double))
536  field.SetValue(obj, jeli.DoubleValue);
537  else if (fieldType == typeof(byte))
538  field.SetValue(obj, jeli.ByteValue);
539  else if (fieldType == typeof(sbyte))
540  field.SetValue(obj, jeli.SByteValue);
541  else if (fieldType == typeof(bool))
542  field.SetValue(obj, jeli.BoolValue);
543  else if (fieldType == typeof(string))
544  field.SetValue(obj, jeli.Value);
545  else if (fieldType == typeof(char))
546  field.SetValue(obj, jeli.CharValue);
547  else if (fieldType.IsEnum)
548  field.SetValue(obj, Enum.Parse(fieldType, jeli.Value));
549  else if (fieldType == typeof(DateTime))
550  field.SetValue(obj, jeli.TimeValue);
551  else if (fieldType == typeof(Vector2))
552  field.SetValue(obj, jeli.Vector2Value);
553  else if (fieldType == typeof(Vector3))
554  field.SetValue(obj, jeli.Vector3Value);
555  else if (fieldType == typeof(Vector4))
556  field.SetValue(obj, jeli.Vector4Value);
557  else if (fieldType == typeof(Quaternion))
558  field.SetValue(obj, jeli.QuaternionValue);
559  else if (fieldType == typeof(Matrix4x4))
560  field.SetValue(obj, jeli.MatrixValue);
561  else if (fieldType == typeof(Ray))
562  field.SetValue(obj, jeli.RayValue);
563  else if (fieldType == typeof(Color))
564  field.SetValue(obj, jeli.ColorValue);
565  else if (fieldType == typeof(GameObject))
566  field.SetValue(obj, jeli.GameObjectValue);
567  }
568  catch { }
569  }
570 
571  private static void NotifySpawn (GameObject go, UJeli details, bool isNew)
572  {
573  SendEnabledMessage(go, "OnSpawn", details, isNew);
574  }
575 
576  private static void NotifyDespawn (GameObject go)
577  {
578  SendEnabledMessage(go, "PreDespawn");
579  SendEnabledMessage(go, "OnDespawn");
580  SendEnabledMessage(go, "PostDespawn");
581  }
582 
583  private static void UnsyncObject (Sync sync)
584  {
585  if (!_isMapSynced)
586  return;
587 
588  if (_timelineManager == null)
589  return;
590 
591  foreach (var timeline in sync._timelines)
592  {
593  _timelineManager.Remove(timeline);
594  }
595 
596  sync._timelines.Clear();
597  sync._essentialTimelines.Clear();
598 
599  _syncedObjects.Remove(sync._objectIndex);
600  _syncingObjects.Remove(sync._objectIndex);
601 
602  foreach (var childSync in sync.GetComponentsInChildren<Sync>())
603  {
604  childSync._objectIndex = -1;
605  }
606  }
607 }
static void DespawnPeerObjects(int peerIndex)
Despawn all objects owned by the peer with the given peer index.
Definition: Sync.Static.cs:854
GameObject Prefab
The prefab from which this object is instantiated.
Definition: SpawnInfo.cs:18
Attribute to use on fields. Marks them for assignment from map settings.
string DetailKey
Gets the detail key used to find the value for the target field.
Marks a MonoBehaviour as a game script with a given script role.
static void SendEnabledMessage(GameObject go, string message, params object[] args)
Invokes a method on all enabled scripts in the GameObject.
Definition: Sync.Static.cs:740
Vector3 Position
The position at which this object was spawned.
Definition: SpawnInfo.cs:22
Attribute to use on fields. Marks them for assignment from spawn details.
static TimelineSynchronizer TimelineSynchronizer
Gets the Janus timeline synchronizer if this is the server.
Definition: Sync.Static.cs:29
void DisconnectFromServer()
Disconnect from the current server.
Definition: GameClient.cs:690
static GameClient Instance
Gets the sole instance of the game client.
Definition: GameClient.cs:136
string SettingKey
Gets the setting key used to find the value for the target field.
Main game client class.
Definition: GameClient.cs:16
int OwnerPeerIndex
The peer index of the peer which owns this object.
Definition: SpawnInfo.cs:30
Vector3 Direction
The direction this object was facing when it was spawned.
Definition: SpawnInfo.cs:26
static GameObject GetObject(int objectIndex)
Get an object by its object index.
Definition: Sync.Static.cs:143
Unity version of Jeli markup class.
Definition: UJeli.cs:10
static TimelineManager TimelineManager
Gets the Janus timeline manager on this peer.
Definition: Sync.Static.cs:21
This class server two main functions: 1) As a MonoBehaviour, it allows for network synchronization of...
Definition: Sync.cs:13
The spawn-time information of a spawned object.
Definition: SpawnInfo.cs:13
UJeli Details
Additional spawn details.
Definition: SpawnInfo.cs:34