JsonInstance.cs 3.8 KB

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