Serialization.cs 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Diagnostics.CodeAnalysis;
  4. using System.IO;
  5. using System.Linq;
  6. using System.Net.WebSockets;
  7. using System.Reflection;
  8. using System.Runtime.InteropServices.ComTypes;
  9. using System.Threading;
  10. using System.Xml.Linq;
  11. using InABox.Clients;
  12. using Newtonsoft.Json;
  13. using Newtonsoft.Json.Linq;
  14. namespace InABox.Core
  15. {
  16. public enum SerializationFormat
  17. {
  18. Json,
  19. Binary
  20. }
  21. public class SerialisationException : Exception
  22. {
  23. public SerialisationException(string message): base(message) { }
  24. }
  25. public interface ISerializeBinary
  26. {
  27. public void SerializeBinary(CoreBinaryWriter writer);
  28. public void DeserializeBinary(CoreBinaryReader reader);
  29. }
  30. public static class Serialization
  31. {
  32. private static JsonSerializerSettings? _serializerSettings;
  33. private static JsonSerializerSettings SerializerSettings(bool indented = true)
  34. {
  35. if (_serializerSettings == null)
  36. {
  37. _serializerSettings = new JsonSerializerSettings
  38. {
  39. DateParseHandling = DateParseHandling.DateTime,
  40. DateFormatHandling = DateFormatHandling.IsoDateFormat,
  41. DateTimeZoneHandling = DateTimeZoneHandling.RoundtripKind
  42. };
  43. _serializerSettings.Converters.Add(new CoreTableJsonConverter());
  44. //serializerSettings.Converters.Add(new DateTimeJsonConverter());
  45. _serializerSettings.Converters.Add(new FilterJsonConverter());
  46. _serializerSettings.Converters.Add(new ColumnJsonConverter());
  47. _serializerSettings.Converters.Add(new SortOrderJsonConverter());
  48. _serializerSettings.Converters.Add(new UserPropertiesJsonConverter());
  49. //_serializerSettings.Converters.Add(new BaseObjectJSONConverter());
  50. _serializerSettings.ConstructorHandling = ConstructorHandling.AllowNonPublicDefaultConstructor;
  51. }
  52. _serializerSettings.Formatting = indented ? Formatting.Indented : Formatting.None;
  53. return _serializerSettings;
  54. }
  55. public static string Serialize(object? o, bool indented = false)
  56. {
  57. var json = JsonConvert.SerializeObject(o, SerializerSettings(indented));
  58. return json;
  59. }
  60. public static void Serialize(object o, Stream stream, bool indented = false)
  61. {
  62. var settings = SerializerSettings(indented);
  63. using (var sw = new StreamWriter(stream))
  64. {
  65. using (JsonWriter writer = new JsonTextWriter(sw))
  66. {
  67. var serializer = JsonSerializer.Create(settings);
  68. serializer.Serialize(writer, o);
  69. }
  70. }
  71. }
  72. public static void DeserializeInto(string json, object target)
  73. {
  74. JsonConvert.PopulateObject(json, target, SerializerSettings());
  75. }
  76. [return: MaybeNull]
  77. public static T Deserialize<T>(Stream? stream, bool strict = false)
  78. {
  79. if (stream == null)
  80. return default;
  81. try
  82. {
  83. var settings = SerializerSettings();
  84. using var sr = new StreamReader(stream);
  85. using JsonReader reader = new JsonTextReader(sr);
  86. var serializer = JsonSerializer.Create(settings);
  87. return serializer.Deserialize<T>(reader);
  88. }
  89. catch (Exception e)
  90. {
  91. if (strict)
  92. throw;
  93. Logger.Send(LogType.Error, ClientFactory.UserID, $"Error in Deserialize<{typeof(T)}>(): {e.Message}");
  94. return default;
  95. }
  96. }
  97. public static object? Deserialize(Type type, Stream? stream)
  98. {
  99. if (stream == null)
  100. return null;
  101. object? result = null;
  102. var settings = SerializerSettings();
  103. using (var sr = new StreamReader(stream))
  104. {
  105. using (JsonReader reader = new JsonTextReader(sr))
  106. {
  107. var serializer = JsonSerializer.Create(settings);
  108. result = serializer.Deserialize(reader, type);
  109. }
  110. }
  111. return result;
  112. }
  113. [return: MaybeNull]
  114. public static T Deserialize<T>(JToken obj, bool strict = false)
  115. {
  116. var ret = default(T);
  117. try
  118. {
  119. var settings = SerializerSettings();
  120. var serializer = JsonSerializer.Create(settings);
  121. return obj.ToObject<T>();
  122. }
  123. catch (Exception)
  124. {
  125. if (strict)
  126. {
  127. throw;
  128. }
  129. if (typeof(T).IsArray)
  130. {
  131. ret = (T)(object)Array.CreateInstance(typeof(T).GetElementType(), 0);
  132. }
  133. else
  134. {
  135. ret = Activator.CreateInstance<T>();
  136. }
  137. }
  138. return ret;
  139. }
  140. [return: MaybeNull]
  141. public static T Deserialize<T>(string? json, bool strict = false) // where T : new()
  142. {
  143. var ret = default(T);
  144. if (string.IsNullOrWhiteSpace(json))
  145. return ret;
  146. try
  147. {
  148. var settings = SerializerSettings();
  149. //if (typeof(T).IsSubclassOf(typeof(BaseObject)))
  150. //{
  151. // ret = Activator.CreateInstance<T>();
  152. // (ret as BaseObject).SetObserving(false);
  153. // JsonConvert.PopulateObject(json, ret, settings);
  154. // (ret as BaseObject).SetObserving(true);
  155. //}
  156. //else
  157. if (typeof(T).IsArray)
  158. {
  159. ret = JsonConvert.DeserializeObject<T>(json, settings);
  160. //object o = Array.CreateInstance(typeof(T).GetElementType(), 0);
  161. //ret = (T)o;
  162. }
  163. else
  164. {
  165. ret = JsonConvert.DeserializeObject<T>(json, settings);
  166. }
  167. }
  168. catch (Exception e)
  169. {
  170. if (strict)
  171. {
  172. throw;
  173. }
  174. if (typeof(T).IsArray)
  175. {
  176. ret = (T)(object)Array.CreateInstance(typeof(T).GetElementType(), 0);
  177. }
  178. else
  179. {
  180. ret = (T)Activator.CreateInstance(typeof(T), true);
  181. }
  182. }
  183. return ret;
  184. }
  185. public static object? Deserialize(Type T, string json) // where T : new()
  186. {
  187. var ret = T.GetDefault();
  188. if (string.IsNullOrWhiteSpace(json))
  189. return ret;
  190. try
  191. {
  192. var settings = SerializerSettings();
  193. //if (typeof(T).IsSubclassOf(typeof(BaseObject)))
  194. //{
  195. // ret = Activator.CreateInstance<T>();
  196. // (ret as BaseObject).SetObserving(false);
  197. // JsonConvert.PopulateObject(json, ret, settings);
  198. // (ret as BaseObject).SetObserving(true);
  199. //}
  200. //else
  201. if (T.IsArray)
  202. {
  203. object o = Array.CreateInstance(T.GetElementType(), 0);
  204. ret = o;
  205. }
  206. else
  207. {
  208. ret = JsonConvert.DeserializeObject(json, T, settings);
  209. }
  210. }
  211. catch (Exception)
  212. {
  213. ret = Activator.CreateInstance(T, true);
  214. }
  215. return ret;
  216. }
  217. #region Binary Serialization
  218. public static byte[] WriteBinary(this ISerializeBinary obj, BinarySerializationSettings settings)
  219. {
  220. using var stream = new MemoryStream();
  221. obj.SerializeBinary(new CoreBinaryWriter(stream, settings));
  222. return stream.ToArray();
  223. }
  224. public static void WriteBinary(this ISerializeBinary obj, Stream stream, BinarySerializationSettings settings)
  225. {
  226. obj.SerializeBinary(new CoreBinaryWriter(stream, settings));
  227. }
  228. public static T ReadBinary<T>(byte[] data, BinarySerializationSettings settings)
  229. where T : ISerializeBinary, new() => (T)ReadBinary(typeof(T), data, settings);
  230. public static T ReadBinary<T>(Stream stream, BinarySerializationSettings settings)
  231. where T : ISerializeBinary, new() => (T)ReadBinary(typeof(T), stream, settings);
  232. public static object ReadBinary(Type T, byte[] data, BinarySerializationSettings settings)
  233. {
  234. using var stream = new MemoryStream(data);
  235. return ReadBinary(T, stream, settings);
  236. }
  237. public static object ReadBinary(Type T, Stream stream, BinarySerializationSettings settings)
  238. {
  239. var obj = (Activator.CreateInstance(T) as ISerializeBinary)!;
  240. obj.DeserializeBinary(new CoreBinaryReader(stream, settings));
  241. return obj;
  242. }
  243. #endregion
  244. }
  245. public class CoreBinaryReader : BinaryReader
  246. {
  247. public BinarySerializationSettings Settings { get; set; }
  248. public CoreBinaryReader(Stream stream, BinarySerializationSettings settings) : base(stream)
  249. {
  250. Settings = settings;
  251. }
  252. }
  253. public class CoreBinaryWriter : BinaryWriter
  254. {
  255. public BinarySerializationSettings Settings { get; set; }
  256. public CoreBinaryWriter(Stream stream, BinarySerializationSettings settings) : base(stream)
  257. {
  258. Settings = settings;
  259. }
  260. }
  261. /// <summary>
  262. /// A class to maintain the consistency of serialisation formats across versions.
  263. /// The design of this is such that specific versions of serialisation have different parameters set,
  264. /// and the versions are maintained as static properties. Please keep the constructor private.
  265. /// </summary>
  266. /// <remarks>
  267. /// Note that <see cref="Latest"/> should always be updated to point to the latest version.
  268. /// <br/>
  269. /// Note also that all versions should have an entry in the <see cref="ConvertVersionString(string)"/> function.
  270. /// <br/>
  271. /// Also, if you create a new format, it would probably be a good idea to add a database update script to get all
  272. /// <see cref="IPackable"/> and <see cref="ISerializeBinary"/> properties and update the version of the format.
  273. /// (Otherwise, we'd basically be nullifying all data that is currently binary serialised.)
  274. /// </remarks>
  275. public class BinarySerializationSettings
  276. {
  277. /// <summary>
  278. /// Should the Info() call return RPC and Rest Ports? This is
  279. /// To workaround a bug in RPCsockets that crash on large uploads
  280. /// </summary>
  281. /// <remarks>
  282. /// True in all serialization versions >= 1.2
  283. /// </remarks>
  284. public bool RPCClientWorkaround { get; set; }
  285. /// <summary>
  286. /// Should reference types include a flag for nullability? (Adds an extra boolean field for whether the value is null or not).
  287. /// </summary>
  288. /// <remarks>
  289. /// True in all serialisation versions >= 1.1.
  290. /// </remarks>
  291. public bool IncludeNullables { get; set; }
  292. public string Version { get; set; }
  293. public static BinarySerializationSettings Latest => V1_2;
  294. public static BinarySerializationSettings V1_0 = new BinarySerializationSettings("1.0")
  295. {
  296. IncludeNullables = false,
  297. RPCClientWorkaround = false
  298. };
  299. public static BinarySerializationSettings V1_1 = new BinarySerializationSettings("1.1")
  300. {
  301. IncludeNullables = true,
  302. RPCClientWorkaround = false
  303. };
  304. public static BinarySerializationSettings V1_2 = new BinarySerializationSettings("1.2")
  305. {
  306. IncludeNullables = true,
  307. RPCClientWorkaround = true
  308. };
  309. public static BinarySerializationSettings ConvertVersionString(string version) => version switch
  310. {
  311. "1.0" => V1_0,
  312. "1.1" => V1_1,
  313. "1.2" => V1_2,
  314. _ => V1_0
  315. };
  316. private BinarySerializationSettings(string version)
  317. {
  318. Version = version;
  319. }
  320. }
  321. public static class SerializationUtils
  322. {
  323. public static void Write(this BinaryWriter writer, Guid guid)
  324. {
  325. writer.Write(guid.ToByteArray());
  326. }
  327. public static Guid ReadGuid(this BinaryReader reader)
  328. {
  329. return new Guid(reader.ReadBytes(16));
  330. }
  331. public static void Write(this BinaryWriter writer, DateTime dateTime)
  332. {
  333. writer.Write(dateTime.Ticks);
  334. }
  335. public static DateTime ReadDateTime(this BinaryReader reader)
  336. {
  337. return new DateTime(reader.ReadInt64());
  338. }
  339. private static bool MatchType<T1>(Type t) => typeof(T1) == t;
  340. private static bool MatchType<T1,T2>(Type t) => (typeof(T1) == t) || (typeof(T2) == t);
  341. /// <summary>
  342. /// Binary serialize a bunch of different types of values. <see cref="WriteBinaryValue(CoreBinaryWriter, Type, object?)"/> and
  343. /// <see cref="ReadBinaryValue(CoreBinaryReader, Type)"/> are inverses of each other.
  344. /// </summary>
  345. /// <remarks>
  346. /// Handles <see cref="byte"/>[], <see cref="Array"/>s of serialisable values, <see cref="Enum"/>, <see cref="bool"/>, <see cref="string"/>,
  347. /// <see cref="Guid"/>, <see cref="byte"/>, <see cref="Int16"/>, <see cref="Int32"/>, <see cref="Int64"/>, <see cref="float"/>, <see cref="double"/>,
  348. /// <see cref="DateTime"/>, <see cref="TimeSpan"/>, <see cref="LoggablePropertyAttribute"/>, <see cref="IPackable"/>, <see cref="Nullable{T}"/>
  349. /// and <see cref="ISerializeBinary"/>.
  350. /// </remarks>
  351. /// <param name="writer"></param>
  352. /// <param name="type"></param>
  353. /// <param name="value"></param>
  354. /// <exception cref="Exception">If an object of <paramref name="type"/> is unable to be serialized.</exception>
  355. public static void WriteBinaryValue(this CoreBinaryWriter writer, Type type, object? value)
  356. {
  357. value ??= CoreUtils.GetDefault(type);
  358. if (value == null)
  359. {
  360. if (MatchType<string>(type))
  361. writer.Write("");
  362. else if (writer.Settings.IncludeNullables && typeof(IPackable).IsAssignableFrom(type))
  363. writer.Write(false);
  364. else if (writer.Settings.IncludeNullables && typeof(ISerializeBinary).IsAssignableFrom(type))
  365. writer.Write(false);
  366. else if (Nullable.GetUnderlyingType(type) is Type t)
  367. writer.Write(false);
  368. else if (MatchType<LoggablePropertyAttribute, object>(type))
  369. writer.Write("");
  370. else
  371. writer.Write(0);
  372. }
  373. else if (MatchType<byte[], object>(type) && value is byte[] bArray)
  374. {
  375. writer.Write(bArray.Length);
  376. writer.Write(bArray);
  377. }
  378. else if (type.IsArray && value is Array array)
  379. {
  380. var elementType = type.GetElementType();
  381. writer.Write(array.Length);
  382. foreach (var val1 in array)
  383. {
  384. WriteBinaryValue(writer, elementType, val1);
  385. }
  386. }
  387. else if (type.IsEnum && value is Enum e)
  388. {
  389. var underlyingType = type.GetEnumUnderlyingType();
  390. WriteBinaryValue(writer, underlyingType, Convert.ChangeType(e, underlyingType));
  391. }
  392. else if (MatchType<bool, object>(type) && value is bool b)
  393. {
  394. writer.Write(b);
  395. }
  396. else if (MatchType<string, object>(type) && value is string str)
  397. writer.Write(str);
  398. else if (MatchType<Guid, object>(type) && value is Guid guid)
  399. writer.Write(guid);
  400. else if (MatchType<byte, object>(type) && value is byte i8)
  401. writer.Write(i8);
  402. else if (MatchType<Int16, object>(type) && value is Int16 i16)
  403. writer.Write(i16);
  404. else if (MatchType<Int32, object>(type) && value is Int32 i32)
  405. writer.Write(i32);
  406. else if (MatchType<Int64, object>(type) && value is Int64 i64)
  407. writer.Write(i64);
  408. else if (MatchType<float, object>(type) && value is float f32)
  409. writer.Write(f32);
  410. else if (MatchType<double, object>(type) && value is double f64)
  411. writer.Write(f64);
  412. else if (MatchType<DateTime, object>(type) && value is DateTime date)
  413. writer.Write(date.Ticks);
  414. else if (MatchType<TimeSpan, object>(type) && value is TimeSpan time)
  415. writer.Write(time.Ticks);
  416. else if (MatchType<LoggablePropertyAttribute, object>(type) && value is LoggablePropertyAttribute lpa)
  417. writer.Write(lpa.Format ?? string.Empty);
  418. else if (typeof(IPackable).IsAssignableFrom(type) && value is IPackable pack)
  419. {
  420. if (writer.Settings.IncludeNullables)
  421. writer.Write(true);
  422. pack.Pack(writer);
  423. }
  424. else if (typeof(ISerializeBinary).IsAssignableFrom(type) && value is ISerializeBinary binary)
  425. {
  426. if (writer.Settings.IncludeNullables)
  427. writer.Write(true);
  428. binary.SerializeBinary(writer);
  429. }
  430. else if (Nullable.GetUnderlyingType(type) is Type t)
  431. {
  432. writer.Write(true);
  433. writer.WriteBinaryValue(t, value);
  434. }
  435. else if (value is UserProperty userprop)
  436. WriteBinaryValue(writer, userprop.Type, userprop.Value);
  437. else
  438. throw new SerialisationException($"Invalid type; Target DataType is {type} and value DataType is {value?.GetType().ToString() ?? "null"}");
  439. }
  440. public static void WriteBinaryValue<T>(this CoreBinaryWriter writer, T value)
  441. => WriteBinaryValue(writer, typeof(T), value);
  442. /// <summary>
  443. /// Binary deserialize a bunch of different types of values. <see cref="WriteBinaryValue(CoreBinaryWriter, Type, object?)"/> and
  444. /// <see cref="ReadBinaryValue(CoreBinaryReader, Type)"/> are inverses of each other.
  445. /// </summary>
  446. /// <remarks>
  447. /// Handles <see cref="byte"/>[], <see cref="Array"/>s of serialisable values, <see cref="Enum"/>, <see cref="bool"/>, <see cref="string"/>,
  448. /// <see cref="Guid"/>, <see cref="byte"/>, <see cref="Int16"/>, <see cref="Int32"/>, <see cref="Int64"/>, <see cref="float"/>, <see cref="double"/>,
  449. /// <see cref="DateTime"/>, <see cref="TimeSpan"/>, <see cref="LoggablePropertyAttribute"/>, <see cref="IPackable"/>, <see cref="Nullable{T}"/>
  450. /// and <see cref="ISerializeBinary"/>.
  451. /// </remarks>
  452. /// <param name="reader"></param>
  453. /// <param name="type"></param>
  454. /// <exception cref="Exception">If an object of <paramref name="type"/> is unable to be deserialized.</exception>
  455. public static object? ReadBinaryValue(this CoreBinaryReader reader, Type type)
  456. {
  457. if (type == typeof(byte[]))
  458. {
  459. var length = reader.ReadInt32();
  460. return reader.ReadBytes(length);
  461. }
  462. else if (type.IsArray)
  463. {
  464. var length = reader.ReadInt32();
  465. var elementType = type.GetElementType();
  466. var array = Array.CreateInstance(elementType, length);
  467. for (int i = 0; i < array.Length; ++i)
  468. {
  469. array.SetValue(ReadBinaryValue(reader, elementType), i);
  470. }
  471. return array;
  472. }
  473. else if (type.IsEnum)
  474. {
  475. var val = ReadBinaryValue(reader, type.GetEnumUnderlyingType());
  476. return Enum.ToObject(type, val);
  477. }
  478. else if (type == typeof(bool))
  479. {
  480. return reader.ReadBoolean();
  481. }
  482. else if (type == typeof(string))
  483. {
  484. return reader.ReadString();
  485. }
  486. else if (type == typeof(Guid))
  487. {
  488. return reader.ReadGuid();
  489. }
  490. else if (type == typeof(byte))
  491. {
  492. return reader.ReadByte();
  493. }
  494. else if (type == typeof(Int16))
  495. {
  496. return reader.ReadInt16();
  497. }
  498. else if (type == typeof(Int32))
  499. {
  500. return reader.ReadInt32();
  501. }
  502. else if (type == typeof(Int64))
  503. {
  504. return reader.ReadInt64();
  505. }
  506. else if (type == typeof(float))
  507. {
  508. return reader.ReadSingle();
  509. }
  510. else if (type == typeof(double))
  511. {
  512. return reader.ReadDouble();
  513. }
  514. else if (type == typeof(DateTime))
  515. {
  516. return new DateTime(reader.ReadInt64());
  517. }
  518. else if (type == typeof(TimeSpan))
  519. {
  520. return new TimeSpan(reader.ReadInt64());
  521. }
  522. else if (type == typeof(LoggablePropertyAttribute))
  523. {
  524. String format = reader.ReadString();
  525. return String.IsNullOrWhiteSpace(format)
  526. ? null
  527. : new LoggablePropertyAttribute() { Format = format };
  528. }
  529. else if (typeof(IPackable).IsAssignableFrom(type))
  530. {
  531. if (!reader.Settings.IncludeNullables || reader.ReadBoolean()) // Note the short-circuit operator preventing reading a boolean.
  532. {
  533. var packable = (Activator.CreateInstance(type) as IPackable)!;
  534. packable.Unpack(reader);
  535. return packable;
  536. }
  537. else
  538. {
  539. return null;
  540. }
  541. }
  542. else if (typeof(ISerializeBinary).IsAssignableFrom(type))
  543. {
  544. if (!reader.Settings.IncludeNullables || reader.ReadBoolean()) // Note the short-circuit operator preventing reading a boolean.
  545. {
  546. var obj = (Activator.CreateInstance(type, true) as ISerializeBinary)!;
  547. obj.DeserializeBinary(reader);
  548. return obj;
  549. }
  550. else
  551. {
  552. return null;
  553. }
  554. }
  555. else if (Nullable.GetUnderlyingType(type) is Type t)
  556. {
  557. var isNull = reader.ReadBoolean();
  558. if (isNull)
  559. {
  560. return null;
  561. }
  562. else
  563. {
  564. return reader.ReadBinaryValue(t);
  565. }
  566. }
  567. else
  568. {
  569. throw new SerialisationException($"Invalid type; Target DataType is {type}");
  570. }
  571. }
  572. public static T ReadBinaryValue<T>(this CoreBinaryReader reader)
  573. {
  574. var result = ReadBinaryValue(reader, typeof(T));
  575. return (result != null ? (T)result : default)!;
  576. }
  577. public static IEnumerable<IProperty> SerializableProperties(Type type, Predicate<IProperty>? filter = null) =>
  578. DatabaseSchema.Properties(type)
  579. .Where(x => (!(x is StandardProperty st) || st.IsSerializable) && (filter?.Invoke(x) ?? true));
  580. private static void WriteOriginalValues<TObject>(this CoreBinaryWriter writer, TObject obj)
  581. where TObject : BaseObject
  582. {
  583. var originalValues = new List<Tuple<Type, string, object?>>();
  584. foreach (var (key, value) in obj.OriginalValueList)
  585. {
  586. if (DatabaseSchema.Property(obj.GetType(), key) is IProperty prop && prop.IsSerializable)
  587. {
  588. originalValues.Add(new Tuple<Type, string, object?>(prop.PropertyType, key, value));
  589. }
  590. }
  591. writer.Write(originalValues.Count);
  592. foreach (var (type, key, value) in originalValues)
  593. {
  594. writer.Write(key);
  595. try
  596. {
  597. writer.WriteBinaryValue(type, value);
  598. }
  599. catch (Exception e)
  600. {
  601. CoreUtils.LogException("", e, "Error serialising OriginalValues");
  602. }
  603. }
  604. }
  605. private static void ReadOriginalValues<TObject>(this CoreBinaryReader reader, TObject obj)
  606. where TObject : BaseObject
  607. {
  608. var nOriginalValues = reader.ReadInt32();
  609. for (int i = 0; i < nOriginalValues; ++i)
  610. {
  611. var key = reader.ReadString();
  612. if (DatabaseSchema.Property(obj.GetType(), key) is IProperty prop)
  613. {
  614. var value = reader.ReadBinaryValue(prop.PropertyType);
  615. obj.OriginalValueList[prop.Name] = value;
  616. }
  617. }
  618. }
  619. public static void WriteObject<TObject>(this CoreBinaryWriter writer, TObject entity, Type type)
  620. where TObject : BaseObject
  621. {
  622. if (!typeof(TObject).IsAssignableFrom(type))
  623. throw new SerialisationException($"{type.EntityName()} is not a subclass of {typeof(TObject).EntityName()}");
  624. var properties = SerializableProperties(type).ToList();
  625. writer.Write(properties.Count);
  626. foreach (var property in properties)
  627. {
  628. writer.Write(property.Name);
  629. writer.WriteBinaryValue(property.PropertyType, property.Getter()(entity));
  630. }
  631. writer.WriteOriginalValues(entity);
  632. }
  633. /// <summary>
  634. /// An implementation of binary serialising a <typeparamref name="TObject"/>; this is the inverse of <see cref="ReadObject{TObject}(CoreBinaryReader)"/>.
  635. /// </summary>
  636. /// <remarks>
  637. /// Also serialises the names of properties along with the values.
  638. /// </remarks>
  639. /// <typeparam name="TObject"></typeparam>
  640. /// <param name="writer"></param>
  641. /// <param name="entity"></param>
  642. public static void WriteObject<TObject>(this CoreBinaryWriter writer, TObject entity)
  643. where TObject : BaseObject, new() => WriteObject(writer, entity, typeof(TObject));
  644. public static TObject ReadObject<TObject>(this CoreBinaryReader reader, Type type)
  645. where TObject : BaseObject
  646. {
  647. if (!typeof(TObject).IsAssignableFrom(type))
  648. throw new SerialisationException($"{type.EntityName()} is not a subclass of {typeof(TObject).EntityName()}");
  649. var obj = (Activator.CreateInstance(type) as TObject)!;
  650. obj.SetObserving(false);
  651. var nProps = reader.ReadInt32();
  652. for (int i = 0; i < nProps; ++i)
  653. {
  654. var propName = reader.ReadString();
  655. var property = DatabaseSchema.Property(type, propName)
  656. ?? throw new SerialisationException($"Property {propName} does not exist on {type.EntityName()}");
  657. property.Setter()(obj, reader.ReadBinaryValue(property.PropertyType));
  658. }
  659. reader.ReadOriginalValues(obj);
  660. obj.SetObserving(true);
  661. return obj;
  662. }
  663. /// <summary>
  664. /// The inverse of <see cref="WriteObject{TObject}(CoreBinaryWriter, TObject)"/>.
  665. /// </summary>
  666. /// <typeparam name="TObject"></typeparam>
  667. /// <param name="reader"></param>
  668. /// <returns></returns>
  669. public static TObject ReadObject<TObject>(this CoreBinaryReader reader)
  670. where TObject : BaseObject, new() => reader.ReadObject<TObject>(typeof(TObject));
  671. /// <summary>
  672. /// An implementation of binary serialising multiple <typeparamref name="TObject"/>s;
  673. /// this is the inverse of <see cref="ReadObjects{TObject}(CoreBinaryReader)"/>.
  674. /// </summary>
  675. /// <remarks>
  676. /// Also serialises the names of properties along with the values.
  677. /// </remarks>
  678. /// <typeparam name="TObject"></typeparam>
  679. /// <param name="writer"></param>
  680. /// <param name="objects"></param>
  681. public static void WriteObjects<TObject>(this CoreBinaryWriter writer, ICollection<TObject>? objects)
  682. where TObject : BaseObject, new() => WriteObjects(writer, typeof(TObject), objects);
  683. public static void WriteObjects<TObject>(this CoreBinaryWriter writer, Type type, ICollection<TObject>? objects, Predicate<IProperty>? filter = null)
  684. where TObject : BaseObject
  685. {
  686. if (!typeof(TObject).IsAssignableFrom(type))
  687. throw new SerialisationException($"{type.EntityName()} is not a subclass of {typeof(TObject).EntityName()}");
  688. var nObjs = objects?.Count ?? 0;
  689. writer.Write(nObjs);
  690. if (nObjs == 0)
  691. {
  692. return;
  693. }
  694. var properties = SerializableProperties(type, filter).ToList();
  695. writer.Write(properties.Count);
  696. foreach (var property in properties)
  697. {
  698. writer.Write(property.Name);
  699. }
  700. if(objects != null)
  701. {
  702. foreach (var obj in objects)
  703. {
  704. foreach (var property in properties)
  705. {
  706. writer.WriteBinaryValue(property.PropertyType, property.Getter()(obj));
  707. }
  708. writer.WriteOriginalValues(obj);
  709. }
  710. }
  711. }
  712. /// <summary>
  713. /// The inverse of <see cref="WriteObjects{TObject}(CoreBinaryWriter, ICollection{TObject})"/>.
  714. /// </summary>
  715. /// <typeparam name="TObject"></typeparam>
  716. /// <param name="reader"></param>
  717. /// <returns></returns>
  718. public static List<TObject> ReadObjects<TObject>(this CoreBinaryReader reader)
  719. where TObject : BaseObject, new()
  720. {
  721. return ReadObjects<TObject>(reader, typeof(TObject));
  722. }
  723. public static List<TObject> ReadObjects<TObject>(this CoreBinaryReader reader, Type type) where TObject : BaseObject
  724. {
  725. if (!typeof(TObject).IsAssignableFrom(type))
  726. throw new SerialisationException($"{type.EntityName()} is not a subclass of {typeof(TObject).EntityName()}");
  727. var objs = new List<TObject>();
  728. var properties = new List<IProperty>();
  729. var nObjs = reader.ReadInt32();
  730. if(nObjs == 0)
  731. {
  732. return objs;
  733. }
  734. var nProps = reader.ReadInt32();
  735. for (int i = 0; i < nProps; ++i)
  736. {
  737. var propertyName = reader.ReadString();
  738. var property = DatabaseSchema.Property(type, propertyName)
  739. ?? throw new SerialisationException($"Property {propertyName} does not exist on {type.EntityName()}");
  740. properties.Add(property);
  741. }
  742. for (int i = 0; i < nObjs; ++i)
  743. {
  744. var obj = (Activator.CreateInstance(type) as TObject)!;
  745. obj.SetObserving(false);
  746. foreach (var property in properties)
  747. {
  748. property.Setter()(obj, reader.ReadBinaryValue(property.PropertyType));
  749. }
  750. reader.ReadOriginalValues(obj);
  751. obj.SetObserving(true);
  752. objs.Add(obj);
  753. }
  754. return objs;
  755. }
  756. }
  757. public class BaseObjectJSONConverter : JsonConverter
  758. {
  759. public override bool CanConvert(Type objectType)
  760. {
  761. return objectType.IsSubclassOf(typeof(BaseObject));
  762. }
  763. public override object? ReadJson(JsonReader reader, Type objectType, object? existingValue, JsonSerializer serializer)
  764. {
  765. var obj = (Activator.CreateInstance(objectType) as BaseObject)!;
  766. obj.SetObserving(false);
  767. if (reader.TokenType == JsonToken.Null)
  768. return null;
  769. var data = JObject.Load(reader);
  770. foreach(var (k, v) in data)
  771. {
  772. if (string.Equals(k, "OriginalValues") && v is JObject originalValues)
  773. {
  774. foreach(var (origk, origv) in originalValues)
  775. {
  776. if (origv is null) continue;
  777. var property = DatabaseSchema.Property(objectType, origk);
  778. if (property is null) continue;
  779. obj.OriginalValueList[origk] = origv.ToObject(property.PropertyType);
  780. }
  781. }
  782. else if(DatabaseSchema.Property(objectType, k) is IProperty prop)
  783. {
  784. prop.Setter()(obj, v?.ToObject(prop.PropertyType));
  785. }
  786. }
  787. obj.SetObserving(true);
  788. return obj;
  789. }
  790. public override void WriteJson(JsonWriter writer, object? value, JsonSerializer serializer)
  791. {
  792. if (!(value is BaseObject obj)) return;
  793. writer.WriteStartObject();
  794. writer.WritePropertyName("OriginalValues");
  795. writer.WriteStartObject();
  796. foreach(var (k, v) in obj.OriginalValueList)
  797. {
  798. writer.WritePropertyName(k);
  799. serializer.Serialize(writer, v);
  800. }
  801. writer.WriteEndObject();
  802. foreach(var property in DatabaseSchema.Properties(obj.GetType()))
  803. {
  804. writer.WritePropertyName(property.Name);
  805. serializer.Serialize(writer, property.Getter()(obj));
  806. }
  807. writer.WriteEndObject();
  808. }
  809. }
  810. }