using System.Collections.Generic; using System.Linq; using Jint.Collections; using Jint.Native.BigInt; using Jint.Native.Boolean; using Jint.Native.Global; using Jint.Native.Number; using Jint.Native.Object; using Jint.Native.String; using Jint.Pooling; using Jint.Runtime; using Jint.Runtime.Descriptors; using Jint.Runtime.Interop; namespace Jint.Native.Json { public class JsonSerializer { private readonly Engine _engine; private ObjectTraverseStack _stack; private string _indent, _gap; private List _propertyList; private JsValue _replacerFunction = Undefined.Instance; private static readonly JsString toJsonProperty = new("toJSON"); public JsonSerializer(Engine engine) { _engine = engine; } public JsValue Serialize(JsValue value, JsValue replacer, JsValue space) { _stack = new ObjectTraverseStack(_engine); // for JSON.stringify(), any function passed as the first argument will return undefined // if the replacer is not defined. The function is not called either. if (value.IsCallable && ReferenceEquals(replacer, Undefined.Instance)) { return Undefined.Instance; } if (replacer.IsObject()) { if (replacer is ICallable) { _replacerFunction = replacer; } else { var replacerObj = replacer.AsObject(); _propertyList = new List(); foreach (var pair in replacerObj.GetOwnProperties()) { var v = replacerObj.UnwrapJsValue(pair.Value); string item = null; if (v.IsString()) { item = v.ToString(); } else if (v.IsNumber()) { item = TypeConverter.ToString(v); } else if (v.IsObject()) { var propertyObj = v.AsObject(); if (propertyObj.Class == ObjectClass.String || propertyObj.Class == ObjectClass.Number) { item = TypeConverter.ToString(v); } } if (item != null && !_propertyList.Contains(item)) { _propertyList.Add(item); } } } } if (space.IsObject()) { var spaceObj = space.AsObject(); if (spaceObj.Class == ObjectClass.Number) { space = TypeConverter.ToNumber(spaceObj); } else if (spaceObj.Class == ObjectClass.String) { space = TypeConverter.ToJsString(spaceObj); } } // defining the gap if (space.IsNumber()) { var number = ((JsNumber) space)._value; if (number > 0) { _gap = new string(' ', (int) System.Math.Min(10, number)); } else { _gap = string.Empty; } } else if (space.IsString()) { var stringSpace = space.ToString(); _gap = stringSpace.Length <= 10 ? stringSpace : stringSpace.Substring(0, 10); } else { _gap = string.Empty; } var wrapper = _engine.Realm.Intrinsics.Object.Construct(Arguments.Empty); wrapper.DefineOwnProperty(JsString.Empty, new PropertyDescriptor(value, PropertyFlag.ConfigurableEnumerableWritable)); return SerializeJSONProperty(JsString.Empty, wrapper); } /// /// https://tc39.es/ecma262/#sec-serializejsonproperty /// private JsValue SerializeJSONProperty(JsValue key, JsValue holder) { var value = holder.Get(key, holder); var isBigInt = value is BigIntInstance || value.IsBigInt(); if (value.IsObject() || isBigInt) { var toJson = value.Get(toJsonProperty, value); if (toJson.IsUndefined() && isBigInt) { toJson = _engine.Realm.Intrinsics.BigInt.PrototypeObject.Get(toJsonProperty); } if (toJson.IsObject()) { if (toJson.AsObject() is ICallable callableToJson) { value = callableToJson.Call(value, Arguments.From(key)); } } } if (!_replacerFunction.IsUndefined()) { var replacerFunctionCallable = (ICallable) _replacerFunction.AsObject(); value = replacerFunctionCallable.Call(holder, Arguments.From(key, value)); } if (value.IsObject()) { var valueObj = value.AsObject(); switch (valueObj) { case NumberInstance numberInstance: value = numberInstance.NumberData; break; case StringInstance stringInstance: value = stringInstance.StringData; break; case BooleanInstance booleanInstance: value = booleanInstance.BooleanData; break; case BigIntInstance bigIntInstance: value = bigIntInstance.BigIntData; break; } } if (ReferenceEquals(value, Null.Instance)) { return JsString.NullString; } if (value.IsBoolean()) { return ((JsBoolean) value)._value ? JsString.TrueString : JsString.FalseString; } if (value.IsString()) { return QuoteJSONString(value.ToString()); } if (value.IsNumber()) { var isFinite = GlobalObject.IsFinite(Undefined.Instance, Arguments.From(value)); if (((JsBoolean) isFinite)._value) { return TypeConverter.ToJsString(value); } return JsString.NullString; } if (value.IsBigInt()) { ExceptionHelper.ThrowTypeError(_engine.Realm, "Do not know how to serialize a BigInt"); } var isCallable = value.IsObject() && value.IsCallable; if (value.IsObject() && isCallable == false) { return SerializesAsArray(value) ? SerializeJSONArray(value) : SerializeJSONObject(value.AsObject()); } return JsValue.Undefined; } private static bool SerializesAsArray(JsValue value) { return value.AsObject().Class == ObjectClass.Array || value is ObjectWrapper { IsArrayLike: true }; } /// /// https://tc39.es/ecma262/#sec-quotejsonstring /// private static string QuoteJSONString(string value) { using var stringBuilder = StringBuilderPool.Rent(); var sb = stringBuilder.Builder; sb.Append("\""); foreach (var c in value) { switch (c) { case '\"': sb.Append("\\\""); break; case '\\': sb.Append("\\\\"); break; case '\b': sb.Append("\\b"); break; case '\f': sb.Append("\\f"); break; case '\n': sb.Append("\\n"); break; case '\r': sb.Append("\\r"); break; case '\t': sb.Append("\\t"); break; default: if (c < 0x20) { sb.Append("\\u"); sb.Append(((int) c).ToString("x4")); } else sb.Append(c); break; } } sb.Append("\""); return sb.ToString(); } /// /// https://tc39.es/ecma262/#sec-serializejsonarray /// private string SerializeJSONArray(JsValue value) { _stack.Enter(value); var stepback = _indent; _indent = _indent + _gap; var partial = new List(); var len = TypeConverter.ToUint32(value.Get(CommonProperties.Length, value)); for (int i = 0; i < len; i++) { var strP = SerializeJSONProperty(i, value); if (strP.IsUndefined()) { strP = JsString.NullString; } partial.Add(strP.ToString()); } if (partial.Count == 0) { _stack.Exit(); return "[]"; } string final; if (_gap == "") { const string separator = ","; var properties = string.Join(separator, partial); final = "[" + properties + "]"; } else { var separator = ",\n" + _indent; var properties = string.Join(separator, partial); final = "[\n" + _indent + properties + "\n" + stepback + "]"; } _stack.Exit(); _indent = stepback; return final; } /// /// https://tc39.es/ecma262/#sec-serializejsonobject /// private string SerializeJSONObject(ObjectInstance value) { string final; _stack.Enter(value); var stepback = _indent; _indent += _gap; var k = _propertyList ?? value.GetOwnProperties() .Where(x => !x.Key.IsSymbol() && x.Value.Enumerable) .Select(x => x.Key); var partial = new List(); foreach (var p in k) { var strP = SerializeJSONProperty(p, value); if (!strP.IsUndefined()) { var member = QuoteJSONString(p.ToString()) + ":"; if (_gap != "") { member += " "; } member += strP.AsString(); // TODO:This could be undefined partial.Add(member); } } if (partial.Count == 0) { final = "{}"; } else { if (_gap == "") { const string separator = ","; var properties = string.Join(separator, partial); final = "{" + properties + "}"; } else { var separator = ",\n" + _indent; var properties = string.Join(separator, partial); final = "{\n" + _indent + properties + "\n" + stepback + "}"; } } _stack.Exit(); _indent = stepback; return final; } } }