123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224 |
- using System.Diagnostics.CodeAnalysis;
- using System.Globalization;
- using System.Runtime.CompilerServices;
- using System.Threading;
- using Jint.Native;
- using Jint.Native.Object;
- using Jint.Runtime;
- using Jint.Runtime.Interop;
- namespace Jint;
- internal static class DefaultObjectConverter
- {
- private static Dictionary<Type, Func<Engine, object, JsValue>> _typeMappers = new()
- {
- { typeof(bool), (engine, v) => (bool)v ? JsBoolean.True : JsBoolean.False },
- { typeof(byte), (engine, v) => JsNumber.Create((byte)v) },
- { typeof(char), (engine, v) => JsString.Create((char)v) },
- { typeof(DateTime), (engine, v) => engine.Realm.Intrinsics.Date.Construct((DateTime)v) },
- { typeof(DateTimeOffset), (engine, v) => engine.Realm.Intrinsics.Date.Construct((DateTimeOffset)v) },
- { typeof(decimal), (engine, v) => (JsValue)(double)(decimal)v },
- { typeof(double), (engine, v) => (JsValue)(double)v },
- { typeof(short), (engine, v) => JsNumber.Create((short)v) },
- { typeof(int), (engine, v) => JsNumber.Create((int)v) },
- { typeof(long), (engine, v) => (JsValue)(long)v },
- { typeof(sbyte), (engine, v) => JsNumber.Create((sbyte)v) },
- { typeof(float), (engine, v) => (JsValue)(float)v },
- { typeof(string), (engine, v) => JsString.Create((string)v) },
- { typeof(ushort), (engine, v) => JsNumber.Create((ushort)v) },
- { typeof(uint), (engine, v) => JsNumber.Create((uint)v) },
- { typeof(ulong), (engine, v) => JsNumber.Create((ulong)v) },
- {
- typeof(System.Text.RegularExpressions.Regex),
- (engine, v) => engine.Realm.Intrinsics.RegExp.Construct((System.Text.RegularExpressions.Regex)v, ((System.Text.RegularExpressions.Regex)v).ToString(), "")
- }
- };
- public static bool TryConvert(Engine engine, object value, Type? type, [NotNullWhen(true)] out JsValue? result)
- {
- result = null;
- Type valueType = ObjectWrapper.GetClrType(value, type);
- var typeMappers = _typeMappers;
- if (typeMappers.TryGetValue(valueType, out var typeMapper))
- {
- result = typeMapper(engine, value);
- }
- else
- {
- if (value is Array a)
- {
- // racy, we don't care, worst case we'll catch up later
- Interlocked.CompareExchange(ref _typeMappers,
- new Dictionary<Type, Func<Engine, object, JsValue>>(typeMappers)
- {
- [valueType] = ConvertArray
- }, typeMappers);
- result = ConvertArray(engine, a);
- return result is not null;
- }
- if (value is IConvertible convertible && TryConvertConvertible(engine, convertible, out result))
- {
- return true;
- }
- if (value is Delegate d)
- {
- result = new DelegateWrapper(engine, d);
- return result is not null;
- }
- if ((engine.Options.ExperimentalFeatures & ExperimentalFeature.TaskInterop) != ExperimentalFeature.None)
- {
- if (value is Task task)
- {
- result = JsValue.ConvertAwaitableToPromise(engine, task);
- return result is not null;
- }
- #if NETSTANDARD2_1_OR_GREATER || NETCOREAPP
- if (value is ValueTask valueTask)
- {
- result = JsValue.ConvertAwaitableToPromise(engine, valueTask);
- return result is not null;
- }
- #endif
- }
- #if NET8_0_OR_GREATER
- if (value is System.Text.Json.Nodes.JsonValue jsonValue)
- {
- result = ConvertSystemTextJsonValue(engine, jsonValue);
- return result is not null;
- }
- #endif
- var t = value.GetType();
- if (!engine.Options.Interop.AllowSystemReflection
- && t.Namespace?.StartsWith("System.Reflection", StringComparison.Ordinal) == true)
- {
- const string Message = "Cannot access System.Reflection namespace, check Engine's interop options";
- Throw.InvalidOperationException(Message);
- }
- if (t.IsEnum)
- {
- var ut = Enum.GetUnderlyingType(t);
- if (ut == typeof(ulong))
- {
- result = JsNumber.Create(Convert.ToDouble(value, CultureInfo.InvariantCulture));
- }
- else
- {
- if (ut == typeof(uint) || ut == typeof(long))
- {
- result = JsNumber.Create(Convert.ToInt64(value, CultureInfo.InvariantCulture));
- }
- else
- {
- result = JsNumber.Create(Convert.ToInt32(value, CultureInfo.InvariantCulture));
- }
- }
- }
- else
- {
- // check global cache, have we already wrapped the value?
- if (engine._objectWrapperCache?.TryGetValue(value, out var cached) == true)
- {
- result = cached;
- }
- else
- {
- var wrapped = engine.Options.Interop.WrapObjectHandler.Invoke(engine, value, type);
- if (ReferenceEquals(wrapped?.GetPrototypeOf(), engine.Realm.Intrinsics.Object.PrototypeObject)
- && engine._typeReferences?.TryGetValue(t, out var typeReference) == true)
- {
- wrapped.SetPrototypeOf(typeReference);
- }
- result = wrapped;
- if (engine.Options.Interop.TrackObjectWrapperIdentity && wrapped is not null)
- {
- engine._objectWrapperCache ??= new ConditionalWeakTable<object, ObjectInstance>();
- engine._objectWrapperCache.Add(value, wrapped);
- }
- }
- }
- // if no known type could be guessed, use the default of wrapping using ObjectWrapper
- }
- return result is not null;
- }
- #if NET8_0_OR_GREATER
- private static JsValue? ConvertSystemTextJsonValue(Engine engine, System.Text.Json.Nodes.JsonNode value)
- {
- return value.GetValueKind() switch
- {
- System.Text.Json.JsonValueKind.Object => JsValue.FromObject(engine, value),
- System.Text.Json.JsonValueKind.Array => JsValue.FromObject(engine, value),
- System.Text.Json.JsonValueKind.String => JsString.Create(value.ToString()),
- #pragma warning disable IL2026, IL3050
- System.Text.Json.JsonValueKind.Number => ((System.Text.Json.Nodes.JsonValue) value).TryGetValue<int>(out var intValue)
- ? JsNumber.Create(intValue)
- : System.Text.Json.JsonSerializer.Deserialize<double>(value),
- #pragma warning restore IL2026, IL3050
- System.Text.Json.JsonValueKind.True => JsBoolean.True,
- System.Text.Json.JsonValueKind.False => JsBoolean.False,
- System.Text.Json.JsonValueKind.Undefined => JsValue.Undefined,
- System.Text.Json.JsonValueKind.Null => JsValue.Null,
- _ => null,
- };
- }
- #endif
- private static bool TryConvertConvertible(Engine engine, IConvertible convertible, [NotNullWhen(true)] out JsValue? result)
- {
- result = convertible.GetTypeCode() switch
- {
- TypeCode.Boolean => convertible.ToBoolean(engine.Options.Culture) ? JsBoolean.True : JsBoolean.False,
- TypeCode.Byte => JsNumber.Create(convertible.ToByte(engine.Options.Culture)),
- TypeCode.Char => JsString.Create(convertible.ToChar(engine.Options.Culture)),
- TypeCode.Double => JsNumber.Create(convertible.ToDouble(engine.Options.Culture)),
- TypeCode.SByte => JsNumber.Create(convertible.ToSByte(engine.Options.Culture)),
- TypeCode.Int16 => JsNumber.Create(convertible.ToInt16(engine.Options.Culture)),
- TypeCode.Int32 => JsNumber.Create(convertible.ToInt32(engine.Options.Culture)),
- TypeCode.UInt16 => JsNumber.Create(convertible.ToUInt16(engine.Options.Culture)),
- TypeCode.Int64 => JsNumber.Create(convertible.ToInt64(engine.Options.Culture)),
- TypeCode.Single => JsNumber.Create(convertible.ToSingle(engine.Options.Culture)),
- TypeCode.String => JsString.Create(convertible.ToString(engine.Options.Culture)),
- TypeCode.UInt32 => JsNumber.Create(convertible.ToUInt32(engine.Options.Culture)),
- TypeCode.UInt64 => JsNumber.Create(convertible.ToUInt64(engine.Options.Culture)),
- TypeCode.DateTime => engine.Realm.Intrinsics.Date.Construct(convertible.ToDateTime(engine.Options.Culture)),
- TypeCode.Decimal => JsNumber.Create(convertible.ToDecimal(engine.Options.Culture)),
- TypeCode.DBNull => JsValue.Null,
- TypeCode.Empty => JsValue.Null,
- _ => null
- };
- return result is not null;
- }
- private static JsArray ConvertArray(Engine e, object v)
- {
- var array = (Array) v;
- var arrayLength = (uint) array.Length;
- var values = new JsValue[arrayLength];
- for (uint i = 0; i < arrayLength; ++i)
- {
- values[i] = JsValue.FromObject(e, array.GetValue(i));
- }
- return new JsArray(e, values);
- }
- }
|