2
0

JsonInstance.cs 4.4 KB

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