JsonInstance.cs 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. using System;
  2. using Jint.Collections;
  3. using Jint.Native.Object;
  4. using Jint.Native.Symbol;
  5. using Jint.Runtime;
  6. using Jint.Runtime.Descriptors;
  7. using Jint.Runtime.Interop;
  8. namespace Jint.Native.Json
  9. {
  10. public sealed class JsonInstance : ObjectInstance
  11. {
  12. private readonly Realm _realm;
  13. internal JsonInstance(
  14. Engine engine,
  15. Realm realm,
  16. ObjectPrototype objectPrototype)
  17. : base(engine, objectClass: ObjectClass.JSON)
  18. {
  19. _realm = realm;
  20. _prototype = objectPrototype;
  21. }
  22. protected override void Initialize()
  23. {
  24. var properties = new PropertyDictionary(2, checkExistingKeys: false)
  25. {
  26. #pragma warning disable 618
  27. ["parse"] = new PropertyDescriptor(new ClrFunctionInstance(Engine, "parse", Parse, 2, PropertyFlag.Configurable), true, false, true),
  28. ["stringify"] = new PropertyDescriptor(new ClrFunctionInstance(Engine, "stringify", Stringify, 3, PropertyFlag.Configurable), true, false, true)
  29. #pragma warning restore 618
  30. };
  31. SetProperties(properties);
  32. var symbols = new SymbolDictionary(1)
  33. {
  34. [GlobalSymbolRegistry.ToStringTag] = new PropertyDescriptor("JSON", false, false, true),
  35. };
  36. SetSymbols(symbols);
  37. }
  38. private static JsValue InternalizeJSONProperty(JsValue holder, JsValue name, ICallable reviver)
  39. {
  40. JsValue temp = holder.Get(name, holder);
  41. if (temp is ObjectInstance val)
  42. {
  43. if (val.IsArray())
  44. {
  45. var i = 0UL;
  46. var len = TypeConverter.ToLength(val.Get(CommonProperties.Length));
  47. while (i < len)
  48. {
  49. var prop = JsString.Create(i);
  50. var newElement = InternalizeJSONProperty(val, prop, reviver);
  51. if (newElement.IsUndefined())
  52. {
  53. val.Delete(prop);
  54. }
  55. else
  56. {
  57. val.CreateDataProperty(prop, newElement);
  58. }
  59. i = i + 1;
  60. }
  61. }
  62. else
  63. {
  64. var keys = val.EnumerableOwnPropertyNames(EnumerableOwnPropertyNamesKind.Key);
  65. foreach (var p in keys)
  66. {
  67. var newElement = InternalizeJSONProperty(val, p, reviver);
  68. if (newElement.IsUndefined())
  69. {
  70. val.Delete(p);
  71. }
  72. else
  73. {
  74. val.CreateDataProperty(p, newElement);
  75. }
  76. }
  77. }
  78. }
  79. return reviver.Call(holder, new[] { name, temp });
  80. }
  81. /// <summary>
  82. /// https://tc39.es/ecma262/#sec-json.parse
  83. /// </summary>
  84. [Obsolete("Method will be made private, use JsonParser directly")]
  85. public JsValue Parse(JsValue thisObject, JsValue[] arguments)
  86. {
  87. var jsonString = TypeConverter.ToString(arguments.At(0));
  88. var reviver = arguments.At(1);
  89. var parser = new JsonParser(_engine);
  90. var unfiltered = parser.Parse(jsonString);
  91. if (reviver.IsCallable)
  92. {
  93. var root = _realm.Intrinsics.Object.Construct(Arguments.Empty);
  94. var rootName = JsString.Empty;
  95. root.CreateDataPropertyOrThrow(rootName, unfiltered);
  96. return InternalizeJSONProperty(root, rootName, (ICallable) reviver);
  97. }
  98. else
  99. {
  100. return unfiltered;
  101. }
  102. }
  103. [Obsolete("Method will be made private, use JsonSerializer directly")]
  104. public JsValue Stringify(JsValue thisObject, JsValue[] arguments)
  105. {
  106. var value = arguments.At(0);
  107. var replacer = arguments.At(1);
  108. var space = arguments.At(2);
  109. if (value.IsUndefined() && replacer.IsUndefined())
  110. {
  111. return Undefined;
  112. }
  113. var serializer = new JsonSerializer(_engine);
  114. return serializer.Serialize(value, replacer, space);
  115. }
  116. }
  117. }