diff --git a/src/libraries/System.Text.Json/gen/JsonSourceGenerator.Emitter.cs b/src/libraries/System.Text.Json/gen/JsonSourceGenerator.Emitter.cs index f18b452c91135..d0f8b2e121df8 100644 --- a/src/libraries/System.Text.Json/gen/JsonSourceGenerator.Emitter.cs +++ b/src/libraries/System.Text.Json/gen/JsonSourceGenerator.Emitter.cs @@ -50,6 +50,7 @@ private sealed partial class Emitter private const string TypeTypeRef = "global::System.Type"; private const string UnsafeTypeRef = "global::System.Runtime.CompilerServices.Unsafe"; private const string NullableTypeRef = "global::System.Nullable"; + private const string ConditionalWeakTableTypeRef = "global::System.Runtime.CompilerServices.ConditionalWeakTable"; private const string EqualityComparerTypeRef = "global::System.Collections.Generic.EqualityComparer"; private const string IListTypeRef = "global::System.Collections.Generic.IList"; private const string KeyValuePairTypeRef = "global::System.Collections.Generic.KeyValuePair"; @@ -70,6 +71,7 @@ private sealed partial class Emitter private const string JsonPropertyInfoTypeRef = "global::System.Text.Json.Serialization.Metadata.JsonPropertyInfo"; private const string JsonPropertyInfoValuesTypeRef = "global::System.Text.Json.Serialization.Metadata.JsonPropertyInfoValues"; private const string JsonTypeInfoTypeRef = "global::System.Text.Json.Serialization.Metadata.JsonTypeInfo"; + private const string JsonTypeInfoResolverTypeRef = "global::System.Text.Json.Serialization.Metadata.IJsonTypeInfoResolver"; private static DiagnosticDescriptor TypeNotSupported { get; } = new DiagnosticDescriptor( id: "SYSLIB1030", @@ -131,14 +133,14 @@ public void Emit() isRootContextDef: true); // Add GetJsonTypeInfo override implementation. - AddSource($"{contextName}.GetJsonTypeInfo.g.cs", GetGetTypeInfoImplementation()); + AddSource($"{contextName}.GetJsonTypeInfo.g.cs", GetGetTypeInfoImplementation(), interfaceImplementation: JsonTypeInfoResolverTypeRef); // Add property name initialization. AddSource($"{contextName}.PropertyNames.g.cs", GetPropertyNameInitialization()); } } - private void AddSource(string fileName, string source, bool isRootContextDef = false) + private void AddSource(string fileName, string source, bool isRootContextDef = false, string? interfaceImplementation = null) { string? generatedCodeAttributeSource = isRootContextDef ? s_generatedCodeAttributeSource : null; @@ -175,7 +177,7 @@ namespace {@namespace} // Add the core implementation for the derived context class. string partialContextImplementation = $@" -{generatedCodeAttributeSource}{declarationList[0]} +{generatedCodeAttributeSource}{declarationList[0]}{(interfaceImplementation is null ? "" : ": " + interfaceImplementation)} {{ {IndentSource(source, Math.Max(1, declarationCount - 1))} }}"; @@ -338,7 +340,7 @@ private static string GenerateForTypeWithUnknownConverter(TypeGenerationSpec typ {{ // Allow nullable handling to forward to the underlying type's converter. converter = {JsonMetadataServicesTypeRef}.GetNullableConverter<{typeCompilableName}>(this.{typeFriendlyName})!; - converter = (({ JsonConverterFactoryTypeRef })converter).CreateConverter(typeToConvert, { OptionsInstanceVariableName })!; + converter = (({JsonConverterFactoryTypeRef})converter).CreateConverter(typeToConvert, {OptionsInstanceVariableName})!; }} else {{ @@ -356,7 +358,7 @@ private static string GenerateForTypeWithUnknownConverter(TypeGenerationSpec typ } metadataInitSource.Append($@" - _{typeFriendlyName} = { JsonMetadataServicesTypeRef }.{ GetCreateValueInfoMethodRef(typeCompilableName)} ({ OptionsInstanceVariableName}, converter); "); + _{typeFriendlyName} = {JsonMetadataServicesTypeRef}.{GetCreateValueInfoMethodRef(typeCompilableName)} ({OptionsInstanceVariableName}, converter); "); return GenerateForType(typeMetadata, metadataInitSource.ToString()); } @@ -704,10 +706,10 @@ private string GeneratePropMetadataInitFunc(TypeGenerationSpec typeGenerationSpe sb.Append($@" -private static {JsonPropertyInfoTypeRef}[] {propInitMethodName}({JsonSerializerContextTypeRef} context) +private {JsonPropertyInfoTypeRef}[] {propInitMethodName}({JsonSerializerContextTypeRef}? context) {{ - {contextTypeRef} {JsonContextVarName} = ({contextTypeRef})context; - {JsonSerializerOptionsTypeRef} options = context.Options; + {contextTypeRef} {JsonContextVarName} = ({contextTypeRef}?)context ?? this; + {JsonSerializerOptionsTypeRef} options = {JsonContextVarName}.Options; {JsonPropertyInfoTypeRef}[] {PropVarName} = {propertyArrayInstantiationValue}; "); @@ -1176,6 +1178,10 @@ private string GetRootJsonContextImplementation() {{ }} +private {contextTypeName}({JsonSerializerOptionsTypeRef} options, bool bindOptionsToContext) : base(options, bindOptionsToContext) +{{ +}} + {GetFetchLogicForRuntimeSpecifiedCustomConverter()}"); if (_generateGetConverterMethodForProperties) @@ -1291,10 +1297,28 @@ private string GetGetTypeInfoImplementation() } } - sb.Append(@" + sb.AppendLine(@" return null!; }"); + // Explicit IJsonTypeInfoResolver implementation + string contextTypeName = _currentContext.ContextType.Name; + + sb.AppendLine(); + sb.Append(@$"{JsonTypeInfoTypeRef}? {JsonTypeInfoResolverTypeRef}.GetTypeInfo({TypeTypeRef} type, {JsonSerializerOptionsTypeRef} options) +{{ + {contextTypeName} context = this; + + if (options != null && {OptionsInstanceVariableName} != options) + {{ + context = (s_resolverCache ??= new()).GetValue(options, static options => new {contextTypeName}(options, bindOptionsToContext: false)); + }} + + return context.GetTypeInfo(type); +}} + +private {ConditionalWeakTableTypeRef}<{JsonSerializerOptionsTypeRef},{contextTypeName}>? s_resolverCache;"); + return sb.ToString(); } diff --git a/src/libraries/System.Text.Json/ref/System.Text.Json.cs b/src/libraries/System.Text.Json/ref/System.Text.Json.cs index 8a5df1d2379c6..3c2466038a5d8 100644 --- a/src/libraries/System.Text.Json/ref/System.Text.Json.cs +++ b/src/libraries/System.Text.Json/ref/System.Text.Json.cs @@ -357,6 +357,13 @@ public JsonSerializerOptions(System.Text.Json.JsonSerializerOptions options) { } public System.Text.Json.JsonCommentHandling ReadCommentHandling { get { throw null; } set { } } public System.Text.Json.Serialization.ReferenceHandler? ReferenceHandler { get { throw null; } set { } } public System.Collections.Generic.IList PolymorphicTypeConfigurations { get { throw null; } } + public System.Text.Json.Serialization.Metadata.IJsonTypeInfoResolver TypeInfoResolver + { + [System.Diagnostics.CodeAnalysis.RequiresUnreferencedCode("JSON serialization and deserialization might require types that cannot be statically analyzed. Use the overload that takes a JsonTypeInfo or JsonSerializerContext, or make sure all of the required types are preserved.")] + [System.Diagnostics.CodeAnalysis.RequiresDynamicCode("JSON serialization and deserialization might require types that cannot be statically analyzed and might need runtime code generation. Use System.Text.Json source generation for native AOT applications.")] + get { throw null; } + set { } + } public System.Text.Json.Serialization.JsonUnknownTypeHandling UnknownTypeHandling { get { throw null; } set { } } public bool WriteIndented { get { throw null; } set { } } public void AddContext() where TContext : System.Text.Json.Serialization.JsonSerializerContext, new() { } @@ -950,12 +957,14 @@ public JsonSerializableAttribute(System.Type type) { } public string? TypeInfoPropertyName { get { throw null; } set { } } public System.Text.Json.Serialization.JsonSourceGenerationMode GenerationMode { get { throw null; } set { } } } - public abstract partial class JsonSerializerContext + public abstract partial class JsonSerializerContext : System.Text.Json.Serialization.Metadata.IJsonTypeInfoResolver { protected JsonSerializerContext(System.Text.Json.JsonSerializerOptions? options) { } + protected JsonSerializerContext(System.Text.Json.JsonSerializerOptions? options, bool bindOptionsToContext) { } protected abstract System.Text.Json.JsonSerializerOptions? GeneratedSerializerOptions { get; } public System.Text.Json.JsonSerializerOptions Options { get { throw null; } } public abstract System.Text.Json.Serialization.Metadata.JsonTypeInfo? GetTypeInfo(System.Type type); + System.Text.Json.Serialization.Metadata.JsonTypeInfo System.Text.Json.Serialization.Metadata.IJsonTypeInfoResolver.GetTypeInfo(Type type, JsonSerializerOptions options) { throw null; } } [System.AttributeUsageAttribute(System.AttributeTargets.Class, AllowMultiple = false)] public sealed partial class JsonSourceGenerationOptionsAttribute : System.Text.Json.Serialization.JsonAttribute @@ -1044,6 +1053,20 @@ protected ReferenceResolver() { } } namespace System.Text.Json.Serialization.Metadata { + public class DefaultJsonTypeInfoResolver : System.Text.Json.Serialization.Metadata.IJsonTypeInfoResolver + { + [System.Diagnostics.CodeAnalysis.RequiresUnreferencedCodeAttribute("JSON serialization and deserialization might require types that cannot be statically analyzed. Use the overload that takes a JsonTypeInfo or JsonSerializerContext, or make sure all of the required types are preserved.")] + [System.Diagnostics.CodeAnalysis.RequiresDynamicCodeAttribute("JSON serialization and deserialization might require types that cannot be statically analyzed and might need runtime code generation. Use System.Text.Json source generation for native AOT applications.")] + public DefaultJsonTypeInfoResolver() { } + + public virtual System.Text.Json.Serialization.Metadata.JsonTypeInfo GetTypeInfo(System.Type type, System.Text.Json.JsonSerializerOptions options) { throw null; } + + public System.Collections.Generic.IList> Modifiers { get; } + } + public interface IJsonTypeInfoResolver + { + System.Text.Json.Serialization.Metadata.JsonTypeInfo? GetTypeInfo(System.Type type, System.Text.Json.JsonSerializerOptions options); + } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public sealed partial class JsonCollectionInfoValues { @@ -1138,10 +1161,17 @@ public JsonParameterInfoValues() { } public System.Type ParameterType { get { throw null; } init { } } public int Position { get { throw null; } init { } } } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public abstract partial class JsonPropertyInfo { internal JsonPropertyInfo() { } + public System.Text.Json.Serialization.JsonConverter? CustomConverter { get { throw null; } set { } } + public System.Func? Get { get { throw null; } set { } } + public string Name { get { throw null; } set { } } + public System.Text.Json.Serialization.JsonNumberHandling? NumberHandling { get { throw null; } set { } } + public System.Type PropertyType { get { throw null; } } + public System.Text.Json.JsonSerializerOptions Options { get { throw null; } } + public System.Action? Set { get { throw null; } set { } } + public System.Func? ShouldSerialize { get { throw null; } set { } } } [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public sealed partial class JsonPropertyInfoValues @@ -1162,15 +1192,38 @@ public JsonPropertyInfoValues() { } public System.Text.Json.Serialization.Metadata.JsonTypeInfo PropertyTypeInfo { get { throw null; } init { } } public System.Action? Setter { get { throw null; } init { } } } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] - public partial class JsonTypeInfo + public static class JsonTypeInfoResolver + { + public static System.Text.Json.Serialization.Metadata.IJsonTypeInfoResolver Combine(params System.Text.Json.Serialization.Metadata.IJsonTypeInfoResolver[] resolvers) { throw null; } + } + public abstract partial class JsonTypeInfo { internal JsonTypeInfo() { } + public System.Text.Json.JsonSerializerOptions Options { get { throw null; } } + public System.Collections.Generic.IList Properties { get { throw null; } } + public System.Type Type { get { throw null; } } + public System.Text.Json.Serialization.JsonConverter Converter { get { throw null; } } + public System.Func? CreateObject { get { throw null; } set { } } + public System.Text.Json.Serialization.Metadata.JsonTypeInfoKind Kind { get { throw null; } } + public System.Text.Json.Serialization.JsonNumberHandling? NumberHandling { get { throw null; } set { } } + public static System.Text.Json.Serialization.Metadata.JsonTypeInfo CreateJsonTypeInfo(System.Text.Json.JsonSerializerOptions options) { throw null; } + [System.Diagnostics.CodeAnalysis.RequiresUnreferencedCode("JSON serialization and deserialization might require types that cannot be statically analyzed and might need runtime code generation. Use generic overload or System.Text.Json source generation for native AOT applications.")] + [System.Diagnostics.CodeAnalysis.RequiresDynamicCode("JSON serialization and deserialization might require types that cannot be statically analyzed and might need runtime code generation. Use generic overload or System.Text.Json source generation for native AOT applications.")] + public static System.Text.Json.Serialization.Metadata.JsonTypeInfo CreateJsonTypeInfo(System.Type type, System.Text.Json.JsonSerializerOptions options) { throw null; } + public JsonPropertyInfo CreateJsonPropertyInfo(Type propertyType, string name) { throw null; } } - [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public abstract partial class JsonTypeInfo : System.Text.Json.Serialization.Metadata.JsonTypeInfo { internal JsonTypeInfo() { } + public new System.Func? CreateObject { get { throw null; } set { } } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] public System.Action? SerializeHandler { get { throw null; } } } + public enum JsonTypeInfoKind + { + None = 0, + Object = 1, + Enumerable = 2, + Dictionary = 3 + } } diff --git a/src/libraries/System.Text.Json/src/Resources/Strings.resx b/src/libraries/System.Text.Json/src/Resources/Strings.resx index 6983f813456a3..fffb04160c73b 100644 --- a/src/libraries/System.Text.Json/src/Resources/Strings.resx +++ b/src/libraries/System.Text.Json/src/Resources/Strings.resx @@ -1,17 +1,17 @@ - @@ -243,6 +243,15 @@ The requested operation requires an element of type '{0}', but the target element has type '{1}'. + + Default TypeInfoResolver and custom TypeInfoResolver after first usage cannot be changed. + + + JsonTypeInfo cannot be changed after first usage. + + + JsonPropertyInfo cannot be changed after first usage. + Max depth must be positive. @@ -390,6 +399,12 @@ The converter '{0}' is not compatible with the type '{1}'. + + TypeInfoResolver expected to return JsonTypeInfo of type '{0}' but returned JsonTypeInfo of type '{1}'. + + + TypeInfoResolver expected to return JsonTypeInfo options bound to the JsonSerializerOptions provided in the argument. + The converter '{0}' wrote too much or not enough. @@ -572,13 +587,13 @@ Metadata for type '{0}' was not provided to the serializer. The serializer method used does not support reflection-based creation of serialization-related type metadata. If using source generation, ensure that all root types passed to the serializer have been indicated with 'JsonSerializableAttribute', along with any types that might be serialized polymorphically. - + Collection is read-only. - + Number was less than 0. - + Destination array was not long enough. @@ -638,4 +653,7 @@ Runtime type '{0}' has a diamond ambiguity between derived types '{1}' and '{2}' of polymorphic type '{3}'. Consider either removing one of the derived types or removing the 'JsonUnknownDerivedTypeHandling.FallBackToNearestAncestor' setting. + + Operation is not possible when Kind is JsonTypeKind.None. + diff --git a/src/libraries/System.Text.Json/src/System.Text.Json.csproj b/src/libraries/System.Text.Json/src/System.Text.Json.csproj index 0e4522699262a..91ff197b847aa 100644 --- a/src/libraries/System.Text.Json/src/System.Text.Json.csproj +++ b/src/libraries/System.Text.Json/src/System.Text.Json.csproj @@ -61,6 +61,7 @@ The System.Text.Json library is built-in as part of the shared framework in .NET + @@ -101,6 +102,7 @@ The System.Text.Json library is built-in as part of the shared framework in .NET + @@ -119,6 +121,12 @@ The System.Text.Json library is built-in as part of the shared framework in .NET + + + + + + @@ -376,12 +384,7 @@ The System.Text.Json library is built-in as part of the shared framework in .NET - - + + diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.KeyCollection.cs b/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.KeyCollection.cs index a0e9edb109bfb..997ee59563900 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.KeyCollection.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.KeyCollection.cs @@ -36,9 +36,9 @@ IEnumerator IEnumerable.GetEnumerator() } } - public void Add(string propertyName) => ThrowHelper.ThrowNotSupportedException_NodeCollectionIsReadOnly(); + public void Add(string propertyName) => ThrowHelper.ThrowNotSupportedException_CollectionIsReadOnly(); - public void Clear() => ThrowHelper.ThrowNotSupportedException_NodeCollectionIsReadOnly(); + public void Clear() => ThrowHelper.ThrowNotSupportedException_CollectionIsReadOnly(); public bool Contains(string propertyName) => _parent.ContainsProperty(propertyName); @@ -46,14 +46,14 @@ public void CopyTo(string[] propertyNameArray, int index) { if (index < 0) { - ThrowHelper.ThrowArgumentOutOfRangeException_NodeArrayIndexNegative(nameof(index)); + ThrowHelper.ThrowArgumentOutOfRangeException_ArrayIndexNegative(nameof(index)); } foreach (KeyValuePair item in _parent) { if (index >= propertyNameArray.Length) { - ThrowHelper.ThrowArgumentException_NodeArrayTooSmall(nameof(propertyNameArray)); + ThrowHelper.ThrowArgumentException_ArrayTooSmall(nameof(propertyNameArray)); } propertyNameArray[index++] = item.Key; @@ -68,7 +68,7 @@ public IEnumerator GetEnumerator() } } - bool ICollection.Remove(string propertyName) => throw ThrowHelper.GetNotSupportedException_NodeCollectionIsReadOnly(); + bool ICollection.Remove(string propertyName) => throw ThrowHelper.GetNotSupportedException_CollectionIsReadOnly(); } } } diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.ValueCollection.cs b/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.ValueCollection.cs index c907454d800e6..c81a67cd29dbf 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.ValueCollection.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.ValueCollection.cs @@ -36,9 +36,9 @@ IEnumerator IEnumerable.GetEnumerator() } } - public void Add(T? jsonNode) => ThrowHelper.ThrowNotSupportedException_NodeCollectionIsReadOnly(); + public void Add(T? jsonNode) => ThrowHelper.ThrowNotSupportedException_CollectionIsReadOnly(); - public void Clear() => ThrowHelper.ThrowNotSupportedException_NodeCollectionIsReadOnly(); + public void Clear() => ThrowHelper.ThrowNotSupportedException_CollectionIsReadOnly(); public bool Contains(T? jsonNode) => _parent.ContainsValue(jsonNode); @@ -46,14 +46,14 @@ public void CopyTo(T?[] nodeArray, int index) { if (index < 0) { - ThrowHelper.ThrowArgumentOutOfRangeException_NodeArrayIndexNegative(nameof(index)); + ThrowHelper.ThrowArgumentOutOfRangeException_ArrayIndexNegative(nameof(index)); } foreach (KeyValuePair item in _parent) { if (index >= nodeArray.Length) { - ThrowHelper.ThrowArgumentException_NodeArrayTooSmall(nameof(nodeArray)); + ThrowHelper.ThrowArgumentException_ArrayTooSmall(nameof(nodeArray)); } nodeArray[index++] = item.Value; @@ -68,7 +68,7 @@ public void CopyTo(T?[] nodeArray, int index) } } - bool ICollection.Remove(T? node) => throw ThrowHelper.GetNotSupportedException_NodeCollectionIsReadOnly(); + bool ICollection.Remove(T? node) => throw ThrowHelper.GetNotSupportedException_CollectionIsReadOnly(); } } } diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.ValueList.cs b/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.ValueList.cs new file mode 100644 index 0000000000000..c428744493b6e --- /dev/null +++ b/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.ValueList.cs @@ -0,0 +1,207 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System.Collections; +using System.Collections.Generic; +using System.Diagnostics; +using System.Diagnostics.CodeAnalysis; + +namespace System.Text.Json +{ + internal sealed partial class JsonPropertyDictionary + { +#if DEBUG + // CreateEditableValueList should be called at most once by JsonTypeInfo + private ValueList? _editableValueList; +#endif + + public ValueList CreateValueList(Func? getKey) + { + ValueList ret = new ValueList(this, getKey); +#if DEBUG + Debug.Assert(_editableValueList == null, "More than one ValueList created"); + return _editableValueList ??= ret; +#else + return ret; +#endif + } + + internal sealed class ValueList : IList + { + private readonly JsonPropertyDictionary _parent; + private Func? _getKey; + private List? _items; + + [MemberNotNullWhen(false, nameof(_getKey))] + [MemberNotNullWhen(false, nameof(_items))] + public bool IsReadOnly => _getKey == null; + public int Count => IsReadOnly ? _parent.Count : _items.Count; + + public T this[int index] + { + get => IsReadOnly ? _parent.List[index].Value! : _items[index]; + set + { + if (IsReadOnly) + ThrowCollectionIsReadOnly(); + + _items[index] = value; + } + } + + public ValueList(JsonPropertyDictionary jsonObject, Func? getKey) + { + // _getKey == null is equivalent to being read-only + _parent = jsonObject; + _getKey = getKey; + + Debug.Assert(!_parent.IsReadOnly, $"{nameof(JsonPropertyDictionary)} is read-only but editable value list is created"); + + if (!IsReadOnly) + { + // We cannot ensure keys won't change while editing therefore we operate on the internal copy. + // Once we're done editing FinishEditingAndMakeReadOnly should be called then we switch to operating directly on _parent + _items = new List(_parent.Count); + foreach (var kv in _parent.List) + { + Debug.Assert(kv.Value != null, $"{nameof(JsonPropertyDictionary)} contains null value"); + _items.Add(kv.Value); + } + } + } + + public void FinishEditingAndMakeReadOnly() + { + Debug.Assert(!IsReadOnly, $"{nameof(FinishEditingAndMakeReadOnly)} called on read-only ValueList"); + + // We do not know if any of the keys needs to be updated therefore we need to re-create cache + _parent.Clear(); + + foreach (var item in _items) + { + _parent.AddValue(_getKey(item), item); + } + + // clearing those so that we don't keep GC from freeing + _items = null; + _getKey = null; + } + + public void Add(T item) + { + if (IsReadOnly) + ThrowCollectionIsReadOnly(); + + _items.Add(item); + } + + public void Clear() + { + if (IsReadOnly) + ThrowCollectionIsReadOnly(); + + _items.Clear(); + } + + public bool Contains(T item) => IsReadOnly ? _parent.ContainsValue(item) : _items.Contains(item); + + public void CopyTo(T[] array, int index) + { + if (index < 0) + { + ThrowHelper.ThrowArgumentOutOfRangeException_ArrayIndexNegative(nameof(index)); + } + + if (IsReadOnly) + { + foreach (KeyValuePair item in _parent) + { + if (index >= array.Length) + { + ThrowHelper.ThrowArgumentException_ArrayTooSmall(nameof(array)); + } + + array[index++] = item.Value!; + } + } + else + { + _items.CopyTo(array, index); + } + } + + public int IndexOf(T item) + { + if (IsReadOnly) + { + int index = 0; + foreach (var kv in _parent.List) + { + if (kv.Value == item) + { + return index; + } + + index++; + } + + return -1; + } + else + { + return _items.IndexOf(item); + } + } + + public void Insert(int index, T item) + { + if (IsReadOnly) + ThrowCollectionIsReadOnly(); + + _items.Insert(index, item); + } + + public bool Remove(T item) + { + if (IsReadOnly) + ThrowCollectionIsReadOnly(); + + return _items.Remove(item); + } + + public void RemoveAt(int index) + { + if (IsReadOnly) + ThrowCollectionIsReadOnly(); + + _items.RemoveAt(index); + } + + public IEnumerator GetEnumerator() + { + if (IsReadOnly) + { + foreach (KeyValuePair item in _parent) + { + yield return item.Value!; + } + } + else + { + foreach (T item in _items) + { + yield return item; + } + } + } + + IEnumerator IEnumerable.GetEnumerator() => GetEnumerator(); + + [DoesNotReturn] + private static void ThrowCollectionIsReadOnly() + { + ThrowHelper.ThrowInvalidOperationException_CollectionIsReadOnly(); + } + } + } +} diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.cs b/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.cs index 8e3180341dba4..27539786b9fc5 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/JsonPropertyDictionary.cs @@ -38,7 +38,7 @@ public void Add(string propertyName, T? value) { if (IsReadOnly) { - ThrowHelper.ThrowNotSupportedException_NodeCollectionIsReadOnly(); + ThrowHelper.ThrowNotSupportedException_CollectionIsReadOnly(); } if (propertyName == null) @@ -53,7 +53,7 @@ public void Add(KeyValuePair property) { if (IsReadOnly) { - ThrowHelper.ThrowNotSupportedException_NodeCollectionIsReadOnly(); + ThrowHelper.ThrowNotSupportedException_CollectionIsReadOnly(); } Add(property.Key, property.Value); @@ -63,7 +63,7 @@ public bool TryAdd(string propertyName, T value) { if (IsReadOnly) { - ThrowHelper.ThrowNotSupportedException_NodeCollectionIsReadOnly(); + ThrowHelper.ThrowNotSupportedException_CollectionIsReadOnly(); } // A check for a null propertyName is not required since this method is only called by internal code. @@ -76,7 +76,7 @@ public void Clear() { if (IsReadOnly) { - ThrowHelper.ThrowNotSupportedException_NodeCollectionIsReadOnly(); + ThrowHelper.ThrowNotSupportedException_CollectionIsReadOnly(); } _propertyList.Clear(); @@ -105,7 +105,7 @@ public bool Remove(string propertyName) { if (IsReadOnly) { - ThrowHelper.ThrowNotSupportedException_NodeCollectionIsReadOnly(); + ThrowHelper.ThrowNotSupportedException_CollectionIsReadOnly(); } if (propertyName == null) @@ -133,14 +133,14 @@ public void CopyTo(KeyValuePair[] array, int index) { if (index < 0) { - ThrowHelper.ThrowArgumentOutOfRangeException_NodeArrayIndexNegative(nameof(index)); + ThrowHelper.ThrowArgumentOutOfRangeException_ArrayIndexNegative(nameof(index)); } foreach (KeyValuePair item in _propertyList) { if (index >= array.Length) { - ThrowHelper.ThrowArgumentException_NodeArrayTooSmall(nameof(array)); + ThrowHelper.ThrowArgumentException_ArrayTooSmall(nameof(array)); } array[index++] = item; @@ -211,7 +211,7 @@ public T? this[string propertyName] { if (IsReadOnly) { - ThrowHelper.ThrowNotSupportedException_NodeCollectionIsReadOnly(); + ThrowHelper.ThrowNotSupportedException_CollectionIsReadOnly(); } if (propertyName == null) @@ -286,7 +286,7 @@ private bool TryAddValue(string propertyName, T? value) { if (IsReadOnly) { - ThrowHelper.ThrowNotSupportedException_NodeCollectionIsReadOnly(); + ThrowHelper.ThrowNotSupportedException_CollectionIsReadOnly(); } CreateDictionaryIfThresholdMet(); @@ -383,7 +383,7 @@ public bool TryRemoveProperty(string propertyName, out T? existing) { if (IsReadOnly) { - ThrowHelper.ThrowNotSupportedException_NodeCollectionIsReadOnly(); + ThrowHelper.ThrowNotSupportedException_CollectionIsReadOnly(); } if (_propertyDictionary != null) diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/ConfigurationList.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/ConfigurationList.cs index fb37d18818b38..5199a93bba29c 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/ConfigurationList.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/ConfigurationList.cs @@ -4,30 +4,25 @@ using System.Collections; using System.Collections.Generic; using System.Diagnostics; +using System.Globalization; namespace System.Text.Json.Serialization { /// - /// A list of configuration items that respects the options class being immutable once (de)serialization occurs. + /// A list of configuration items that can be locked for modification /// - internal sealed class ConfigurationList : IList + internal abstract class ConfigurationList : IList { private readonly List _list; - private readonly JsonSerializerOptions _options; - public Action? OnElementAdded { get; set; } - - public ConfigurationList(JsonSerializerOptions options) + public ConfigurationList(IList? source = null) { - _options = options; - _list = new List(); + _list = source is null ? new List() : new List(source); } - public ConfigurationList(JsonSerializerOptions options, IList source) - { - _options = options; - _list = new List(source is ConfigurationList cl ? cl._list : source); - } + protected abstract bool IsLockedInstance { get; } + protected abstract void VerifyMutable(); + protected virtual void OnItemAdded(TItem item) { } public TItem this[int index] { @@ -42,15 +37,15 @@ public TItem this[int index] throw new ArgumentNullException(nameof(value)); } - _options.VerifyMutable(); + VerifyMutable(); _list[index] = value; - OnElementAdded?.Invoke(value); + OnItemAdded(value); } } public int Count => _list.Count; - public bool IsReadOnly => false; + public bool IsReadOnly => IsLockedInstance; public void Add(TItem item) { @@ -59,14 +54,14 @@ public void Add(TItem item) ThrowHelper.ThrowArgumentNullException(nameof(item)); } - _options.VerifyMutable(); + VerifyMutable(); _list.Add(item); - OnElementAdded?.Invoke(item); + OnItemAdded(item); } public void Clear() { - _options.VerifyMutable(); + VerifyMutable(); _list.Clear(); } @@ -97,20 +92,20 @@ public void Insert(int index, TItem item) ThrowHelper.ThrowArgumentNullException(nameof(item)); } - _options.VerifyMutable(); + VerifyMutable(); _list.Insert(index, item); - OnElementAdded?.Invoke(item); + OnItemAdded(item); } public bool Remove(TItem item) { - _options.VerifyMutable(); + VerifyMutable(); return _list.Remove(item); } public void RemoveAt(int index) { - _options.VerifyMutable(); + VerifyMutable(); _list.RemoveAt(index); } diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/CastingConverter.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/CastingConverter.cs new file mode 100644 index 0000000000000..88915a2c471c3 --- /dev/null +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/CastingConverter.cs @@ -0,0 +1,74 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System.Diagnostics; +using System.Diagnostics.CodeAnalysis; +using System.Text.Json.Serialization.Metadata; + +namespace System.Text.Json.Serialization.Converters +{ + /// + /// Converter wrapper which casts SourceType into TargetType + /// + internal sealed class CastingConverter : JsonConverter + { + private JsonConverter _sourceConverter; + + internal override Type? KeyType => _sourceConverter.KeyType; + internal override Type? ElementType => _sourceConverter.ElementType; + + public override bool HandleNull => _sourceConverter.HandleNull; + internal override ConverterStrategy ConverterStrategy => _sourceConverter.ConverterStrategy; + + internal CastingConverter(JsonConverter sourceConverter) : base(initialize: false) + { + _sourceConverter = sourceConverter; + Initialize(); + + IsInternalConverter = sourceConverter.IsInternalConverter; + IsInternalConverterForNumberType = sourceConverter.IsInternalConverterForNumberType; + RequiresReadAhead = sourceConverter.RequiresReadAhead; + CanUseDirectReadOrWrite = sourceConverter.CanUseDirectReadOrWrite; + } + + public override TargetType? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) + => ConvertToTarget(_sourceConverter.Read(ref reader, typeToConvert, options)); + + public override void Write(Utf8JsonWriter writer, TargetType value, JsonSerializerOptions options) + => _sourceConverter.Write(writer, ConvertToSource(value), options); + + internal override bool OnTryRead(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options, ref ReadStack state, out TargetType? value) + { + bool ret = _sourceConverter.OnTryRead(ref reader, typeToConvert, options, ref state, out SourceType? sourceValue); + value = ConvertToTarget(sourceValue); + return ret; + } + + internal override bool OnTryWrite(Utf8JsonWriter writer, TargetType value, JsonSerializerOptions options, ref WriteStack state) + => _sourceConverter.OnTryWrite(writer, ConvertToSource(value), options, ref state); + + public override TargetType ReadAsPropertyName(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) + => ConvertToTarget(_sourceConverter.ReadAsPropertyName(ref reader, typeToConvert, options)); + + internal override TargetType ReadAsPropertyNameCore(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) + => ConvertToTarget(_sourceConverter.ReadAsPropertyNameCore(ref reader, typeToConvert, options)); + + public override void WriteAsPropertyName(Utf8JsonWriter writer, TargetType value, JsonSerializerOptions options) + => _sourceConverter.WriteAsPropertyName(writer, ConvertToSource(value), options); + + internal override void WriteAsPropertyNameCore(Utf8JsonWriter writer, TargetType value, JsonSerializerOptions options, bool isWritingExtensionDataProperty) + => _sourceConverter.WriteAsPropertyNameCore(writer, ConvertToSource(value), options, isWritingExtensionDataProperty); + + internal override TargetType ReadNumberWithCustomHandling(ref Utf8JsonReader reader, JsonNumberHandling handling, JsonSerializerOptions options) + => ConvertToTarget(_sourceConverter.ReadNumberWithCustomHandling(ref reader, handling, options)); + + internal override void WriteNumberWithCustomHandling(Utf8JsonWriter writer, TargetType value, JsonNumberHandling handling) + => _sourceConverter.WriteNumberWithCustomHandling(writer, ConvertToSource(value), handling); + + private static SourceType ConvertToSource(TargetType? targetValue) + => (SourceType)(object?)targetValue!; + + private static TargetType ConvertToTarget(SourceType? sourceValue) + => (TargetType)(object?)sourceValue!; + } +} diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Collection/JsonCollectionConverter.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Collection/JsonCollectionConverter.cs index 5b32b9cc83a8f..2ee506e720496 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Collection/JsonCollectionConverter.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Collection/JsonCollectionConverter.cs @@ -41,7 +41,7 @@ protected virtual void CreateCollection(ref Utf8JsonReader reader, ref ReadStack } } - state.Current.ReturnValue = typeInfo.CreateObject()!; + state.Current.ReturnValue = typeInfo.CreateObject(); Debug.Assert(state.Current.ReturnValue is TCollection); } @@ -49,7 +49,7 @@ protected virtual void ConvertCollection(ref ReadStack state, JsonSerializerOpti protected static JsonConverter GetElementConverter(JsonTypeInfo elementTypeInfo) { - JsonConverter converter = (JsonConverter)elementTypeInfo.PropertyInfoForTypeInfo.ConverterBase; + JsonConverter converter = (JsonConverter)elementTypeInfo.Converter; Debug.Assert(converter != null); // It should not be possible to have a null converter at this point. return converter; @@ -57,7 +57,7 @@ protected static JsonConverter GetElementConverter(JsonTypeInfo elemen protected static JsonConverter GetElementConverter(ref WriteStack state) { - JsonConverter converter = (JsonConverter)state.Current.JsonPropertyInfo!.ConverterBase; + JsonConverter converter = (JsonConverter)state.Current.JsonPropertyInfo!.EffectiveConverter; Debug.Assert(converter != null); // It should not be possible to have a null converter at this point. return converter; diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Collection/JsonDictionaryConverter.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Collection/JsonDictionaryConverter.cs index 721f1c64d0faf..78660b8f6aa8c 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Collection/JsonDictionaryConverter.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Collection/JsonDictionaryConverter.cs @@ -70,7 +70,7 @@ protected virtual void CreateCollection(ref Utf8JsonReader reader, ref ReadStack protected static JsonConverter GetConverter(JsonTypeInfo typeInfo) { - JsonConverter converter = (JsonConverter)typeInfo.PropertyInfoForTypeInfo.ConverterBase; + JsonConverter converter = (JsonConverter)typeInfo.Converter; Debug.Assert(converter != null); // It should not be possible to have a null converter at this point. return converter; diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Collection/StackOrQueueConverter.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Collection/StackOrQueueConverter.cs index 5832092cbb043..59f3f3c336ec9 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Collection/StackOrQueueConverter.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Collection/StackOrQueueConverter.cs @@ -21,7 +21,7 @@ protected sealed override void Add(in object? value, ref ReadStack state) protected sealed override void CreateCollection(ref Utf8JsonReader reader, ref ReadStack state, JsonSerializerOptions options) { JsonTypeInfo typeInfo = state.Current.JsonTypeInfo; - JsonTypeInfo.ConstructorDelegate? constructorDelegate = typeInfo.CreateObject; + Func? constructorDelegate = typeInfo.CreateObject; if (constructorDelegate == null) { diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/JsonMetadataServicesConverter.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/JsonMetadataServicesConverter.cs index b597c4d16da6a..a0efc0a884331 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/JsonMetadataServicesConverter.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/JsonMetadataServicesConverter.cs @@ -67,7 +67,7 @@ internal override bool OnTryWrite(Utf8JsonWriter writer, T value, JsonSerializer jsonTypeInfo is JsonTypeInfo info && info.SerializeHandler != null && !state.CurrentContainsMetadata && // Do not use the fast path if state needs to write metadata. - info.Options.JsonSerializerContext?.CanUseSerializationLogic == true) + info.Options.SerializerContext?.CanUseSerializationLogic == true) { info.SerializeHandler(writer, value); return true; diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Object/ObjectDefaultConverter.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Object/ObjectDefaultConverter.cs index cc8216363a4da..19a14dac9980f 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Object/ObjectDefaultConverter.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Object/ObjectDefaultConverter.cs @@ -36,7 +36,7 @@ internal override bool OnTryRead(ref Utf8JsonReader reader, Type typeToConvert, ThrowHelper.ThrowNotSupportedException_DeserializeNoConstructor(jsonTypeInfo.Type, ref reader, ref state); } - obj = jsonTypeInfo.CreateObject!()!; + obj = jsonTypeInfo.CreateObject()!; if (obj is IJsonOnDeserializing onDeserializing) { @@ -132,7 +132,7 @@ internal override bool OnTryRead(ref Utf8JsonReader reader, Type typeToConvert, ThrowHelper.ThrowNotSupportedException_DeserializeNoConstructor(jsonTypeInfo.Type, ref reader, ref state); } - obj = jsonTypeInfo.CreateObject!()!; + obj = jsonTypeInfo.CreateObject()!; if (state.Current.MetadataPropertyNames.HasFlag(MetadataPropertyName.Id)) { @@ -206,7 +206,7 @@ internal override bool OnTryRead(ref Utf8JsonReader reader, Type typeToConvert, if (state.Current.PropertyState < StackFramePropertyState.ReadValue) { - if (!jsonPropertyInfo.ShouldDeserialize) + if (!jsonPropertyInfo.CanDeserialize) { if (!reader.TrySkip()) { @@ -301,7 +301,7 @@ internal sealed override bool OnTryWrite( for (int i = 0; i < properties.Count; i++) { JsonPropertyInfo jsonPropertyInfo = properties[i].Value!; - if (jsonPropertyInfo.ShouldSerialize) + if (jsonPropertyInfo.CanSerialize) { // Remember the current property for JsonPath support if an exception is thrown. state.Current.JsonPropertyInfo = jsonPropertyInfo; @@ -317,7 +317,7 @@ internal sealed override bool OnTryWrite( // Write extension data after the normal properties. JsonPropertyInfo? dataExtensionProperty = jsonTypeInfo.DataExtensionProperty; - if (dataExtensionProperty?.ShouldSerialize == true) + if (dataExtensionProperty?.CanSerialize == true) { // Remember the current property for JsonPath support if an exception is thrown. state.Current.JsonPropertyInfo = dataExtensionProperty; @@ -355,14 +355,14 @@ internal sealed override bool OnTryWrite( { JsonPropertyInfo? jsonPropertyInfo = propertyList![state.Current.EnumeratorIndex].Value; Debug.Assert(jsonPropertyInfo != null); - if (jsonPropertyInfo.ShouldSerialize) + if (jsonPropertyInfo.CanSerialize) { state.Current.JsonPropertyInfo = jsonPropertyInfo; state.Current.NumberHandling = jsonPropertyInfo.EffectiveNumberHandling; if (!jsonPropertyInfo.GetMemberAndWriteJson(obj!, ref state, writer)) { - Debug.Assert(jsonPropertyInfo.ConverterBase.ConverterStrategy != ConverterStrategy.Value); + Debug.Assert(jsonPropertyInfo.EffectiveConverter.ConverterStrategy != ConverterStrategy.Value); return false; } @@ -384,7 +384,7 @@ internal sealed override bool OnTryWrite( if (state.Current.EnumeratorIndex == propertyList.Count) { JsonPropertyInfo? dataExtensionProperty = jsonTypeInfo.DataExtensionProperty; - if (dataExtensionProperty?.ShouldSerialize == true) + if (dataExtensionProperty?.CanSerialize == true) { // Remember the current property for JsonPath support if an exception is thrown. state.Current.JsonPropertyInfo = dataExtensionProperty; @@ -434,7 +434,7 @@ protected static void ReadPropertyValue( bool useExtensionProperty) { // Skip the property if not found. - if (!jsonPropertyInfo.ShouldDeserialize) + if (!jsonPropertyInfo.CanDeserialize) { reader.Skip(); } @@ -464,7 +464,7 @@ protected static bool ReadAheadPropertyValue(ref ReadStack state, ref Utf8JsonRe if (!state.Current.UseExtensionProperty) { - if (!SingleValueReadWithReadAhead(jsonPropertyInfo.ConverterBase.RequiresReadAhead, ref reader, ref state)) + if (!SingleValueReadWithReadAhead(jsonPropertyInfo.EffectiveConverter.RequiresReadAhead, ref reader, ref state)) { return false; } diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Object/ObjectWithParameterizedConstructorConverter.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Object/ObjectWithParameterizedConstructorConverter.cs index a3aaf6f298fc9..dd29fe61670e5 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Object/ObjectWithParameterizedConstructorConverter.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Converters/Object/ObjectWithParameterizedConstructorConverter.cs @@ -289,7 +289,7 @@ private void ReadConstructorArguments(ref ReadStack state, ref Utf8JsonReader re out _, createExtensionProperty: false); - if (jsonPropertyInfo.ShouldDeserialize) + if (jsonPropertyInfo.CanDeserialize) { ArgumentState argumentState = state.Current.CtorArgumentState!; @@ -454,7 +454,7 @@ private static bool HandlePropertyWithContinuation( { if (state.Current.PropertyState < StackFramePropertyState.ReadValue) { - if (!jsonPropertyInfo.ShouldDeserialize) + if (!jsonPropertyInfo.CanDeserialize) { if (!reader.TrySkip()) { @@ -488,7 +488,7 @@ private static bool HandlePropertyWithContinuation( } } - Debug.Assert(jsonPropertyInfo.ShouldDeserialize); + Debug.Assert(jsonPropertyInfo.CanDeserialize); // Ensure that the cache has enough capacity to add this property. diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonConverter.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonConverter.cs index fd6acba560a81..f4f6c63ebaa26 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonConverter.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonConverter.cs @@ -1,8 +1,10 @@ // Licensed to the .NET Foundation under one or more agreements. // The .NET Foundation licenses this file to you under the MIT license. +using System.Diagnostics; using System.Diagnostics.CodeAnalysis; using System.Reflection; +using System.Text.Json.Serialization.Converters; using System.Text.Json.Serialization.Metadata; namespace System.Text.Json.Serialization @@ -69,6 +71,13 @@ internal virtual void ReadElementAndSetProperty( internal abstract JsonParameterInfo CreateJsonParameterInfo(); + internal virtual JsonConverter CreateCastingConverter() + { + // This can only happen for factory converters and we should always expand factory here. + ThrowHelper.ThrowInvalidOperationException_ConverterCanConvertMultipleTypes(typeof(TargetType), this); + return null!; + } + internal abstract Type? ElementType { get; } internal abstract Type? KeyType { get; } diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonConverterOfT.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonConverterOfT.cs index 98e477f95d398..96e043c5612f0 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonConverterOfT.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonConverterOfT.cs @@ -17,6 +17,19 @@ public abstract partial class JsonConverter : JsonConverter /// When overidden, constructs a new instance. /// protected internal JsonConverter() + { + Initialize(); + } + + internal JsonConverter(bool initialize) + { + if (initialize) + { + Initialize(); + } + } + + internal void Initialize() { IsValueType = typeof(T).IsValueType; IsInternalConverter = GetType().Assembly == typeof(JsonConverter).Assembly; @@ -64,6 +77,11 @@ internal sealed override JsonParameterInfo CreateJsonParameterInfo() return new JsonParameterInfo(); } + internal sealed override JsonConverter CreateCastingConverter() + { + return new CastingConverter(this); + } + internal override Type? KeyType => null; internal override Type? ElementType => null; @@ -318,7 +336,7 @@ value is not null && state.Current.PolymorphicSerializationState != PolymorphicSerializationState.PolymorphicReEntryStarted) { JsonTypeInfo jsonTypeInfo = state.PeekNestedJsonTypeInfo(); - Debug.Assert(jsonTypeInfo.PropertyInfoForTypeInfo.ConverterBase.TypeToConvert == TypeToConvert); + Debug.Assert(jsonTypeInfo.Converter.TypeToConvert == TypeToConvert); bool canBePolymorphic = CanBePolymorphic || jsonTypeInfo.PolymorphicTypeResolver is not null; JsonConverter? polymorphicConverter = canBePolymorphic ? @@ -528,7 +546,11 @@ public abstract void Write( /// Method should be overridden in custom converters of types used in deserialized dictionary keys. public virtual T ReadAsPropertyName(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) { - if (!IsInternalConverter && options.TryGetDefaultSimpleConverter(TypeToConvert, out JsonConverter? defaultConverter)) + if (!IsInternalConverter && + options.SerializerContext is null && // For consistency do not return any default converters for + // options instances linked to a JsonSerializerContext, + // even if the default converters might have been rooted. + DefaultJsonTypeInfoResolver.TryGetDefaultSimpleConverter(TypeToConvert, out JsonConverter? defaultConverter)) { // .NET 5 backward compatibility: hardcode the default converter for primitive key serialization. Debug.Assert(defaultConverter.IsInternalConverter && defaultConverter is JsonConverter); @@ -562,7 +584,11 @@ internal virtual T ReadAsPropertyNameCore(ref Utf8JsonReader reader, Type typeTo /// Method should be overridden in custom converters of types used in serialized dictionary keys. public virtual void WriteAsPropertyName(Utf8JsonWriter writer, T value, JsonSerializerOptions options) { - if (!IsInternalConverter && options.TryGetDefaultSimpleConverter(TypeToConvert, out JsonConverter? defaultConverter)) + if (!IsInternalConverter && + options.SerializerContext is null && // For consistency do not return any default converters for + // options instances linked to a JsonSerializerContext, + // even if the default converters might have been rooted. + DefaultJsonTypeInfoResolver.TryGetDefaultSimpleConverter(TypeToConvert, out JsonConverter? defaultConverter)) { // .NET 5 backward compatibility: hardcode the default converter for primitive key serialization. Debug.Assert(defaultConverter.IsInternalConverter && defaultConverter is JsonConverter); diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.HandlePropertyName.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.HandlePropertyName.cs index 04aa45166638d..f51a2cac4bbc3 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.HandlePropertyName.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.HandlePropertyName.cs @@ -121,12 +121,16 @@ internal static void CreateDataExtensionProperty( genericArgs[1].UnderlyingSystemType == typeof(JsonElement) || genericArgs[1].UnderlyingSystemType == typeof(Nodes.JsonNode)); #endif - if (jsonPropertyInfo.JsonTypeInfo.CreateObject == null) + + Func? createObjectForExtensionDataProp = jsonPropertyInfo.JsonTypeInfo.CreateObject + ?? jsonPropertyInfo.JsonTypeInfo.CreateObjectForExtensionDataProperty; + + if (createObjectForExtensionDataProp == null) { // Avoid a reference to the JsonNode type for trimming if (jsonPropertyInfo.PropertyType.FullName == JsonTypeInfo.JsonObjectTypeName) { - extensionData = jsonPropertyInfo.ConverterBase.CreateObject(options); + extensionData = jsonPropertyInfo.EffectiveConverter.CreateObject(options); } else { @@ -135,7 +139,7 @@ internal static void CreateDataExtensionProperty( } else { - extensionData = jsonPropertyInfo.JsonTypeInfo.CreateObject(); + extensionData = createObjectForExtensionDataProp(); } jsonPropertyInfo.SetExtensionDictionaryAsObject(obj, extensionData); diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.Helpers.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.Helpers.cs index 29869dda32cfd..6a157f9ba3f73 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.Helpers.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.Helpers.cs @@ -35,7 +35,7 @@ public static partial class JsonSerializer state.Initialize(jsonTypeInfo); TValue? value; - JsonConverter jsonConverter = jsonTypeInfo.PropertyInfoForTypeInfo.ConverterBase; + JsonConverter jsonConverter = jsonTypeInfo.Converter; // For performance, the code below is a lifted ReadCore() above. if (jsonConverter is JsonConverter converter) diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.Stream.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.Stream.cs index 55ba3ef0a2496..16415a1cdb634 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.Stream.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.Stream.cs @@ -440,7 +440,7 @@ private static async IAsyncEnumerable CreateAsyncEnumerableDeserializer< ref bufferState, ref jsonReaderState, ref readStack, - queueTypeInfo.PropertyInfoForTypeInfo.ConverterBase, + queueTypeInfo.Converter, options); if (readStack.Current.ReturnValue is Queue queue) @@ -469,7 +469,7 @@ private static async IAsyncEnumerable CreateAsyncEnumerableDeserializer< ReadStack readStack = default; jsonTypeInfo.EnsureConfigured(); readStack.Initialize(jsonTypeInfo, supportContinuation: true); - JsonConverter converter = readStack.Current.JsonPropertyInfo!.ConverterBase; + JsonConverter converter = readStack.Current.JsonPropertyInfo!.EffectiveConverter; var jsonReaderState = new JsonReaderState(options.GetReaderOptions()); try @@ -500,7 +500,7 @@ private static async IAsyncEnumerable CreateAsyncEnumerableDeserializer< ReadStack readStack = default; jsonTypeInfo.EnsureConfigured(); readStack.Initialize(jsonTypeInfo, supportContinuation: true); - JsonConverter converter = readStack.Current.JsonPropertyInfo!.ConverterBase; + JsonConverter converter = readStack.Current.JsonPropertyInfo!.EffectiveConverter; var jsonReaderState = new JsonReaderState(options.GetReaderOptions()); try diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.Utf8JsonReader.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.Utf8JsonReader.cs index d98fd97672bdb..5fb3adb99ba6b 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.Utf8JsonReader.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Read.Utf8JsonReader.cs @@ -422,7 +422,7 @@ public static partial class JsonSerializer var newReader = new Utf8JsonReader(rentedSpan, originalReaderOptions); - JsonConverter jsonConverter = state.Current.JsonPropertyInfo!.ConverterBase; + JsonConverter jsonConverter = state.Current.JsonPropertyInfo!.EffectiveConverter; TValue? value = ReadCore(jsonConverter, ref newReader, jsonTypeInfo.Options, ref state); // The reader should have thrown if we have remaining bytes. diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Write.Helpers.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Write.Helpers.cs index e9d3932df45f3..42da3e131bd4d 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Write.Helpers.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializer.Write.Helpers.cs @@ -41,7 +41,7 @@ private static void WriteUsingGeneratedSerializer(Utf8JsonWriter writer, if (jsonTypeInfo.HasSerialize && jsonTypeInfo is JsonTypeInfo typedInfo && - typedInfo.Options.JsonSerializerContext?.CanUseSerializationLogic == true) + typedInfo.Options.SerializerContext?.CanUseSerializationLogic == true) { Debug.Assert(typedInfo.SerializeHandler != null); typedInfo.SerializeHandler(writer, value); @@ -59,15 +59,15 @@ private static void WriteUsingSerializer(Utf8JsonWriter writer, in TValu Debug.Assert(!jsonTypeInfo.HasSerialize || jsonTypeInfo is not JsonTypeInfo || - jsonTypeInfo.Options.JsonSerializerContext == null || - !jsonTypeInfo.Options.JsonSerializerContext.CanUseSerializationLogic, + jsonTypeInfo.Options.SerializerContext == null || + !jsonTypeInfo.Options.SerializerContext.CanUseSerializationLogic, "Incorrect method called. WriteUsingGeneratedSerializer() should have been called instead."); WriteStack state = default; jsonTypeInfo.EnsureConfigured(); state.Initialize(jsonTypeInfo, supportContinuation: false, supportAsync: false); - JsonConverter converter = jsonTypeInfo.PropertyInfoForTypeInfo.ConverterBase; + JsonConverter converter = jsonTypeInfo.Converter; Debug.Assert(converter != null); Debug.Assert(jsonTypeInfo.Options != null); diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerContext.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerContext.cs index 0b8385b5210d6..6d87444a463c2 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerContext.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerContext.cs @@ -1,6 +1,7 @@ // Licensed to the .NET Foundation under one or more agreements. // The .NET Foundation licenses this file to you under the MIT license. +using System.Diagnostics; using System.Text.Json.Serialization.Metadata; namespace System.Text.Json.Serialization @@ -8,7 +9,7 @@ namespace System.Text.Json.Serialization /// /// Provides metadata about a set of types that is relevant to JSON serialization. /// - public abstract partial class JsonSerializerContext + public abstract partial class JsonSerializerContext : IJsonTypeInfoResolver { private bool? _canUseSerializationLogic; @@ -19,9 +20,9 @@ public abstract partial class JsonSerializerContext /// when instanciating the context, then a new instance is bound and returned. /// /// - /// The instance cannot be mutated once it is bound with the context instance. + /// The instance cannot be mutated once it is bound to the context instance. /// - public JsonSerializerOptions Options => _options ??= new JsonSerializerOptions { JsonSerializerContext = this }; + public JsonSerializerOptions Options => _options ??= new JsonSerializerOptions { TypeInfoResolver = this }; /// /// Indicates whether pre-generated serialization logic for types in the context @@ -80,11 +81,32 @@ internal bool CanUseSerializationLogic /// or until is called, where a new options instance is created and bound. /// protected JsonSerializerContext(JsonSerializerOptions? options) + : this(options, bindOptionsToContext: true) + { + } + + /// + /// Creates an instance of and optionally binds it with the indicated . + /// + /// The run time provided options for the context instance. + /// Specify whether the options instance should be bound to the new context. + /// + /// If no instance options are passed, then no options are set until the context is bound using , + /// or until is called, where a new options instance is created and bound. + /// + protected JsonSerializerContext(JsonSerializerOptions? options, bool bindOptionsToContext) { if (options != null) { - options.JsonSerializerContext = this; - _options = options; + if (bindOptionsToContext) + { + options.TypeInfoResolver = this; + Debug.Assert(_options == options, "options.TypeInfoResolver setter did not assign options"); + } + else + { + _options = options; + } } } @@ -94,5 +116,16 @@ protected JsonSerializerContext(JsonSerializerOptions? options) /// The type to fetch metadata about. /// The metadata for the specified type, or if the context has no metadata for the type. public abstract JsonTypeInfo? GetTypeInfo(Type type); + + JsonTypeInfo? IJsonTypeInfoResolver.GetTypeInfo(Type type, JsonSerializerOptions options) + { + if (options != null && _options != options) + { + // TODO is this the appropriate exception message to throw? + ThrowHelper.ThrowInvalidOperationException_SerializerContextOptionsImmutable(); + } + + return GetTypeInfo(type); + } } } diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerOptions.Caching.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerOptions.Caching.cs index 482cdc0466fb0..a0a2de2adaded 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerOptions.Caching.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerOptions.Caching.cs @@ -28,7 +28,6 @@ internal JsonTypeInfo GetOrAddJsonTypeInfo(Type type) if (_cachingContext == null) { InitializeCachingContext(); - Debug.Assert(_cachingContext != null); } return _cachingContext.GetOrAddJsonTypeInfo(type); @@ -71,13 +70,11 @@ internal void ClearCaches() _lastTypeInfo = null; } + [MemberNotNull(nameof(_cachingContext))] private void InitializeCachingContext() { + _isLockedInstance = true; _cachingContext = TrackedCachingContexts.GetOrCreate(this); - if (IsInitializedForReflectionSerializer) - { - _cachingContext.Options.IsInitializedForReflectionSerializer = true; - } } /// @@ -129,6 +126,7 @@ internal static class TrackedCachingContexts public static CachingContext GetOrCreate(JsonSerializerOptions options) { + Debug.Assert(options._isLockedInstance, "Cannot create caching contexts for mutable JsonSerializerOptions instances"); ConcurrentDictionary> cache = s_cache; if (cache.TryGetValue(options, out WeakReference? wr) && wr.TryGetTarget(out CachingContext? ctx)) @@ -167,7 +165,7 @@ public static CachingContext GetOrCreate(JsonSerializerOptions options) { // Copy fields ignored by the copy constructor // but are necessary to determine equivalence. - _serializerContext = options._serializerContext, + _typeInfoResolver = options._typeInfoResolver, }; Debug.Assert(key._cachingContext == null); @@ -269,6 +267,7 @@ private sealed class EqualityComparer : IEqualityComparer public bool Equals(JsonSerializerOptions? left, JsonSerializerOptions? right) { Debug.Assert(left != null && right != null); + return left._dictionaryKeyPolicy == right._dictionaryKeyPolicy && left._jsonPropertyNamingPolicy == right._jsonPropertyNamingPolicy && @@ -287,11 +286,9 @@ public bool Equals(JsonSerializerOptions? left, JsonSerializerOptions? right) left._includeFields == right._includeFields && left._propertyNameCaseInsensitive == right._propertyNameCaseInsensitive && left._writeIndented == right._writeIndented && - left._serializerContext == right._serializerContext && + NormalizeResolver(left._typeInfoResolver) == NormalizeResolver(right._typeInfoResolver) && CompareLists(left._converters, right._converters) && -#pragma warning disable CA2252 // This API requires opting into preview features CompareLists(left._polymorphicTypeConfigurations, right._polymorphicTypeConfigurations); -#pragma warning restore CA2252 // This API requires opting into preview features static bool CompareLists(ConfigurationList left, ConfigurationList right) { @@ -334,11 +331,9 @@ public int GetHashCode(JsonSerializerOptions options) hc.Add(options._includeFields); hc.Add(options._propertyNameCaseInsensitive); hc.Add(options._writeIndented); - hc.Add(options._serializerContext); + hc.Add(NormalizeResolver(options._typeInfoResolver)); GetHashCode(ref hc, options._converters); -#pragma warning disable CA2252 // This API requires opting into preview features GetHashCode(ref hc, options._polymorphicTypeConfigurations); -#pragma warning restore CA2252 // This API requires opting into preview features static void GetHashCode(ref HashCode hc, ConfigurationList list) { @@ -351,6 +346,10 @@ static void GetHashCode(ref HashCode hc, ConfigurationList list) return hc.ToHashCode(); } + // An options instance might be locked but not initialized for reflection serialization yet. + private static IJsonTypeInfoResolver? NormalizeResolver(IJsonTypeInfoResolver? resolver) + => resolver ?? DefaultJsonTypeInfoResolver.DefaultInstance; + #if !NETCOREAPP /// /// Polyfill for System.HashCode. diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerOptions.Converters.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerOptions.Converters.cs index d41325d738138..a9c4c257c5f9b 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerOptions.Converters.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerOptions.Converters.cs @@ -5,7 +5,6 @@ using System.Diagnostics; using System.Diagnostics.CodeAnalysis; using System.Reflection; -using System.Runtime.ExceptionServices; using System.Text.Json.Reflection; using System.Text.Json.Serialization; using System.Text.Json.Serialization.Converters; @@ -19,127 +18,6 @@ namespace System.Text.Json /// public sealed partial class JsonSerializerOptions { - // The global list of built-in simple converters. - private static Dictionary? s_defaultSimpleConverters; - - // The global list of built-in converters that override CanConvert(). - private static JsonConverter[]? s_defaultFactoryConverters; - - // Stores the JsonTypeInfo factory, which requires unreferenced code and must be rooted by the reflection-based serializer. - private static Func? s_typeInfoCreationFunc; - - [RequiresUnreferencedCode(JsonSerializer.SerializationUnreferencedCodeMessage)] - [RequiresDynamicCode(JsonSerializer.SerializationRequiresDynamicCodeMessage)] - private static void RootReflectionSerializerDependencies() - { - // s_typeInfoCreationFunc is the last field assigned. - // Use it as the sentinel to ensure that all dependencies are initialized. - if (Volatile.Read(ref s_typeInfoCreationFunc) is null) - { - s_defaultSimpleConverters = GetDefaultSimpleConverters(); - s_defaultFactoryConverters = GetDefaultFactoryConverters(); - // Explicitly ensure that the previous fields are initialized along with this one. - Volatile.Write(ref s_typeInfoCreationFunc, CreateJsonTypeInfo); - } - - [RequiresUnreferencedCode(JsonSerializer.SerializationUnreferencedCodeMessage)] - [RequiresDynamicCode(JsonSerializer.SerializationRequiresDynamicCodeMessage)] - static JsonTypeInfo CreateJsonTypeInfo(Type type, JsonSerializerOptions options) - { - JsonTypeInfo.ValidateType(type, null, null, options); - - MethodInfo methodInfo = typeof(JsonSerializerOptions).GetMethod(nameof(CreateReflectionJsonTypeInfo), BindingFlags.NonPublic | BindingFlags.Instance)!; -#if NETCOREAPP - return (JsonTypeInfo)methodInfo.MakeGenericMethod(type).Invoke(options, BindingFlags.NonPublic | BindingFlags.DoNotWrapExceptions, null, null, null)!; -#else - try - { - return (JsonTypeInfo)methodInfo.MakeGenericMethod(type).Invoke(options, null)!; - } - catch (TargetInvocationException ex) - { - // Some of the validation is done during construction (i.e. validity of JsonConverter, inner types etc.) - // therefore we need to unwrap TargetInvocationException for better user experience - ExceptionDispatchInfo.Capture(ex.InnerException).Throw(); - throw null!; - } -#endif - } - } - - [RequiresUnreferencedCode(JsonSerializer.SerializationUnreferencedCodeMessage)] - [RequiresDynamicCode(JsonSerializer.SerializationRequiresDynamicCodeMessage)] - private JsonTypeInfo CreateReflectionJsonTypeInfo() - { - return new ReflectionJsonTypeInfo(this); - } - - [RequiresUnreferencedCode(JsonSerializer.SerializationUnreferencedCodeMessage)] - [RequiresDynamicCode(JsonSerializer.SerializationRequiresDynamicCodeMessage)] - private static JsonConverter[] GetDefaultFactoryConverters() - { - return new JsonConverter[] - { - // Check for disallowed types. - new UnsupportedTypeConverterFactory(), - // Nullable converter should always be next since it forwards to any nullable type. - new NullableConverterFactory(), - new EnumConverterFactory(), - new JsonNodeConverterFactory(), - new FSharpTypeConverterFactory(), - // IAsyncEnumerable takes precedence over IEnumerable. - new IAsyncEnumerableConverterFactory(), - // IEnumerable should always be second to last since they can convert any IEnumerable. - new IEnumerableConverterFactory(), - // Object should always be last since it converts any type. - new ObjectConverterFactory() - }; - } - - private static Dictionary GetDefaultSimpleConverters() - { - const int NumberOfSimpleConverters = 26; - var converters = new Dictionary(NumberOfSimpleConverters); - - // Use a dictionary for simple converters. - // When adding to this, update NumberOfSimpleConverters above. - Add(JsonMetadataServices.BooleanConverter); - Add(JsonMetadataServices.ByteConverter); - Add(JsonMetadataServices.ByteArrayConverter); - Add(JsonMetadataServices.CharConverter); - Add(JsonMetadataServices.DateTimeConverter); - Add(JsonMetadataServices.DateTimeOffsetConverter); -#if NETCOREAPP - Add(JsonMetadataServices.DateOnlyConverter); - Add(JsonMetadataServices.TimeOnlyConverter); -#endif - Add(JsonMetadataServices.DoubleConverter); - Add(JsonMetadataServices.DecimalConverter); - Add(JsonMetadataServices.GuidConverter); - Add(JsonMetadataServices.Int16Converter); - Add(JsonMetadataServices.Int32Converter); - Add(JsonMetadataServices.Int64Converter); - Add(JsonMetadataServices.JsonElementConverter); - Add(JsonMetadataServices.JsonDocumentConverter); - Add(JsonMetadataServices.ObjectConverter); - Add(JsonMetadataServices.SByteConverter); - Add(JsonMetadataServices.SingleConverter); - Add(JsonMetadataServices.StringConverter); - Add(JsonMetadataServices.TimeSpanConverter); - Add(JsonMetadataServices.UInt16Converter); - Add(JsonMetadataServices.UInt32Converter); - Add(JsonMetadataServices.UInt64Converter); - Add(JsonMetadataServices.UriConverter); - Add(JsonMetadataServices.VersionConverter); - - Debug.Assert(converters.Count <= NumberOfSimpleConverters); - - return converters; - - void Add(JsonConverter converter) => - converters.Add(converter.TypeToConvert, converter); - } - /// /// The list of custom converters. /// @@ -156,11 +34,11 @@ void Add(JsonConverter converter) => /// public IList PolymorphicTypeConfigurations => _polymorphicTypeConfigurations; - internal JsonConverter GetConverterFromMember(Type? parentClassType, Type propertyType, MemberInfo? memberInfo) + // This may return factory converter + internal JsonConverter? GetCustomConverterFromMember(Type? parentClassType, Type typeToConvert, MemberInfo? memberInfo) { - JsonConverter converter = null!; + JsonConverter? converter = null; - // Priority 1: attempt to get converter from JsonConverterAttribute on property. if (memberInfo != null) { Debug.Assert(parentClassType != null); @@ -170,24 +48,41 @@ internal JsonConverter GetConverterFromMember(Type? parentClassType, Type proper if (converterAttribute != null) { - converter = GetConverterFromAttribute(converterAttribute, typeToConvert: propertyType, classTypeAttributeIsOn: parentClassType!, memberInfo); + converter = GetConverterFromAttribute(converterAttribute, typeToConvert, classTypeAttributeIsOn: parentClassType!, memberInfo); } } - if (converter == null) - { - converter = GetConverterInternal(propertyType); - Debug.Assert(converter != null); - } + return converter; + } + internal JsonConverter GetConverterForType(Type typeToConvert) + { + JsonConverter converter = GetConverterInternal(typeToConvert); + Debug.Assert(converter != null); + + converter = ExpandFactoryConverter(converter, typeToConvert); + + CheckConverterNullabilityIsSameAsPropertyType(converter, typeToConvert); + + return converter; + } + + [return: NotNullIfNotNull("converter")] + internal JsonConverter? ExpandFactoryConverter(JsonConverter? converter, Type typeToConvert) + { if (converter is JsonConverterFactory factory) { - converter = factory.GetConverterInternal(propertyType, this); + converter = factory.GetConverterInternal(typeToConvert, this); // A factory cannot return null; GetConverterInternal checked for that. Debug.Assert(converter != null); } + return converter; + } + + internal static void CheckConverterNullabilityIsSameAsPropertyType(JsonConverter converter, Type propertyType) + { // User has indicated that either: // a) a non-nullable-struct handling converter should handle a nullable struct type or // b) a nullable-struct handling converter should handle a non-nullable struct type. @@ -201,8 +96,6 @@ internal JsonConverter GetConverterFromMember(Type? parentClassType, Type proper { ThrowHelper.ThrowInvalidOperationException_ConverterCanConvertMultipleTypes(propertyType, converter); } - - return converter; } /// @@ -228,7 +121,7 @@ public JsonConverter GetConverter(Type typeToConvert) ThrowHelper.ThrowArgumentNullException(nameof(typeToConvert)); } - RootReflectionSerializerDependencies(); + DefaultJsonTypeInfoResolver.RootDefaultInstance(); return GetConverterInternal(typeToConvert); } @@ -243,12 +136,12 @@ internal JsonConverter GetConverterInternal(Type typeToConvert) return GetConverterFromType(typeToConvert); } - private JsonConverter GetConverterFromType(Type typeToConvert) + internal JsonConverter GetConverterFromType(Type typeToConvert) { Debug.Assert(typeToConvert != null); // Priority 1: If there is a JsonSerializerContext, fetch the converter from there. - JsonConverter? converter = _serializerContext?.GetTypeInfo(typeToConvert)?.PropertyInfoForTypeInfo?.ConverterBase; + JsonConverter? converter = SerializerContext?.GetTypeInfo(typeToConvert)?.Converter; // Priority 2: Attempt to get custom converter added at runtime. // Currently there is not a way at runtime to override the [JsonConverter] when applied to a property. @@ -276,35 +169,7 @@ private JsonConverter GetConverterFromType(Type typeToConvert) // Priority 4: Attempt to get built-in converter. if (converter == null) { - if (s_defaultSimpleConverters == null || s_defaultFactoryConverters == null) - { - // (De)serialization using serializer's options-based methods has not yet occurred, so the built-in converters are not rooted. - // Even though source-gen code paths do not call this method , we do not root all the - // built-in converters here since we fetch converters for any type included for source generation from the binded context (Priority 1). - Debug.Assert(s_defaultSimpleConverters == null); - Debug.Assert(s_defaultFactoryConverters == null); - ThrowHelper.ThrowNotSupportedException_BuiltInConvertersNotRooted(typeToConvert); - return null!; - } - - if (s_defaultSimpleConverters.TryGetValue(typeToConvert, out JsonConverter? foundConverter)) - { - converter = foundConverter; - } - else - { - foreach (JsonConverter item in s_defaultFactoryConverters) - { - if (item.CanConvert(typeToConvert)) - { - converter = item; - break; - } - } - - // Since the object and IEnumerable converters cover all types, we should have a converter. - Debug.Assert(converter != null); - } + converter = DefaultJsonTypeInfoResolver.GetDefaultConverter(typeToConvert); } // Allow redirection for generic types or the enum converter. @@ -376,21 +241,6 @@ private JsonConverter GetConverterFromAttribute(JsonConverterAttribute converter return converter; } - internal bool TryGetDefaultSimpleConverter(Type typeToConvert, [NotNullWhen(true)] out JsonConverter? converter) - { - if (_serializerContext == null && // For consistency do not return any default converters for - // options instances linked to a JsonSerializerContext, - // even if the default converters might have been rooted. - s_defaultSimpleConverters != null && - s_defaultSimpleConverters.TryGetValue(typeToConvert, out converter)) - { - return true; - } - - converter = null; - return false; - } - private static Attribute? GetAttributeThatCanHaveMultiple(Type classType, Type attributeType, MemberInfo memberInfo) { object[] attributes = memberInfo.GetCustomAttributes(attributeType, inherit: false); diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerOptions.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerOptions.cs index e444cc9c27b07..45ae91d6abd88 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerOptions.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/JsonSerializerOptions.cs @@ -1,6 +1,7 @@ // Licensed to the .NET Foundation under one or more agreements. // The .NET Foundation licenses this file to you under the MIT license. +using System.Collections.Generic; using System.ComponentModel; using System.Diagnostics; using System.Diagnostics.CodeAnalysis; @@ -9,7 +10,6 @@ using System.Text.Json.Nodes; using System.Text.Json.Serialization; using System.Text.Json.Serialization.Metadata; -using System.Threading; namespace System.Text.Json { @@ -35,7 +35,7 @@ public sealed partial class JsonSerializerOptions public static JsonSerializerOptions Default { get; } = CreateDefaultImmutableInstance(); // For any new option added, adding it to the options copied in the copy constructor below must be considered. - private JsonSerializerContext? _serializerContext; + private IJsonTypeInfoResolver? _typeInfoResolver; private MemberAccessor? _memberAccessorStrategy; private JsonNamingPolicy? _dictionaryKeyPolicy; private JsonNamingPolicy? _jsonPropertyNamingPolicy; @@ -43,9 +43,7 @@ public sealed partial class JsonSerializerOptions private ReferenceHandler? _referenceHandler; private JavaScriptEncoder? _encoder; private ConfigurationList _converters; -#pragma warning disable CA2252 // This API requires opting into preview features private ConfigurationList _polymorphicTypeConfigurations; -#pragma warning restore CA2252 // This API requires opting into preview features private JsonIgnoreCondition _defaultIgnoreCondition; private JsonNumberHandling _numberHandling; private JsonUnknownTypeHandling _unknownTypeHandling; @@ -60,20 +58,15 @@ public sealed partial class JsonSerializerOptions private bool _propertyNameCaseInsensitive; private bool _writeIndented; + private bool _isLockedInstance; + /// /// Constructs a new instance. /// public JsonSerializerOptions() { - _converters = new ConfigurationList(this); - -#pragma warning disable CA2252 // This API requires opting into preview features - _polymorphicTypeConfigurations = new ConfigurationList(this) - { - OnElementAdded = static config => { config.IsAssignedToOptionsInstance = true; } - }; -#pragma warning restore CA2252 // This API requires opting into preview features - + _converters = new ConverterList(this); + _polymorphicTypeConfigurations = new PolymorphicConfigurationList(this); TrackOptionsInstance(this); } @@ -96,10 +89,8 @@ public JsonSerializerOptions(JsonSerializerOptions options) _jsonPropertyNamingPolicy = options._jsonPropertyNamingPolicy; _readCommentHandling = options._readCommentHandling; _referenceHandler = options._referenceHandler; - _converters = new ConfigurationList(this, options._converters); -#pragma warning disable CA2252 // This API requires opting into preview features - _polymorphicTypeConfigurations = new ConfigurationList(this, options._polymorphicTypeConfigurations); -#pragma warning restore CA2252 // This API requires opting into preview features + _converters = new ConverterList(this, options._converters); + _polymorphicTypeConfigurations = new PolymorphicConfigurationList(this, options._polymorphicTypeConfigurations); _encoder = options._encoder; _defaultIgnoreCondition = options._defaultIgnoreCondition; _numberHandling = options._numberHandling; @@ -114,7 +105,9 @@ public JsonSerializerOptions(JsonSerializerOptions options) _includeFields = options._includeFields; _propertyNameCaseInsensitive = options._propertyNameCaseInsensitive; _writeIndented = options._writeIndented; - + // Preserve backward compatibility with .NET 6 + // This should almost certainly be changed, cf. https://github.com/dotnet/aspnetcore/issues/38720 + _typeInfoResolver = options._typeInfoResolver is JsonSerializerContext ? null : options._typeInfoResolver; EffectiveMaxDepth = options.EffectiveMaxDepth; ReferenceHandlingStrategy = options.ReferenceHandlingStrategy; @@ -166,10 +159,48 @@ public JsonSerializerOptions(JsonSerializerDefaults defaults) : this() { VerifyMutable(); TContext context = new(); - _serializerContext = context; + _typeInfoResolver = context; + _isLockedInstance = true; context._options = this; } + /// + /// Gets or sets JsonTypeInfo resolver. + /// + public IJsonTypeInfoResolver TypeInfoResolver + { + [RequiresUnreferencedCode(JsonSerializer.SerializationUnreferencedCodeMessage)] + [RequiresDynamicCode(JsonSerializer.SerializationRequiresDynamicCodeMessage)] + get + { + return _typeInfoResolver ?? DefaultJsonTypeInfoResolver.RootDefaultInstance(); + } + set + { + VerifyMutable(); + + if (value is null) + { + throw new ArgumentNullException(nameof(value)); + } + + if (value is JsonSerializerContext ctx) + { + if (ctx._options != null && ctx._options != this) + { + // TODO evaluate if this is the appropriate behaviour; + ThrowHelper.ThrowInvalidOperationException_SerializerContextOptionsImmutable(); + } + + // Associate options instance with context and lock for further modification + ctx._options = this; + _isLockedInstance = true; + } + + _typeInfoResolver = value; + } + } + /// /// Defines whether an extra comma at the end of a list of JSON values in an object or array /// is allowed (and ignored) within the JSON payload being deserialized. @@ -559,15 +590,7 @@ public ReferenceHandler? ReferenceHandler } } - internal JsonSerializerContext? JsonSerializerContext - { - get => _serializerContext; - set - { - VerifyMutable(); - _serializerContext = value; - } - } + internal JsonSerializerContext? SerializerContext => _typeInfoResolver as JsonSerializerContext; // The cached value used to determine if ReferenceHandler should use Preserve or IgnoreCycles semanitcs or None of them. internal ReferenceHandlingStrategy ReferenceHandlingStrategy = ReferenceHandlingStrategy.None; @@ -596,37 +619,43 @@ internal MemberAccessor MemberAccessorStrategy } } - /// - /// Whether the options instance has been primed for reflection-based serialization. - /// - internal bool IsInitializedForReflectionSerializer; + internal bool IsInitializedForReflectionSerializer { get; private set; } + // Effective resolver, populated when enacting reflection-based fallback + // Should not be taken into account when calculating options equality. + private IJsonTypeInfoResolver? _effectiveJsonTypeInfoResolver; /// /// Initializes the converters for the reflection-based serializer. - /// must be checked before calling. /// [RequiresUnreferencedCode(JsonSerializer.SerializationUnreferencedCodeMessage)] [RequiresDynamicCode(JsonSerializer.SerializationRequiresDynamicCodeMessage)] internal void InitializeForReflectionSerializer() { - RootReflectionSerializerDependencies(); - Volatile.Write(ref IsInitializedForReflectionSerializer, true); - if (_cachingContext != null) + if (_typeInfoResolver is JsonSerializerContext ctx) + { + // .NET 6 backward compatibility; use fallback to reflection serialization + // TODO: Consider removing this behaviour (needs to be filed as a breaking change). + _effectiveJsonTypeInfoResolver = JsonTypeInfoResolver.Combine(ctx, DefaultJsonTypeInfoResolver.RootDefaultInstance()); + } + else { - _cachingContext.Options.IsInitializedForReflectionSerializer = true; + _typeInfoResolver ??= DefaultJsonTypeInfoResolver.RootDefaultInstance(); } + + if (_cachingContext != null && _cachingContext.Options != this) + { + // We're using a shared caching context deriving from a different options instance; + // for coherence ensure that it has been opted in for reflection-based serialization as well. + _cachingContext.Options.InitializeForReflectionSerializer(); + } + + IsInitializedForReflectionSerializer = true; } private JsonTypeInfo GetJsonTypeInfoFromContextOrCreate(Type type) { - JsonTypeInfo? info = _serializerContext?.GetTypeInfo(type); - if (info == null && IsInitializedForReflectionSerializer) - { - Debug.Assert( - s_typeInfoCreationFunc != null, - "Reflection-based JsonTypeInfo creator should be initialized if IsInitializedForReflectionSerializer is true."); - info = s_typeInfoCreationFunc(type, this); - } + IJsonTypeInfoResolver? resolver = _effectiveJsonTypeInfoResolver ?? _typeInfoResolver; + JsonTypeInfo? info = resolver?.GetTypeInfo(type, this); if (info == null) { @@ -634,6 +663,16 @@ private JsonTypeInfo GetJsonTypeInfoFromContextOrCreate(Type type) return null!; } + if (info.Type != type) + { + ThrowHelper.ThrowInvalidOperationException_ResolverTypeNotCompatible(type, info.Type); + } + + if (info.Options != this) + { + ThrowHelper.ThrowInvalidOperationException_ResolverTypeInfoOptionsNotCompatible(); + } + info.EnsureConfigured(); return info; } @@ -681,16 +720,44 @@ internal JsonWriterOptions GetWriterOptions() internal void VerifyMutable() { - if (_cachingContext != null || _serializerContext != null) + if (_isLockedInstance) + { + ThrowHelper.ThrowInvalidOperationException_SerializerOptionsImmutable(_typeInfoResolver as JsonSerializerContext); + } + } + + private sealed class ConverterList : ConfigurationList + { + private readonly JsonSerializerOptions _options; + + public ConverterList(JsonSerializerOptions options, IList? source = null) + : base(source) { - ThrowHelper.ThrowInvalidOperationException_SerializerOptionsImmutable(_serializerContext); + _options = options; } + + protected override bool IsLockedInstance => _options._isLockedInstance; + protected override void VerifyMutable() => _options.VerifyMutable(); + } + + private sealed class PolymorphicConfigurationList : ConfigurationList + { + private readonly JsonSerializerOptions _options; + + public PolymorphicConfigurationList(JsonSerializerOptions options, IList? source = null) + : base(source) + { + _options = options; + } + + protected override bool IsLockedInstance => _options._isLockedInstance; + protected override void VerifyMutable() => _options.VerifyMutable(); + protected override void OnItemAdded(JsonPolymorphicTypeConfiguration config) => config.IsAssignedToOptionsInstance = true; } private static JsonSerializerOptions CreateDefaultImmutableInstance() { - var options = new JsonSerializerOptions(); - options.InitializeCachingContext(); // eagerly initialize caching context to close type for modification. + var options = new JsonSerializerOptions { _isLockedInstance = true }; return options; } } diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/CustomJsonTypeInfoOfT.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/CustomJsonTypeInfoOfT.cs new file mode 100644 index 0000000000000..db4761754ba50 --- /dev/null +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/CustomJsonTypeInfoOfT.cs @@ -0,0 +1,36 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System.Collections.Generic; +using System.Diagnostics; +using System.Reflection; +using System.Text.Json.Serialization.Converters; + +namespace System.Text.Json.Serialization.Metadata +{ + /// + /// Creates and initializes serialization metadata for a type. + /// + /// + internal sealed class CustomJsonTypeInfo : JsonTypeInfo + { + /// + /// Creates serialization metadata for a type using a simple converter. + /// + internal CustomJsonTypeInfo(JsonSerializerOptions options) + : base(GetEffectiveConverter( + typeof(T), + parentClassType: null, // A TypeInfo never has a "parent" class. + memberInfo: null, // A TypeInfo never has a "parent" property. + options), + options) + { + } + + internal override JsonParameterInfoValues[] GetParameterInfoValues() + { + // Parametrized constructors not supported yet for custom types + return Array.Empty(); + } + } +} diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/DefaultJsonTypeInfoResolver.Converters.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/DefaultJsonTypeInfoResolver.Converters.cs new file mode 100644 index 0000000000000..1ab5bff497acf --- /dev/null +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/DefaultJsonTypeInfoResolver.Converters.cs @@ -0,0 +1,126 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System.Collections.Generic; +using System.Diagnostics; +using System.Diagnostics.CodeAnalysis; +using System.Text.Json.Serialization.Converters; + +namespace System.Text.Json.Serialization.Metadata +{ + public partial class DefaultJsonTypeInfoResolver + { + private static Dictionary? s_defaultSimpleConverters; + private static JsonConverterFactory[]? s_defaultFactoryConverters; + + [RequiresUnreferencedCode(JsonSerializer.SerializationUnreferencedCodeMessage)] + [RequiresDynamicCode(JsonSerializer.SerializationRequiresDynamicCodeMessage)] + private static JsonConverterFactory[] GetDefaultFactoryConverters() + { + return new JsonConverterFactory[] + { + // Check for disallowed types. + new UnsupportedTypeConverterFactory(), + // Nullable converter should always be next since it forwards to any nullable type. + new NullableConverterFactory(), + new EnumConverterFactory(), + new JsonNodeConverterFactory(), + new FSharpTypeConverterFactory(), + // IAsyncEnumerable takes precedence over IEnumerable. + new IAsyncEnumerableConverterFactory(), + // IEnumerable should always be second to last since they can convert any IEnumerable. + new IEnumerableConverterFactory(), + // Object should always be last since it converts any type. + new ObjectConverterFactory() + }; + } + + private static Dictionary GetDefaultSimpleConverters() + { + const int NumberOfSimpleConverters = 26; + var converters = new Dictionary(NumberOfSimpleConverters); + + // Use a dictionary for simple converters. + // When adding to this, update NumberOfSimpleConverters above. + Add(JsonMetadataServices.BooleanConverter); + Add(JsonMetadataServices.ByteConverter); + Add(JsonMetadataServices.ByteArrayConverter); + Add(JsonMetadataServices.CharConverter); + Add(JsonMetadataServices.DateTimeConverter); + Add(JsonMetadataServices.DateTimeOffsetConverter); +#if NETCOREAPP + Add(JsonMetadataServices.DateOnlyConverter); + Add(JsonMetadataServices.TimeOnlyConverter); +#endif + Add(JsonMetadataServices.DoubleConverter); + Add(JsonMetadataServices.DecimalConverter); + Add(JsonMetadataServices.GuidConverter); + Add(JsonMetadataServices.Int16Converter); + Add(JsonMetadataServices.Int32Converter); + Add(JsonMetadataServices.Int64Converter); + Add(JsonMetadataServices.JsonElementConverter); + Add(JsonMetadataServices.JsonDocumentConverter); + Add(JsonMetadataServices.ObjectConverter); + Add(JsonMetadataServices.SByteConverter); + Add(JsonMetadataServices.SingleConverter); + Add(JsonMetadataServices.StringConverter); + Add(JsonMetadataServices.TimeSpanConverter); + Add(JsonMetadataServices.UInt16Converter); + Add(JsonMetadataServices.UInt32Converter); + Add(JsonMetadataServices.UInt64Converter); + Add(JsonMetadataServices.UriConverter); + Add(JsonMetadataServices.VersionConverter); + + Debug.Assert(converters.Count <= NumberOfSimpleConverters); + + return converters; + + void Add(JsonConverter converter) => + converters.Add(converter.TypeToConvert, converter); + } + + internal static JsonConverter GetDefaultConverter(Type typeToConvert) + { + if (s_defaultSimpleConverters == null || s_defaultFactoryConverters == null) + { + // (De)serialization using serializer's options-based methods has not yet occurred, so the built-in converters are not rooted. + // Even though source-gen code paths do not call this method , we do not root all the + // built-in converters here since we fetch converters for any type included for source generation from the binded context (Priority 1). + ThrowHelper.ThrowNotSupportedException_BuiltInConvertersNotRooted(typeToConvert); + return null!; + } + + JsonConverter? converter; + if (s_defaultSimpleConverters.TryGetValue(typeToConvert, out converter)) + { + return converter; + } + else + { + foreach (JsonConverter item in s_defaultFactoryConverters) + { + if (item.CanConvert(typeToConvert)) + { + converter = item; + break; + } + } + + // Since the object and IEnumerable converters cover all types, we should have a converter. + Debug.Assert(converter != null); + return converter; + } + } + + internal static bool TryGetDefaultSimpleConverter(Type typeToConvert, [NotNullWhen(true)] out JsonConverter? converter) + { + if (s_defaultSimpleConverters is null) + { + converter = null; + return false; + } + + return s_defaultSimpleConverters.TryGetValue(typeToConvert, out converter); + } + } +} diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/DefaultJsonTypeInfoResolver.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/DefaultJsonTypeInfoResolver.cs new file mode 100644 index 0000000000000..d0baa8e2e18dc --- /dev/null +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/DefaultJsonTypeInfoResolver.cs @@ -0,0 +1,138 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System.Collections.Generic; +using System.Diagnostics.CodeAnalysis; +using System.Reflection; +using System.Runtime.ExceptionServices; +using System.Threading; + +namespace System.Text.Json.Serialization.Metadata +{ + /// + /// Default JsonTypeInfo resolver. + /// + public partial class DefaultJsonTypeInfoResolver : IJsonTypeInfoResolver + { + private bool _mutable; + + /// + /// Constructs DefaultJsonTypeInfoResolver. + /// + [RequiresUnreferencedCode(JsonSerializer.SerializationUnreferencedCodeMessage)] + [RequiresDynamicCode(JsonSerializer.SerializationRequiresDynamicCodeMessage)] + public DefaultJsonTypeInfoResolver() : this(mutable: true) + { + } + + [RequiresUnreferencedCode(JsonSerializer.SerializationUnreferencedCodeMessage)] + [RequiresDynamicCode(JsonSerializer.SerializationRequiresDynamicCodeMessage)] + private DefaultJsonTypeInfoResolver(bool mutable) + { + _mutable = mutable; + + s_defaultFactoryConverters ??= GetDefaultFactoryConverters(); + s_defaultSimpleConverters ??= GetDefaultSimpleConverters(); + } + + /// + public virtual JsonTypeInfo GetTypeInfo(Type type, JsonSerializerOptions options) + { + if (type == null) + { + throw new ArgumentNullException(nameof(type)); + } + + if (options == null) + { + throw new ArgumentNullException(nameof(options)); + } + + _mutable = false; + + JsonTypeInfo.ValidateType(type, null, null, options); + JsonTypeInfo typeInfo = CreateJsonTypeInfo(type, options); + + if (_modifiers != null) + { + foreach (Action modifier in _modifiers) + { + modifier(typeInfo); + } + } + + return typeInfo; + } + + [UnconditionalSuppressMessage("ReflectionAnalysis", "IL2026:RequiresUnreferencedCode", + Justification = "The ctor is marked RequiresUnreferencedCode.")] + [UnconditionalSuppressMessage("AotAnalysis", "IL3050:RequiresDynamicCode", + Justification = "The ctor is marked RequiresDynamicCode.")] + private JsonTypeInfo CreateJsonTypeInfo(Type type, JsonSerializerOptions options) + { + MethodInfo methodInfo = typeof(DefaultJsonTypeInfoResolver).GetMethod(nameof(CreateReflectionJsonTypeInfo), BindingFlags.NonPublic | BindingFlags.Static)!; +#if NETCOREAPP + return (JsonTypeInfo)methodInfo.MakeGenericMethod(type).Invoke(null, BindingFlags.NonPublic | BindingFlags.DoNotWrapExceptions, null, new[] { options }, null)!; +#else + try + { + return (JsonTypeInfo)methodInfo.MakeGenericMethod(type).Invoke(null, new[] { options })!; + } + catch (TargetInvocationException ex) + { + // Some of the validation is done during construction (i.e. validity of JsonConverter, inner types etc.) + // therefore we need to unwrap TargetInvocationException for better user experience + ExceptionDispatchInfo.Capture(ex.InnerException).Throw(); + throw null!; + } +#endif + } + + [RequiresUnreferencedCode(JsonSerializer.SerializationUnreferencedCodeMessage)] + [RequiresDynamicCode(JsonSerializer.SerializationRequiresDynamicCodeMessage)] + private static JsonTypeInfo CreateReflectionJsonTypeInfo(JsonSerializerOptions options) => new ReflectionJsonTypeInfo(options); + + /// + /// List of JsonTypeInfo modifiers. Modifying callbacks are called consecutively after initial resolution + /// and cannot be changed after GetTypeInfo is called. + /// + public IList> Modifiers => _modifiers ??= new ModifierCollection(this); + private ModifierCollection? _modifiers; + + private sealed class ModifierCollection : ConfigurationList> + { + private readonly DefaultJsonTypeInfoResolver _resolver; + + public ModifierCollection(DefaultJsonTypeInfoResolver resolver) + { + _resolver = resolver; + } + + protected override bool IsLockedInstance => !_resolver._mutable; + protected override void VerifyMutable() + { + if (!_resolver._mutable) + { + ThrowHelper.ThrowInvalidOperationException_TypeInfoResolverImmutable(); + } + } + } + + internal static DefaultJsonTypeInfoResolver? DefaultInstance => s_defaultInstance; + private static DefaultJsonTypeInfoResolver? s_defaultInstance; + + [RequiresUnreferencedCode(JsonSerializer.SerializationUnreferencedCodeMessage)] + [RequiresDynamicCode(JsonSerializer.SerializationRequiresDynamicCodeMessage)] + internal static DefaultJsonTypeInfoResolver RootDefaultInstance() + { + if (s_defaultInstance is DefaultJsonTypeInfoResolver result) + { + return result; + } + + var newInstance = new DefaultJsonTypeInfoResolver(mutable: false); + DefaultJsonTypeInfoResolver? originalInstance = Interlocked.CompareExchange(ref s_defaultInstance, newInstance, comparand: null); + return originalInstance ?? newInstance; + } + } +} diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/IJsonTypeInfoResolver.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/IJsonTypeInfoResolver.cs new file mode 100644 index 0000000000000..22f74387153d4 --- /dev/null +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/IJsonTypeInfoResolver.cs @@ -0,0 +1,24 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System; +using System.Collections.Generic; +using System.Text; +using System.Threading.Tasks; + +namespace System.Text.Json.Serialization.Metadata +{ + /// + /// Exposes method for resolving Type into JsonTypeInfo for given options. + /// + public interface IJsonTypeInfoResolver + { + /// + /// Resolves Type into JsonTypeInfo which defines serialization and deserialization logic. + /// + /// Type to be resolved. + /// JsonSerializerOptions instance defining resolution parameters. + /// Returns JsonTypeInfo instance or null if the resolver cannot produce metadata for this type. + JsonTypeInfo? GetTypeInfo(Type type, JsonSerializerOptions options); + } +} diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonMetadataServices.Converters.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonMetadataServices.Converters.cs index 4b605b7175e59..e39751e1603e5 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonMetadataServices.Converters.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonMetadataServices.Converters.cs @@ -264,7 +264,7 @@ public static JsonConverter GetEnumConverter(JsonSerializerOptions options ThrowHelper.ThrowArgumentNullException(nameof(underlyingTypeInfo)); } - JsonConverter? underlyingConverter = underlyingTypeInfo.PropertyInfoForTypeInfo?.ConverterBase as JsonConverter; + JsonConverter? underlyingConverter = underlyingTypeInfo.Converter as JsonConverter; if (underlyingConverter == null) { throw new InvalidOperationException(SR.Format(SR.SerializationConverterNotCompatible, underlyingConverter, typeof(T))); diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonMetadataServices.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonMetadataServices.cs index 848e45f23c51f..b2954c11e4af2 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonMetadataServices.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonMetadataServices.cs @@ -49,16 +49,6 @@ public static JsonPropertyInfo CreatePropertyInfo(JsonSerializerOptions optio throw new ArgumentException(nameof(propertyInfo.PropertyName)); } - JsonConverter? converter = propertyInfo.Converter; - if (converter == null) - { - converter = propertyTypeInfo.PropertyInfoForTypeInfo.ConverterBase as JsonConverter; - if (converter == null) - { - throw new InvalidOperationException(SR.Format(SR.ConverterForPropertyMustBeValid, declaringType, propertyName, typeof(T))); - } - } - if (!propertyInfo.IsProperty && propertyInfo.IsVirtual) { throw new InvalidOperationException(SR.Format(SR.FieldCannotBeVirtual, nameof(propertyInfo.IsProperty), nameof(propertyInfo.IsVirtual))); diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonParameterInfo.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonParameterInfo.cs index 91a25e0b41393..4d9403b81d022 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonParameterInfo.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonParameterInfo.cs @@ -61,7 +61,7 @@ public virtual void Initialize(JsonParameterInfoValues parameterInfo, JsonProper PropertyType = matchingProperty.PropertyType; NameAsUtf8Bytes = matchingProperty.NameAsUtf8Bytes!; - ConverterBase = matchingProperty.ConverterBase; + ConverterBase = matchingProperty.EffectiveConverter; IgnoreDefaultValuesOnRead = matchingProperty.IgnoreDefaultValuesOnRead; NumberHandling = matchingProperty.EffectiveNumberHandling; MatchingPropertyCanBeNull = matchingProperty.PropertyTypeCanBeNull; diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonPropertyInfo.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonPropertyInfo.cs index 8c9236cbdd35b..ea1e76a429814 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonPropertyInfo.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonPropertyInfo.cs @@ -13,7 +13,6 @@ namespace System.Text.Json.Serialization.Metadata /// Provides JSON serialization-related metadata about a property or field. /// [DebuggerDisplay("{DebuggerDisplay,nq}")] - [EditorBrowsable(EditorBrowsableState.Never)] public abstract class JsonPropertyInfo { internal static readonly JsonPropertyInfo s_missingProperty = GetPropertyPlaceholder(); @@ -22,7 +21,72 @@ public abstract class JsonPropertyInfo internal ConverterStrategy ConverterStrategy; - internal abstract JsonConverter ConverterBase { get; set; } + /// + /// Converter resolved from PropertyType and not taking in consideration any custom attributes or custom settings. + /// - for reflection we store the original value since we need it in order to construct typed JsonPropertyInfo + /// - for source gen it remains null, we will initialize it only if someone used resolver to remove CustomConverter + /// + internal JsonConverter? NonCustomConverter { get; set; } + + /// + /// Converter after applying CustomConverter (i.e. JsonConverterAttribute) + /// + internal abstract JsonConverter EffectiveConverter { get; set; } + + /// + /// Custom converter override at the property level, equivalent to JsonConverterAttribute annotation + /// + public JsonConverter? CustomConverter + { + get => _customConverter; + set + { + CheckMutable(); + _customConverter = value; + } + } + + private JsonConverter? _customConverter; + + /// + /// Getter delegate. Property cannot be serialized without it. + /// + public Func? Get + { + get => _untypedGet; + set => SetGetter(value); + } + + /// + /// Setter delegate. Property cannot be deserialized without it. + /// + public Action? Set + { + get => _untypedSet; + set => SetSetter(value); + } + + private protected Func? _untypedGet; + private protected Action? _untypedSet; + + private protected abstract void SetGetter(Delegate? getter); + private protected abstract void SetSetter(Delegate? setter); + + /// + /// Decides if property with given declaring object and property value should be serialized. + /// If not set it is equivalent to always returning true. + /// + public Func? ShouldSerialize + { + get => _shouldSerialize; + set + { + CheckMutable(); + _shouldSerialize = value; + } + } + + private Func? _shouldSerialize; internal JsonPropertyInfo() { @@ -33,8 +97,8 @@ internal static JsonPropertyInfo GetPropertyPlaceholder() JsonPropertyInfo info = new JsonPropertyInfo(); Debug.Assert(!info.IsForTypeInfo); - Debug.Assert(!info.ShouldDeserialize); - Debug.Assert(!info.ShouldSerialize); + Debug.Assert(!info.CanDeserialize); + Debug.Assert(!info.CanSerialize); info.Name = string.Empty; @@ -57,29 +121,42 @@ internal static JsonPropertyInfo CreateIgnoredPropertyPlaceholder( jsonPropertyInfo.IsVirtual = isVirtual; jsonPropertyInfo.DeterminePropertyName(); - Debug.Assert(!jsonPropertyInfo.ShouldDeserialize); - Debug.Assert(!jsonPropertyInfo.ShouldSerialize); + Debug.Assert(!jsonPropertyInfo.CanDeserialize); + Debug.Assert(!jsonPropertyInfo.CanSerialize); return jsonPropertyInfo; } - internal Type PropertyType { get; set; } = null!; + /// + /// Type associated with JsonPropertyInfo + /// + public Type PropertyType { get; private protected set; } = null!; + + private protected void CheckMutable() + { + if (_isConfigured) + { + ThrowHelper.ThrowInvalidOperationException_PropertyInfoImmutable(); + } + } private bool _isConfigured; - internal void EnsureConfigured() + internal void EnsureConfigured(JsonTypeInfo typeInfo) { if (_isConfigured) { return; } - Configure(); + Configure(typeInfo); _isConfigured = true; } - internal virtual void Configure() + internal virtual void Configure(JsonTypeInfo typeInfo) { + DeclaringTypeNumberHandling = typeInfo.NumberHandling; + if (!IsForTypeInfo) { CacheNameAsUtf8BytesAndEscapedNameSection(); @@ -90,6 +167,8 @@ internal virtual void Configure() return; } + SetEffectiveConverter(); + if (IsForTypeInfo) { DetermineNumberHandlingForTypeInfo(); @@ -103,6 +182,8 @@ internal virtual void Configure() } } + internal abstract void SetEffectiveConverter(); + internal void GetPolicies() { Debug.Assert(MemberInfo != null); @@ -161,7 +242,7 @@ internal void CacheNameAsUtf8BytesAndEscapedNameSection() internal void DetermineSerializationCapabilities(JsonIgnoreCondition? ignoreCondition) { - Debug.Assert(MemberType == MemberTypes.Property || MemberType == MemberTypes.Field); + Debug.Assert(MemberType == MemberTypes.Property || MemberType == MemberTypes.Field || MemberType == default); if ((ConverterStrategy & (ConverterStrategy.Enumerable | ConverterStrategy.Dictionary)) == 0) { @@ -176,22 +257,22 @@ internal void DetermineSerializationCapabilities(JsonIgnoreCondition? ignoreCond : !Options.IgnoreReadOnlyFields); // We serialize if there is a getter + not ignoring readonly properties. - ShouldSerialize = HasGetter && (HasSetter || serializeReadOnlyProperty); + CanSerialize = HasGetter && (HasSetter || serializeReadOnlyProperty); // We deserialize if there is a setter. - ShouldDeserialize = HasSetter; + CanDeserialize = HasSetter; } else { if (HasGetter) { - Debug.Assert(ConverterBase != null); + Debug.Assert(EffectiveConverter != null); - ShouldSerialize = true; + CanSerialize = true; if (HasSetter) { - ShouldDeserialize = true; + CanDeserialize = true; } } } @@ -249,7 +330,7 @@ internal void DetermineIgnoreCondition(JsonIgnoreCondition? ignoreCondition) internal void DetermineNumberHandlingForTypeInfo() { - if (DeclaringTypeNumberHandling != null && DeclaringTypeNumberHandling != JsonNumberHandling.Strict && !ConverterBase.IsInternalConverter) + if (DeclaringTypeNumberHandling != null && DeclaringTypeNumberHandling != JsonNumberHandling.Strict && !EffectiveConverter.IsInternalConverter) { ThrowHelper.ThrowInvalidOperationException_NumberHandlingOnPropertyInvalid(this); } @@ -276,8 +357,8 @@ internal void DetermineNumberHandlingForProperty() if (numberHandlingIsApplicable) { - // Priority 1: Get handling from attribute on property/field, or its parent class type. - JsonNumberHandling? handling = NumberHandling ?? DeclaringTypeNumberHandling; + // Priority 1: Get handling from attribute on property/field, it's parent class type or property type. + JsonNumberHandling? handling = NumberHandling ?? DeclaringTypeNumberHandling ?? JsonTypeInfo.NumberHandling; // Priority 2: Get handling from JsonSerializerOptions instance. if (!handling.HasValue && Options.NumberHandling != JsonNumberHandling.Strict) @@ -295,21 +376,21 @@ internal void DetermineNumberHandlingForProperty() private bool NumberHandingIsApplicable() { - if (ConverterBase.IsInternalConverterForNumberType) + if (EffectiveConverter.IsInternalConverterForNumberType) { return true; } Type potentialNumberType; - if (!ConverterBase.IsInternalConverter || + if (!EffectiveConverter.IsInternalConverter || ((ConverterStrategy.Enumerable | ConverterStrategy.Dictionary) & ConverterStrategy) == 0) { potentialNumberType = PropertyType; } else { - Debug.Assert(ConverterBase.ElementType != null); - potentialNumberType = ConverterBase.ElementType; + Debug.Assert(EffectiveConverter.ElementType != null); + potentialNumberType = EffectiveConverter.ElementType; } potentialNumberType = Nullable.GetUnderlyingType(potentialNumberType) ?? potentialNumberType; @@ -349,16 +430,16 @@ internal string GetDebugInfo(int indent = 0) sb.AppendLine($"{ind} NameAsUtf8.Length: {(NameAsUtf8Bytes?.Length ?? -1)},"); sb.AppendLine($"{ind} IsConfigured: {_isConfigured},"); sb.AppendLine($"{ind} IsIgnored: {IsIgnored},"); - sb.AppendLine($"{ind} ShouldSerialize: {ShouldSerialize},"); - sb.AppendLine($"{ind} ShouldDeserialize: {ShouldDeserialize},"); + sb.AppendLine($"{ind} CanSerialize: {CanSerialize},"); + sb.AppendLine($"{ind} CanDeserialize: {CanDeserialize},"); sb.AppendLine($"{ind}}}"); return sb.ToString(); } #endif - internal bool HasGetter { get; set; } - internal bool HasSetter { get; set; } + internal bool HasGetter => _untypedGet is not null; + internal bool HasSetter => _untypedSet is not null; internal abstract void Initialize( Type parentClassType, @@ -369,7 +450,8 @@ internal abstract void Initialize( JsonConverter converter, JsonIgnoreCondition? ignoreCondition, JsonSerializerOptions options, - JsonTypeInfo? jsonTypeInfo = null); + JsonTypeInfo? jsonTypeInfo = null, + bool isCustomProperty = false); internal bool IgnoreDefaultValuesOnRead { get; private set; } internal bool IgnoreDefaultValuesOnWrite { get; private set; } @@ -385,12 +467,28 @@ internal abstract void Initialize( // 3) EscapedNameSection. The escaped verson of NameAsUtf8Bytes plus the wrapping quotes and a trailing colon. Used during serialization. /// - /// The unescaped name of the property. - /// Is either the actual CLR property name, + /// The name of the property. + /// It's either the actual CLR property name, /// the value specified in JsonPropertyNameAttribute, - /// or the value returned from PropertyNamingPolicy(clrPropertyName). + /// or the value returned from PropertyNamingPolicy. /// - internal string Name { get; set; } = null!; + public string Name + { + get => _name; + set + { + CheckMutable(); + + if (value == null) + { + throw new ArgumentNullException(nameof(value)); + } + + _name = value; + } + } + + private string _name = null!; /// /// Utf8 version of Name. @@ -402,7 +500,10 @@ internal abstract void Initialize( /// internal byte[] EscapedNameSection { get; set; } = null!; - internal JsonSerializerOptions Options { get; set; } = null!; // initialized in Init method + /// + /// Options associated with JsonPropertyInfo + /// + public JsonSerializerOptions Options { get; internal set; } = null!; // initialized in Init method /// /// The property order. @@ -441,7 +542,7 @@ internal bool ReadJsonAndAddExtensionProperty( { // Avoid a type reference to JsonObject and its converter to support trimming. Debug.Assert(propValue is Nodes.JsonObject); - ConverterBase.ReadElementAndSetProperty(propValue, state.Current.JsonPropertyNameAsString!, ref reader, Options, ref state); + EffectiveConverter.ReadElementAndSetProperty(propValue, state.Current.JsonPropertyNameAsString!, ref reader, Options, ref state); } return true; @@ -453,7 +554,7 @@ JsonConverter GetDictionaryValueConverter(Type dictionaryValueType) if (dictionaryValueInfo != null) { // Fast path when there is a generic type such as Dictionary<,>. - converter = dictionaryValueInfo.PropertyInfoForTypeInfo.ConverterBase; + converter = dictionaryValueInfo.Converter; } else { @@ -528,9 +629,9 @@ internal JsonTypeInfo JsonTypeInfo internal abstract void SetExtensionDictionaryAsObject(object obj, object? extensionDict); - internal bool ShouldSerialize { get; set; } + internal bool CanSerialize { get; set; } - internal bool ShouldDeserialize { get; set; } + internal bool CanDeserialize { get; set; } internal bool IsIgnored { get; set; } @@ -557,7 +658,17 @@ internal JsonTypeInfo JsonTypeInfo /// /// Number handling specific to this property, i.e. set by attribute /// - internal JsonNumberHandling? NumberHandling { get; set; } + public JsonNumberHandling? NumberHandling + { + get => _numberHandling; + set + { + CheckMutable(); + _numberHandling = value; + } + } + + private JsonNumberHandling? _numberHandling; /// /// Number handling after considering options and declaring type number handling diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonPropertyInfoOfT.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonPropertyInfoOfT.cs index 51bf1f0a343bf..b9a97e3bcbf6d 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonPropertyInfoOfT.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonPropertyInfoOfT.cs @@ -27,16 +27,75 @@ internal sealed class JsonPropertyInfo : JsonPropertyInfo // the property's type, we track that and whether the property type can be null. private bool _propertyTypeEqualsTypeToConvert; - internal Func? Get { get; set; } + private Func? _typedGet; + private Action? _typedSet; - internal Action? Set { get; set; } + internal new Func? Get + { + get => _typedGet; + set => SetGetter(value); + } + + internal new Action? Set + { + get => _typedSet; + set => SetSetter(value); + } + + private protected override void SetGetter(Delegate? getter) + { + Debug.Assert(getter is null or Func or Func); + + CheckMutable(); + + if (getter is null) + { + _typedGet = null; + _untypedGet = null; + } + else if (getter is Func typedGetter) + { + _typedGet = typedGetter; + _untypedGet = getter is Func untypedGet ? untypedGet : obj => typedGetter(obj); + } + else + { + Func untypedGet = (Func)getter; + _typedGet = (obj => (T)untypedGet(obj)!); + _untypedGet = untypedGet; + } + } + + private protected override void SetSetter(Delegate? setter) + { + Debug.Assert(setter is null or Action or Action); + + CheckMutable(); + + if (setter is null) + { + _typedSet = null; + _untypedSet = null; + } + else if (setter is Action typedSetter) + { + _typedSet = typedSetter; + _untypedSet = setter is Action untypedSet ? untypedSet : (obj, value) => typedSetter(obj, (T)value!); + } + else + { + Action untypedSet = (Action)setter; + _typedSet = ((obj, value) => untypedSet(obj, (T)value!)); + _untypedSet = untypedSet; + } + } internal override object? DefaultValue => default(T); - public JsonConverter Converter { get; internal set; } = null!; + internal JsonConverter TypedEffectiveConverter { get; set; } = null!; internal override void Initialize( - Type parentClassType, + Type declaringType, Type declaredPropertyType, ConverterStrategy converterStrategy, MemberInfo? memberInfo, @@ -44,7 +103,8 @@ internal override void Initialize( JsonConverter converter, JsonIgnoreCondition? ignoreCondition, JsonSerializerOptions options, - JsonTypeInfo? jsonTypeInfo = null) + JsonTypeInfo? jsonTypeInfo = null, + bool isCustomProperty = false) { Debug.Assert(converter != null); @@ -55,9 +115,9 @@ internal override void Initialize( JsonTypeInfo = jsonTypeInfo; } - ConverterBase = converter; + NonCustomConverter = converter; Options = options; - DeclaringType = parentClassType; + DeclaringType = declaringType; MemberInfo = memberInfo; IsVirtual = isVirtual; IgnoreCondition = ignoreCondition; @@ -73,14 +133,12 @@ internal override void Initialize( MethodInfo? getMethod = propertyInfo.GetMethod; if (getMethod != null && (getMethod.IsPublic || useNonPublicAccessors)) { - HasGetter = true; Get = options.MemberAccessorStrategy.CreatePropertyGetter(propertyInfo); } MethodInfo? setMethod = propertyInfo.SetMethod; if (setMethod != null && (setMethod.IsPublic || useNonPublicAccessors)) { - HasSetter = true; Set = options.MemberAccessorStrategy.CreatePropertySetter(propertyInfo); } @@ -93,12 +151,10 @@ internal override void Initialize( { Debug.Assert(fieldInfo.IsPublic); - HasGetter = true; Get = options.MemberAccessorStrategy.CreateFieldGetter(fieldInfo); if (!fieldInfo.IsInitOnly) { - HasSetter = true; Set = options.MemberAccessorStrategy.CreateFieldSetter(fieldInfo); } @@ -116,11 +172,9 @@ internal override void Initialize( GetPolicies(); } - else + else if (!isCustomProperty) { IsForTypeInfo = true; - HasGetter = true; - HasSetter = true; } } @@ -155,60 +209,76 @@ internal void InitializeForSourceGen(JsonSerializerOptions options, JsonProperty JsonTypeInfo propertyTypeInfo = propertyInfo.PropertyTypeInfo; Type declaringType = propertyInfo.DeclaringType; - JsonConverter? converter = propertyInfo.Converter; - if (converter == null) + JsonConverter? typedCustomConverter = propertyInfo.Converter; + CustomConverter = typedCustomConverter; + + JsonConverter? typedNonCustomConverter = propertyTypeInfo.Converter as JsonConverter; + NonCustomConverter = typedNonCustomConverter; + JsonConverter? typedEffectiveConverter = typedCustomConverter ?? typedNonCustomConverter; + if (typedEffectiveConverter == null) { - converter = propertyTypeInfo.PropertyInfoForTypeInfo.ConverterBase as JsonConverter; - if (converter == null) - { - throw new InvalidOperationException(SR.Format(SR.ConverterForPropertyMustBeValid, declaringType, ClrName, typeof(T))); - } + throw new InvalidOperationException(SR.Format(SR.ConverterForPropertyMustBeValid, declaringType, ClrName, typeof(T))); } - ConverterBase = converter; - if (propertyInfo.IgnoreCondition == JsonIgnoreCondition.Always) { IsIgnored = true; - Debug.Assert(!ShouldSerialize); - Debug.Assert(!ShouldDeserialize); + Debug.Assert(!CanSerialize); + Debug.Assert(!CanDeserialize); } else { Get = propertyInfo.Getter!; Set = propertyInfo.Setter; - HasGetter = Get != null; - HasSetter = Set != null; JsonTypeInfo = propertyTypeInfo; DeclaringType = declaringType; IgnoreCondition = propertyInfo.IgnoreCondition; MemberType = propertyInfo.IsProperty ? MemberTypes.Property : MemberTypes.Field; - ConverterStrategy = Converter!.ConverterStrategy; + ConverterStrategy = typedEffectiveConverter.ConverterStrategy; NumberHandling = propertyInfo.NumberHandling; } } - internal override void Configure() + internal override void Configure(JsonTypeInfo typeInfo) { - base.Configure(); + base.Configure(typeInfo); if (!IsForTypeInfo && !IsIgnored) { - _converterIsExternalAndPolymorphic = !ConverterBase.IsInternalConverter && PropertyType != ConverterBase.TypeToConvert; + _converterIsExternalAndPolymorphic = !EffectiveConverter.IsInternalConverter && PropertyType != EffectiveConverter.TypeToConvert; _propertyTypeEqualsTypeToConvert = typeof(T) == PropertyType; } } - internal override JsonConverter ConverterBase + internal override void SetEffectiveConverter() + { + JsonConverter? customConverter = CustomConverter; + if (customConverter != null) + { + customConverter = Options.ExpandFactoryConverter(customConverter, PropertyType); + JsonSerializerOptions.CheckConverterNullabilityIsSameAsPropertyType(customConverter, PropertyType); + } + + JsonConverter effectiveConverter = customConverter ?? NonCustomConverter ?? Options.GetConverterForType(PropertyType); + if (effectiveConverter.TypeToConvert == PropertyType) + { + EffectiveConverter = effectiveConverter; + } + else + { + EffectiveConverter = effectiveConverter.CreateCastingConverter(); + } + } + + internal override JsonConverter EffectiveConverter { get { - return Converter; + return TypedEffectiveConverter; } set { - Debug.Assert(value is JsonConverter); - Converter = (JsonConverter)value; + TypedEffectiveConverter = (JsonConverter)value; } } @@ -227,11 +297,21 @@ internal override bool GetMemberAndWriteJson(object obj, ref WriteStack state, U { T value = Get!(obj); + if (ShouldSerialize != null) + { + if (!ShouldSerialize(obj, value)) + { + // We return true here. + // False means that there is not enough data. + return true; + } + } + if ( #if NETCOREAPP !typeof(T).IsValueType && // treated as a constant by recent versions of the JIT. #else - !Converter.IsValueType && + !TypedEffectiveConverter.IsValueType && #endif Options.ReferenceHandlingStrategy == ReferenceHandlingStrategy.IgnoreCycles && value is not null && @@ -280,7 +360,7 @@ value is not null && { Debug.Assert(PropertyTypeCanBeNull); - if (Converter.HandleNullOnWrite) + if (TypedEffectiveConverter.HandleNullOnWrite) { if (state.Current.PropertyState < StackFramePropertyState.Name) { @@ -289,10 +369,10 @@ value is not null && } int originalDepth = writer.CurrentDepth; - Converter.Write(writer, value, Options); + TypedEffectiveConverter.Write(writer, value, Options); if (originalDepth != writer.CurrentDepth) { - ThrowHelper.ThrowJsonException_SerializationConverterWrite(Converter); + ThrowHelper.ThrowJsonException_SerializationConverterWrite(TypedEffectiveConverter); } } else @@ -310,7 +390,7 @@ value is not null && writer.WritePropertyNameSection(EscapedNameSection); } - return Converter.TryWrite(writer, value, Options, ref state); + return TypedEffectiveConverter.TryWrite(writer, value, Options, ref state); } } @@ -319,13 +399,23 @@ internal override bool GetMemberAndWriteJsonExtensionData(object obj, ref WriteS bool success; T value = Get!(obj); + if (ShouldSerialize != null) + { + if (!ShouldSerialize(obj, value)) + { + // We return true here. + // False means that there is not enough data. + return true; + } + } + if (value == null) { success = true; } else { - success = Converter.TryWriteDataExtensionProperty(writer, value, Options, ref state); + success = TypedEffectiveConverter.TryWriteDataExtensionProperty(writer, value, Options, ref state); } return success; @@ -336,11 +426,11 @@ internal override bool ReadJsonAndSetMember(object obj, ref ReadStack state, ref bool success; bool isNullToken = reader.TokenType == JsonTokenType.Null; - if (isNullToken && !Converter.HandleNullOnRead && !state.IsContinuation) + if (isNullToken && !TypedEffectiveConverter.HandleNullOnRead && !state.IsContinuation) { if (!PropertyTypeCanBeNull) { - ThrowHelper.ThrowJsonException_DeserializeUnableToConvertValue(Converter.TypeToConvert); + ThrowHelper.ThrowJsonException_DeserializeUnableToConvertValue(TypedEffectiveConverter.TypeToConvert); } Debug.Assert(default(T) == null); @@ -353,7 +443,7 @@ internal override bool ReadJsonAndSetMember(object obj, ref ReadStack state, ref success = true; } - else if (Converter.CanUseDirectReadOrWrite && state.Current.NumberHandling == null) + else if (TypedEffectiveConverter.CanUseDirectReadOrWrite && state.Current.NumberHandling == null) { // CanUseDirectReadOrWrite == false when using streams Debug.Assert(!state.IsContinuation); @@ -361,7 +451,7 @@ internal override bool ReadJsonAndSetMember(object obj, ref ReadStack state, ref if (!isNullToken || !IgnoreDefaultValuesOnRead || !PropertyTypeCanBeNull) { // Optimize for internal converters by avoiding the extra call to TryRead. - T? fastValue = Converter.Read(ref reader, PropertyType, Options); + T? fastValue = TypedEffectiveConverter.Read(ref reader, PropertyType, Options); Set!(obj, fastValue!); } @@ -372,7 +462,7 @@ internal override bool ReadJsonAndSetMember(object obj, ref ReadStack state, ref success = true; if (!isNullToken || !IgnoreDefaultValuesOnRead || !PropertyTypeCanBeNull || state.IsContinuation) { - success = Converter.TryRead(ref reader, PropertyType, Options, ref state, out T? value); + success = TypedEffectiveConverter.TryRead(ref reader, PropertyType, Options, ref state, out T? value); if (success) { #if !DEBUG @@ -405,11 +495,11 @@ internal override bool ReadJsonAsObject(ref ReadStack state, ref Utf8JsonReader { bool success; bool isNullToken = reader.TokenType == JsonTokenType.Null; - if (isNullToken && !Converter.HandleNullOnRead && !state.IsContinuation) + if (isNullToken && !TypedEffectiveConverter.HandleNullOnRead && !state.IsContinuation) { if (!PropertyTypeCanBeNull) { - ThrowHelper.ThrowJsonException_DeserializeUnableToConvertValue(Converter.TypeToConvert); + ThrowHelper.ThrowJsonException_DeserializeUnableToConvertValue(TypedEffectiveConverter.TypeToConvert); } value = default(T); @@ -418,17 +508,17 @@ internal override bool ReadJsonAsObject(ref ReadStack state, ref Utf8JsonReader else { // Optimize for internal converters by avoiding the extra call to TryRead. - if (Converter.CanUseDirectReadOrWrite && state.Current.NumberHandling == null) + if (TypedEffectiveConverter.CanUseDirectReadOrWrite && state.Current.NumberHandling == null) { // CanUseDirectReadOrWrite == false when using streams Debug.Assert(!state.IsContinuation); - value = Converter.Read(ref reader, PropertyType, Options); + value = TypedEffectiveConverter.Read(ref reader, PropertyType, Options); success = true; } else { - success = Converter.TryRead(ref reader, PropertyType, Options, ref state, out T? typedValue); + success = TypedEffectiveConverter.TryRead(ref reader, PropertyType, Options, ref state, out T? typedValue); value = typedValue; } } diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfo.Cache.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfo.Cache.cs index 5688406084063..059e3279decb8 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfo.Cache.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfo.Cache.cs @@ -10,7 +10,7 @@ namespace System.Text.Json.Serialization.Metadata { - public partial class JsonTypeInfo + public abstract partial class JsonTypeInfo { /// /// Cached typeof(object). It is faster to cache this than to call typeof(object) multiple times. @@ -59,7 +59,9 @@ internal static JsonPropertyInfo CreateProperty( JsonConverter converter, JsonSerializerOptions options, JsonIgnoreCondition? ignoreCondition = null, - JsonTypeInfo? jsonTypeInfo = null) + JsonTypeInfo? jsonTypeInfo = null, + JsonConverter? customConverter = null, + bool isCustomProperty = false) { // Create the JsonPropertyInfo instance. JsonPropertyInfo jsonPropertyInfo = converter.CreateJsonPropertyInfo(); @@ -73,7 +75,10 @@ internal static JsonPropertyInfo CreateProperty( converter, ignoreCondition, options, - jsonTypeInfo); + jsonTypeInfo, + isCustomProperty: isCustomProperty); + + jsonPropertyInfo.CustomConverter = customConverter; return jsonPropertyInfo; } diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfo.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfo.cs index 3d8932b19fd05..1b76242d9614d 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfo.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfo.cs @@ -14,18 +14,66 @@ namespace System.Text.Json.Serialization.Metadata /// /// Provides JSON serialization-related metadata about a type. /// - /// This API is for use by the output of the System.Text.Json source generator and should not be called directly. [DebuggerDisplay("{DebuggerDisplay,nq}")] - [EditorBrowsable(EditorBrowsableState.Never)] - public partial class JsonTypeInfo + public abstract partial class JsonTypeInfo { internal const string JsonObjectTypeName = "System.Text.Json.Nodes.JsonObject"; - internal delegate object? ConstructorDelegate(); - internal delegate T ParameterizedConstructorDelegate(TArg0 arg0, TArg1 arg1, TArg2 arg2, TArg3 arg3); - internal ConstructorDelegate? CreateObject { get; set; } + private JsonPropertyDictionary.ValueList? _properties; + + /// + /// Object constructor. If set to null type is not deserializable. + /// + public Func? CreateObject + { + get => _createObject; + set + { + SetCreateObject(value); + } + } + + private protected abstract void SetCreateObject(Delegate? createObject, bool useForExtensionDataProperty = false); + private protected Func? _createObject; + + // this is only assigned if Kind == None + internal Func? CreateObjectForExtensionDataProperty { get; set; } + + /// + /// Gets JsonPropertyInfo list. Only applicable when Kind is Object. + /// + public IList Properties + { + get + { + if (_properties != null) + { + return _properties; + } + + if (Kind == JsonTypeInfoKind.Object) + { + // We need to ensure SourceGen had a chance to add properties + LateAddProperties(); + } + + PropertyCache ??= CreatePropertyCache(capacity: 0); + + if (_isConfigured || Kind != JsonTypeInfoKind.Object) + { + // We do not pass getKey to ensure list is read-only + _properties = PropertyCache.CreateValueList(getKey: null); + } + else + { + _properties = PropertyCache.CreateValueList(getKey: (prop) => prop.Name); + } + + return _properties; + } + } internal object? CreateObjectWithArgs { get; set; } @@ -51,7 +99,7 @@ internal void ValidateCanBeUsedForDeserialization() { if (ThrowOnDeserialize) { - ThrowHelper.ThrowInvalidOperationException_NoMetadataForTypeProperties(Options.JsonSerializerContext, Type); + ThrowHelper.ThrowInvalidOperationException_NoMetadataForTypeProperties(Options.SerializerContext, Type); } } @@ -134,9 +182,30 @@ internal JsonTypeInfo? KeyTypeInfo internal Type? KeyType { get; set; } - internal JsonSerializerOptions Options { get; set; } + /// + /// Options associated with JsonTypeInfo + /// + public JsonSerializerOptions Options { get; private set; } - internal Type Type { get; private set; } + /// + /// Type associated with JsonTypeInfo + /// + public Type Type { get; private set; } + + /// + /// Converter associated with the type for the given options instance + /// + public JsonConverter Converter + // For JsonTypeInfo CustomConverter is always null + // while NonCustomConverter always contains final converter. + // This property can be used before JsonTypeInfo is configured (especially in SourceGen case) + // therefore it's safer to return NonCustomConverter rather than EffectiveConverter. + => PropertyInfoForTypeInfo.NonCustomConverter!; + + /// + /// Determines the kind of contract metadata current JsonTypeInfo instance is customizing + /// + public JsonTypeInfoKind Kind { get; private set; } /// /// The JsonPropertyInfo for this JsonTypeInfo. It is used to obtain the converter for the TypeInfo. @@ -162,7 +231,20 @@ internal JsonTypeInfo? KeyTypeInfo internal DefaultValueHolder DefaultValueHolder => _defaultValueHolder ??= DefaultValueHolder.CreateHolder(Type); private DefaultValueHolder? _defaultValueHolder; - internal JsonNumberHandling? NumberHandling { get; set; } + /// + /// Type specific value overriding JsonSerializerOptions NumberHandling. For DefaultJsonTypeInfoResolver it is equivalent to JsonNumberHandlingAttribute value. + /// + public JsonNumberHandling? NumberHandling + { + get => _numberHandling; + set + { + CheckMutable(); + _numberHandling = value; + } + } + + private JsonNumberHandling? _numberHandling; internal JsonTypeInfo(Type type, JsonConverter converter, JsonSerializerOptions options) { @@ -192,6 +274,16 @@ internal JsonTypeInfo(Type type, JsonConverter converter, JsonSerializerOptions Debug.Fail($"Unexpected class type: {PropertyInfoForTypeInfo.ConverterStrategy}"); throw new InvalidOperationException(); } + + Kind = GetTypeInfoKind(type, PropertyInfoForTypeInfo.ConverterStrategy); + } + + private protected void CheckMutable() + { + if (_isConfigured) + { + ThrowHelper.ThrowInvalidOperationException_TypeInfoImmutable(); + } } private volatile bool _isConfigured; @@ -216,34 +308,50 @@ internal void EnsureConfigured() internal virtual void Configure() { Debug.Assert(Monitor.IsEntered(_configureLock), "Configure called directly, use EnsureConfigured which locks this method"); - JsonConverter converter = PropertyInfoForTypeInfo.ConverterBase; - Debug.Assert(PropertyInfoForTypeInfo.ConverterStrategy == PropertyInfoForTypeInfo.ConverterBase.ConverterStrategy, - $"ConverterStrategy from PropertyInfoForTypeInfo.ConverterStrategy ({PropertyInfoForTypeInfo.ConverterStrategy}) does not match converter's ({PropertyInfoForTypeInfo.ConverterBase.ConverterStrategy})"); + + PropertyInfoForTypeInfo.EnsureConfigured(this); + + JsonConverter converter = Converter; + Debug.Assert(PropertyInfoForTypeInfo.ConverterStrategy == Converter.ConverterStrategy, + $"ConverterStrategy from PropertyInfoForTypeInfo.ConverterStrategy ({PropertyInfoForTypeInfo.ConverterStrategy}) does not match converter's ({Converter.ConverterStrategy})"); converter.ConfigureJsonTypeInfo(this, Options); - PropertyInfoForTypeInfo.DeclaringTypeNumberHandling = NumberHandling; - PropertyInfoForTypeInfo.EnsureConfigured(); - // Source gen currently when initializes properties - // also assigns JsonPropertyInfo's JsonTypeInfo which causes SO if there are any - // cycles in the object graph. For that reason properties cannot be added immediately. - // This is a no-op for ReflectionJsonTypeInfo - LateAddProperties(); + if (_properties != null) + { + // If user tried to access Properties for something else than JsonTypeInfoKind.Object + // Properties will already be read-only + if (!_properties.IsReadOnly) + { + _properties.FinishEditingAndMakeReadOnly(); + } + } + else + { + // Resolver didn't modify properties + + // Source gen currently when initializes properties + // also assigns JsonPropertyInfo's JsonTypeInfo which causes SO if there are any + // cycles in the object graph. For that reason properties cannot be added immediately. + // This is a no-op for ReflectionJsonTypeInfo + LateAddProperties(); + } - DataExtensionProperty?.EnsureConfigured(); + DataExtensionProperty?.EnsureConfigured(this); - if (converter.ConverterStrategy == ConverterStrategy.Object && PropertyCache != null) + if (converter.ConverterStrategy == ConverterStrategy.Object) { + PropertyCache ??= CreatePropertyCache(capacity: 0); + foreach (var jsonPropertyInfoKv in PropertyCache.List) { JsonPropertyInfo jsonPropertyInfo = jsonPropertyInfoKv.Value!; - jsonPropertyInfo.DeclaringTypeNumberHandling = NumberHandling; - jsonPropertyInfo.EnsureConfigured(); + jsonPropertyInfo.EnsureConfigured(this); } if (converter.ConstructorIsParameterized) { - InitializeConstructorParameters(GetParameterInfoValues(), sourceGenMode: Options.JsonSerializerContext != null); + InitializeConstructorParameters(GetParameterInfoValues(), sourceGenMode: Options.SerializerContext != null); } } } @@ -290,13 +398,66 @@ internal string GetDebugInfo() internal virtual void LateAddProperties() { } - internal virtual JsonParameterInfoValues[] GetParameterInfoValues() + /// + /// Creates JsonTypeInfo + /// + /// Type for which JsonTypeInfo stores metadata for + /// Options associated with JsonTypeInfo + /// JsonTypeInfo instance + public static JsonTypeInfo CreateJsonTypeInfo(JsonSerializerOptions options) + { + return new CustomJsonTypeInfo(options); + } + + private static MethodInfo? s_createJsonTypeInfo; + + /// + /// Creates JsonTypeInfo + /// + /// Type for which JsonTypeInfo stores metadata for + /// Options associated with JsonTypeInfo + /// JsonTypeInfo instance + [RequiresUnreferencedCode("JSON serialization and deserialization might require types that cannot be statically analyzed and might need runtime code generation. Use generic overload or System.Text.Json source generation for native AOT applications.")] + [RequiresDynamicCode("JSON serialization and deserialization might require types that cannot be statically analyzed and might need runtime code generation. Use generic overload or System.Text.Json source generation for native AOT applications.")] + public static JsonTypeInfo CreateJsonTypeInfo(Type type, JsonSerializerOptions options) + { + s_createJsonTypeInfo ??= typeof(JsonTypeInfo).GetMethod(nameof(CreateJsonTypeInfo), new Type[] { typeof(JsonSerializerOptions) })!; + return (JsonTypeInfo)s_createJsonTypeInfo.MakeGenericMethod(type) + .Invoke(null, new object[] { options })!; + } + + /// + /// Creates JsonPropertyInfo + /// + /// Type of the property + /// Name of the property + /// JsonPropertyInfo instance + public JsonPropertyInfo CreateJsonPropertyInfo(Type propertyType, string name) { - // If JsonTypeInfo becomes abstract this should be abstract as well - Debug.Fail("This should never be called."); - return null!; + ValidateType(propertyType, Type, null, Options); + + JsonConverter converter = GetConverter(propertyType, + parentClassType: null, + memberInfo: null, + options: Options, + out _); + + JsonPropertyInfo propertyInfo = CreateProperty( + declaredPropertyType: propertyType, + memberInfo: null, + parentClassType: Type, + isVirtual: false, + converter: converter, + options: Options, + isCustomProperty: true); + + propertyInfo.Name = name; + + return propertyInfo; } + internal abstract JsonParameterInfoValues[] GetParameterInfoValues(); + internal void CacheMember(JsonPropertyInfo jsonPropertyInfo, JsonPropertyDictionary? propertyCache, ref Dictionary? ignoredMembers) { string memberName = jsonPropertyInfo.ClrName!; @@ -506,6 +667,42 @@ internal bool IsValidDataExtensionProperty(JsonPropertyInfo jsonPropertyInfo) return typeIsValid && Options.GetConverterInternal(memberType) != null; } + internal JsonPropertyDictionary CreatePropertyCache(int capacity) + { + return new JsonPropertyDictionary(Options.PropertyNameCaseInsensitive, capacity); + } + + // This method gets the runtime information for a given type or property. + // The runtime information consists of the following: + // - class type, + // - element type (if the type is a collection), + // - the converter (either native or custom), if one exists. + internal static JsonConverter GetConverter( + Type type, + Type? parentClassType, + MemberInfo? memberInfo, + JsonSerializerOptions options, + out JsonConverter? customConverter) + { + Debug.Assert(type != null); + Debug.Assert(!IsInvalidForSerialization(type), $"Type `{type.FullName}` should already be validated."); + customConverter = parentClassType != null ? options.GetCustomConverterFromMember(parentClassType, type, memberInfo) : null; + return options.GetConverterForType(type); + } + + internal static JsonConverter GetEffectiveConverter( + Type type, + Type? parentClassType, + MemberInfo? memberInfo, + JsonSerializerOptions options) + { + JsonConverter converter = GetConverter(type, parentClassType, memberInfo, options, out JsonConverter? customConverter); + + customConverter = options.ExpandFactoryConverter(customConverter, type); + + return customConverter ?? converter; + } + private static JsonParameterInfo CreateConstructorParameter( JsonParameterInfoValues parameterInfo, JsonPropertyInfo jsonPropertyInfo, @@ -517,7 +714,7 @@ private static JsonParameterInfo CreateConstructorParameter( return JsonParameterInfo.CreateIgnoredParameterPlaceholder(parameterInfo, jsonPropertyInfo, sourceGenMode); } - JsonConverter converter = jsonPropertyInfo.ConverterBase; + JsonConverter converter = jsonPropertyInfo.EffectiveConverter; JsonParameterInfo jsonParameterInfo = converter.CreateJsonParameterInfo(); jsonParameterInfo.Initialize(parameterInfo, jsonPropertyInfo, options); @@ -525,6 +722,23 @@ private static JsonParameterInfo CreateConstructorParameter( return jsonParameterInfo; } + private static JsonTypeInfoKind GetTypeInfoKind(Type type, ConverterStrategy converterStrategy) + { + // System.Object is semi-polimorphic and will not respect Properties + if (type == typeof(object)) + { + return JsonTypeInfoKind.None; + } + + return converterStrategy switch + { + ConverterStrategy.Object => JsonTypeInfoKind.Object, + ConverterStrategy.Enumerable => JsonTypeInfoKind.Enumerable, + ConverterStrategy.Dictionary => JsonTypeInfoKind.Dictionary, + _ => JsonTypeInfoKind.None + }; + } + [DebuggerBrowsable(DebuggerBrowsableState.Never)] private string DebuggerDisplay => $"ConverterStrategy.{PropertyInfoForTypeInfo.ConverterStrategy}, {Type.Name}"; } diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfoKind.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfoKind.cs new file mode 100644 index 0000000000000..200773d83218f --- /dev/null +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfoKind.cs @@ -0,0 +1,28 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +namespace System.Text.Json.Serialization.Metadata +{ + /// + /// Determines the kind of contract metadata a given JsonTypeInfo instance is customizing + /// + public enum JsonTypeInfoKind + { + /// + /// Type is either a primitive value or uses a custom converter. JsonTypeInfo metadata does not apply here. + /// + None = 0, + /// + /// Type is serialized as object with properties + /// + Object = 1, + /// + /// Type is serialized as a collection with elements + /// + Enumerable = 2, + /// + /// Type is serialized as a dictionary with key/value pair entries + /// + Dictionary = 3 + } +} diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfoOfT.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfoOfT.cs index f5e3e341bbceb..1d0af2d935f88 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfoOfT.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfoOfT.cs @@ -10,11 +10,71 @@ namespace System.Text.Json.Serialization.Metadata /// Provides JSON serialization-related metadata about a type. /// /// The generic definition of the type. - [EditorBrowsable(EditorBrowsableState.Never)] public abstract class JsonTypeInfo : JsonTypeInfo { private Action? _serialize; + private Func? _typedCreateObject; + + /// + /// Function for creating object before properties are set. If set to null type is not deserializable. + /// + public new Func? CreateObject + { + get => _typedCreateObject; + set + { + SetCreateObject(value); + } + } + + private protected override void SetCreateObject(Delegate? createObject, bool useForExtensionDataProperty = false) + { + Debug.Assert(createObject is null or Func or Func); + + CheckMutable(); + + Func? untypedCreateObject; + Func? typedCreateObject; + + if (createObject is null) + { + untypedCreateObject = null; + typedCreateObject = null; + } + else if (createObject is Func typedDelegate) + { + typedCreateObject = typedDelegate; + untypedCreateObject = createObject is Func untypedDelegate ? untypedDelegate : () => typedDelegate()!; + } + else + { + Debug.Assert(createObject is Func); + untypedCreateObject = (Func)createObject; + typedCreateObject = () => (T)untypedCreateObject(); + } + + + if (Kind != JsonTypeInfoKind.None) + { + _createObject = untypedCreateObject; + _typedCreateObject = typedCreateObject; + } + else + { + if (useForExtensionDataProperty) + { + CreateObjectForExtensionDataProperty = untypedCreateObject; + } + else + { + Debug.Assert(_createObject == null); + Debug.Assert(_typedCreateObject == null); + ThrowHelper.ThrowInvalidOperationException_JsonTypeInfoOperationNotPossibleForKindNone(); + } + } + } + internal JsonTypeInfo(JsonConverter converter, JsonSerializerOptions options) : base(typeof(T), converter, options) { } @@ -24,6 +84,7 @@ internal JsonTypeInfo(JsonConverter converter, JsonSerializerOptions options) /// values specified at design time. /// /// The writer is not flushed after writing. + [EditorBrowsable(EditorBrowsableState.Never)] public Action? SerializeHandler { get @@ -32,6 +93,7 @@ public Action? SerializeHandler } private protected set { + CheckMutable(); _serialize = value; HasSerialize = value != null; } diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfoResolver.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfoResolver.cs new file mode 100644 index 0000000000000..bf390ef774267 --- /dev/null +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/JsonTypeInfoResolver.cs @@ -0,0 +1,54 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System; +using System.Collections.Generic; +using System.Text; +using System.Threading.Tasks; + +namespace System.Text.Json.Serialization.Metadata +{ + /// + /// Contains utilities for IJsonTypeInfoResolver + /// + public static class JsonTypeInfoResolver + { + /// + /// Combines multiple IJsonTypeInfoResolvers + /// + /// + /// + /// + /// All resolvers except last one should return null when they do not know how to create JsonTypeInfo for a given type. + /// Last resolver on the list should return non-null for most of the types unless explicit type blocking is desired. + /// + public static IJsonTypeInfoResolver Combine(params IJsonTypeInfoResolver[] resolvers) + { + return new CombiningJsonTypeInfoResolver(resolvers); + } + + private sealed class CombiningJsonTypeInfoResolver : IJsonTypeInfoResolver + { + private readonly IJsonTypeInfoResolver[] _resolvers; + + public CombiningJsonTypeInfoResolver(IJsonTypeInfoResolver[] resolvers) + { + _resolvers = resolvers.AsSpan().ToArray(); + } + + public JsonTypeInfo? GetTypeInfo(Type type, JsonSerializerOptions options) + { + foreach (IJsonTypeInfoResolver resolver in _resolvers) + { + JsonTypeInfo? typeInfo = resolver.GetTypeInfo(type, options); + if (typeInfo != null) + { + return typeInfo; + } + } + + return null; + } + } + } +} diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/MemberAccessor.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/MemberAccessor.cs index 9e3fb51b28498..d685d241628bf 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/MemberAccessor.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/MemberAccessor.cs @@ -9,7 +9,7 @@ namespace System.Text.Json.Serialization.Metadata { internal abstract class MemberAccessor { - public abstract JsonTypeInfo.ConstructorDelegate? CreateConstructor( + public abstract Func? CreateConstructor( [DynamicallyAccessedMembers(DynamicallyAccessedMemberTypes.PublicConstructors)] Type classType); public abstract Func? CreateParameterizedConstructor(ConstructorInfo constructor); diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionEmitCachingMemberAccessor.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionEmitCachingMemberAccessor.cs index d35e6e9e2cdae..2e0e5e94b2a87 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionEmitCachingMemberAccessor.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionEmitCachingMemberAccessor.cs @@ -23,7 +23,7 @@ internal sealed partial class ReflectionEmitCachingMemberAccessor : MemberAccess Justification = "Parent method annotation does not flow to lambda method, cf. https://github.com/dotnet/roslyn/issues/46646")] static (_) => s_sourceAccessor.CreateAddMethodDelegate()); - public override JsonTypeInfo.ConstructorDelegate? CreateConstructor([DynamicallyAccessedMembers(DynamicallyAccessedMemberTypes.PublicConstructors)] Type classType) + public override Func? CreateConstructor([DynamicallyAccessedMembers(DynamicallyAccessedMemberTypes.PublicConstructors)] Type classType) => s_cache.GetOrAdd((nameof(CreateConstructor), classType, null), [UnconditionalSuppressMessage("ReflectionAnalysis", "IL2077:UnrecognizedReflectionPattern", Justification = "Cannot apply DynamicallyAccessedMembersAttribute to tuple properties.")] diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionEmitMemberAccessor.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionEmitMemberAccessor.cs index 6a6f612363f6d..5a5500a1cb0d6 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionEmitMemberAccessor.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionEmitMemberAccessor.cs @@ -13,7 +13,7 @@ namespace System.Text.Json.Serialization.Metadata [RequiresDynamicCode(JsonSerializer.SerializationRequiresDynamicCodeMessage)] internal sealed class ReflectionEmitMemberAccessor : MemberAccessor { - public override JsonTypeInfo.ConstructorDelegate? CreateConstructor( + public override Func? CreateConstructor( [DynamicallyAccessedMembers(DynamicallyAccessedMemberTypes.PublicConstructors)] Type type) { Debug.Assert(type != null); @@ -59,7 +59,7 @@ internal sealed class ReflectionEmitMemberAccessor : MemberAccessor generator.Emit(OpCodes.Ret); - return (JsonTypeInfo.ConstructorDelegate)dynamicMethod.CreateDelegate(typeof(JsonTypeInfo.ConstructorDelegate)); + return (Func)dynamicMethod.CreateDelegate(typeof(Func)); } public override Func? CreateParameterizedConstructor(ConstructorInfo constructor) => diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionJsonTypeInfoOfT.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionJsonTypeInfoOfT.cs index 43d18ae473c6a..1c323c7cd1f97 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionJsonTypeInfoOfT.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionJsonTypeInfoOfT.cs @@ -19,7 +19,7 @@ internal sealed class ReflectionJsonTypeInfo : JsonTypeInfo [RequiresDynamicCode(JsonSerializer.SerializationRequiresDynamicCodeMessage)] internal ReflectionJsonTypeInfo(JsonSerializerOptions options) : this( - GetConverter( + GetEffectiveConverter( typeof(T), parentClassType: null, // A TypeInfo never has a "parent" class. memberInfo: null, // A TypeInfo never has a "parent" property. @@ -40,17 +40,17 @@ internal ReflectionJsonTypeInfo(JsonConverter converter, JsonSerializerOptions o AddPropertiesAndParametersUsingReflection(); } - CreateObject = Options.MemberAccessorStrategy.CreateConstructor(typeof(T)); + SetCreateObject(Options.MemberAccessorStrategy.CreateConstructor(typeof(T)), useForExtensionDataProperty: true); } [UnconditionalSuppressMessage("ReflectionAnalysis", "IL2026:RequiresUnreferencedCode", - Justification = "The ctor is marked as RequiresUnreferencedCode")] + Justification = "The ctor is marked as RequiresUnreferencedCode")] [UnconditionalSuppressMessage("AotAnalysis", "IL3050:RequiresDynamicCode", Justification = "The ctor is marked RequiresDynamicCode.")] internal override void Configure() { base.Configure(); - PropertyInfoForTypeInfo.ConverterBase.ConfigureJsonTypeInfoUsingReflection(this, Options); + Converter.ConfigureJsonTypeInfoUsingReflection(this, Options); } [RequiresUnreferencedCode(JsonSerializer.SerializationUnreferencedCodeMessage)] @@ -72,7 +72,7 @@ private void AddPropertiesAndParametersUsingReflection() // PropertyCache is not accessed by other threads until the current JsonTypeInfo instance // is finished initializing and added to the cache on JsonSerializerOptions. // Default 'capacity' to the common non-polymorphic + property case. - PropertyCache = new JsonPropertyDictionary(Options.PropertyNameCaseInsensitive, capacity: properties.Length); + PropertyCache = CreatePropertyCache(capacity: properties.Length); // We start from the most derived type. Type? currentType = Type; @@ -212,11 +212,13 @@ private static JsonPropertyInfo AddProperty( ValidateType(memberType, parentClassType, memberInfo, options); + JsonConverter? customConverter; JsonConverter converter = GetConverter( memberType, parentClassType, memberInfo, - options); + options, + out customConverter); return CreateProperty( declaredPropertyType: memberType, @@ -225,7 +227,8 @@ private static JsonPropertyInfo AddProperty( isVirtual, converter, options, - ignoreCondition); + ignoreCondition, + customConverter: customConverter); } private static JsonNumberHandling? GetNumberHandlingForType(Type type) @@ -236,22 +239,6 @@ private static JsonPropertyInfo AddProperty( return numberHandlingAttribute?.Handling; } - // This method gets the runtime information for a given type or property. - // The runtime information consists of the following: - // - class type, - // - element type (if the type is a collection), - // - the converter (either native or custom), if one exists. - private static JsonConverter GetConverter( - Type type, - Type? parentClassType, - MemberInfo? memberInfo, - JsonSerializerOptions options) - { - Debug.Assert(type != null); - Debug.Assert(!IsInvalidForSerialization(type), $"Type `{type.FullName}` should already be validated."); - return options.GetConverterFromMember(parentClassType, type, memberInfo); - } - private static bool PropertyIsOverridenAndIgnored( string currentMemberName, Type currentMemberType, @@ -270,7 +257,7 @@ private static bool PropertyIsOverridenAndIgnored( internal override JsonParameterInfoValues[] GetParameterInfoValues() { - ParameterInfo[] parameters = PropertyInfoForTypeInfo.ConverterBase.ConstructorInfo!.GetParameters(); + ParameterInfo[] parameters = Converter.ConstructorInfo!.GetParameters(); return GetParameterInfoArray(parameters); } diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionMemberAccessor.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionMemberAccessor.cs index 41527aa309366..9914033a11bed 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionMemberAccessor.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/ReflectionMemberAccessor.cs @@ -22,7 +22,7 @@ public ConstructorContext([DynamicallyAccessedMembers(DynamicallyAccessedMemberT => Activator.CreateInstance(_type, nonPublic: false); } - public override JsonTypeInfo.ConstructorDelegate? CreateConstructor( + public override Func? CreateConstructor( [DynamicallyAccessedMembers(DynamicallyAccessedMemberTypes.PublicConstructors)] Type type) { Debug.Assert(type != null); @@ -38,7 +38,7 @@ public ConstructorContext([DynamicallyAccessedMembers(DynamicallyAccessedMemberT return null; } - return new ConstructorContext(type).CreateInstance; + return new ConstructorContext(type).CreateInstance!; } public override Func? CreateParameterizedConstructor(ConstructorInfo constructor) diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/SourceGenJsonTypeInfoOfT.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/SourceGenJsonTypeInfoOfT.cs index d724c39236fef..f86c091c00f48 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/SourceGenJsonTypeInfoOfT.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/Metadata/SourceGenJsonTypeInfoOfT.cs @@ -34,12 +34,12 @@ public SourceGenJsonTypeInfo(JsonSerializerOptions options, JsonObjectInfoValues } else { - SetCreateObjectFunc(objectInfo.ObjectCreator); + SetCreateObject(objectInfo.ObjectCreator, useForExtensionDataProperty: true); } - PropInitFunc = objectInfo.PropertyMetadataInitializer; SerializeHandler = objectInfo.SerializeHandler; + NumberHandling = objectInfo.NumberHandling; } @@ -61,11 +61,12 @@ public SourceGenJsonTypeInfo( KeyTypeInfo = collectionInfo.KeyInfo; ElementTypeInfo = collectionInfo.ElementInfo ?? throw new ArgumentNullException(nameof(collectionInfo.ElementInfo)); + Debug.Assert(Kind != JsonTypeInfoKind.None); NumberHandling = collectionInfo.NumberHandling; SerializeHandler = collectionInfo.SerializeHandler; CreateObjectWithArgs = createObjectWithArgs; AddMethodDelegate = addFunc; - SetCreateObjectFunc(collectionInfo.ObjectCreator); + CreateObject = collectionInfo.ObjectCreator; } private static JsonConverter GetConverter(JsonObjectInfoValues objectInfo) @@ -99,9 +100,9 @@ internal override void LateAddProperties() internal override JsonParameterInfoValues[] GetParameterInfoValues() { - JsonSerializerContext? context = Options.JsonSerializerContext; + JsonSerializerContext? context = Options.SerializerContext; JsonParameterInfoValues[] array; - if (context == null || CtorParamInitFunc == null || (array = CtorParamInitFunc()) == null) + if (CtorParamInitFunc == null || (array = CtorParamInitFunc()) == null) { ThrowHelper.ThrowInvalidOperationException_NoMetadataForTypeCtorParams(context, Type); return null!; @@ -117,22 +118,22 @@ internal void AddPropertiesUsingSourceGenInfo() return; } - JsonSerializerContext? context = Options.JsonSerializerContext; + JsonSerializerContext? context = Options.SerializerContext; JsonPropertyInfo[] array; - if (context == null || PropInitFunc == null || (array = PropInitFunc(context)) == null) + if (PropInitFunc == null || (array = PropInitFunc(context!)) == null) { if (typeof(T) == typeof(object)) { return; } - if (PropertyInfoForTypeInfo.ConverterBase.ElementType != null) + if (Converter.ElementType != null) { // Nullable<> or F# optional converter's strategy is set to element's strategy return; } - if (SerializeHandler != null && Options.JsonSerializerContext?.CanUseSerializationLogic == true) + if (SerializeHandler != null && Options.SerializerContext?.CanUseSerializationLogic == true) { ThrowOnDeserialize = true; return; @@ -143,7 +144,7 @@ internal void AddPropertiesUsingSourceGenInfo() } Dictionary? ignoredMembers = null; - JsonPropertyDictionary propertyCache = new(Options.PropertyNameCaseInsensitive, array.Length); + JsonPropertyDictionary propertyCache = CreatePropertyCache(capacity: array.Length); for (int i = 0; i < array.Length; i++) { @@ -181,13 +182,5 @@ internal void AddPropertiesUsingSourceGenInfo() PropertyCache = propertyCache; } - - private void SetCreateObjectFunc(Func? createObjectFunc) - { - if (createObjectFunc != null) - { - CreateObject = () => createObjectFunc(); - } - } } } diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/ReadStack.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/ReadStack.cs index 8f0e9ec9292cd..2ca7109af2886 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/ReadStack.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/ReadStack.cs @@ -226,7 +226,7 @@ public JsonConverter InitializePolymorphicReEntry(JsonTypeInfo derivedJsonTypeIn Current.PolymorphicSerializationState = PolymorphicSerializationState.PolymorphicReEntryStarted; SetConstructorArgumentState(); - return derivedJsonTypeInfo.PropertyInfoForTypeInfo.ConverterBase; + return derivedJsonTypeInfo.Converter; } @@ -241,7 +241,7 @@ public JsonConverter ResumePolymorphicReEntry() // Swap out the two values as we resume the polymorphic converter (Current.JsonTypeInfo, Current.PolymorphicJsonTypeInfo) = (Current.PolymorphicJsonTypeInfo, Current.JsonTypeInfo); Current.PolymorphicSerializationState = PolymorphicSerializationState.PolymorphicReEntryStarted; - return Current.JsonTypeInfo.PropertyInfoForTypeInfo.ConverterBase; + return Current.JsonTypeInfo.Converter; } /// @@ -382,20 +382,20 @@ public JsonTypeInfo GetTopJsonTypeInfoWithParameterizedConstructor() for (int i = 0; i < _count - 1; i++) { - if (_stack[i].JsonTypeInfo.PropertyInfoForTypeInfo.ConverterBase.ConstructorIsParameterized) + if (_stack[i].JsonTypeInfo.Converter.ConstructorIsParameterized) { return _stack[i].JsonTypeInfo; } } - Debug.Assert(Current.JsonTypeInfo.PropertyInfoForTypeInfo.ConverterBase.ConstructorIsParameterized); + Debug.Assert(Current.JsonTypeInfo.Converter.ConstructorIsParameterized); return Current.JsonTypeInfo; } [MethodImpl(MethodImplOptions.AggressiveInlining)] private void SetConstructorArgumentState() { - if (Current.JsonTypeInfo.PropertyInfoForTypeInfo.ConverterBase.ConstructorIsParameterized) + if (Current.JsonTypeInfo.Converter.ConstructorIsParameterized) { // A zero index indicates a new stack frame. if (Current.CtorArgumentStateIndex == 0) diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/WriteStack.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/WriteStack.cs index 81e00d7147f52..0918f09f39379 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/WriteStack.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/WriteStack.cs @@ -158,7 +158,7 @@ internal JsonConverter Initialize(JsonTypeInfo jsonTypeInfo, bool supportContinu SupportContinuation = supportContinuation; SupportAsync = supportAsync; - return jsonTypeInfo.PropertyInfoForTypeInfo.ConverterBase; + return jsonTypeInfo.Converter; } /// diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/WriteStackFrame.cs b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/WriteStackFrame.cs index 0e9fc8c63fbc2..98127e2370372 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/WriteStackFrame.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/Serialization/WriteStackFrame.cs @@ -129,7 +129,7 @@ public JsonConverter InitializePolymorphicReEntry(Type runtimeType, JsonSerializ } PolymorphicSerializationState = PolymorphicSerializationState.PolymorphicReEntryStarted; - return PolymorphicJsonTypeInfo.ConverterBase; + return PolymorphicJsonTypeInfo.EffectiveConverter; } /// @@ -141,7 +141,7 @@ public JsonConverter InitializePolymorphicReEntry(JsonTypeInfo derivedJsonTypeIn PolymorphicJsonTypeInfo = derivedJsonTypeInfo.PropertyInfoForTypeInfo; PolymorphicSerializationState = PolymorphicSerializationState.PolymorphicReEntryStarted; - return PolymorphicJsonTypeInfo.ConverterBase; + return PolymorphicJsonTypeInfo.EffectiveConverter; } /// @@ -152,7 +152,7 @@ public JsonConverter ResumePolymorphicReEntry() Debug.Assert(PolymorphicSerializationState == PolymorphicSerializationState.PolymorphicReEntrySuspended); Debug.Assert(PolymorphicJsonTypeInfo is not null); PolymorphicSerializationState = PolymorphicSerializationState.PolymorphicReEntryStarted; - return PolymorphicJsonTypeInfo.ConverterBase; + return PolymorphicJsonTypeInfo.EffectiveConverter; } /// diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/ThrowHelper.Node.cs b/src/libraries/System.Text.Json/src/System/Text/Json/ThrowHelper.Node.cs index 364c100282412..e056d1b1d1bad 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/ThrowHelper.Node.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/ThrowHelper.Node.cs @@ -14,18 +14,6 @@ public static void ThrowArgumentException_NodeValueNotAllowed(string paramName) throw new ArgumentException(SR.NodeValueNotAllowed, paramName); } - [DoesNotReturn] - public static void ThrowArgumentException_NodeArrayTooSmall(string paramName) - { - throw new ArgumentException(SR.NodeArrayTooSmall, paramName); - } - - [DoesNotReturn] - public static void ThrowArgumentOutOfRangeException_NodeArrayIndexNegative(string paramName) - { - throw new ArgumentOutOfRangeException(paramName, SR.NodeArrayIndexNegative); - } - [DoesNotReturn] public static void ThrowArgumentException_DuplicateKey(string propertyName) { @@ -51,14 +39,14 @@ public static void ThrowInvalidOperationException_NodeElementCannotBeObjectOrArr } [DoesNotReturn] - public static void ThrowNotSupportedException_NodeCollectionIsReadOnly() + public static void ThrowNotSupportedException_CollectionIsReadOnly() { - throw GetNotSupportedException_NodeCollectionIsReadOnly(); + throw GetNotSupportedException_CollectionIsReadOnly(); } - public static NotSupportedException GetNotSupportedException_NodeCollectionIsReadOnly() + public static NotSupportedException GetNotSupportedException_CollectionIsReadOnly() { - return new NotSupportedException(SR.NodeCollectionIsReadOnly); + return new NotSupportedException(SR.CollectionIsReadOnly); } } } diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/ThrowHelper.Serialization.cs b/src/libraries/System.Text.Json/src/System/Text/Json/ThrowHelper.Serialization.cs index 28db4eee742b8..ba3de9326e57d 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/ThrowHelper.Serialization.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/ThrowHelper.Serialization.cs @@ -104,6 +104,18 @@ public static void ThrowInvalidOperationException_SerializationConverterNotCompa throw new InvalidOperationException(SR.Format(SR.SerializationConverterNotCompatible, converterType, type)); } + [DoesNotReturn] + public static void ThrowInvalidOperationException_ResolverTypeNotCompatible(Type requestedType, Type actualType) + { + throw new InvalidOperationException(SR.Format(SR.ResolverTypeNotCompatible, requestedType, actualType)); + } + + [DoesNotReturn] + public static void ThrowInvalidOperationException_ResolverTypeInfoOptionsNotCompatible() + { + throw new InvalidOperationException(SR.ResolverTypeInfoOptionsNotCompatible); + } + [DoesNotReturn] public static void ThrowInvalidOperationException_SerializationConverterOnAttributeInvalid(Type classType, MemberInfo? memberInfo) { @@ -139,6 +151,30 @@ public static void ThrowInvalidOperationException_SerializerOptionsImmutable(Jso throw new InvalidOperationException(message); } + [DoesNotReturn] + public static void ThrowInvalidOperationException_SerializerContextOptionsImmutable() + { + throw new InvalidOperationException(SR.SerializerContextOptionsImmutable); + } + + [DoesNotReturn] + public static void ThrowInvalidOperationException_TypeInfoResolverImmutable() + { + throw new InvalidOperationException(SR.TypeInfoResolverImmutable); + } + + [DoesNotReturn] + public static void ThrowInvalidOperationException_TypeInfoImmutable() + { + throw new InvalidOperationException(SR.TypeInfoImmutable); + } + + [DoesNotReturn] + public static void ThrowInvalidOperationException_PropertyInfoImmutable() + { + throw new InvalidOperationException(SR.PropertyInfoImmutable); + } + [DoesNotReturn] public static void ThrowInvalidOperationException_SerializerPropertyNameConflict(Type type, JsonPropertyInfo jsonPropertyInfo) { @@ -239,6 +275,18 @@ public static void ThrowNotSupportedException_ObjectWithParameterizedCtorRefMeta ThrowNotSupportedException(ref state, reader, ex); } + [DoesNotReturn] + public static void ThrowInvalidOperationException_JsonTypeInfoOperationNotPossibleForKindNone() + { + throw new InvalidOperationException(SR.JsonTypeInfoOperationNotPossibleForKindNone); + } + + [DoesNotReturn] + public static void ThrowInvalidOperationException_CollectionIsReadOnly() + { + throw new InvalidOperationException(SR.CollectionIsReadOnly); + } + [DoesNotReturn] public static void ReThrowWithPath(ref ReadStack state, JsonReaderException ex) { diff --git a/src/libraries/System.Text.Json/src/System/Text/Json/ThrowHelper.cs b/src/libraries/System.Text.Json/src/System/Text/Json/ThrowHelper.cs index e7f151dc9df2b..9af92adb3808d 100644 --- a/src/libraries/System.Text.Json/src/System/Text/Json/ThrowHelper.cs +++ b/src/libraries/System.Text.Json/src/System/Text/Json/ThrowHelper.cs @@ -35,6 +35,18 @@ public static void ThrowArgumentOutOfRangeException_CommentEnumMustBeInRange(str throw GetArgumentOutOfRangeException(parameterName, SR.CommentHandlingMustBeValid); } + [DoesNotReturn] + public static void ThrowArgumentOutOfRangeException_ArrayIndexNegative(string paramName) + { + throw new ArgumentOutOfRangeException(paramName, SR.ArrayIndexNegative); + } + + [DoesNotReturn] + public static void ThrowArgumentException_ArrayTooSmall(string paramName) + { + throw new ArgumentException(SR.ArrayTooSmall, paramName); + } + private static ArgumentException GetArgumentException(string message) { return new ArgumentException(message); diff --git a/src/libraries/System.Text.Json/tests/System.Text.Json.SourceGeneration.Tests/JsonSerializerContextTests.cs b/src/libraries/System.Text.Json/tests/System.Text.Json.SourceGeneration.Tests/JsonSerializerContextTests.cs index ea171415963d0..bf56fd4791805 100644 --- a/src/libraries/System.Text.Json/tests/System.Text.Json.SourceGeneration.Tests/JsonSerializerContextTests.cs +++ b/src/libraries/System.Text.Json/tests/System.Text.Json.SourceGeneration.Tests/JsonSerializerContextTests.cs @@ -4,6 +4,7 @@ using System.Collections.Generic; using System.Reflection; using System.Text.Json.Serialization; +using System.Text.Json.Serialization.Metadata; using Microsoft.DotNet.RemoteExecutor; using Xunit; @@ -37,25 +38,20 @@ public static void Converters_AndTypeInfoCreator_NotRooted_WhenMetadataNotPresen Assert.Contains("JsonSerializerOptions", exAsStr); // This test uses reflection to: - // - Access JsonSerializerOptions.s_defaultSimpleConverters - // - Access JsonSerializerOptions.s_defaultFactoryConverters - // - Access JsonSerializerOptions.s_typeInfoCreationFunc + // - Access DefaultJsonTypeInfoResolver.s_defaultSimpleConverters + // - Access DefaultJsonTypeInfoResolver.s_defaultFactoryConverters // // If any of them changes, this test will need to be kept in sync. // Confirm built-in converters not set. - AssertFieldNull("s_defaultSimpleConverters", optionsInstance: null); - AssertFieldNull("s_defaultFactoryConverters", optionsInstance: null); + AssertFieldNull("s_defaultSimpleConverters"); + AssertFieldNull("s_defaultFactoryConverters"); - // Confirm type info dynamic creator not set. - AssertFieldNull("s_typeInfoCreationFunc", optionsInstance: null); - - static void AssertFieldNull(string fieldName, JsonSerializerOptions? optionsInstance) + static void AssertFieldNull(string fieldName) { - BindingFlags bindingFlags = BindingFlags.NonPublic | (optionsInstance == null ? BindingFlags.Static : BindingFlags.Instance); - FieldInfo fieldInfo = typeof(JsonSerializerOptions).GetField(fieldName, bindingFlags); + FieldInfo fieldInfo = typeof(DefaultJsonTypeInfoResolver).GetField(fieldName, BindingFlags.Static | BindingFlags.NonPublic); Assert.NotNull(fieldInfo); - Assert.Null(fieldInfo.GetValue(optionsInstance)); + Assert.Null(fieldInfo.GetValue(null)); } }).Dispose(); } @@ -72,6 +68,73 @@ public static void SupportsPositionalRecords() Assert.Equal("Doe", person.LastName); } + [Fact] + public static void CombiningContexts_ResolveJsonTypeInfo() + { + // Basic smoke test establishing combination of JsonSerializerContext classes. + IJsonTypeInfoResolver combined = JsonTypeInfoResolver.Combine(NestedContext.Default, PersonJsonContext.Default); + var options = new JsonSerializerOptions { TypeInfoResolver = combined }; + + JsonTypeInfo messageInfo = combined.GetTypeInfo(typeof(JsonMessage), options); + Assert.IsAssignableFrom>(messageInfo); + Assert.Same(options, messageInfo.Options); + + JsonTypeInfo personInfo = combined.GetTypeInfo(typeof(Person), options); + Assert.IsAssignableFrom>(personInfo); + Assert.Same(options, personInfo.Options); + } + + [Fact] + public static void CombiningContexts_ResolveJsonTypeInfo_DifferentCasing() + { + IJsonTypeInfoResolver combined = JsonTypeInfoResolver.Combine(NestedContext.Default, PersonJsonContext.Default); + var options = new JsonSerializerOptions + { + TypeInfoResolver = combined, + PropertyNamingPolicy = JsonNamingPolicy.CamelCase, + }; + + Assert.NotSame(JsonNamingPolicy.CamelCase, NestedContext.Default.Options.PropertyNamingPolicy); + Assert.Same(JsonNamingPolicy.CamelCase, PersonJsonContext.Default.Options.PropertyNamingPolicy); + + JsonTypeInfo messageInfo = combined.GetTypeInfo(typeof(JsonMessage), options); + Assert.Equal(2, messageInfo.Properties.Count); + Assert.Equal("message", messageInfo.Properties[0].Name); + Assert.Equal("length", messageInfo.Properties[1].Name); + + JsonTypeInfo personInfo = combined.GetTypeInfo(typeof(Person), options); + Assert.Equal(2, personInfo.Properties.Count); + Assert.Equal("firstName", personInfo.Properties[0].Name); + Assert.Equal("lastName", personInfo.Properties[1].Name); + } + + [Theory] + [MemberData(nameof(GetCombiningContextsData))] + public static void CombiningContexts_Serialization(T value, string expectedJson) + { + // Basic smoke test establishing combination of JsonSerializerContext classes. + IJsonTypeInfoResolver combined = JsonTypeInfoResolver.Combine(NestedContext.Default, PersonJsonContext.Default); + var options = new JsonSerializerOptions { TypeInfoResolver = combined }; + + JsonTypeInfo typeInfo = (JsonTypeInfo)combined.GetTypeInfo(typeof(T), options)!; + + string json = JsonSerializer.Serialize(value, typeInfo); + JsonTestHelper.AssertJsonEqual(expectedJson, json); + + json = JsonSerializer.Serialize(value, options); + JsonTestHelper.AssertJsonEqual(expectedJson, json); + + JsonSerializer.Deserialize(json, typeInfo); + JsonSerializer.Deserialize(json, options); + } + + public static IEnumerable GetCombiningContextsData() + { + yield return WrapArgs(new JsonMessage { Message = "Hi" }, """{ "Message" : "Hi", "Length" : 2 }"""); + yield return WrapArgs(new Person("John", "Doe"), """{ "FirstName" : "John", "LastName" : "Doe" }"""); + static object[] WrapArgs(T value, string expectedJson) => new object[] { value, expectedJson }; + } + [JsonSerializable(typeof(JsonMessage))] internal partial class NestedContext : JsonSerializerContext { } diff --git a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/CacheTests.cs b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/CacheTests.cs index c16738ce5056d..e53b33c3b4dda 100644 --- a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/CacheTests.cs +++ b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/CacheTests.cs @@ -371,6 +371,7 @@ public static void JsonSerializerOptions_EqualityComparer_ChangingAnySettingShou yield return (GetProp(nameof(JsonSerializerOptions.UnknownTypeHandling)), JsonUnknownTypeHandling.JsonNode); yield return (GetProp(nameof(JsonSerializerOptions.WriteIndented)), true); yield return (GetProp(nameof(JsonSerializerOptions.ReferenceHandler)), ReferenceHandler.Preserve); + yield return (GetProp(nameof(JsonSerializerOptions.TypeInfoResolver)), new DefaultJsonTypeInfoResolver()); static PropertyInfo GetProp(string name) { diff --git a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/CustomConverterTests/CustomConverterTests.cs b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/CustomConverterTests/CustomConverterTests.cs index e7055c2db1a67..792e6873f7422 100644 --- a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/CustomConverterTests/CustomConverterTests.cs +++ b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/CustomConverterTests/CustomConverterTests.cs @@ -196,20 +196,20 @@ public static void GetConverter_Poco_WriteThrowsNotSupportedException() // for reflection-based serialization should throw NotSupportedException // since it can't resolve reflection-based metadata. Assert.Throws(() => converter.Write(writer, value, options)); - Debug.Assert(writer.BytesCommitted + writer.BytesPending == 0); + Assert.Equal(0, writer.BytesCommitted + writer.BytesPending); JsonSerializer.Serialize(42, options); // Same operation should succeed when instance has been primed. converter.Write(writer, value, options); - Debug.Assert(writer.BytesCommitted + writer.BytesPending > 0); + Assert.NotEqual(0, writer.BytesCommitted + writer.BytesPending); writer.Reset(); // State change should not leak into unrelated options instances. var options2 = new JsonSerializerOptions(); options2.AddContext(); Assert.Throws(() => converter.Write(writer, value, options2)); - Debug.Assert(writer.BytesCommitted + writer.BytesPending == 0); + Assert.Equal(0, writer.BytesCommitted + writer.BytesPending); }).Dispose(); } diff --git a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/JsonSerializerWrapper.Reflection.cs b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/JsonSerializerWrapper.Reflection.cs index 2e3912a7572e9..7f6b4ebd56dd8 100644 --- a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/JsonSerializerWrapper.Reflection.cs +++ b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/JsonSerializerWrapper.Reflection.cs @@ -572,7 +572,6 @@ private static class JsonSerializerOptionsSmallBufferMapper { private static readonly ConditionalWeakTable s_smallBufferMap = new(); private static readonly JsonSerializerOptions s_DefaultOptionsWithSmallBuffer = new JsonSerializerOptions { DefaultBufferSize = 1 }; - private static readonly FieldInfo s_optionsContextField = typeof(JsonSerializerOptions).GetField("_serializerContext", BindingFlags.NonPublic | BindingFlags.Instance); public static JsonSerializerOptions ResolveOptionsInstanceWithSmallBuffer(JsonSerializerOptions? options) { @@ -592,20 +591,15 @@ public static JsonSerializerOptions ResolveOptionsInstanceWithSmallBuffer(JsonSe return resolvedValue; } - JsonSerializerOptions smallBufferCopy = new JsonSerializerOptions(options) { DefaultBufferSize = 1 }; - CopyJsonSerializerContext(options, smallBufferCopy); + JsonSerializerOptions smallBufferCopy = new JsonSerializerOptions(options) + { + // Copy the resolver explicitly until https://github.com/dotnet/aspnetcore/issues/38720 is resolved. + TypeInfoResolver = options.TypeInfoResolver, + DefaultBufferSize = 1, + }; s_smallBufferMap.Add(options, smallBufferCopy); return smallBufferCopy; } - - private static void CopyJsonSerializerContext(JsonSerializerOptions source, JsonSerializerOptions target) - { - JsonSerializerContext context = (JsonSerializerContext)s_optionsContextField.GetValue(source); - if (context != null) - { - s_optionsContextField.SetValue(target, context); - } - } } } } diff --git a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/DefaultJsonTypeInfoResolverTests.JsonPropertyInfo.cs b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/DefaultJsonTypeInfoResolverTests.JsonPropertyInfo.cs new file mode 100644 index 0000000000000..9cbed0b9017e5 --- /dev/null +++ b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/DefaultJsonTypeInfoResolverTests.JsonPropertyInfo.cs @@ -0,0 +1,489 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System; +using System.Collections.Generic; +using System.Reflection; +using System.Text; +using System.Text.Json.Serialization.Metadata; +using System.Text.Json.Tests; +using System.Threading.Tasks; +using Xunit; + +namespace System.Text.Json.Serialization.Tests +{ + public static partial class DefaultJsonTypeInfoResolverTests + { + [Fact] + public static void JsonPropertyInfoOptionsAreSet() + { + JsonSerializerOptions options = new(); + JsonTypeInfo typeInfo = JsonTypeInfo.CreateJsonTypeInfo(typeof(MyClass), options); + CreatePropertyAndCheckOptions(options, typeInfo); + + typeInfo = JsonTypeInfo.CreateJsonTypeInfo(options); + CreatePropertyAndCheckOptions(options, typeInfo); + + typeInfo = options.TypeInfoResolver.GetTypeInfo(typeof(MyClass), options); + CreatePropertyAndCheckOptions(options, typeInfo); + + static void CreatePropertyAndCheckOptions(JsonSerializerOptions expectedOptions, JsonTypeInfo typeInfo) + { + JsonPropertyInfo propertyInfo = typeInfo.CreateJsonPropertyInfo(typeof(string), "test"); + Assert.Same(expectedOptions, propertyInfo.Options); + } + } + + [Theory] + [InlineData(typeof(string))] + [InlineData(typeof(int))] + [InlineData(typeof(MyClass))] + public static void JsonPropertyInfoPropertyTypeIsSetWhenUsingCreateJsonPropertyInfo(Type propertyType) + { + JsonSerializerOptions options = new(); + JsonTypeInfo typeInfo = JsonTypeInfo.CreateJsonTypeInfo(typeof(MyClass), options); + JsonPropertyInfo propertyInfo = typeInfo.CreateJsonPropertyInfo(propertyType, "test"); + + Assert.Equal(propertyType, propertyInfo.PropertyType); + } + + [Fact] + public static void JsonPropertyInfoPropertyTypeIsSet() + { + JsonSerializerOptions options = new(); + JsonTypeInfo typeInfo = options.TypeInfoResolver.GetTypeInfo(typeof(MyClass), options); + JsonPropertyInfo propertyInfo = typeInfo.Properties[0]; + Assert.Equal(typeof(string), propertyInfo.PropertyType); + } + + [Theory] + [InlineData(typeof(string))] + [InlineData(typeof(int))] + [InlineData(typeof(MyClass))] + public static void JsonPropertyInfoNameIsSetAndIsMutableWhenUsingCreateJsonPropertyInfo(Type propertyType) + { + JsonSerializerOptions options = new(); + JsonTypeInfo typeInfo = JsonTypeInfo.CreateJsonTypeInfo(typeof(MyClass), options); + JsonPropertyInfo propertyInfo = typeInfo.CreateJsonPropertyInfo(propertyType, "test"); + + Assert.Equal("test", propertyInfo.Name); + + propertyInfo.Name = "foo"; + Assert.Equal("foo", propertyInfo.Name); + + Assert.Throws(() => propertyInfo.Name = null); + } + + [Fact] + public static void JsonPropertyInfoNameIsSetAndIsMutableForDefaultResolver() + { + JsonSerializerOptions options = new(); + JsonTypeInfo typeInfo = options.TypeInfoResolver.GetTypeInfo(typeof(MyClass), options); + JsonPropertyInfo propertyInfo = typeInfo.Properties[0]; + + Assert.Equal(nameof(MyClass.Value), propertyInfo.Name); + + propertyInfo.Name = "foo"; + Assert.Equal("foo", propertyInfo.Name); + + Assert.Throws(() => propertyInfo.Name = null); + } + + [Fact] + public static void JsonPropertyInfoForDefaultResolverHasNamingPoliciesRulesApplied() + { + JsonSerializerOptions options = new(); + options.PropertyNamingPolicy = JsonNamingPolicy.CamelCase; + JsonTypeInfo typeInfo = options.TypeInfoResolver.GetTypeInfo(typeof(MyClass), options); + JsonPropertyInfo propertyInfo = typeInfo.Properties[0]; + + Assert.Equal(nameof(MyClass.Value).ToLowerInvariant(), propertyInfo.Name); + + // explicitly setting does not change casing + propertyInfo.Name = "Foo"; + Assert.Equal("Foo", propertyInfo.Name); + } + + [Fact] + public static void JsonPropertyInfoCustomConverterIsNullWhenUsingCreateJsonPropertyInfo() + { + JsonSerializerOptions options = new(); + JsonTypeInfo typeInfo = options.TypeInfoResolver.GetTypeInfo(typeof(TestClassWithCustomConverterOnProperty), options); + JsonPropertyInfo propertyInfo = typeInfo.CreateJsonPropertyInfo(typeof(MyClass), "test"); + + Assert.Null(propertyInfo.CustomConverter); + } + + [Fact] + public static void JsonPropertyInfoCustomConverterIsNotNullForPropertyWithCustomConverter() + { + JsonSerializerOptions options = new(); + JsonTypeInfo typeInfo = options.TypeInfoResolver.GetTypeInfo(typeof(TestClassWithCustomConverterOnProperty), options); + JsonPropertyInfo propertyInfo = typeInfo.Properties[0]; + + Assert.NotNull(propertyInfo.CustomConverter); + Assert.IsType(propertyInfo.CustomConverter); + } + + [Fact] + public static void JsonPropertyInfoCustomConverterSetToNullIsRespected() + { + JsonSerializerOptions options = new(); + DefaultJsonTypeInfoResolver r = new(); + r.Modifiers.Add(ti => + { + if (ti.Type == typeof(TestClassWithCustomConverterOnProperty)) + { + JsonPropertyInfo propertyInfo = ti.Properties[0]; + Assert.NotNull(propertyInfo.CustomConverter); + Assert.IsType(propertyInfo.CustomConverter); + propertyInfo.CustomConverter = null; + } + }); + + options.TypeInfoResolver = r; + + TestClassWithCustomConverterOnProperty obj = new() + { + MyClassProperty = new MyClass() { Value = "SomeValue" }, + }; + + string json = JsonSerializer.Serialize(obj, options); + Assert.Equal("""{"MyClassProperty":{"Value":"SomeValue","Thing":null}}""", json); + + TestClassWithCustomConverterOnProperty deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(obj.MyClassProperty.Value, deserialized.MyClassProperty.Value); + } + + [Fact] + public static void JsonPropertyInfoCustomConverterIsRespected() + { + JsonSerializerOptions options = new(); + DefaultJsonTypeInfoResolver r = new(); + r.Modifiers.Add(ti => + { + if (ti.Type == typeof(TestClassWithCustomConverterOnProperty)) + { + JsonPropertyInfo propertyInfo = ti.Properties[0]; + Assert.NotNull(propertyInfo.CustomConverter); + Assert.IsType(propertyInfo.CustomConverter); + propertyInfo.CustomConverter = new MyClassCustomConverter("test_"); + } + }); + + options.TypeInfoResolver = r; + + TestClassWithCustomConverterOnProperty obj = new() + { + MyClassProperty = new MyClass() { Value = "SomeValue" }, + }; + + string json = JsonSerializer.Serialize(obj, options); + Assert.Equal("""{"MyClassProperty":"test_SomeValue"}""", json); + + TestClassWithCustomConverterOnProperty deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(obj.MyClassProperty.Value, deserialized.MyClassProperty.Value); + } + + [Fact] + public static void JsonPropertyInfoGetIsNullAndMutableWhenUsingCreateJsonPropertyInfo() + { + JsonSerializerOptions options = new(); + JsonTypeInfo typeInfo = options.TypeInfoResolver.GetTypeInfo(typeof(TestClassWithCustomConverterOnProperty), options); + JsonPropertyInfo propertyInfo = typeInfo.CreateJsonPropertyInfo(typeof(MyClass), "test"); + Assert.Null(propertyInfo.Get); + Func get = (obj) => + { + throw new NotImplementedException(); + }; + + propertyInfo.Get = get; + Assert.Same(get, propertyInfo.Get); + } + + [Fact] + public static void JsonPropertyInfoGetIsNotNullForDefaultResolver() + { + JsonSerializerOptions options = new(); + JsonTypeInfo typeInfo = options.TypeInfoResolver.GetTypeInfo(typeof(TestClassWithCustomConverterOnProperty), options); + JsonPropertyInfo propertyInfo = typeInfo.Properties[0]; + + Assert.NotNull(propertyInfo.Get); + + TestClassWithCustomConverterOnProperty obj = new(); + + Assert.Null(propertyInfo.Get(obj)); + + obj.MyClassProperty = new MyClass(); + Assert.Same(obj.MyClassProperty, propertyInfo.Get(obj)); + + MyClass sentinel = new(); + Func get = (obj) => sentinel; + propertyInfo.Get = get; + Assert.Same(get, propertyInfo.Get); + Assert.Same(sentinel, propertyInfo.Get(obj)); + } + + [Fact] + public static void JsonPropertyInfoGetPropertyNotSerializableButDeserializableWhenNull() + { + JsonSerializerOptions options = new(); + DefaultJsonTypeInfoResolver r = new(); + r.Modifiers.Add(ti => + { + if (ti.Type == typeof(TestClassWithCustomConverterOnProperty)) + { + JsonPropertyInfo propertyInfo = ti.Properties[0]; + propertyInfo.Get = null; + } + }); + + options.TypeInfoResolver = r; + + TestClassWithCustomConverterOnProperty obj = new() + { + MyClassProperty = new MyClass() { Value = "SomeValue" }, + }; + + string json = JsonSerializer.Serialize(obj, options); + Assert.Equal("{}", json); + + json = """{"MyClassProperty":"SomeValue"}"""; + TestClassWithCustomConverterOnProperty deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(obj.MyClassProperty.Value, deserialized.MyClassProperty.Value); + } + + [Theory] + [InlineData(true)] + [InlineData(false)] + public static void JsonPropertyInfoGetIsRespected(bool useCustomConverter) + { + TestClassWithCustomConverterOnProperty obj = new() + { + MyClassProperty = new MyClass() { Value = "SomeValue" }, + }; + + MyClass substitutedValue = new MyClass() { Value = "SomeOtherValue" }; + + bool getterCalled = false; + + JsonSerializerOptions options = new(); + DefaultJsonTypeInfoResolver r = new(); + r.Modifiers.Add(ti => + { + if (ti.Type == typeof(TestClassWithCustomConverterOnProperty)) + { + JsonPropertyInfo propertyInfo = ti.Properties[0]; + if (!useCustomConverter) + { + propertyInfo.CustomConverter = null; + } + + propertyInfo.Get = (o) => + { + Assert.Same(obj, o); + Assert.False(getterCalled); + getterCalled = true; + return substitutedValue; + }; + } + }); + + options.TypeInfoResolver = r; + + string json = JsonSerializer.Serialize(obj, options); + if (useCustomConverter) + { + Assert.Equal("""{"MyClassProperty":"SomeOtherValue"}""", json); + } + else + { + Assert.Equal("""{"MyClassProperty":{"Value":"SomeOtherValue","Thing":null}}""", json); + } + + TestClassWithCustomConverterOnProperty deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(substitutedValue.Value, deserialized.MyClassProperty.Value); + + Assert.True(getterCalled); + } + + [Fact] + public static void JsonPropertyInfoSetIsNullAndMutableWhenUsingCreateJsonPropertyInfo() + { + JsonSerializerOptions options = new(); + JsonTypeInfo typeInfo = options.TypeInfoResolver.GetTypeInfo(typeof(TestClassWithCustomConverterOnProperty), options); + JsonPropertyInfo propertyInfo = typeInfo.CreateJsonPropertyInfo(typeof(MyClass), "test"); + Assert.Null(propertyInfo.Set); + Action set = (obj, val) => + { + throw new NotImplementedException(); + }; + + propertyInfo.Set = set; + Assert.Same(set, propertyInfo.Set); + } + + [Fact] + public static void JsonPropertyInfoSetIsNotNullForDefaultResolver() + { + JsonSerializerOptions options = new(); + JsonTypeInfo typeInfo = options.TypeInfoResolver.GetTypeInfo(typeof(TestClassWithCustomConverterOnProperty), options); + JsonPropertyInfo propertyInfo = typeInfo.Properties[0]; + + Assert.NotNull(propertyInfo.Set); + + TestClassWithCustomConverterOnProperty obj = new(); + + MyClass value = new MyClass(); + propertyInfo.Set(obj, value); + Assert.Same(value, obj.MyClassProperty); + + MyClass sentinel = new(); + Action set = (o, value) => + { + Assert.Same(obj, o); + Assert.Same(sentinel, value); + obj.MyClassProperty = sentinel; + }; + + propertyInfo.Set = set; + Assert.Same(set, propertyInfo.Set); + + propertyInfo.Set(obj, sentinel); + Assert.Same(obj.MyClassProperty, sentinel); + } + + [Fact] + public static void JsonPropertyInfoSetPropertyDeserializableButNotSerializableWhenNull() + { + JsonSerializerOptions options = new(); + DefaultJsonTypeInfoResolver r = new(); + r.Modifiers.Add(ti => + { + if (ti.Type == typeof(TestClassWithCustomConverterOnProperty)) + { + JsonPropertyInfo propertyInfo = ti.Properties[0]; + propertyInfo.Set = null; + } + }); + + options.TypeInfoResolver = r; + + TestClassWithCustomConverterOnProperty obj = new() + { + MyClassProperty = new MyClass() { Value = "SomeValue" }, + }; + + string json = JsonSerializer.Serialize(obj, options); + Assert.Equal("""{"MyClassProperty":"SomeValue"}""", json); + + TestClassWithCustomConverterOnProperty deserialized = JsonSerializer.Deserialize(json, options); + Assert.Null(deserialized.MyClassProperty); + } + + [Theory] + [InlineData(true)] + [InlineData(false)] + public static void JsonPropertyInfoSetIsRespected(bool useCustomConverter) + { + TestClassWithCustomConverterOnProperty obj = new() + { + MyClassProperty = new MyClass() { Value = "SomeValue" }, + }; + + MyClass substitutedValue = new MyClass() { Value = "SomeOtherValue" }; + bool setterCalled = false; + + JsonSerializerOptions options = new(); + DefaultJsonTypeInfoResolver r = new(); + r.Modifiers.Add(ti => + { + if (ti.Type == typeof(TestClassWithCustomConverterOnProperty)) + { + JsonPropertyInfo propertyInfo = ti.Properties[0]; + if (!useCustomConverter) + { + propertyInfo.CustomConverter = null; + } + + propertyInfo.Set = (o, val) => + { + var testClass = (TestClassWithCustomConverterOnProperty)o; + Assert.IsType(val); + MyClass myClass = (MyClass)val; + Assert.Equal(obj.MyClassProperty.Value, myClass.Value); + + testClass.MyClassProperty = substitutedValue; + Assert.False(setterCalled); + setterCalled = true; + }; + } + }); + + options.TypeInfoResolver = r; + + string json = JsonSerializer.Serialize(obj, options); + if (useCustomConverter) + { + Assert.Equal("""{"MyClassProperty":"SomeValue"}""", json); + } + else + { + Assert.Equal("""{"MyClassProperty":{"Value":"SomeValue","Thing":null}}""", json); + } + + TestClassWithCustomConverterOnProperty deserialized = JsonSerializer.Deserialize(json, options); + Assert.Same(substitutedValue, deserialized.MyClassProperty); + Assert.True(setterCalled); + } + + private class TestClassWithCustomConverterOnProperty + { + [JsonConverter(typeof(MyClassConverterOriginal))] + public MyClass MyClassProperty { get; set; } + } + + private class MyClassConverterOriginal : JsonConverter + { + public override MyClass? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) + { + if (reader.TokenType != JsonTokenType.String) + throw new InvalidOperationException($"Wrong token type: {reader.TokenType}"); + + MyClass myClass = new MyClass(); + myClass.Value = reader.GetString(); + return myClass; + } + + public override void Write(Utf8JsonWriter writer, MyClass value, JsonSerializerOptions options) + { + writer.WriteStringValue(value.Value); + } + } + + private class MyClassCustomConverter : JsonConverter + { + private string _prefix; + + public MyClassCustomConverter(string prefix) + { + _prefix = prefix; + } + + public override MyClass? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) + { + if (reader.TokenType != JsonTokenType.String) + throw new InvalidOperationException($"Wrong token type: {reader.TokenType}"); + + MyClass myClass = new MyClass(); + myClass.Value = reader.GetString().Substring(_prefix.Length); + return myClass; + } + + public override void Write(Utf8JsonWriter writer, MyClass value, JsonSerializerOptions options) + { + writer.WriteStringValue(_prefix + value.Value); + } + } + } +} diff --git a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/DefaultJsonTypeInfoResolverTests.JsonTypeInfo.cs b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/DefaultJsonTypeInfoResolverTests.JsonTypeInfo.cs new file mode 100644 index 0000000000000..272efcc2332a7 --- /dev/null +++ b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/DefaultJsonTypeInfoResolverTests.JsonTypeInfo.cs @@ -0,0 +1,497 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System; +using System.Collections.Generic; +using System.Reflection; +using System.Text; +using System.Text.Json.Serialization.Metadata; +using System.Text.Json.Tests; +using System.Threading.Tasks; +using Xunit; + +namespace System.Text.Json.Serialization.Tests +{ + public static partial class DefaultJsonTypeInfoResolverTests + { + [Theory] + [InlineData(typeof(object))] + [InlineData(typeof(int))] + [InlineData(typeof(string))] + [InlineData(typeof(SomeClass))] + [InlineData(typeof(StructWithFourArgs))] + [InlineData(typeof(Dictionary))] + [InlineData(typeof(DictionaryWrapper))] + [InlineData(typeof(List))] + [InlineData(typeof(ListWrapper))] + public static void TypeInfoPropertiesDefaults(Type type) + { + DefaultJsonTypeInfoResolver r = new(); + JsonSerializerOptions o = new(); + o.Converters.Add(new CustomThrowingConverter()); + + JsonTypeInfo ti = r.GetTypeInfo(type, o); + + Assert.Same(o, ti.Options); + Assert.NotNull(ti.Properties); + + if (ti.Kind == JsonTypeInfoKind.None) + { + Assert.Null(ti.CreateObject); + Assert.Throws(() => ti.CreateObject = () => Activator.CreateInstance(type)); + } + else + { + Assert.NotNull(ti.CreateObject); + Func createObj = () => Activator.CreateInstance(type); + ti.CreateObject = createObj; + Assert.Same(createObj, ti.CreateObject); + } + + JsonPropertyInfo property = ti.CreateJsonPropertyInfo(typeof(string), "foo"); + Assert.NotNull(property); + + if (ti.Kind == JsonTypeInfoKind.Object) + { + Assert.InRange(ti.Properties.Count, 1, 10); + Assert.False(ti.Properties.IsReadOnly); + ti.Properties.Add(property); + ti.Properties.Remove(property); + } + else + { + Assert.Equal(0, ti.Properties.Count); + Assert.True(ti.Properties.IsReadOnly); + Assert.Throws(() => ti.Properties.Add(property)); + Assert.Throws(() => ti.Properties.Insert(0, property)); + Assert.Throws(() => ti.Properties.Clear()); + } + + Assert.Null(ti.NumberHandling); + JsonNumberHandling numberHandling = JsonNumberHandling.AllowReadingFromString | JsonNumberHandling.WriteAsString; + ti.NumberHandling = numberHandling; + Assert.Equal(numberHandling, ti.NumberHandling); + + InvokeGeneric(type, nameof(TypeInfoPropertiesDefaults_Generic), ti); + } + + private static void TypeInfoPropertiesDefaults_Generic(JsonTypeInfo ti) + { + if (ti.Kind == JsonTypeInfoKind.None) + { + Assert.Null(ti.CreateObject); + Assert.Throws(() => ti.CreateObject = () => (T)Activator.CreateInstance(typeof(T))); + } + else + { + bool createObjCalled = false; + Assert.NotNull(ti.CreateObject); + Func createObj = () => + { + createObjCalled = true; + return default(T); + }; + + ti.CreateObject = createObj; + Assert.Same(createObj, ti.CreateObject); + + JsonTypeInfo untyped = ti; + if (typeof(T).IsValueType) + { + Assert.NotSame(createObj, untyped.CreateObject); + } + else + { + Assert.Same(createObj, untyped.CreateObject); + } + + Assert.Same(untyped.CreateObject, untyped.CreateObject); + Assert.Same(createObj, ti.CreateObject); + untyped.CreateObject(); + Assert.True(createObjCalled); + + ti.CreateObject = null; + Assert.Null(ti.CreateObject); + Assert.Null(untyped.CreateObject); + + bool untypedCreateObjCalled = false; + Func untypedCreateObj = () => + { + untypedCreateObjCalled = true; + return default(T); + }; + untyped.CreateObject = untypedCreateObj; + Assert.Same(untypedCreateObj, untyped.CreateObject); + Assert.Same(ti.CreateObject, ti.CreateObject); + Assert.NotSame(untypedCreateObj, ti.CreateObject); + + ti.CreateObject(); + Assert.True(untypedCreateObjCalled); + + untyped.CreateObject = null; + Assert.Null(ti.CreateObject); + Assert.Null(untyped.CreateObject); + } + } + + [Fact] + public static void TypeInfoKindNoneNumberHandlingDirect() + { + DefaultJsonTypeInfoResolver r = new(); + r.Modifiers.Add((ti) => + { + if (ti.Type == typeof(int)) + { + ti.NumberHandling = JsonNumberHandling.WriteAsString | JsonNumberHandling.AllowReadingFromString; + } + }); + + JsonSerializerOptions o = new(); + o.TypeInfoResolver = r; + + string json = JsonSerializer.Serialize(13, o); + Assert.Equal(@"""13""", json); + + var deserialized = JsonSerializer.Deserialize(json, o); + Assert.Equal(13, deserialized); + } + + [Fact] + public static void TypeInfoKindNoneNumberHandlingDirectThroughObject() + { + DefaultJsonTypeInfoResolver r = new(); + r.Modifiers.Add((ti) => + { + if (ti.Type == typeof(int)) + { + ti.NumberHandling = JsonNumberHandling.WriteAsString | JsonNumberHandling.AllowReadingFromString; + } + }); + + JsonSerializerOptions o = new(); + o.TypeInfoResolver = r; + + string json = JsonSerializer.Serialize(13, o); + Assert.Equal(@"""13""", json); + + var deserialized = JsonSerializer.Deserialize(json, o); + Assert.Equal("13", ((JsonElement)deserialized).GetString()); + } + + [Fact] + public static void TypeInfoKindNoneNumberHandling() + { + DefaultJsonTypeInfoResolver r = new(); + r.Modifiers.Add((ti) => + { + if (ti.Type == typeof(int) || ti.Type == typeof(object)) + { + ti.NumberHandling = JsonNumberHandling.WriteAsString | JsonNumberHandling.AllowReadingFromString; + } + }); + + JsonSerializerOptions o = new(); + o.TypeInfoResolver = r; + + SomeClass testObj = new SomeClass() + { + ObjProp = 45, + IntProp = 13, + }; + + string json = JsonSerializer.Serialize(testObj, o); + Assert.Equal(@"{""ObjProp"":""45"",""IntProp"":""13""}", json); + + var deserialized = JsonSerializer.Deserialize(json, o); + Assert.Equal(testObj.ObjProp.ToString(), ((JsonElement)deserialized.ObjProp).GetString()); + Assert.Equal(testObj.IntProp, deserialized.IntProp); + } + + [Fact] + public static void RecursiveTypeNumberHandling() + { + DefaultJsonTypeInfoResolver r = new(); + r.Modifiers.Add((ti) => + { + if (ti.Type == typeof(SomeRecursiveClass)) + { + ti.NumberHandling = JsonNumberHandling.WriteAsString | JsonNumberHandling.AllowReadingFromString; + } + }); + + JsonSerializerOptions o = new(); + o.TypeInfoResolver = r; + + SomeRecursiveClass testObj = new SomeRecursiveClass() + { + IntProp = 13, + RecursiveProperty = new SomeRecursiveClass() + { + IntProp = 14, + }, + }; + + string json = JsonSerializer.Serialize(testObj, o); + Assert.Equal(@"{""IntProp"":""13"",""RecursiveProperty"":{""IntProp"":""14"",""RecursiveProperty"":null}}", json); + + var deserialized = JsonSerializer.Deserialize(json, o); + Assert.Equal(testObj.IntProp, deserialized.IntProp); + Assert.NotNull(testObj.RecursiveProperty); + Assert.Equal(testObj.RecursiveProperty.IntProp, deserialized.RecursiveProperty.IntProp); + Assert.Null(testObj.RecursiveProperty.RecursiveProperty); + } + + [Theory] + [InlineData(typeof(SomeClass), typeof(object))] + [InlineData(typeof(object), typeof(string))] + [InlineData(typeof(object), typeof(int))] + [InlineData(typeof(string), typeof(int))] + [InlineData(typeof(int), typeof(string))] + [InlineData(typeof(int), typeof(double))] + public static void TypeInfoOfWrongTypeOnObject(Type expectedType, Type actualType) + { + DefaultJsonTypeInfoResolver dr = new(); + TestResolver r = new((type, options) => + { + if (type == expectedType) + { + return dr.GetTypeInfo(actualType, options); + } + + return dr.GetTypeInfo(type, options); + }); + + JsonSerializerOptions o = new(); + o.TypeInfoResolver = r; + + SomeClass testObj = new() + { + ObjProp = "test", + }; + + Assert.Throws(() => JsonSerializer.Serialize(testObj, o)); + } + + [Fact] + public static void TypeInfoOfWrongOptions() + { + JsonSerializerOptions wrongOptions = new(); + DefaultJsonTypeInfoResolver dr = new(); + TestResolver r = new((type, options) => + { + if (type == typeof(int)) + { + return dr.GetTypeInfo(type, wrongOptions); + } + + return dr.GetTypeInfo(type, options); + }); + + JsonSerializerOptions o = new(); + o.TypeInfoResolver = r; + + SomeClass testObj = new() + { + IntProp = 17, + }; + + Assert.Throws(() => JsonSerializer.Serialize(testObj, o)); + } + + [Theory] + [InlineData(typeof(SomeClass), typeof(object))] + [InlineData(typeof(object), typeof(string))] + [InlineData(typeof(object), typeof(int))] + [InlineData(typeof(int), typeof(string))] + [InlineData(typeof(int), typeof(double))] + public static void TypeInfoOfWrongTypeDirectCall(Type expectedType, Type actualType) + { + DefaultJsonTypeInfoResolver dr = new(); + TestResolver r = new((type, options) => + { + if (type == expectedType) + { + return dr.GetTypeInfo(actualType, options); + } + + return dr.GetTypeInfo(type, options); + }); + + JsonSerializerOptions o = new(); + o.TypeInfoResolver = r; + + object testObj = Activator.CreateInstance(expectedType); + + Assert.Throws(() => JsonSerializer.Serialize(testObj, expectedType, o)); + } + + [Theory] + [MemberData(nameof(GetTypeInfoTestData))] + public static void TypeInfoIsImmutableAfterFirstUsage(T testObj) + { + JsonTypeInfo untyped = null; + DefaultJsonTypeInfoResolver dr = new(); + TestResolver r = new((typeToResolve, options) => + { + var ret = dr.GetTypeInfo(typeToResolve, options); + if (typeToResolve == typeof(T)) + { + Assert.Null(untyped); + untyped = ret; + } + + return ret; + }); + + JsonSerializerOptions o = new(); + o.TypeInfoResolver = r; + + Assert.NotNull(JsonSerializer.Serialize(testObj, typeof(T), o)); + Assert.NotNull(untyped); + + JsonTypeInfo typeInfo = (JsonTypeInfo)untyped; + + if (typeInfo.Kind == JsonTypeInfoKind.None) + { + Assert.Null(typeInfo.CreateObject); + Assert.Null(untyped.CreateObject); + } + else + { + Assert.NotNull(typeInfo.CreateObject); + Assert.NotNull(untyped.CreateObject); + } + + Assert.Null(typeInfo.NumberHandling); + + TestTypeInfoImmutability(typeInfo); + } + + private static void TestTypeInfoImmutability(JsonTypeInfo typeInfo) + { + JsonTypeInfo untyped = typeInfo; + Assert.Equal(typeof(T), typeInfo.Type); + Assert.True(typeInfo.Converter.CanConvert(typeof(T))); + + JsonPropertyInfo prop = typeInfo.CreateJsonPropertyInfo(typeof(string), "foo"); + Assert.Throws(() => untyped.CreateObject = untyped.CreateObject); + Assert.Throws(() => typeInfo.CreateObject = typeInfo.CreateObject); + Assert.Throws(() => typeInfo.NumberHandling = typeInfo.NumberHandling); + Assert.Throws(() => typeInfo.Properties.Clear()); + Assert.Throws(() => typeInfo.Properties.Add(prop)); + Assert.Throws(() => typeInfo.Properties.Insert(0, prop)); + + foreach (var property in typeInfo.Properties) + { + Assert.NotNull(property.PropertyType); + Assert.Null(property.CustomConverter); + Assert.NotNull(property.Name); + Assert.NotNull(property.Get); + Assert.NotNull(property.Set); + Assert.Null(property.ShouldSerialize); + Assert.Null(typeInfo.NumberHandling); + + Assert.Throws(() => property.CustomConverter = property.CustomConverter); + Assert.Throws(() => property.Name = property.Name); + Assert.Throws(() => property.Get = property.Get); + Assert.Throws(() => property.Set = property.Set); + Assert.Throws(() => property.ShouldSerialize = property.ShouldSerialize); + Assert.Throws(() => property.NumberHandling = property.NumberHandling); + } + } + + [Theory] + [InlineData(typeof(object), JsonTypeInfoKind.None)] + [InlineData(typeof(string), JsonTypeInfoKind.None)] + [InlineData(typeof(int), JsonTypeInfoKind.None)] + [InlineData(typeof(SomeRecursiveClass) /* custom converter */, JsonTypeInfoKind.None)] + [InlineData(typeof(SomeClass), JsonTypeInfoKind.Object)] + [InlineData(typeof(DefaultJsonTypeInfoResolverTests), JsonTypeInfoKind.Object)] + [InlineData(typeof(StructWithFourArgs), JsonTypeInfoKind.Object)] + [InlineData(typeof(Dictionary), JsonTypeInfoKind.Dictionary)] + [InlineData(typeof(DictionaryWrapper), JsonTypeInfoKind.Dictionary)] + [InlineData(typeof(List), JsonTypeInfoKind.Enumerable)] + [InlineData(typeof(ListWrapper), JsonTypeInfoKind.Enumerable)] + [InlineData(typeof(int[]), JsonTypeInfoKind.Enumerable)] + public static void JsonTypeInfoKindIsReportedCorrectly(Type type, JsonTypeInfoKind expectedJsonTypeInfoKind) + { + InvokeGeneric(type, nameof(JsonTypeInfoKindIsReportedCorrectly_Generic), expectedJsonTypeInfoKind); + } + + private static void JsonTypeInfoKindIsReportedCorrectly_Generic(JsonTypeInfoKind expectedJsonTypeInfoKind) + { + DefaultJsonTypeInfoResolver r = new(); + JsonSerializerOptions o = new(); + o.Converters.Add(new CustomThrowingConverter()); + JsonTypeInfo ti = r.GetTypeInfo(typeof(T), o); + Assert.Equal(expectedJsonTypeInfoKind, ti.Kind); + + ti = JsonTypeInfo.CreateJsonTypeInfo(typeof(T), o); + Assert.Equal(expectedJsonTypeInfoKind, ti.Kind); + + ti = JsonTypeInfo.CreateJsonTypeInfo(o); + Assert.Equal(expectedJsonTypeInfoKind, ti.Kind); + } + + [Theory] + [InlineData(typeof(object))] + [InlineData(typeof(string))] + [InlineData(typeof(int))] + [InlineData(typeof(SomeRecursiveClass))] + [InlineData(typeof(SomeClass))] + [InlineData(typeof(DefaultJsonTypeInfoResolverTests))] + [InlineData(typeof(StructWithFourArgs))] + [InlineData(typeof(Dictionary))] + [InlineData(typeof(DictionaryWrapper))] + [InlineData(typeof(List))] + [InlineData(typeof(ListWrapper))] + [InlineData(typeof(int[]))] + public static void CreateJsonTypeInfo(Type type) + { + InvokeGeneric(type, nameof(CreateJsonTypeInfo_Generic)); + } + + private static void CreateJsonTypeInfo_Generic() + { + TestCreateJsonTypeInfo((o) => (JsonTypeInfo)JsonTypeInfo.CreateJsonTypeInfo(typeof(T), o)); + TestCreateJsonTypeInfo((o) => JsonTypeInfo.CreateJsonTypeInfo(o)); + + static void TestCreateJsonTypeInfo(Func> getTypeInfo) + { + JsonSerializerOptions o = new(); + TestCreateJsonTypeInfoInstance(o, getTypeInfo(o)); + + o = new JsonSerializerOptions(); + var conv = new DummyConverter(); + o.Converters.Add(conv); + JsonTypeInfo ti = getTypeInfo(o); + Assert.Same(conv, ti.Converter); + Assert.Equal(JsonTypeInfoKind.None, ti.Kind); + TestCreateJsonTypeInfoInstance(o, ti); + } + + static void TestCreateJsonTypeInfoInstance(JsonSerializerOptions o, JsonTypeInfo ti) + { + Assert.Equal(typeof(T), ti.Type); + Assert.NotNull(ti.Converter); + Assert.True(ti.Converter.CanConvert(typeof(T))); + + JsonSerializer.Serialize(default(T), ti); + + JsonTypeInfo untyped = ti; + Assert.Null(ti.CreateObject); + Assert.Null(untyped.CreateObject); + + TestTypeInfoImmutability(ti); + } + } + + public static IEnumerable GetTypeInfoTestData() + { + yield return new object[] { "test" }; + yield return new object[] { 13 }; + yield return new object[] { new SomeClass { IntProp = 17 } }; + yield return new object[] { new SomeRecursiveClass() }; + } + } +} diff --git a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/DefaultJsonTypeInfoResolverTests.JsonTypeInfoApis.cs b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/DefaultJsonTypeInfoResolverTests.JsonTypeInfoApis.cs new file mode 100644 index 0000000000000..ce22ff0b8d711 --- /dev/null +++ b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/DefaultJsonTypeInfoResolverTests.JsonTypeInfoApis.cs @@ -0,0 +1,35 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System; +using System.Collections.Generic; +using System.Text; +using System.Text.Json.Serialization.Metadata; +using System.Threading.Tasks; +using Xunit; + +namespace System.Text.Json.Serialization.Tests +{ + // TODO: ensure converter rooting happens for APIs taking JsonTypeInfo + public static partial class DefaultJsonTypeInfoResolverTests + { + [Theory] + [InlineData("value", @"""value""")] + [InlineData(5, @"5")] + [MemberData(nameof(JsonSerializerSerializeWithTypeInfoOfT_TestData))] + public static void JsonSerializerSerializeWithTypeInfoOfT(T testObj, string expectedJson) + { + DefaultJsonTypeInfoResolver r = new(); + JsonSerializerOptions o = new(); + o.TypeInfoResolver = r; + JsonTypeInfo typeInfo = (JsonTypeInfo)r.GetTypeInfo(typeof(T), o); + string json = JsonSerializer.Serialize(testObj, typeInfo); + Assert.Equal(expectedJson, json); + } + + public static IEnumerable JsonSerializerSerializeWithTypeInfoOfT_TestData() + { + yield return new object[] { new SomeClass() { IntProp = 15, ObjProp = 17m }, @"{""ObjProp"":17,""IntProp"":15}" }; + } + } +} diff --git a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/DefaultJsonTypeInfoResolverTests.cs b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/DefaultJsonTypeInfoResolverTests.cs new file mode 100644 index 0000000000000..c293f30b380c0 --- /dev/null +++ b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/DefaultJsonTypeInfoResolverTests.cs @@ -0,0 +1,228 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System; +using System.Collections.Generic; +using System.Reflection; +using System.Text; +using System.Text.Json.Serialization.Metadata; +using System.Threading.Tasks; +using Xunit; + +namespace System.Text.Json.Serialization.Tests +{ + public static partial class DefaultJsonTypeInfoResolverTests + { + [Fact] + public static void GetTypeInfoNullArguments() + { + DefaultJsonTypeInfoResolver r = new(); + Assert.Throws(() => r.GetTypeInfo(null, null)); + Assert.Throws(() => r.GetTypeInfo(null, new JsonSerializerOptions())); + Assert.Throws(() => r.GetTypeInfo(typeof(string), null)); + } + + [Fact] + public static void ModifiersIsEmptyNonCastableIList() + { + DefaultJsonTypeInfoResolver r = new(); + Assert.NotNull(r.Modifiers); + Assert.Null(r.Modifiers as List>); + Assert.False(r.Modifiers.GetType().IsPublic); + Assert.Empty(r.Modifiers); + Assert.Equal(0, r.Modifiers.Count); + } + + [Fact] + public static void ModifiersAreMutableAndInterfaceIsImplementedCorrectly() + { + DefaultJsonTypeInfoResolver r = new(); + Assert.Same(r.Modifiers, r.Modifiers); + var mods = r.Modifiers; + + Action el0 = (ti) => { }; + Action el1 = (ti) => { }; + Action el2 = (ti) => { }; + Assert.NotSame(el0, el1); + Assert.NotSame(el1, el2); + IEnumerator> enumerator; + + Assert.Equal(0, mods.Count); + Assert.False(mods.IsReadOnly); + Assert.Throws(() => mods[-1]); + Assert.Throws(() => mods[0]); + Assert.Throws(() => mods[1]); + + using (enumerator = mods.GetEnumerator()) + { + Assert.False(enumerator.MoveNext()); + } + + mods.Add(el0); + Assert.Equal(1, mods.Count); + Assert.Throws(() => mods[-1]); + Assert.Same(el0, mods[0]); + Assert.Throws(() => mods[1]); + + using (enumerator = mods.GetEnumerator()) + { + Assert.True(enumerator.MoveNext()); + Assert.Same(el0, enumerator.Current); + Assert.False(enumerator.MoveNext()); + } + + mods.Clear(); + Assert.Equal(0, mods.Count); + + using (enumerator = mods.GetEnumerator()) + { + Assert.False(enumerator.MoveNext()); + } + + mods.Insert(0, el0); + Assert.Equal(1, mods.Count); + Assert.Same(el0, mods[0]); + Assert.True(mods.Remove(el0)); + Assert.Equal(0, mods.Count); + + mods.Insert(0, el1); + mods.Insert(0, el0); + Assert.Equal(2, mods.Count); + Assert.Same(el0, mods[0]); + Assert.Same(el1, mods[1]); + Assert.False(mods.Remove(el2)); + mods.RemoveAt(1); + Assert.Equal(1, mods.Count); + Assert.Same(el0, mods[0]); + } + + [Fact] + public static void EmptyModifiersAreImmutableAfterFirstUsage() + { + DefaultJsonTypeInfoResolver r = new(); + IList> mods = r.Modifiers; + + Assert.NotNull(r.GetTypeInfo(typeof(string), new JsonSerializerOptions())); + + Assert.True(mods.IsReadOnly); + Assert.Same(mods, r.Modifiers); + Assert.Equal(0, mods.Count); + + Assert.Throws(() => mods.Add((ti) => { })); + Assert.Throws(() => mods.Insert(0, (ti) => { })); + } + + [Fact] + public static void NonEmptyModifiersAreImmutableAfterFirstUsage() + { + DefaultJsonTypeInfoResolver r = new(); + IList> mods = r.Modifiers; + Action el0 = (ti) => { }; + Action el1 = (ti) => { }; + mods.Add(el0); + mods.Add(el1); + + Assert.NotNull(r.GetTypeInfo(typeof(string), new JsonSerializerOptions())); + + Assert.True(mods.IsReadOnly); + Assert.Same(mods, r.Modifiers); + Assert.Equal(2, mods.Count); + + Assert.Throws(() => mods.Add((ti) => { })); + Assert.Throws(() => mods.Insert(0, (ti) => { })); + Assert.Throws(() => mods.Remove(el0)); + Assert.Throws(() => mods.RemoveAt(0)); + } + + [Fact] + public static void ModifiersAreCalledAndModifyTypeInfos() + { + DefaultJsonTypeInfoResolver r = new(); + JsonTypeInfo storedTypeInfo = null; + bool createObjectCalled = false; + bool secondModifierCalled = false; + r.Modifiers.Add((ti) => + { + Assert.Null(storedTypeInfo); + storedTypeInfo = ti; + + // marker that test has modified something + ti.CreateObject = () => + { + Assert.False(createObjectCalled); + createObjectCalled = true; + + // we don't care what's returned as it won't be used by deserialization + return null; + }; + }); + + r.Modifiers.Add((ti) => + { + // this proves we've been called after first modifier + Assert.NotNull(storedTypeInfo); + Assert.Same(storedTypeInfo, ti); + secondModifierCalled = true; + }); + + JsonTypeInfo returnedTypeInfo = r.GetTypeInfo(typeof(InvalidOperationException), new JsonSerializerOptions()); + + Assert.NotNull(storedTypeInfo); + Assert.Same(storedTypeInfo, returnedTypeInfo); + + Assert.False(createObjectCalled); + // we call our previously set marker + storedTypeInfo.CreateObject(); + + Assert.True(createObjectCalled); + Assert.True(secondModifierCalled); + } + + private static void InvokeGeneric(Type type, string methodName, params object[] args) + { + typeof(DefaultJsonTypeInfoResolverTests) + .GetMethod(methodName, BindingFlags.NonPublic | BindingFlags.Static) + .MakeGenericMethod(type) + .Invoke(null, args); + } + + private class SomeClass + { + public object ObjProp { get; set; } + public int IntProp { get; set; } + } + + private class CustomThrowingConverter : JsonConverter + { + public override T? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) => throw new NotImplementedException(); + public override void Write(Utf8JsonWriter writer, T value, JsonSerializerOptions options) => throw new NotImplementedException(); + } + + private class DummyConverter : JsonConverter + { + public override T? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) => default(T); + public override void Write(Utf8JsonWriter writer, T value, JsonSerializerOptions options) { } + } + + private class SomeRecursiveClass + { + public int IntProp { get; set; } + public SomeRecursiveClass RecursiveProperty { get; set; } + } + + private class TestResolver : IJsonTypeInfoResolver + { + private Func _getTypeInfo; + + public TestResolver(Func getTypeInfo) + { + _getTypeInfo = getTypeInfo; + } + + public JsonTypeInfo GetTypeInfo(Type type, JsonSerializerOptions options) + { + return _getTypeInfo(type, options); + } + } + } +} diff --git a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/MetadataTests.Options.cs b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/MetadataTests.Options.cs index 1a092379ee891..c1f53298ac5df 100644 --- a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/MetadataTests.Options.cs +++ b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/MetadataTests/MetadataTests.Options.cs @@ -55,9 +55,9 @@ public void AddContextOverwritesOptionsForFreshContext() // Those options are overwritten when context is binded via options.AddContext(); JsonSerializerOptions options = new(); options.AddContext(); // No error. - FieldInfo contextField = typeof(JsonSerializerOptions).GetField("_serializerContext", BindingFlags.NonPublic | BindingFlags.Instance); - Assert.NotNull(contextField); - Assert.Same(options, ((JsonSerializerContext)contextField.GetValue(options)).Options); + FieldInfo resolverField = typeof(JsonSerializerOptions).GetField("_typeInfoResolver", BindingFlags.NonPublic | BindingFlags.Instance); + Assert.NotNull(resolverField); + Assert.Same(options, ((JsonSerializerContext)resolverField.GetValue(options)).Options); } [Fact] diff --git a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/OptionsTests.cs b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/OptionsTests.cs index 064f5d9c94894..951441d8b482c 100644 --- a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/OptionsTests.cs +++ b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/OptionsTests.cs @@ -6,6 +6,7 @@ using System.IO; using System.Reflection; using System.Text.Encodings.Web; +using System.Text.Json.Serialization.Metadata; using System.Text.Unicode; using Xunit; @@ -31,21 +32,28 @@ public static void SetOptionsFail() { var options = new JsonSerializerOptions(); - // Verify these do not throw. - options.Converters.Clear(); - TestConverter tc = new TestConverter(); - options.Converters.Add(tc); - options.Converters.Insert(0, new TestConverter()); - options.Converters.Remove(tc); - options.Converters.RemoveAt(0); + TestIListNonThrowingOperationsWhenMutable(options.Converters, () => new TestConverter()); + + // Verify TypeInfoResolver throws on null resolver + Assert.Throws(() => options.TypeInfoResolver = null); + + // Verify default TypeInfoResolver throws + Action tiModifier = (ti) => { }; + Assert.Throws(() => (options.TypeInfoResolver as DefaultJsonTypeInfoResolver).Modifiers.Clear()); + Assert.Throws(() => (options.TypeInfoResolver as DefaultJsonTypeInfoResolver).Modifiers.Add(tiModifier)); + Assert.Throws(() => (options.TypeInfoResolver as DefaultJsonTypeInfoResolver).Modifiers.Insert(0, tiModifier)); + + // Now set DefaultTypeInfoResolver + options.TypeInfoResolver = new DefaultJsonTypeInfoResolver(); + TestIListNonThrowingOperationsWhenMutable((options.TypeInfoResolver as DefaultJsonTypeInfoResolver).Modifiers, () => (ti) => { }); // Add one item for later. + TestConverter tc = new TestConverter(); options.Converters.Add(tc); + (options.TypeInfoResolver as DefaultJsonTypeInfoResolver).Modifiers.Add(tiModifier); - // Verify converter collection throws on null adds. - Assert.Throws(() => options.Converters.Add(null)); - Assert.Throws(() => options.Converters.Insert(0, null)); - Assert.Throws(() => options.Converters[0] = null); + TestIListThrowingOperationsWhenMutable(options.Converters); + TestIListThrowingOperationsWhenMutable((options.TypeInfoResolver as DefaultJsonTypeInfoResolver).Modifiers); // Perform serialization. JsonSerializer.Deserialize("1", options); @@ -62,14 +70,8 @@ public static void SetOptionsFail() Assert.Equal(JsonCommentHandling.Disallow, options.ReadCommentHandling); Assert.False(options.WriteIndented); - Assert.Equal(tc, options.Converters[0]); - Assert.True(options.Converters.Contains(tc)); - options.Converters.CopyTo(new JsonConverter[1] { null }, 0); - Assert.Equal(1, options.Converters.Count); - Assert.False(options.Converters.Equals(tc)); - Assert.NotNull(options.Converters.GetEnumerator()); - Assert.Equal(0, options.Converters.IndexOf(tc)); - Assert.False(options.Converters.IsReadOnly); + TestIListNonThrowingOperationsWhenImmutable(options.Converters, tc); + TestIListNonThrowingOperationsWhenImmutable((options.TypeInfoResolver as DefaultJsonTypeInfoResolver).Modifiers, tiModifier); // Setters should always throw; we don't check to see if the value is the same or not. Assert.Throws(() => options.AllowTrailingCommas = options.AllowTrailingCommas); @@ -82,13 +84,102 @@ public static void SetOptionsFail() Assert.Throws(() => options.PropertyNamingPolicy = options.PropertyNamingPolicy); Assert.Throws(() => options.ReadCommentHandling = options.ReadCommentHandling); Assert.Throws(() => options.WriteIndented = options.WriteIndented); + Assert.Throws(() => options.TypeInfoResolver = options.TypeInfoResolver); - Assert.Throws(() => options.Converters[0] = tc); - Assert.Throws(() => options.Converters.Clear()); - Assert.Throws(() => options.Converters.Add(tc)); - Assert.Throws(() => options.Converters.Insert(0, new TestConverter())); - Assert.Throws(() => options.Converters.Remove(tc)); - Assert.Throws(() => options.Converters.RemoveAt(0)); + TestIListThrowingOperationsWhenImmutable(options.Converters, tc); + TestIListThrowingOperationsWhenImmutable((options.TypeInfoResolver as DefaultJsonTypeInfoResolver).Modifiers, tiModifier); + + static void TestIListNonThrowingOperationsWhenMutable(IList list, Func newT) + { + list.Clear(); + T el = newT(); + list.Add(el); + Assert.Equal(1, list.Count); + list.Insert(0, newT()); + Assert.Equal(2, list.Count); + list.Remove(el); + Assert.Equal(1, list.Count); + list.RemoveAt(0); + Assert.Equal(0, list.Count); + Assert.False(list.IsReadOnly, "List should not be read-only"); + } + + static void TestIListThrowingOperationsWhenMutable(IList list) where T : class + { + // Verify collection throws on null adds. + Assert.Throws(() => list.Add(null)); + Assert.Throws(() => list.Insert(0, null)); + Assert.Throws(() => list[0] = null); + } + + static void TestIListNonThrowingOperationsWhenImmutable(IList list, T onlyElement) + { + Assert.Equal(onlyElement, list[0]); + Assert.True(list.Contains(onlyElement)); + list.CopyTo(new T[1] { default(T) }, 0); + Assert.Equal(1, list.Count); + Assert.False(list.Equals(onlyElement)); + Assert.NotNull(list.GetEnumerator()); + Assert.Equal(0, list.IndexOf(onlyElement)); + Assert.True(list.IsReadOnly, "List should be read-only"); + } + + static void TestIListThrowingOperationsWhenImmutable(IList list, T firstElement) + { + Assert.Throws(() => list[0] = firstElement); + Assert.Throws(() => list.Clear()); + Assert.Throws(() => list.Add(firstElement)); + Assert.Throws(() => list.Insert(0, firstElement)); + Assert.Throws(() => list.Remove(firstElement)); + Assert.Throws(() => list.RemoveAt(0)); + } + } + + [Fact] + public static void TypeInfoResolverIsNotNullAndCorrectType() + { + var options = new JsonSerializerOptions(); + Assert.NotNull(options.TypeInfoResolver); + Assert.IsType(options.TypeInfoResolver); + Assert.Same(options.TypeInfoResolver, options.TypeInfoResolver); + } + + [Fact] + public static void TypeInfoResolverCannotBeSetAfterAddingContext() + { + var options = new JsonSerializerOptions(); + options.AddContext(); + Assert.IsType(options.TypeInfoResolver); + Assert.Throws(() => options.TypeInfoResolver = new DefaultJsonTypeInfoResolver()); + } + + [Fact] + public static void TypeInfoResolverCannotBeSetOnOptionsCreatedFromContext() + { + var context = new JsonContext(); + var options = context.Options; + Assert.Same(context, options.TypeInfoResolver); + Assert.Throws(() => options.TypeInfoResolver = new DefaultJsonTypeInfoResolver()); + } + + [Fact] + public static void WhenAddingContextTypeInfoResolverAsContextOptionsAreSameAsOptions() + { + var options = new JsonSerializerOptions(); + options.AddContext(); + Assert.Same(options, (options.TypeInfoResolver as JsonContext).Options); + } + + [Fact] + public static void TypeInfoResolverCannotBeSetAfterContextIsSetThroughTypeInfoResolver() + { + var options = new JsonSerializerOptions(); + IJsonTypeInfoResolver resolver = new JsonContext(); + options.TypeInfoResolver = resolver; + Assert.Same(resolver, options.TypeInfoResolver); + + resolver = new DefaultJsonTypeInfoResolver(); + Assert.Throws(() => options.TypeInfoResolver = resolver); } [Fact] @@ -607,6 +698,10 @@ private static JsonSerializerOptions GetFullyPopulatedOptionsInstance() { options.ReferenceHandler = ReferenceHandler.Preserve; } + else if (propertyType == typeof(IJsonTypeInfoResolver)) + { + options.TypeInfoResolver = new DefaultJsonTypeInfoResolver(); + } else if (propertyType.IsValueType) { options.ReadCommentHandling = JsonCommentHandling.Disallow; diff --git a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/TypeInfoResolverFunctionalTests.cs b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/TypeInfoResolverFunctionalTests.cs new file mode 100644 index 0000000000000..afeed4f0e1fc5 --- /dev/null +++ b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/Serialization/TypeInfoResolverFunctionalTests.cs @@ -0,0 +1,534 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System.Text.Json.Serialization.Metadata; +using Xunit; + +namespace System.Text.Json.Serialization.Tests +{ + public static class TypeInfoResolverFunctionalTests + { + [Fact] + public static void AddPrefixToEveryPropertyOfClass() + { + DefaultJsonTypeInfoResolver resolver = new(); + resolver.Modifiers.Add((ti) => + { + if (ti.Type == typeof(TestClass)) + { + Assert.Equal(JsonTypeInfoKind.Object, ti.Kind); + foreach (var prop in ti.Properties) + { + prop.Name = "renamed_" + prop.Name; + } + } + }); + + JsonSerializerOptions options = new JsonSerializerOptions(); + options.IncludeFields = true; + options.TypeInfoResolver = resolver; + + TestClass originalObj = new TestClass() + { + TestField = "test value", + TestProperty = 42, + }; + + string json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""renamed_TestProperty"":42,""renamed_TestField"":""test value""}", json); + + TestClass deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(originalObj.TestProperty, deserialized.TestProperty); + } + + [Fact] + public static void AppendCharacterWhenSerializingField() + { + DefaultJsonTypeInfoResolver resolver = new(); + resolver.Modifiers.Add((ti) => + { + if (ti.Type == typeof(TestClass)) + { + Assert.Equal(JsonTypeInfoKind.Object, ti.Kind); + // Because IncludeFields is false + Assert.Equal(1, ti.Properties.Count); + JsonPropertyInfo field = ti.CreateJsonPropertyInfo(typeof(string), "TestField"); + field.Get = (o) => + { + var obj = (TestClass)o; + return obj.TestField + "X"; + }; + field.Set = (o, val) => + { + var obj = (TestClass)o; + var value = (string)val; + // We append 'X' on serialization + // therefore on deserialization we remove last character + obj.TestField = value.Substring(0, value.Length - 1); + }; + ti.Properties.Add(field); + } + }); + + JsonSerializerOptions options = new JsonSerializerOptions(); + options.TypeInfoResolver = resolver; + + TestClass originalObj = new TestClass() + { + TestField = "test value", + TestProperty = 42, + }; + + string json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""TestProperty"":42,""TestField"":""test valueX""}", json); + + TestClass deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(originalObj.TestProperty, deserialized.TestProperty); + } + + [Fact] + public static void DoNotSerializeValue42() + { + DefaultJsonTypeInfoResolver resolver = new(); + resolver.Modifiers.Add((ti) => + { + if (ti.Type == typeof(TestClass)) + { + Assert.Equal(JsonTypeInfoKind.Object, ti.Kind); + foreach (var prop in ti.Properties) + { + if (prop.PropertyType == typeof(int)) + { + prop.ShouldSerialize = (o, val) => + { + return (int)val != 42; + }; + } + } + } + }); + + JsonSerializerOptions options = new JsonSerializerOptions(); + options.IncludeFields = true; + options.TypeInfoResolver = resolver; + + TestClass originalObj = new TestClass() + { + TestField = "test value", + TestProperty = 43, + }; + + string json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""TestProperty"":43,""TestField"":""test value""}", json); + + originalObj.TestProperty = 42; + json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""TestField"":""test value""}", json); + + TestClass deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(0, deserialized.TestProperty); + } + + [Fact] + public static void DoNotSerializePropertyWithNameButDeserializeIt() + { + DefaultJsonTypeInfoResolver resolver = new(); + resolver.Modifiers.Add((ti) => + { + if (ti.Type == typeof(TestClass)) + { + Assert.Equal(JsonTypeInfoKind.Object, ti.Kind); + foreach (var prop in ti.Properties) + { + if (prop.Name == nameof(TestClass.TestProperty)) + { + prop.Get = null; + } + } + } + }); + + JsonSerializerOptions options = new JsonSerializerOptions(); + options.IncludeFields = true; + options.TypeInfoResolver = resolver; + + TestClass originalObj = new TestClass() + { + TestField = "test value", + TestProperty = 42, + }; + + string json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""TestField"":""test value""}", json); + + json = @"{""TestProperty"":42,""TestField"":""test value""}"; + + TestClass deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(originalObj.TestProperty, deserialized.TestProperty); + } + + [Fact] + public static void DoNotDeserializePropertyWithNameButSerializeIt() + { + DefaultJsonTypeInfoResolver resolver = new(); + resolver.Modifiers.Add((ti) => + { + if (ti.Type == typeof(TestClass)) + { + Assert.Equal(JsonTypeInfoKind.Object, ti.Kind); + foreach (var prop in ti.Properties) + { + if (prop.Name == nameof(TestClass.TestProperty)) + { + prop.Set = null; + } + } + } + }); + + JsonSerializerOptions options = new JsonSerializerOptions(); + options.IncludeFields = true; + options.TypeInfoResolver = resolver; + + TestClass originalObj = new TestClass() + { + TestField = "test value", + TestProperty = 42, + }; + + string json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""TestProperty"":42,""TestField"":""test value""}", json); + + TestClass deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(0, deserialized.TestProperty); + } + + [Fact] + public static void SetCustomNumberHandlingForAProperty() + { + DefaultJsonTypeInfoResolver resolver = new(); + resolver.Modifiers.Add((ti) => + { + if (ti.Type == typeof(TestClass)) + { + Assert.Equal(JsonTypeInfoKind.Object, ti.Kind); + foreach (var prop in ti.Properties) + { + if (prop.Name == nameof(TestClass.TestProperty)) + { + prop.NumberHandling = JsonNumberHandling.AllowReadingFromString | JsonNumberHandling.WriteAsString; + } + } + } + }); + + JsonSerializerOptions options = new JsonSerializerOptions(); + options.IncludeFields = true; + options.TypeInfoResolver = resolver; + + TestClass originalObj = new TestClass() + { + TestField = "test value", + TestProperty = 42, + }; + + string json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""TestProperty"":""42"",""TestField"":""test value""}", json); + + TestClass deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(originalObj.TestProperty, deserialized.TestProperty); + } + + [Fact] + public static void SetCustomConverterForAProperty() + { + DefaultJsonTypeInfoResolver resolver = new(); + resolver.Modifiers.Add((ti) => + { + if (ti.Type == typeof(TestClass)) + { + Assert.Equal(JsonTypeInfoKind.Object, ti.Kind); + foreach (var prop in ti.Properties) + { + if (prop.Name == nameof(TestClass.TestProperty)) + { + prop.CustomConverter = new PlusOneConverter(); + } + } + } + }); + + JsonSerializerOptions options = new JsonSerializerOptions(); + options.IncludeFields = true; + options.TypeInfoResolver = resolver; + + TestClass originalObj = new TestClass() + { + TestField = "test value", + TestProperty = 42, + }; + + string json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""TestProperty"":43,""TestField"":""test value""}", json); + + TestClass deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(originalObj.TestProperty, deserialized.TestProperty); + } + + [Fact] + public static void UntypedCreateObjectWithDefaults() + { + DefaultJsonTypeInfoResolver resolver = new(); + resolver.Modifiers.Add((ti) => + { + if (ti.Type == typeof(TestClass)) + { + ti.CreateObject = () => + { + return new TestClass() + { + TestField = "test value", + TestProperty = 42, + }; + }; + } + }); + + JsonSerializerOptions options = new JsonSerializerOptions(); + options.IncludeFields = true; + options.TypeInfoResolver = resolver; + + TestClass originalObj = new TestClass() + { + TestField = "test value 2", + TestProperty = 45, + }; + + string json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""TestProperty"":45,""TestField"":""test value 2""}", json); + + TestClass deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(originalObj.TestProperty, deserialized.TestProperty); + + json = @"{}"; + deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal("test value", deserialized.TestField); + Assert.Equal(42, deserialized.TestProperty); + + json = @"{""TestField"":""test value 2""}"; + deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(42, deserialized.TestProperty); + } + + [Fact] + public static void TypedCreateObjectWithDefaults() + { + DefaultJsonTypeInfoResolver resolver = new(); + resolver.Modifiers.Add((ti) => + { + if (ti.Type == typeof(TestClass)) + { + JsonTypeInfo typedTi = ti as JsonTypeInfo; + Assert.NotNull(typedTi); + typedTi.CreateObject = () => + { + return new TestClass() + { + TestField = "test value", + TestProperty = 42, + }; + }; + } + }); + + JsonSerializerOptions options = new JsonSerializerOptions(); + options.IncludeFields = true; + options.TypeInfoResolver = resolver; + + TestClass originalObj = new TestClass() + { + TestField = "test value 2", + TestProperty = 45, + }; + + string json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""TestProperty"":45,""TestField"":""test value 2""}", json); + + TestClass deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(originalObj.TestProperty, deserialized.TestProperty); + + json = @"{}"; + deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal("test value", deserialized.TestField); + Assert.Equal(42, deserialized.TestProperty); + + json = @"{""TestField"":""test value 2""}"; + deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(42, deserialized.TestProperty); + } + + [Fact] + public static void SetCustomNumberHandlingForAType() + { + DefaultJsonTypeInfoResolver resolver = new(); + resolver.Modifiers.Add((ti) => + { + if (ti.Type == typeof(TestClass)) + { + Assert.Equal(JsonTypeInfoKind.Object, ti.Kind); + ti.NumberHandling = JsonNumberHandling.AllowReadingFromString | JsonNumberHandling.WriteAsString; + } + }); + + JsonSerializerOptions options = new JsonSerializerOptions(); + options.IncludeFields = true; + options.TypeInfoResolver = resolver; + + TestClass originalObj = new TestClass() + { + TestField = "test value", + TestProperty = 42, + }; + + string json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""TestProperty"":""42"",""TestField"":""test value""}", json); + + TestClass deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(originalObj.TestProperty, deserialized.TestProperty); + } + + [Fact] + public static void CombineCustomResolverWithDefault() + { + TestResolver resolver = new TestResolver((Type type, JsonSerializerOptions options) => + { + if (type != typeof(TestClass)) + return null; + + JsonTypeInfo ti = JsonTypeInfo.CreateJsonTypeInfo(options); + ti.CreateObject = () => new TestClass() + { + TestField = string.Empty, + TestProperty = 42, + }; + + JsonPropertyInfo field = ti.CreateJsonPropertyInfo(typeof(string), "MyTestField"); + field.Get = (o) => + { + TestClass obj = (TestClass)o; + return obj.TestField ?? string.Empty; + }; + + field.Set = (o, val) => + { + TestClass obj = (TestClass)o; + string value = (string?)val ?? string.Empty; + obj.TestField = value; + }; + + field.ShouldSerialize = (o, val) => (string)val != string.Empty; + + JsonPropertyInfo prop = ti.CreateJsonPropertyInfo(typeof(int), "MyTestProperty"); + prop.Get = (o) => + { + TestClass obj = (TestClass)o; + return obj.TestProperty; + }; + + prop.Set = (o, val) => + { + TestClass obj = (TestClass)o; + obj.TestProperty = (int)val; + }; + + prop.ShouldSerialize = (o, val) => (int)val != 42; + + ti.Properties.Add(field); + ti.Properties.Add(prop); + return ti; + }); + + JsonSerializerOptions options = new JsonSerializerOptions(); + options.IncludeFields = true; + options.TypeInfoResolver = JsonTypeInfoResolver.Combine(resolver, options.TypeInfoResolver); + + TestClass originalObj = new TestClass() + { + TestField = "test value", + TestProperty = 45, + }; + + string json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""MyTestField"":""test value"",""MyTestProperty"":45}", json); + + TestClass deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(originalObj.TestProperty, deserialized.TestProperty); + + originalObj.TestField = null; + json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""MyTestProperty"":45}", json); + + originalObj.TestField = string.Empty; + json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""MyTestProperty"":45}", json); + + deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(originalObj.TestProperty, deserialized.TestProperty); + + originalObj.TestField = "test value"; + originalObj.TestProperty = 42; + json = JsonSerializer.Serialize(originalObj, options); + Assert.Equal(@"{""MyTestField"":""test value""}", json); + deserialized = JsonSerializer.Deserialize(json, options); + Assert.Equal(originalObj.TestField, deserialized.TestField); + Assert.Equal(originalObj.TestProperty, deserialized.TestProperty); + } + + internal class TestClass + { + public int TestProperty { get; set; } + public string TestField; + } + + internal class TestResolver : IJsonTypeInfoResolver + { + private Func _getTypeInfo; + + public TestResolver(Func getTypeInfo) + { + _getTypeInfo = getTypeInfo; + } + + public JsonTypeInfo? GetTypeInfo(Type type, JsonSerializerOptions options) => _getTypeInfo(type, options); + } + + // adds one on write, subtracts one on read + internal class PlusOneConverter : JsonConverter + { + public override int Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options) + { + Assert.Equal(typeof(int), typeToConvert); + return reader.GetInt32() - 1; + } + + public override void Write(Utf8JsonWriter writer, int value, JsonSerializerOptions options) + { + writer.WriteNumberValue(value + 1); + } + } + } +} diff --git a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/System.Text.Json.Tests.csproj b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/System.Text.Json.Tests.csproj index f775496500b1a..c0bd69ae1d8ec 100644 --- a/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/System.Text.Json.Tests.csproj +++ b/src/libraries/System.Text.Json/tests/System.Text.Json.Tests/System.Text.Json.Tests.csproj @@ -170,6 +170,10 @@ + + + + @@ -196,6 +200,7 @@ + diff --git a/src/libraries/apicompat/ApiCompatBaseline.NetCoreAppLatestStable.txt b/src/libraries/apicompat/ApiCompatBaseline.NetCoreAppLatestStable.txt index 30e8d88d4acff..ff2c3ef870c04 100644 --- a/src/libraries/apicompat/ApiCompatBaseline.NetCoreAppLatestStable.txt +++ b/src/libraries/apicompat/ApiCompatBaseline.NetCoreAppLatestStable.txt @@ -178,4 +178,6 @@ CannotRemoveAttribute : Attribute 'System.Runtime.Versioning.UnsupportedOSPlatfo CannotRemoveAttribute : Attribute 'System.Runtime.Versioning.UnsupportedOSPlatformAttribute' exists on 'System.Security.Cryptography.TripleDES' in the contract but not the implementation. Compat issues with assembly System.Security.Cryptography.X509Certificates: CannotChangeAttribute : Attribute 'System.Runtime.Versioning.UnsupportedOSPlatformAttribute' on 'System.Security.Cryptography.X509Certificates.PublicKey.GetDSAPublicKey()' changed from '[UnsupportedOSPlatformAttribute("ios")]' in the contract to '[UnsupportedOSPlatformAttribute("browser")]' in the implementation. -Total Issues: 167 +Compat issues with assembly System.Text.Json: +CannotMakeTypeAbstract : Type 'System.Text.Json.Serialization.Metadata.JsonTypeInfo' is abstract in the implementation but is not abstract in the contract. +Total Issues: 168