JsonInstance.cs 3.8 KB

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