From 2b40ac06a47809458c69f1a853269691bdc8c12f Mon Sep 17 00:00:00 2001 From: maumar Date: Mon, 1 Apr 2024 00:35:20 -0700 Subject: [PATCH] Introduce liftable constants to shaper to prepare for precompilation. Fix materializer code to replace non-primitive constants with liftable constants. Precompilation is opt-in in QueryCompilationContext, switched on for relational, off for everything else. Architecture, design and initial implementation done by @roji, polish done by @maumar Part of #25009 --- EFCore.sln.DotSettings | 1 + .../Query/Internal/InMemoryQueryExpression.cs | 1 - ...ntityFrameworkRelationalServicesBuilder.cs | 5 + .../IRelationalLiftableConstantFactory.cs | 28 + .../Query/Internal/BufferedDataReader.cs | 5 +- .../Internal/FromSqlQueryingEnumerable.cs | 36 + .../GroupBySingleQueryingEnumerable.cs | 12 +- .../GroupBySplitQueryingEnumerable.cs | 12 +- ...ionalProjectionBindingExpressionVisitor.cs | 8 +- .../Internal/SingleQueryingEnumerable.cs | 34 + .../Query/Internal/SplitQueryingEnumerable.cs | 38 + ...lLiftableConstantExpressionDependencies.cs | 35 + .../RelationalLiftableConstantFactory.cs | 39 ++ .../RelationalLiftableConstantProcessor.cs | 44 ++ ...onalMaterializerLiftableConstantContext.cs | 18 + ...ocessingExpressionVisitor.ClientMethods.cs | 212 +++++- ...sitor.ShaperProcessingExpressionVisitor.cs | 661 +++++++++++++----- ...alShapedQueryCompilingExpressionVisitor.cs | 216 ++++-- ...yCompilingExpressionVisitorDependencies.cs | 9 +- ...lationalSqlTranslatingExpressionVisitor.cs | 18 +- .../Query/SqlExpressions/SelectExpression.cs | 94 +-- src/EFCore.Relational/Storage/ReaderColumn.cs | 20 +- .../Storage/ReaderColumn`.cs | 8 +- .../SqlServerJsonHierarchyIdReaderWriter.cs | 7 + ...SqlServerJsonSqlHierarchyIdReaderWriter.cs | 7 + .../SqlServerJsonGeometryWktReaderWriter.cs | 7 + .../SqlServerQueryCompilationContext.cs | 3 + ...qlServerSqlTranslatingExpressionVisitor.cs | 38 +- .../SqliteServiceCollectionExtensions.cs | 1 + .../Internal/SqliteQueryCompilationContext.cs | 30 + .../SqliteQueryCompilationContextFactory.cs | 46 ++ ...yableMethodTranslatingExpressionVisitor.cs | 1 - .../SqliteJsonByteArrayReaderWriter.cs | 7 + .../SqliteJsonDateTimeOffsetReaderWriter.cs | 7 + .../SqliteJsonDateTimeReaderWriter.cs | 7 + .../Internal/SqliteJsonDecimalReaderWriter.cs | 7 + .../Internal/SqliteJsonGuidReaderWriter.cs | 7 + .../Json/SqliteJsonGeometryWktReaderWriter.cs | 7 + .../ListOfNullableValueTypesComparer.cs | 9 +- .../ListOfValueTypesComparer.cs | 9 +- src/EFCore/ChangeTracking/ValueComparer.cs | 5 + src/EFCore/ChangeTracking/ValueComparer`.cs | 98 ++- src/EFCore/EFCore.csproj | 1 + .../EntityFrameworkServicesBuilder.cs | 7 +- src/EFCore/Query/ILiftableConstantFactory.cs | 28 + .../Query/ILiftableConstantProcessor.cs | 55 ++ .../Internal/EntityMaterializerSource.cs | 9 +- ...ingExpressionVisitor.ExpressionVisitors.cs | 17 +- .../Query/LiftableConstantExpression.cs | 104 +++ .../LiftableConstantExpressionDependencies.cs | 35 + .../LiftableConstantExpressionHelpers.cs | 292 ++++++++ src/EFCore/Query/LiftableConstantFactory.cs | 37 + src/EFCore/Query/LiftableConstantProcessor.cs | 486 +++++++++++++ .../MaterializerLiftableConstantContext.cs | 15 + src/EFCore/Query/QueryCompilationContext.cs | 101 ++- .../QueryCompilationContextDependencies.cs | 14 + .../Query/ReplacingExpressionVisitor.cs | 12 +- .../ShapedQueryCompilingExpressionVisitor.cs | 216 ++++-- ...yCompilingExpressionVisitorDependencies.cs | 38 +- .../Query/StructuralTypeShaperExpression.cs | 33 +- .../Storage/Json/JsonBoolReaderWriter.cs | 15 +- .../Storage/Json/JsonByteArrayReaderWriter.cs | 7 + .../Storage/Json/JsonByteReaderWriter.cs | 7 + .../Storage/Json/JsonCastValueReaderWriter.cs | 8 + .../Storage/Json/JsonCharReaderWriter.cs | 7 + ...CollectionOfNullableStructsReaderWriter.cs | 9 + .../JsonCollectionOfReferencesReaderWriter.cs | 8 + .../JsonCollectionOfStructsReaderWriter.cs | 8 + .../Json/JsonConvertedValueReaderWriter.cs | 11 + .../Storage/Json/JsonDateOnlyReaderWriter.cs | 7 + .../Json/JsonDateTimeOffsetReaderWriter.cs | 7 + .../Storage/Json/JsonDateTimeReaderWriter.cs | 7 + .../Storage/Json/JsonDecimalReaderWriter.cs | 7 + .../Storage/Json/JsonDoubleReaderWriter.cs | 7 + .../Storage/Json/JsonFloatReaderWriter.cs | 7 + .../Storage/Json/JsonGuidReaderWriter.cs | 7 + .../Storage/Json/JsonInt16ReaderWriter.cs | 7 + .../Storage/Json/JsonInt32ReaderWriter.cs | 7 + .../Storage/Json/JsonInt64ReaderWriter.cs | 7 + .../Storage/Json/JsonNullReaderWriter.cs | 7 + .../Storage/Json/JsonSByteReaderWriter.cs | 7 + .../Json/JsonSignedEnumReaderWriter.cs | 7 + .../Storage/Json/JsonStringReaderWriter.cs | 7 + .../Storage/Json/JsonTimeOnlyReaderWriter.cs | 7 + .../Storage/Json/JsonTimeSpanReaderWriter.cs | 7 + .../Storage/Json/JsonUInt16ReaderWriter.cs | 7 + .../Storage/Json/JsonUInt32ReaderWriter.cs | 7 + .../Storage/Json/JsonUInt64ReaderWriter.cs | 7 + .../Json/JsonUnsignedEnumReaderWriter.cs | 7 + .../Storage/Json/JsonValueReaderWriter.cs | 7 + .../Json/JsonWarningEnumReaderWriter.cs | 7 + .../ValueConversion/BoolToStringConverter.cs | 36 +- .../CollectionToJsonStringConverter.cs | 38 +- .../Internal/StringNumberConverter.cs | 7 +- .../ValueConversion/StringToBytesConverter.cs | 38 +- .../Storage/ValueConversion/ValueConverter.cs | 7 + .../ValueConversion/ValueConverter`.cs | 28 + src/Shared/ExpressionExtensions.cs | 30 +- ...AdvancedMappingsQueryRelationalTestBase.cs | 3 +- .../Query/AdHocJsonQueryTestBase.cs | 3 +- .../Query/AdHocQuerySplittingQueryTestBase.cs | 3 +- .../Query/JsonQueryTestBase.cs | 6 +- .../OwnedEntityQueryRelationalTestBase.cs | 6 +- .../Query/UdfDbFunctionTestBase.cs | 12 +- .../Query/Internal/BufferedDataReaderTest.cs | 9 +- .../RelationalApiConsistencyTest.cs | 20 +- .../ApiConsistencyTestBase.cs | 10 + .../JsonTypesTestBase.cs | 7 + .../Query/AdHocMiscellaneousQueryTestBase.cs | 18 +- .../Query/ComplexNavigationsQueryTestBase.cs | 12 +- .../Query/GearsOfWarQueryTestBase.cs | 17 +- .../Query/NorthwindGroupByQueryTestBase.cs | 6 +- .../NorthwindMiscellaneousQueryTestBase.cs | 3 +- .../Query/OwnedEntityQueryTestBase.cs | 6 +- .../Query/OwnedQueryTestBase.cs | 3 +- .../Scaffolding/CompiledModelTestBase.cs | 5 + .../SpatialTestBase.cs | 6 +- .../AdHocMiscellaneousQuerySqlServerTest.cs | 2 +- ...dPrimitiveCollectionsQuerySqlServerTest.cs | 16 +- .../Query/QueryLoggingSqlServerTest.cs | 4 +- .../Query/RawSqlServerTest.cs | 35 +- .../SpatialQuerySqlServerGeographyTest.cs | 2 +- .../SpatialQuerySqlServerGeometryTest.cs | 2 +- .../Baselines/BigModel/ManyTypesEntityType.cs | 4 +- .../ManyTypesEntityType.cs | 4 +- ...lServerValueGenerationScenariosTestBase.cs | 2 +- .../Baselines/BigModel/ManyTypesEntityType.cs | 4 +- .../ManyTypesEntityType.cs | 4 +- .../Metadata/Internal/PropertyTest.cs | 33 +- .../Query/EntityMaterializerSourceTest.cs | 3 +- ...vigationExpandingExpressionVisitorTests.cs | 2 + 131 files changed, 3506 insertions(+), 541 deletions(-) create mode 100644 src/EFCore.Relational/Query/IRelationalLiftableConstantFactory.cs create mode 100644 src/EFCore.Relational/Query/RelationalLiftableConstantExpressionDependencies.cs create mode 100644 src/EFCore.Relational/Query/RelationalLiftableConstantFactory.cs create mode 100644 src/EFCore.Relational/Query/RelationalLiftableConstantProcessor.cs create mode 100644 src/EFCore.Relational/Query/RelationalMaterializerLiftableConstantContext.cs create mode 100644 src/EFCore.Sqlite.Core/Query/Internal/SqliteQueryCompilationContext.cs create mode 100644 src/EFCore.Sqlite.Core/Query/Internal/SqliteQueryCompilationContextFactory.cs create mode 100644 src/EFCore/Query/ILiftableConstantFactory.cs create mode 100644 src/EFCore/Query/ILiftableConstantProcessor.cs create mode 100644 src/EFCore/Query/LiftableConstantExpression.cs create mode 100644 src/EFCore/Query/LiftableConstantExpressionDependencies.cs create mode 100644 src/EFCore/Query/LiftableConstantExpressionHelpers.cs create mode 100644 src/EFCore/Query/LiftableConstantFactory.cs create mode 100644 src/EFCore/Query/LiftableConstantProcessor.cs create mode 100644 src/EFCore/Query/MaterializerLiftableConstantContext.cs diff --git a/EFCore.sln.DotSettings b/EFCore.sln.DotSettings index d1ddc281f56..6881961da98 100644 --- a/EFCore.sln.DotSettings +++ b/EFCore.sln.DotSettings @@ -309,6 +309,7 @@ The .NET Foundation licenses this file to you under the MIT license. True True True + True True True True diff --git a/src/EFCore.InMemory/Query/Internal/InMemoryQueryExpression.cs b/src/EFCore.InMemory/Query/Internal/InMemoryQueryExpression.cs index 5d833b29426..791e04c0593 100644 --- a/src/EFCore.InMemory/Query/Internal/InMemoryQueryExpression.cs +++ b/src/EFCore.InMemory/Query/Internal/InMemoryQueryExpression.cs @@ -2,7 +2,6 @@ // The .NET Foundation licenses this file to you under the MIT license. using Microsoft.EntityFrameworkCore.InMemory.Internal; -using Microsoft.EntityFrameworkCore.Internal; using ExpressionExtensions = Microsoft.EntityFrameworkCore.Infrastructure.ExpressionExtensions; namespace Microsoft.EntityFrameworkCore.InMemory.Query.Internal; diff --git a/src/EFCore.Relational/Infrastructure/EntityFrameworkRelationalServicesBuilder.cs b/src/EFCore.Relational/Infrastructure/EntityFrameworkRelationalServicesBuilder.cs index 93cd941c2f5..6d4ad282930 100644 --- a/src/EFCore.Relational/Infrastructure/EntityFrameworkRelationalServicesBuilder.cs +++ b/src/EFCore.Relational/Infrastructure/EntityFrameworkRelationalServicesBuilder.cs @@ -63,6 +63,7 @@ public static readonly IDictionary RelationalServi { typeof(IQuerySqlGeneratorFactory), new ServiceCharacteristics(ServiceLifetime.Singleton) }, { typeof(IModificationCommandFactory), new ServiceCharacteristics(ServiceLifetime.Singleton) }, { typeof(ISqlAliasManagerFactory), new ServiceCharacteristics(ServiceLifetime.Singleton) }, + { typeof(IRelationalLiftableConstantFactory), new ServiceCharacteristics(ServiceLifetime.Singleton) }, { typeof(ICommandBatchPreparer), new ServiceCharacteristics(ServiceLifetime.Scoped) }, { typeof(IModificationCommandBatchFactory), new ServiceCharacteristics(ServiceLifetime.Scoped) }, { typeof(IRelationalSqlTranslatingExpressionVisitorFactory), new ServiceCharacteristics(ServiceLifetime.Scoped) }, @@ -189,6 +190,9 @@ public override EntityFrameworkServicesBuilder TryAddCoreServices() TryAdd(); TryAdd(); TryAdd(); + TryAdd(p => p.GetRequiredService()); + TryAdd(); + TryAdd(); ServiceCollectionMap.GetInfrastructure() .AddDependencySingleton() @@ -204,6 +208,7 @@ public override EntityFrameworkServicesBuilder TryAddCoreServices() .AddDependencySingleton() .AddDependencySingleton() .AddDependencySingleton() + .AddDependencySingleton() .AddDependencyScoped() .AddDependencyScoped() .AddDependencyScoped() diff --git a/src/EFCore.Relational/Query/IRelationalLiftableConstantFactory.cs b/src/EFCore.Relational/Query/IRelationalLiftableConstantFactory.cs new file mode 100644 index 00000000000..f93fc4340e4 --- /dev/null +++ b/src/EFCore.Relational/Query/IRelationalLiftableConstantFactory.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. + +using System.Diagnostics.CodeAnalysis; + +namespace Microsoft.EntityFrameworkCore.Query; + +/// +/// This is an experimental API used by the Entity Framework Core feature and it is not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +[Experimental(EFDiagnostics.PrecompiledQueryExperimental)] +public interface IRelationalLiftableConstantFactory : ILiftableConstantFactory +{ + /// + /// This is an experimental API used by the Entity Framework Core feature and it is not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + LiftableConstantExpression CreateLiftableConstant( + object? originalValue, + Expression> resolverExpression, + string variableName, + Type type); +} diff --git a/src/EFCore.Relational/Query/Internal/BufferedDataReader.cs b/src/EFCore.Relational/Query/Internal/BufferedDataReader.cs index 620bdc9a5bd..8c8ceec058c 100644 --- a/src/EFCore.Relational/Query/Internal/BufferedDataReader.cs +++ b/src/EFCore.Relational/Query/Internal/BufferedDataReader.cs @@ -836,10 +836,7 @@ public ulong GetUInt64(int ordinal) public object GetValue(int ordinal) => GetFieldValue(ordinal); -#pragma warning disable IDE0060 // Remove unused parameter - public static int GetValues(object[] values) -#pragma warning restore IDE0060 // Remove unused parameter - => throw new NotSupportedException(); + public static int GetValues(object[] values) => throw new NotSupportedException(); public T GetFieldValue(int ordinal) => (_columnTypeCases[ordinal]) switch diff --git a/src/EFCore.Relational/Query/Internal/FromSqlQueryingEnumerable.cs b/src/EFCore.Relational/Query/Internal/FromSqlQueryingEnumerable.cs index 817b2614048..4893afefe89 100644 --- a/src/EFCore.Relational/Query/Internal/FromSqlQueryingEnumerable.cs +++ b/src/EFCore.Relational/Query/Internal/FromSqlQueryingEnumerable.cs @@ -6,6 +6,42 @@ namespace Microsoft.EntityFrameworkCore.Query.Internal; +/// +/// This is an internal API that supports the Entity Framework Core infrastructure and not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +public static class FromSqlQueryingEnumerable +{ + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public static FromSqlQueryingEnumerable Create( + RelationalQueryContext relationalQueryContext, + RelationalCommandCache relationalCommandCache, + IReadOnlyList? readerColumns, + IReadOnlyList columnNames, + Func shaper, + Type contextType, + bool standAloneStateManager, + bool detailedErrorsEnabled, + bool threadSafetyChecksEnabled) + => new( + relationalQueryContext, + relationalCommandCache, + readerColumns, + columnNames, + shaper, + contextType, + standAloneStateManager, + detailedErrorsEnabled, + threadSafetyChecksEnabled); +} + /// /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to /// the same compatibility standards as public APIs. It may be changed or removed without notice in diff --git a/src/EFCore.Relational/Query/Internal/GroupBySingleQueryingEnumerable.cs b/src/EFCore.Relational/Query/Internal/GroupBySingleQueryingEnumerable.cs index b53b4dc5d7c..73d4960150f 100644 --- a/src/EFCore.Relational/Query/Internal/GroupBySingleQueryingEnumerable.cs +++ b/src/EFCore.Relational/Query/Internal/GroupBySingleQueryingEnumerable.cs @@ -20,7 +20,7 @@ public class GroupBySingleQueryingEnumerable private readonly IReadOnlyList? _readerColumns; private readonly Func _keySelector; private readonly Func _keyIdentifier; - private readonly IReadOnlyList _keyIdentifierValueComparers; + private readonly IReadOnlyList> _keyIdentifierValueComparers; private readonly Func _elementSelector; private readonly Type _contextType; private readonly IDiagnosticsLogger _queryLogger; @@ -40,7 +40,7 @@ public GroupBySingleQueryingEnumerable( IReadOnlyList? readerColumns, Func keySelector, Func keyIdentifier, - IReadOnlyList keyIdentifierValueComparers, + IReadOnlyList> keyIdentifierValueComparers, Func elementSelector, Type contextType, bool standAloneStateManager, @@ -139,12 +139,12 @@ IEnumerator IEnumerable.GetEnumerator() => GetEnumerator(); } - private static bool CompareIdentifiers(IReadOnlyList valueComparers, object[] left, object[] right) + private static bool CompareIdentifiers(IReadOnlyList> valueComparers, object[] left, object[] right) { // Ignoring size check on all for perf as they should be same unless bug in code. for (var i = 0; i < left.Length; i++) { - if (!valueComparers[i].Equals(left[i], right[i])) + if (!valueComparers[i](left[i], right[i])) { return false; } @@ -160,7 +160,7 @@ private sealed class Enumerator : IEnumerator> private readonly IReadOnlyList? _readerColumns; private readonly Func _keySelector; private readonly Func _keyIdentifier; - private readonly IReadOnlyList _keyIdentifierValueComparers; + private readonly IReadOnlyList> _keyIdentifierValueComparers; private readonly Func _elementSelector; private readonly Type _contextType; private readonly IDiagnosticsLogger _queryLogger; @@ -344,7 +344,7 @@ private sealed class AsyncEnumerator : IAsyncEnumerator? _readerColumns; private readonly Func _keySelector; private readonly Func _keyIdentifier; - private readonly IReadOnlyList _keyIdentifierValueComparers; + private readonly IReadOnlyList> _keyIdentifierValueComparers; private readonly Func _elementSelector; private readonly Type _contextType; private readonly IDiagnosticsLogger _queryLogger; diff --git a/src/EFCore.Relational/Query/Internal/GroupBySplitQueryingEnumerable.cs b/src/EFCore.Relational/Query/Internal/GroupBySplitQueryingEnumerable.cs index c8e21e4f499..7c610c47b3a 100644 --- a/src/EFCore.Relational/Query/Internal/GroupBySplitQueryingEnumerable.cs +++ b/src/EFCore.Relational/Query/Internal/GroupBySplitQueryingEnumerable.cs @@ -20,7 +20,7 @@ public class GroupBySplitQueryingEnumerable private readonly IReadOnlyList? _readerColumns; private readonly Func _keySelector; private readonly Func _keyIdentifier; - private readonly IReadOnlyList _keyIdentifierValueComparers; + private readonly IReadOnlyList> _keyIdentifierValueComparers; private readonly Func _elementSelector; private readonly Action? _relatedDataLoaders; private readonly Func? _relatedDataLoadersAsync; @@ -42,7 +42,7 @@ public GroupBySplitQueryingEnumerable( IReadOnlyList? readerColumns, Func keySelector, Func keyIdentifier, - IReadOnlyList keyIdentifierValueComparers, + IReadOnlyList> keyIdentifierValueComparers, Func elementSelector, Action? relatedDataLoaders, Func? relatedDataLoadersAsync, @@ -145,12 +145,12 @@ IEnumerator IEnumerable.GetEnumerator() => GetEnumerator(); } - private static bool CompareIdentifiers(IReadOnlyList valueComparers, object[] left, object[] right) + private static bool CompareIdentifiers(IReadOnlyList> valueComparers, object[] left, object[] right) { // Ignoring size check on all for perf as they should be same unless bug in code. for (var i = 0; i < left.Length; i++) { - if (!valueComparers[i].Equals(left[i], right[i])) + if (!valueComparers[i](left[i], right[i])) { return false; } @@ -166,7 +166,7 @@ private sealed class Enumerator : IEnumerator> private readonly IReadOnlyList? _readerColumns; private readonly Func _keySelector; private readonly Func _keyIdentifier; - private readonly IReadOnlyList _keyIdentifierValueComparers; + private readonly IReadOnlyList> _keyIdentifierValueComparers; private readonly Func _elementSelector; private readonly Action? _relatedDataLoaders; private readonly Type _contextType; @@ -340,7 +340,7 @@ private sealed class AsyncEnumerator : IAsyncEnumerator? _readerColumns; private readonly Func _keySelector; private readonly Func _keyIdentifier; - private readonly IReadOnlyList _keyIdentifierValueComparers; + private readonly IReadOnlyList> _keyIdentifierValueComparers; private readonly Func _elementSelector; private readonly Func? _relatedDataLoaders; private readonly Type _contextType; diff --git a/src/EFCore.Relational/Query/Internal/RelationalProjectionBindingExpressionVisitor.cs b/src/EFCore.Relational/Query/Internal/RelationalProjectionBindingExpressionVisitor.cs index 2d9ef438560..42e34d8c4c5 100644 --- a/src/EFCore.Relational/Query/Internal/RelationalProjectionBindingExpressionVisitor.cs +++ b/src/EFCore.Relational/Query/Internal/RelationalProjectionBindingExpressionVisitor.cs @@ -653,7 +653,13 @@ private ProjectionBindingExpression AddClientProjection(Expression expression, T return new ProjectionBindingExpression(_selectExpression, existingIndex, type); } - private static T GetParameterValue(QueryContext queryContext, string parameterName) + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public static T GetParameterValue(QueryContext queryContext, string parameterName) #pragma warning restore IDE0052 // Remove unread private members => (T)queryContext.ParameterValues[parameterName]!; diff --git a/src/EFCore.Relational/Query/Internal/SingleQueryingEnumerable.cs b/src/EFCore.Relational/Query/Internal/SingleQueryingEnumerable.cs index 53a3c372294..560179079fe 100644 --- a/src/EFCore.Relational/Query/Internal/SingleQueryingEnumerable.cs +++ b/src/EFCore.Relational/Query/Internal/SingleQueryingEnumerable.cs @@ -6,6 +6,40 @@ namespace Microsoft.EntityFrameworkCore.Query.Internal; +/// +/// This is an internal API that supports the Entity Framework Core infrastructure and not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +public static class SingleQueryingEnumerable +{ + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public static SingleQueryingEnumerable Create( + RelationalQueryContext relationalQueryContext, + RelationalCommandCache relationalCommandCache, + IReadOnlyList? readerColumns, + Func shaper, + Type contextType, + bool standAloneStateManager, + bool detailedErrorsEnabled, + bool threadSafetyChecksEnabled) + => new( + relationalQueryContext, + relationalCommandCache, + readerColumns, + shaper, + contextType, + standAloneStateManager, + detailedErrorsEnabled, + threadSafetyChecksEnabled); +} + /// /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to /// the same compatibility standards as public APIs. It may be changed or removed without notice in diff --git a/src/EFCore.Relational/Query/Internal/SplitQueryingEnumerable.cs b/src/EFCore.Relational/Query/Internal/SplitQueryingEnumerable.cs index c87544d1120..1575352ebbd 100644 --- a/src/EFCore.Relational/Query/Internal/SplitQueryingEnumerable.cs +++ b/src/EFCore.Relational/Query/Internal/SplitQueryingEnumerable.cs @@ -6,6 +6,44 @@ namespace Microsoft.EntityFrameworkCore.Query.Internal; +/// +/// This is an internal API that supports the Entity Framework Core infrastructure and not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +public static class SplitQueryingEnumerable +{ + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public static SplitQueryingEnumerable Create( + RelationalQueryContext relationalQueryContext, + RelationalCommandCache relationalCommandCache, + IReadOnlyList? readerColumns, + Func shaper, + Action? relatedDataLoaders, + Func? relatedDataLoadersAsync, + Type contextType, + bool standAloneStateManager, + bool detailedErrorsEnabled, + bool threadSafetyChecksEnabled) + => new( + relationalQueryContext, + relationalCommandCache, + readerColumns, + shaper, + relatedDataLoaders, + relatedDataLoadersAsync, + contextType, + standAloneStateManager, + detailedErrorsEnabled, + threadSafetyChecksEnabled); +} + /// /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to /// the same compatibility standards as public APIs. It may be changed or removed without notice in diff --git a/src/EFCore.Relational/Query/RelationalLiftableConstantExpressionDependencies.cs b/src/EFCore.Relational/Query/RelationalLiftableConstantExpressionDependencies.cs new file mode 100644 index 00000000000..6d8f445db2c --- /dev/null +++ b/src/EFCore.Relational/Query/RelationalLiftableConstantExpressionDependencies.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.Diagnostics.CodeAnalysis; + +namespace Microsoft.EntityFrameworkCore.Query; + +/// +/// +/// Service dependencies parameter class for +/// +/// +/// This type is typically used by database providers (and other extensions). It is generally +/// not used in application code. +/// +/// +/// +/// +/// Do not construct instances of this class directly from either provider or application code as the +/// constructor signature may change as new dependencies are added. Instead, use this type in +/// your constructor so that an instance will be created and injected automatically by the +/// dependency injection container. To create an instance with some dependent services replaced, +/// first resolve the object from the dependency injection container, then replace selected +/// services using the C# 'with' operator. Do not call the constructor at any point in this process. +/// +/// +/// The service lifetime is . This means a single instance +/// is used by many instances. The implementation must be thread-safe. +/// This service cannot depend on services registered as . +/// +/// +[Experimental(EFDiagnostics.PrecompiledQueryExperimental)] +public sealed record RelationalLiftableConstantExpressionDependencies +{ +} diff --git a/src/EFCore.Relational/Query/RelationalLiftableConstantFactory.cs b/src/EFCore.Relational/Query/RelationalLiftableConstantFactory.cs new file mode 100644 index 00000000000..d2d1155fe92 --- /dev/null +++ b/src/EFCore.Relational/Query/RelationalLiftableConstantFactory.cs @@ -0,0 +1,39 @@ +// 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.CodeAnalysis; + +namespace Microsoft.EntityFrameworkCore.Query; + +/// +/// This is an experimental API used by the Entity Framework Core feature and it is not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +[Experimental(EFDiagnostics.PrecompiledQueryExperimental)] +public class RelationalLiftableConstantFactory( + LiftableConstantExpressionDependencies dependencies, + RelationalLiftableConstantExpressionDependencies relationalDependencies) : LiftableConstantFactory(dependencies), IRelationalLiftableConstantFactory +{ + /// + /// This is an experimental API used by the Entity Framework Core feature and it is not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public virtual RelationalLiftableConstantExpressionDependencies RelationalDependencies { get; } = relationalDependencies; + + /// + /// This is an experimental API used by the Entity Framework Core feature and it is not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public virtual LiftableConstantExpression CreateLiftableConstant( + object? originalValue, + Expression> resolverExpression, + string variableName, + Type type) + => new(originalValue, resolverExpression, variableName, type); +} diff --git a/src/EFCore.Relational/Query/RelationalLiftableConstantProcessor.cs b/src/EFCore.Relational/Query/RelationalLiftableConstantProcessor.cs new file mode 100644 index 00000000000..1e942372ab0 --- /dev/null +++ b/src/EFCore.Relational/Query/RelationalLiftableConstantProcessor.cs @@ -0,0 +1,44 @@ +// 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.CodeAnalysis; + +namespace Microsoft.EntityFrameworkCore.Query; + +/// +/// This is an experimental API used by the Entity Framework Core feature and it is not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +[Experimental(EFDiagnostics.PrecompiledQueryExperimental)] +public class RelationalLiftableConstantProcessor : LiftableConstantProcessor +{ + private readonly RelationalMaterializerLiftableConstantContext _relationalMaterializerLiftableConstantContext; + + /// + /// This is an experimental API used by the Entity Framework Core feature and it is not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public RelationalLiftableConstantProcessor( + ShapedQueryCompilingExpressionVisitorDependencies dependencies, + RelationalShapedQueryCompilingExpressionVisitorDependencies relationalDependencies) + : base(dependencies) + => _relationalMaterializerLiftableConstantContext = new(dependencies, relationalDependencies); + + /// + protected override ConstantExpression InlineConstant(LiftableConstantExpression liftableConstant) + { + if (liftableConstant.ResolverExpression is Expression> + resolverExpression) + { + var resolver = resolverExpression.Compile(preferInterpretation: true); + var value = resolver(_relationalMaterializerLiftableConstantContext); + return Expression.Constant(value, liftableConstant.Type); + } + + return base.InlineConstant(liftableConstant); + } +} diff --git a/src/EFCore.Relational/Query/RelationalMaterializerLiftableConstantContext.cs b/src/EFCore.Relational/Query/RelationalMaterializerLiftableConstantContext.cs new file mode 100644 index 00000000000..e54e2b507a2 --- /dev/null +++ b/src/EFCore.Relational/Query/RelationalMaterializerLiftableConstantContext.cs @@ -0,0 +1,18 @@ +// 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.CodeAnalysis; + +namespace Microsoft.EntityFrameworkCore.Query; + +/// +/// This is an experimental API used by the Entity Framework Core feature and it is not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +[Experimental(EFDiagnostics.PrecompiledQueryExperimental)] +public record RelationalMaterializerLiftableConstantContext( + ShapedQueryCompilingExpressionVisitorDependencies Dependencies, + RelationalShapedQueryCompilingExpressionVisitorDependencies RelationalDependencies) + : MaterializerLiftableConstantContext(Dependencies); diff --git a/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitor.ShaperProcessingExpressionVisitor.ClientMethods.cs b/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitor.ShaperProcessingExpressionVisitor.ClientMethods.cs index 33b01fd885a..07df5d5c415 100644 --- a/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitor.ShaperProcessingExpressionVisitor.ClientMethods.cs +++ b/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitor.ShaperProcessingExpressionVisitor.ClientMethods.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; using System.Runtime.CompilerServices; using System.Text.Json; using Microsoft.EntityFrameworkCore.Internal; @@ -11,7 +12,14 @@ namespace Microsoft.EntityFrameworkCore.Query; public partial class RelationalShapedQueryCompilingExpressionVisitor { - private sealed partial class ShaperProcessingExpressionVisitor : ExpressionVisitor + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public sealed partial class ShaperProcessingExpressionVisitor : ExpressionVisitor { private static readonly MethodInfo ThrowReadValueExceptionMethod = typeof(ShaperProcessingExpressionVisitor).GetTypeInfo().GetDeclaredMethod(nameof(ThrowReadValueException))!; @@ -71,8 +79,15 @@ private static readonly MethodInfo MaterializeJsonEntityCollectionMethodInfo private static readonly MethodInfo InverseCollectionFixupMethod = typeof(ShaperProcessingExpressionVisitor).GetTypeInfo().GetDeclaredMethod(nameof(InverseCollectionFixup))!; + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// [MethodImpl(MethodImplOptions.AggressiveInlining)] - private static TValue ThrowReadValueException( + [EntityFrameworkInternal] + public static TValue ThrowReadValueException( Exception exception, object? value, Type expectedType, @@ -122,7 +137,14 @@ private static TValue ThrowExtractJsonPropertyException(Exception except exception); } - private static void IncludeReference( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static void IncludeReference( QueryContext queryContext, TEntity entity, TIncludedEntity? relatedEntity, @@ -160,7 +182,14 @@ private static void IncludeReference } } - private static void InitializeIncludeCollection( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static void InitializeIncludeCollection( int collectionId, QueryContext queryContext, DbDataReader dbDataReader, @@ -201,7 +230,14 @@ private static void InitializeIncludeCollection( resultCoordinator.SetSingleQueryCollectionContext(collectionId, collectionMaterializationContext); } - private static void PopulateIncludeCollection( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static void PopulateIncludeCollection( int collectionId, QueryContext queryContext, DbDataReader dbDataReader, @@ -209,9 +245,9 @@ private static void PopulateIncludeCollection Func parentIdentifier, Func outerIdentifier, Func selfIdentifier, - IReadOnlyList parentIdentifierValueComparers, - IReadOnlyList outerIdentifierValueComparers, - IReadOnlyList selfIdentifierValueComparers, + IReadOnlyList> parentIdentifierValueComparers, + IReadOnlyList> outerIdentifierValueComparers, + IReadOnlyList> selfIdentifierValueComparers, Func innerShaper, INavigationBase? inverseNavigation, Action fixup, @@ -319,7 +355,14 @@ void GenerateCurrentElementIfPending() } } - private static void InitializeSplitIncludeCollection( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static void InitializeSplitIncludeCollection( int collectionId, QueryContext queryContext, DbDataReader parentDataReader, @@ -358,7 +401,14 @@ private static void InitializeSplitIncludeCollection resultCoordinator.SetSplitQueryCollectionContext(collectionId, splitQueryCollectionContext); } - private static void PopulateSplitIncludeCollection( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static void PopulateSplitIncludeCollection( int collectionId, RelationalQueryContext queryContext, IExecutionStrategy executionStrategy, @@ -367,7 +417,7 @@ private static void PopulateSplitIncludeCollection childIdentifier, - IReadOnlyList identifierValueComparers, + IReadOnlyList> identifierValueComparers, Func innerShaper, Action? relatedDataLoaders, INavigationBase? inverseNavigation, @@ -442,7 +492,14 @@ static RelationalDataReader InitializeReader( } } - private static async Task PopulateSplitIncludeCollectionAsync( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static async Task PopulateSplitIncludeCollectionAsync( int collectionId, RelationalQueryContext queryContext, IExecutionStrategy executionStrategy, @@ -451,7 +508,7 @@ private static async Task PopulateSplitIncludeCollectionAsync childIdentifier, - IReadOnlyList identifierValueComparers, + IReadOnlyList> identifierValueComparers, Func innerShaper, Func? relatedDataLoaders, INavigationBase? inverseNavigation, @@ -538,7 +595,14 @@ static async Task InitializeReaderAsync( } } - private static TCollection InitializeCollection( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static TCollection InitializeCollection( int collectionId, QueryContext queryContext, DbDataReader dbDataReader, @@ -560,7 +624,14 @@ private static TCollection InitializeCollection( return (TCollection)collection; } - private static void PopulateCollection( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static void PopulateCollection( int collectionId, QueryContext queryContext, DbDataReader dbDataReader, @@ -568,9 +639,9 @@ private static void PopulateCollection( Func parentIdentifier, Func outerIdentifier, Func selfIdentifier, - IReadOnlyList parentIdentifierValueComparers, - IReadOnlyList outerIdentifierValueComparers, - IReadOnlyList selfIdentifierValueComparers, + IReadOnlyList> parentIdentifierValueComparers, + IReadOnlyList> outerIdentifierValueComparers, + IReadOnlyList> selfIdentifierValueComparers, Func innerShaper) where TRelatedEntity : TElement where TCollection : class, ICollection @@ -590,8 +661,8 @@ private static void PopulateCollection( } if (!CompareIdentifiers( - outerIdentifierValueComparers, - outerIdentifier(queryContext, dbDataReader), collectionMaterializationContext.OuterIdentifier)) + outerIdentifierValueComparers, + outerIdentifier(queryContext, dbDataReader), collectionMaterializationContext.OuterIdentifier)) { // Outer changed so collection has ended. Materialize last element. GenerateCurrentElementIfPending(); @@ -616,8 +687,8 @@ private static void PopulateCollection( if (collectionMaterializationContext.SelfIdentifier != null) { if (CompareIdentifiers( - selfIdentifierValueComparers, - innerKey, collectionMaterializationContext.SelfIdentifier)) + selfIdentifierValueComparers, + innerKey, collectionMaterializationContext.SelfIdentifier)) { // repeated row for current element // If it is pending materialization then it may have nested elements @@ -673,7 +744,14 @@ void GenerateCurrentElementIfPending() } } - private static TCollection InitializeSplitCollection( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static TCollection InitializeSplitCollection( int collectionId, QueryContext queryContext, DbDataReader parentDataReader, @@ -691,7 +769,14 @@ private static TCollection InitializeSplitCollection( return (TCollection)collection; } - private static void PopulateSplitCollection( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static void PopulateSplitCollection( int collectionId, RelationalQueryContext queryContext, IExecutionStrategy executionStrategy, @@ -700,7 +785,7 @@ private static void PopulateSplitCollection childIdentifier, - IReadOnlyList identifierValueComparers, + IReadOnlyList> identifierValueComparers, Func innerShaper, Action? relatedDataLoaders) where TRelatedEntity : TElement @@ -770,7 +855,14 @@ static RelationalDataReader InitializeReader( dataReaderContext.HasNext = false; } - private static async Task PopulateSplitCollectionAsync( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static async Task PopulateSplitCollectionAsync( int collectionId, RelationalQueryContext queryContext, IExecutionStrategy executionStrategy, @@ -779,7 +871,7 @@ private static async Task PopulateSplitCollectionAsync childIdentifier, - IReadOnlyList identifierValueComparers, + IReadOnlyList> identifierValueComparers, Func innerShaper, Func? relatedDataLoaders) where TRelatedEntity : TElement @@ -861,7 +953,14 @@ static async Task InitializeReaderAsync( dataReaderContext.HasNext = false; } - private static TEntity? MaterializeJsonEntity( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static TEntity? MaterializeJsonEntity( QueryContext queryContext, object[] keyPropertyValues, JsonReaderData? jsonReaderData, @@ -900,7 +999,14 @@ static async Task InitializeReaderAsync( return result; } - private static TResult? MaterializeJsonEntityCollection( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static TResult? MaterializeJsonEntityCollection( QueryContext queryContext, object[] keyPropertyValues, JsonReaderData? jsonReaderData, @@ -967,7 +1073,14 @@ static async Task InitializeReaderAsync( return result; } - private static void IncludeJsonEntityReference( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static void IncludeJsonEntityReference( QueryContext queryContext, object[] keyPropertyValues, JsonReaderData? jsonReaderData, @@ -991,7 +1104,14 @@ private static void IncludeJsonEntityReference( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static void IncludeJsonEntityCollection( QueryContext queryContext, object[] keyPropertyValues, JsonReaderData? jsonReaderData, @@ -1058,7 +1178,31 @@ private static void IncludeJsonEntityCollection[] taskFactories) + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static bool Any(IEnumerable source) + { + foreach (var _ in source) + { + return true; + } + + return false; + } + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static async Task TaskAwaiter(Func[] taskFactories) { for (var i = 0; i < taskFactories.Length; i++) { @@ -1066,12 +1210,12 @@ private static async Task TaskAwaiter(Func[] taskFactories) } } - private static bool CompareIdentifiers(IReadOnlyList valueComparers, object[] left, object[] right) + private static bool CompareIdentifiers(IReadOnlyList> valueComparers, object[] left, object[] right) { // Ignoring size check on all for perf as they should be same unless bug in code. for (var i = 0; i < left.Length; i++) { - if (!valueComparers[i].Equals(left[i], right[i])) + if (!valueComparers[i](left[i], right[i])) { return false; } diff --git a/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitor.ShaperProcessingExpressionVisitor.cs b/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitor.ShaperProcessingExpressionVisitor.cs index b6f5aa95364..ecc5c893c8d 100644 --- a/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitor.ShaperProcessingExpressionVisitor.cs +++ b/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitor.ShaperProcessingExpressionVisitor.cs @@ -2,6 +2,7 @@ // The .NET Foundation licenses this file to you under the MIT license. using System.Diagnostics.CodeAnalysis; +using System.Text.Encodings.Web; using System.Text.Json; using Microsoft.EntityFrameworkCore.ChangeTracking.Internal; using Microsoft.EntityFrameworkCore.Metadata.Internal; @@ -14,7 +15,13 @@ namespace Microsoft.EntityFrameworkCore.Query; public partial class RelationalShapedQueryCompilingExpressionVisitor { - private sealed partial class ShaperProcessingExpressionVisitor : ExpressionVisitor + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public sealed partial class ShaperProcessingExpressionVisitor : ExpressionVisitor { /// /// Reading database values @@ -22,6 +29,12 @@ private sealed partial class ShaperProcessingExpressionVisitor : ExpressionVisit private static readonly MethodInfo IsDbNullMethod = typeof(DbDataReader).GetRuntimeMethod(nameof(DbDataReader.IsDBNull), [typeof(int)])!; + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// public static readonly MethodInfo GetFieldValueMethod = typeof(DbDataReader).GetRuntimeMethod(nameof(DbDataReader.GetFieldValue), [typeof(int)])!; @@ -77,6 +90,15 @@ private static readonly MethodInfo Utf8JsonReaderGetStringMethod private static readonly MethodInfo EnumParseMethodInfo = typeof(Enum).GetMethod(nameof(Enum.Parse), [typeof(Type), typeof(string)])!; + private static readonly MethodInfo ReadColumnCreateMethod + = typeof(ReaderColumn).GetMethod(nameof(ReaderColumn.Create))!; + + private readonly static MethodInfo PropertyGetJsonValueReaderWriterMethod = + typeof(IReadOnlyProperty).GetMethod(nameof(IReadOnlyProperty.GetJsonValueReaderWriter), [])!; + + private readonly static MethodInfo PropertyGetTypeMappingMethod = + typeof(IReadOnlyProperty).GetMethod(nameof(IReadOnlyProperty.GetTypeMapping), [])!; + private readonly RelationalShapedQueryCompilingExpressionVisitor _parentVisitor; private readonly ISet? _tags; private readonly bool _isTracking; @@ -160,6 +182,12 @@ private readonly Dictionary /// private readonly Dictionary _jsonArrayNonConstantElementAccessMap = new(); + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// public ShaperProcessingExpressionVisitor( RelationalShapedQueryCompilingExpressionVisitor parentVisitor, SelectExpression selectExpression, @@ -172,7 +200,6 @@ public ShaperProcessingExpressionVisitor( _resultCoordinatorParameter = Parameter( splitQuery ? typeof(SplitQueryResultCoordinator) : typeof(SingleQueryResultCoordinator), "resultCoordinator"); _executionStrategyParameter = splitQuery ? Parameter(typeof(IExecutionStrategy), "executionStrategy") : null; - _selectExpression = selectExpression; _tags = tags; _dataReaderParameter = Parameter(typeof(DbDataReader), "dataReader"); @@ -247,9 +274,15 @@ private ShaperProcessingExpressionVisitor( _selectExpression.ApplyTags(_tags); } + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// public LambdaExpression ProcessRelationalGroupingResult( RelationalGroupByResultExpression relationalGroupByResultExpression, - out RelationalCommandCache relationalCommandCache, + out Expression relationalCommandCache, out IReadOnlyList? readerColumns, out LambdaExpression keySelector, out LambdaExpression keyIdentifier, @@ -277,9 +310,15 @@ public LambdaExpression ProcessRelationalGroupingResult( ref collectionId); } + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// public LambdaExpression ProcessShaper( Expression shaperExpression, - out RelationalCommandCache? relationalCommandCache, + out Expression relationalCommandCache, out IReadOnlyList? readerColumns, out LambdaExpression? relatedDataLoaders, ref int collectionId) @@ -293,12 +332,7 @@ public LambdaExpression ProcessShaper( _expressions.Add(result); result = Block(_variables, _expressions); - relationalCommandCache = new RelationalCommandCache( - _parentVisitor.Dependencies.MemoryCache, - _parentVisitor.RelationalDependencies.QuerySqlGeneratorFactory, - _parentVisitor.RelationalDependencies.RelationalParameterBasedSqlProcessorFactory, - _selectExpression, - _parentVisitor._useRelationalNulls); + relationalCommandCache = _parentVisitor.CreateRelationalCommandCacheExpression(_selectExpression); readerColumns = _readerColumns; return Lambda( @@ -320,13 +354,8 @@ public LambdaExpression ProcessShaper( result = Block(_variables, _expressions); relationalCommandCache = _generateCommandCache - ? new RelationalCommandCache( - _parentVisitor.Dependencies.MemoryCache, - _parentVisitor.RelationalDependencies.QuerySqlGeneratorFactory, - _parentVisitor.RelationalDependencies.RelationalParameterBasedSqlProcessorFactory, - _selectExpression, - _parentVisitor._useRelationalNulls) - : null; + ? _parentVisitor.CreateRelationalCommandCacheExpression(_selectExpression) + : Constant(null, typeof(RelationalCommandCache)); readerColumns = _readerColumns; return Lambda( @@ -408,13 +437,8 @@ public LambdaExpression ProcessShaper( } relationalCommandCache = _generateCommandCache - ? new RelationalCommandCache( - _parentVisitor.Dependencies.MemoryCache, - _parentVisitor.RelationalDependencies.QuerySqlGeneratorFactory, - _parentVisitor.RelationalDependencies.RelationalParameterBasedSqlProcessorFactory, - _selectExpression, - _parentVisitor._useRelationalNulls) - : null; + ? _parentVisitor.CreateRelationalCommandCacheExpression(_selectExpression) + : Constant(null, typeof(RelationalCommandCache));; readerColumns = _readerColumns; collectionId = _collectionId; @@ -428,6 +452,12 @@ public LambdaExpression ProcessShaper( } } + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// protected override Expression VisitBinary(BinaryExpression binaryExpression) { switch (binaryExpression) @@ -452,8 +482,16 @@ protected override Expression VisitBinary(BinaryExpression binaryExpression) ? value : propertyMap.Values.Max() + 1; - var updatedExpression = newExpression.Update( - new[] { Constant(ValueBuffer.Empty), newExpression.Arguments[1] }); + var updatedExpression = newExpression.Update( + new[] + { + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + ValueBuffer.Empty, + static _ => ValueBuffer.Empty, + "emptyValueBuffer", + typeof(ValueBuffer)), + newExpression.Arguments[1] + }); return Assign(binaryExpression.Left, updatedExpression); } @@ -465,7 +503,15 @@ protected override Expression VisitBinary(BinaryExpression binaryExpression) _jsonMaterializationContextToJsonReaderDataAndKeyValuesParameterMapping[parameterExpression] = mappedParameter; var updatedExpression = newExpression.Update( - new[] { Constant(ValueBuffer.Empty), newExpression.Arguments[1] }); + new[] + { + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + ValueBuffer.Empty, + static _ => ValueBuffer.Empty, + "emptyValueBuffer", + typeof(ValueBuffer)), + newExpression.Arguments[1] + }); return Assign(binaryExpression.Left, updatedExpression); } @@ -497,6 +543,12 @@ protected override Expression VisitBinary(BinaryExpression binaryExpression) return base.VisitBinary(binaryExpression); } + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// protected override Expression VisitExtension(Expression extensionExpression) { switch (extensionExpression) @@ -597,7 +649,7 @@ protected override Expression VisitExtension(Expression extensionExpression) } else { - var entityParameter = Parameter(shaper.Type); + var entityParameter = Parameter(shaper.Type, "entity"); _variables.Add(entityParameter); if (shaper.StructuralType is IEntityType entityType && entityType.GetMappingStrategy() == RelationalAnnotationNames.TpcMappingStrategy) @@ -718,7 +770,7 @@ when GetProjectionIndex(collectionResultExpression.ProjectionBindingExpression) var projection = _selectExpression.Projection[projectionIndex]; var nullable = IsNullableProjection(projection); - var valueParameter = Parameter(projectionBindingExpression.Type); + var valueParameter = Parameter(projectionBindingExpression.Type, "value" + (_variables.Count + 1)); _variables.Add(valueParameter); _expressions.Add( @@ -769,13 +821,13 @@ when GetProjectionIndex(collectionResultExpression.ProjectionBindingExpression) _readerColumns) .ProcessShaper(relationalCollectionShaperExpression.InnerShaper, out _, out _, out _, ref _collectionId); - var entityType = entity.Type; + var entityClrType = entity.Type; var navigation = includeExpression.Navigation; - var includingEntityType = navigation.DeclaringEntityType.ClrType; - if (includingEntityType != entityType - && includingEntityType.IsAssignableFrom(entityType)) + var includingEntityClrType = navigation.DeclaringEntityType.ClrType; + if (includingEntityClrType != entityClrType + && includingEntityClrType.IsAssignableFrom(entityClrType)) { - includingEntityType = entityType; + includingEntityClrType = entityClrType; } _inline = true; @@ -799,51 +851,60 @@ when GetProjectionIndex(collectionResultExpression.ProjectionBindingExpression) _includeExpressions.Add( Call( - InitializeIncludeCollectionMethodInfo.MakeGenericMethod(entityType, includingEntityType), + InitializeIncludeCollectionMethodInfo.MakeGenericMethod(entityClrType, includingEntityClrType), collectionIdConstant, QueryCompilationContext.QueryContextParameter, _dataReaderParameter, _resultCoordinatorParameter, entity, - Constant(parentIdentifierLambda.Compile()), - Constant(outerIdentifierLambda.Compile()), - Constant(navigation), - Constant( - navigation.IsShadowProperty() - ? null - : navigation.GetCollectionAccessor(), typeof(IClrCollectionAccessor)), + parentIdentifierLambda, + outerIdentifierLambda, + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + navigation, + LiftableConstantExpressionHelpers.BuildNavigationAccessLambda(navigation), + navigation.Name + "Navigation", + typeof(INavigationBase)), + navigation.IsShadowProperty() + ? Constant(null, typeof(IClrCollectionAccessor)) + : _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + navigation.GetCollectionAccessor(), + LiftableConstantExpressionHelpers.BuildClrCollectionAccessorLambda(navigation), + navigation.Name + "NavigationCollectionAccessor", + typeof(IClrCollectionAccessor)), Constant(_isTracking), #pragma warning disable EF1001 // Internal EF Core API usage. Constant(includeExpression.SetLoaded))); #pragma warning restore EF1001 // Internal EF Core API usage. - var relatedEntityType = innerShaper.ReturnType; + var relatedEntityClrType = innerShaper.ReturnType; var inverseNavigation = navigation.Inverse; _collectionPopulatingExpressions!.Add( Call( - PopulateIncludeCollectionMethodInfo.MakeGenericMethod(includingEntityType, relatedEntityType), + PopulateIncludeCollectionMethodInfo.MakeGenericMethod(includingEntityClrType, relatedEntityClrType), collectionIdConstant, QueryCompilationContext.QueryContextParameter, _dataReaderParameter, _resultCoordinatorParameter, - Constant(parentIdentifierLambda.Compile()), - Constant(outerIdentifierLambda.Compile()), - Constant(selfIdentifierLambda.Compile()), - Constant( - relationalCollectionShaperExpression.ParentIdentifierValueComparers, - typeof(IReadOnlyList)), - Constant( - relationalCollectionShaperExpression.OuterIdentifierValueComparers, - typeof(IReadOnlyList)), - Constant( - relationalCollectionShaperExpression.SelfIdentifierValueComparers, - typeof(IReadOnlyList)), - Constant(innerShaper.Compile()), - Constant(inverseNavigation, typeof(INavigationBase)), - Constant( - GenerateFixup( - includingEntityType, relatedEntityType, navigation, inverseNavigation).Compile()), + parentIdentifierLambda, + outerIdentifierLambda, + selfIdentifierLambda, + NewArrayInit( + typeof(Func), + relationalCollectionShaperExpression.ParentIdentifierValueComparers.Select(vc => vc.ObjectEqualsExpression)), + NewArrayInit( + typeof(Func), + relationalCollectionShaperExpression.OuterIdentifierValueComparers.Select(vc => vc.ObjectEqualsExpression)), + NewArrayInit( + typeof(Func), + relationalCollectionShaperExpression.SelfIdentifierValueComparers.Select(vc => vc.ObjectEqualsExpression)), + innerShaper, + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + inverseNavigation, + LiftableConstantExpressionHelpers.BuildNavigationAccessLambda(inverseNavigation), + (inverseNavigation?.Name ?? "null") + "InverseNavigation", + typeof(INavigationBase)), + GenerateFixup(includingEntityClrType, relatedEntityClrType, navigation, inverseNavigation), Constant(_isTracking))); } else if (includeExpression.NavigationExpression is RelationalSplitCollectionShaperExpression @@ -862,11 +923,11 @@ when GetProjectionIndex(collectionResultExpression.ProjectionBindingExpression) var entityType = entity.Type; var navigation = includeExpression.Navigation; - var includingEntityType = navigation.DeclaringEntityType.ClrType; - if (includingEntityType != entityType - && includingEntityType.IsAssignableFrom(entityType)) + var includingEntityClrType = navigation.DeclaringEntityType.ClrType; + if (includingEntityClrType != entityType + && includingEntityClrType.IsAssignableFrom(entityType)) { - includingEntityType = entityType; + includingEntityClrType = entityType; } _inline = true; @@ -889,48 +950,57 @@ when GetProjectionIndex(collectionResultExpression.ProjectionBindingExpression) _includeExpressions.Add( Call( - InitializeSplitIncludeCollectionMethodInfo.MakeGenericMethod(entityType, includingEntityType), + InitializeSplitIncludeCollectionMethodInfo.MakeGenericMethod(entityType, includingEntityClrType), collectionIdConstant, QueryCompilationContext.QueryContextParameter, _dataReaderParameter, _resultCoordinatorParameter, entity, - Constant(parentIdentifierLambda.Compile()), - Constant(navigation), - Constant(navigation.GetCollectionAccessor()), + parentIdentifierLambda, + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + navigation, + LiftableConstantExpressionHelpers.BuildNavigationAccessLambda(navigation), + navigation.Name + "Navigation", + typeof(INavigationBase)), + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + navigation.GetCollectionAccessor(), + LiftableConstantExpressionHelpers.BuildClrCollectionAccessorLambda(navigation), + navigation.Name + "NavigationCollectionAccessor", + typeof(IClrCollectionAccessor)), Constant(_isTracking), #pragma warning disable EF1001 // Internal EF Core API usage. Constant(includeExpression.SetLoaded))); #pragma warning restore EF1001 // Internal EF Core API usage. - var relatedEntityType = innerShaper.ReturnType; + var relatedEntityClrType = innerShaper.ReturnType; var inverseNavigation = navigation.Inverse; _collectionPopulatingExpressions!.Add( Call( (_isAsync ? PopulateSplitIncludeCollectionAsyncMethodInfo : PopulateSplitIncludeCollectionMethodInfo) - .MakeGenericMethod(includingEntityType, relatedEntityType), + .MakeGenericMethod(includingEntityClrType, relatedEntityClrType), collectionIdConstant, Convert(QueryCompilationContext.QueryContextParameter, typeof(RelationalQueryContext)), _executionStrategyParameter!, - Constant(relationalCommandCache), - Constant(readerColumns, typeof(IReadOnlyList)), + relationalCommandCache, + CreateReaderColumnsExpression(readerColumns, _parentVisitor.Dependencies.LiftableConstantFactory), Constant(_detailedErrorsEnabled), _resultCoordinatorParameter, - Constant(childIdentifierLambda.Compile()), - Constant( - relationalSplitCollectionShaperExpression.IdentifierValueComparers, - typeof(IReadOnlyList)), - Constant(innerShaper.Compile()), - Constant( - relatedDataLoaders?.Compile(), + childIdentifierLambda, + NewArrayInit( + typeof(Func), + relationalSplitCollectionShaperExpression.IdentifierValueComparers.Select(vc => vc.ObjectEqualsExpression)), + innerShaper, + relatedDataLoaders ?? (Expression)Constant(null, _isAsync ? typeof(Func) : typeof(Action)), - Constant(inverseNavigation, typeof(INavigationBase)), - Constant( - GenerateFixup( - includingEntityType, relatedEntityType, navigation, inverseNavigation).Compile()), + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + inverseNavigation, + LiftableConstantExpressionHelpers.BuildNavigationAccessLambda(inverseNavigation), + (inverseNavigation?.Name ?? "null") + "InverseNavigation", + typeof(INavigationBase)), + GenerateFixup(includingEntityClrType, relatedEntityClrType, navigation, inverseNavigation), Constant(_isTracking))); } else @@ -982,11 +1052,17 @@ when GetProjectionIndex(collectionResultExpression.ProjectionBindingExpression) QueryCompilationContext.QueryContextParameter, entity, navigationExpression, - Constant(navigation), - Constant(inverseNavigation, typeof(INavigationBase)), - Constant( - GenerateFixup( - includingType, relatedEntityType, navigation, inverseNavigation).Compile()), + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + navigation, + LiftableConstantExpressionHelpers.BuildNavigationAccessLambda(navigation), + navigation.Name + "Navigation", + typeof(INavigation)), + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + inverseNavigation, + LiftableConstantExpressionHelpers.BuildNavigationAccessLambda(inverseNavigation), + (inverseNavigation?.Name ?? "null") + "InverseNavigation", + typeof(INavigation)), + GenerateFixup(includingType, relatedEntityType, navigation, inverseNavigation), Constant(_isTracking)); _includeExpressions.Add(updatedExpression); @@ -1042,9 +1118,13 @@ when GetProjectionIndex(collectionResultExpression.ProjectionBindingExpression) QueryCompilationContext.QueryContextParameter, _dataReaderParameter, _resultCoordinatorParameter, - Constant(parentIdentifierLambda.Compile()), - Constant(outerIdentifierLambda.Compile()), - Constant(collectionAccessor, typeof(IClrCollectionAccessor))))); + parentIdentifierLambda, + outerIdentifierLambda, + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + collectionAccessor, + LiftableConstantExpressionHelpers.BuildClrCollectionAccessorLambda(navigation), + (navigation?.Name ?? "null") + "ClrCollectionAccessor", + typeof(IClrCollectionAccessor))))); _valuesArrayInitializers!.Add(collectionParameter); accessor = Convert( @@ -1060,19 +1140,19 @@ when GetProjectionIndex(collectionResultExpression.ProjectionBindingExpression) QueryCompilationContext.QueryContextParameter, _dataReaderParameter, _resultCoordinatorParameter, - Constant(parentIdentifierLambda.Compile()), - Constant(outerIdentifierLambda.Compile()), - Constant(selfIdentifierLambda.Compile()), - Constant( - relationalCollectionShaperExpression.ParentIdentifierValueComparers, - typeof(IReadOnlyList)), - Constant( - relationalCollectionShaperExpression.OuterIdentifierValueComparers, - typeof(IReadOnlyList)), - Constant( - relationalCollectionShaperExpression.SelfIdentifierValueComparers, - typeof(IReadOnlyList)), - Constant(innerShaper.Compile()))); + parentIdentifierLambda, + outerIdentifierLambda, + selfIdentifierLambda, + NewArrayInit( + typeof(Func), + relationalCollectionShaperExpression.ParentIdentifierValueComparers.Select(vc => vc.ObjectEqualsExpression)), + NewArrayInit( + typeof(Func), + relationalCollectionShaperExpression.OuterIdentifierValueComparers.Select(vc => vc.ObjectEqualsExpression)), + NewArrayInit( + typeof(Func), + relationalCollectionShaperExpression.SelfIdentifierValueComparers.Select(vc => vc.ObjectEqualsExpression)), + innerShaper)); _variableShaperMapping[relationalCollectionShaperExpression] = accessor; } @@ -1121,6 +1201,7 @@ when GetProjectionIndex(collectionResultExpression.ProjectionBindingExpression) var collectionParameter = Parameter(collectionType); _variables.Add(collectionParameter); + _expressions.Add( Assign( collectionParameter, @@ -1130,8 +1211,12 @@ when GetProjectionIndex(collectionResultExpression.ProjectionBindingExpression) QueryCompilationContext.QueryContextParameter, _dataReaderParameter, _resultCoordinatorParameter, - Constant(parentIdentifierLambda.Compile()), - Constant(collectionAccessor, typeof(IClrCollectionAccessor))))); + parentIdentifierLambda, + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + collectionAccessor, + LiftableConstantExpressionHelpers.BuildClrCollectionAccessorLambda(navigation), + (navigation?.Name ?? "null") + "CollectionAccessor", + typeof(IClrCollectionAccessor))))); _valuesArrayInitializers!.Add(collectionParameter); accessor = Convert( @@ -1147,20 +1232,20 @@ when GetProjectionIndex(collectionResultExpression.ProjectionBindingExpression) collectionIdConstant, Convert(QueryCompilationContext.QueryContextParameter, typeof(RelationalQueryContext)), _executionStrategyParameter!, - Constant(relationalCommandCache), - Constant(readerColumns, typeof(IReadOnlyList)), + relationalCommandCache, + CreateReaderColumnsExpression(readerColumns, _parentVisitor.Dependencies.LiftableConstantFactory), Constant(_detailedErrorsEnabled), _resultCoordinatorParameter, - Constant(childIdentifierLambda.Compile()), - Constant( - relationalSplitCollectionShaperExpression.IdentifierValueComparers, - typeof(IReadOnlyList)), - Constant(innerShaper.Compile()), - Constant( - relatedDataLoaders?.Compile(), - _isAsync + childIdentifierLambda, + NewArrayInit( + typeof(Func), + relationalSplitCollectionShaperExpression.IdentifierValueComparers.Select(vc => vc.ObjectEqualsExpression)), + innerShaper, + relatedDataLoaders == null + ? Constant(null, _isAsync ? typeof(Func) - : typeof(Action)))); + : typeof(Action)) + : relatedDataLoaders)); _variableShaperMapping[relationalSplitCollectionShaperExpression] = accessor; } @@ -1170,6 +1255,9 @@ when GetProjectionIndex(collectionResultExpression.ProjectionBindingExpression) case GroupByShaperExpression: throw new InvalidOperationException(RelationalStrings.ClientGroupByNotSupported); + + case LiftableConstantExpression: + return extensionExpression; } return base.VisitExtension(extensionExpression); @@ -1190,6 +1278,12 @@ Expression CompensateForCollectionMaterialization(ParameterExpression parameter, } } + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// protected override Expression VisitMethodCall(MethodCallExpression methodCallExpression) { if (methodCallExpression.Method.IsGenericMethod @@ -1329,7 +1423,7 @@ private Expression CreateJsonShapers( ReferenceEqual(Constant(null), shaperCollectionParameter), IsFalse( Call( - typeof(EnumerableExtensions).GetMethod(nameof(EnumerableExtensions.Any))!, + typeof(ShaperProcessingExpressionVisitor).GetMethod(nameof(Any))!, shaperCollectionParameter))), shaperEntityParameter .MakeMemberAccess(ownedNavigation.GetMemberInfo(forMaterialization: true, forSet: true)) @@ -1396,7 +1490,8 @@ private Expression CreateJsonShapers( innerShapersMap, innerFixupMap, trackingInnerFixupMap, - _queryLogger).Rewrite(entityShaperMaterializer); + _queryLogger, + _parentVisitor.Dependencies.LiftableConstantFactory).Rewrite(entityShaperMaterializer); var entityShaperMaterializerVariable = Variable( entityShaperMaterializer.Type, @@ -1494,7 +1589,11 @@ private Expression CreateJsonShapers( QueryCompilationContext.QueryContextParameter, keyValuesParameter, jsonReaderDataParameter, - Constant(navigation), + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + navigation, + LiftableConstantExpressionHelpers.BuildNavigationAccessLambda(navigation), + navigation.Name + "Navigation", + typeof(INavigation)), shaperLambda); return materializeJsonEntityCollectionMethodCall; @@ -1520,10 +1619,14 @@ private sealed class JsonEntityMaterializerRewriter : ExpressionVisitor private readonly IDictionary _innerFixupMap; private readonly IDictionary _trackingInnerFixupMap; private readonly IDiagnosticsLogger _queryLogger; + private readonly ILiftableConstantFactory _liftableConstantFactory; private static readonly PropertyInfo JsonEncodedTextEncodedUtf8BytesProperty = typeof(JsonEncodedText).GetProperty(nameof(JsonEncodedText.EncodedUtf8Bytes))!; + private static readonly MethodInfo JsonEncodedTextEncodeMethod + = typeof(JsonEncodedText).GetMethod(nameof(JsonEncodedText.Encode), [typeof(string), typeof(JavaScriptEncoder)])!; + // keep track which variable corresponds to which navigation - we need that info for fixup // which happens at the end (after we read everything to guarantee that we can instantiate the entity private readonly Dictionary _navigationVariableMap = new(); @@ -1535,7 +1638,8 @@ public JsonEntityMaterializerRewriter( IDictionary innerShapersMap, IDictionary innerFixupMap, IDictionary trackingInnerFixupMap, - IDiagnosticsLogger queryLogger) + IDiagnosticsLogger queryLogger, + ILiftableConstantFactory liftableConstantFactory) { _entityType = entityType; _isTracking = isTracking; @@ -1544,6 +1648,7 @@ public JsonEntityMaterializerRewriter( _innerFixupMap = innerFixupMap; _trackingInnerFixupMap = trackingInnerFixupMap; _queryLogger = queryLogger; + _liftableConstantFactory = liftableConstantFactory; } public BlockExpression Rewrite(BlockExpression jsonEntityShaperMaterializer) @@ -1554,10 +1659,15 @@ protected override Expression VisitSwitch(SwitchExpression switchExpression) if (switchExpression.SwitchValue.Type == typeof(IEntityType) && switchExpression is { - Cases: [{ TestValues: [ConstantExpression onlyValue], Body: BlockExpression body }] + Cases: + [ + { + Body: BlockExpression { Expressions.Count: > 0 } body, + TestValues: [Expression onlyValueExpression] + } + ] } - && onlyValue.Value == _entityType - && body.Expressions.Count > 0) + && onlyValueExpression.GetConstantValue() == _entityType) { var valueBufferTryReadValueMethodsToProcess = new ValueBufferTryReadValueMethodsFinder(_entityType).FindValueBufferTryReadValueMethods(body); @@ -1647,7 +1757,11 @@ protected override Expression VisitSwitch(SwitchExpression switchExpression) New( JsonReaderManagerConstructor, _jsonReaderDataParameter, - Constant(_queryLogger))), + _liftableConstantFactory.CreateLiftableConstant( + _queryLogger, + static c => c.Dependencies.QueryLogger, + "queryLogger", + typeof(IDiagnosticsLogger)))), // tokenType = jsonReaderManager.CurrentReader.TokenType Assign( tokenTypeVariable, @@ -1759,8 +1873,8 @@ void ProcessFixup(IDictionary fixupMap) foreach (var valueBufferTryReadValueMethodToProcess in valueBufferTryReadValueMethodsToProcess) { - var property = (IProperty)((ConstantExpression)valueBufferTryReadValueMethodToProcess.Arguments[2]).Value!; - + var property = valueBufferTryReadValueMethodToProcess.Arguments[2].GetConstantValue(); + var jsonPropertyName = property.GetJsonPropertyName()!; testExpressions.Add( Call( Field( @@ -1768,7 +1882,18 @@ void ProcessFixup(IDictionary fixupMap) Utf8JsonReaderManagerCurrentReaderField), Utf8JsonReaderValueTextEqualsMethod, Property( - Constant(JsonEncodedText.Encode(property.GetJsonPropertyName()!)), + _liftableConstantFactory.CreateLiftableConstant( + JsonEncodedText.Encode(jsonPropertyName), + Lambda>( + Convert( + Call( + JsonEncodedTextEncodeMethod, + Constant(jsonPropertyName), + Default(typeof(JavaScriptEncoder))), + typeof(object)), + Parameter(typeof(MaterializerLiftableConstantContext), "_")), + jsonPropertyName + "EncodedProperty", + typeof(JsonEncodedText)), JsonEncodedTextEncodedUtf8BytesProperty))); var propertyVariable = Variable(valueBufferTryReadValueMethodToProcess.Type); @@ -1794,6 +1919,7 @@ void ProcessFixup(IDictionary fixupMap) foreach (var innerShaperMapElement in _innerShapersMap) { + var innerShaperMapElementKey = innerShaperMapElement.Key; testExpressions.Add( Call( Field( @@ -1801,7 +1927,18 @@ void ProcessFixup(IDictionary fixupMap) Utf8JsonReaderManagerCurrentReaderField), Utf8JsonReaderValueTextEqualsMethod, Property( - Constant(JsonEncodedText.Encode(innerShaperMapElement.Key)), + _liftableConstantFactory.CreateLiftableConstant( + JsonEncodedText.Encode(innerShaperMapElementKey), + Lambda>( + Convert( + Call( + JsonEncodedTextEncodeMethod, + Constant(innerShaperMapElementKey), + Default(typeof(JavaScriptEncoder))), + typeof(object)), + Parameter(typeof(MaterializerLiftableConstantContext), "_")), + innerShaperMapElementKey + "EncodedNavigation", + typeof(JsonEncodedText)), JsonEncodedTextEncodedUtf8BytesProperty))); var propertyVariable = Variable(innerShaperMapElement.Value.Type); @@ -1813,7 +1950,15 @@ void ProcessFixup(IDictionary fixupMap) var captureState = Call(managerVariable, Utf8JsonReaderManagerCaptureStateMethod); var assignment = Assign(propertyVariable, innerShaperMapElement.Value); var managerRecreation = Assign( - managerVariable, New(JsonReaderManagerConstructor, _jsonReaderDataParameter, Constant(_queryLogger))); + managerVariable, + New( + JsonReaderManagerConstructor, + _jsonReaderDataParameter, + _liftableConstantFactory.CreateLiftableConstant( + _queryLogger, + static c => c.Dependencies.QueryLogger, + "queryLogger", + typeof(IDiagnosticsLogger)))); readExpressions.Add( Block( @@ -2032,7 +2177,7 @@ protected override Expression VisitMethodCall(MethodCallExpression methodCallExp if (methodCallExpression.Method.IsGenericMethod && methodCallExpression.Method.GetGenericMethodDefinition() == Infrastructure.ExpressionExtensions.ValueBufferTryReadValueMethod - && ((ConstantExpression)methodCallExpression.Arguments[2]).Value is IProperty property + && methodCallExpression.Arguments[2].GetConstantValue() is IProperty property && _nonKeyProperties.Contains(property)) { _valueBufferTryReadValueMethods.Add(methodCallExpression); @@ -2115,7 +2260,7 @@ private bool IsPropertyAssignment( if (methodCallExpression.Method.IsGenericMethod && methodCallExpression.Method.GetGenericMethodDefinition() == Infrastructure.ExpressionExtensions.ValueBufferTryReadValueMethod - && ((ConstantExpression)methodCallExpression.Arguments[2]).Value is IProperty prop + && methodCallExpression.Arguments[2].GetConstantValue() is IProperty prop && _propertyAssignmentMap.TryGetValue(prop, out var param)) { property = prop; @@ -2174,7 +2319,15 @@ private bool IsPropertyAssignment( Default(typeof(JsonReaderData))), Block( Assign( - jsonReaderManagerVariable, New(JsonReaderManagerConstructor, jsonReaderDataVariable, Constant(_queryLogger))), + jsonReaderManagerVariable, + New( + JsonReaderManagerConstructor, + jsonReaderDataVariable, + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + _queryLogger, + static c => c.Dependencies.QueryLogger, + "queryLogger", + typeof(IDiagnosticsLogger)))), Call(jsonReaderManagerVariable, Utf8JsonReaderManagerMoveNextMethod), Call(jsonReaderManagerVariable, Utf8JsonReaderManagerCaptureStateMethod))); @@ -2308,10 +2461,13 @@ protected override Expression VisitBinary(BinaryExpression binaryExpression) Left: MethodCallExpression { Method: { IsGenericMethod: true } method, - Arguments: [_, _, ConstantExpression { Value: IProperty property }] + Arguments: [_, _, Expression leftExpression] }, - Right: ConstantExpression { Value: null } + Right: Expression rightExpression } + && leftExpression.TryGetNonNullConstantValue(out var property) + && rightExpression is ConstantExpression or LiftableConstantExpression + && rightExpression.GetConstantValue() == null && method.GetGenericMethodDefinition() == Infrastructure.ExpressionExtensions.ValueBufferTryReadValueMethod) { return _mappedProperties.Contains(property) @@ -2327,8 +2483,9 @@ protected override Expression VisitMethodCall(MethodCallExpression methodCallExp if (methodCallExpression is { Method: { IsGenericMethod: true } method, - Arguments: [_, _, ConstantExpression { Value: IProperty property }] + Arguments: [_, _, Expression argumentExpression] } + && argumentExpression.TryGetNonNullConstantValue(out var property) && method.GetGenericMethodDefinition() == Infrastructure.ExpressionExtensions.ValueBufferTryReadValueMethod && !_mappedProperties.Contains(property)) { @@ -2339,7 +2496,7 @@ protected override Expression VisitMethodCall(MethodCallExpression methodCallExp } } - private static LambdaExpression GenerateFixup( + private LambdaExpression GenerateFixup( Type entityType, Type relatedEntityType, INavigationBase navigation, @@ -2401,7 +2558,14 @@ private static LambdaExpression GenerateReferenceFixupForJson( return Lambda(Block(typeof(void), expressions), entityParameter, relatedEntityParameter); } - private static void InverseCollectionFixup( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static void InverseCollectionFixup( ICollection collection, TEntity entity, Action elementFixup) @@ -2418,7 +2582,7 @@ private static Expression AssignReferenceNavigation( INavigationBase navigation) => entity.MakeMemberAccess(navigation.GetMemberInfo(forMaterialization: true, forSet: true)).Assign(relatedEntity); - private static Expression GetOrCreateCollectionObjectLambda( + private Expression GetOrCreateCollectionObjectLambda( Type entityType, INavigationBase navigation) { @@ -2428,19 +2592,27 @@ private static Expression GetOrCreateCollectionObjectLambda( Block( typeof(void), Call( - Constant(navigation.GetCollectionAccessor()), + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + navigation.GetCollectionAccessor(), + LiftableConstantExpressionHelpers.BuildClrCollectionAccessorLambda(navigation), + navigation.Name + "NavigationCollectionAccessor", + typeof(IClrCollectionAccessor)), CollectionAccessorGetOrCreateMethodInfo, prm, Constant(true))), prm); } - private static Expression AddToCollectionNavigation( + private Expression AddToCollectionNavigation( ParameterExpression entity, ParameterExpression relatedEntity, INavigationBase navigation) => Call( - Constant(navigation.GetCollectionAccessor()), + _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + navigation.GetCollectionAccessor(), + LiftableConstantExpressionHelpers.BuildClrCollectionAccessorLambda(navigation), + navigation.Name + "NavigationCollectionAccessor", + typeof(IClrCollectionAccessor)), CollectionAccessorAddMethodInfo, entity, relatedEntity, @@ -2508,7 +2680,7 @@ Expression valueExpression Lambda( bufferedReaderLambdaExpression, dbDataReader, - _indexMapParameter ?? Parameter(typeof(int[]))).Compile()); + _indexMapParameter ?? Parameter(typeof(int[]), "indexMap"))); } valueExpression = Call( @@ -2523,17 +2695,66 @@ Expression valueExpression var converter = typeMapping.Converter; + var converterExpression = default(Expression); if (converter != null) { - if (valueExpression.Type != converter.ProviderClrType) + // if IProperty is available, we can reliably get the converter from the model and then incorporate FromProvider(Typed) delegate + // into the expression. This way we have consistent behavior between precompiled and normal queries (same code path) + // however, if IProperty is not available, we could try to get TypeMapping from TypeMappingSource based on ClrType, but that may + // return incorrect mapping. So for that case we would prefer to incorporate the FromProvider lambda, like we used to do before AOT + // and only resort to unreliable TypeMappingSource lookup, if the converter expression captures "forbidden" constant + var requiresLiftableConstant = new ConstantValidator().RequiresLiftableConstant(converter.ConvertFromProviderExpression.Body); + if (property != null || requiresLiftableConstant) { - valueExpression = Convert(valueExpression, converter.ProviderClrType); + var typeMappingExpression = CreateTypeMappingExpression(property, type, typeMapping, _parentVisitor.Dependencies.LiftableConstantFactory); + converterExpression = Property(typeMappingExpression, nameof(CoreTypeMapping.Converter)); + + var converterType = converter.GetType(); + var typedConverterType = converterType.GetGenericTypeImplementations(typeof(ValueConverter<,>)).FirstOrDefault(); + var invocationExpression = default(Expression); + + // TODO: do we even need to do this check? can we ever have a custom ValueConverter that is not generic? + if (typedConverterType != null) + { + if (converterExpression.Type != converter.GetType()) + { + converterExpression = Convert(converterExpression, converter.GetType()); + } + + if (valueExpression.Type != converter.ProviderClrType) + { + valueExpression = Convert(valueExpression, converter.ProviderClrType); + } + + invocationExpression = Invoke( + Property( + converterExpression, + nameof(ValueConverter.ConvertFromProviderTyped)), + valueExpression); + } + else + { + invocationExpression = Invoke( + Property( + converterExpression, + nameof(ValueConverter.ConvertFromProvider)), + Convert(valueExpression, typeof(object))); + } + + valueExpression = invocationExpression; } + else + { + if (valueExpression.Type != converter.ProviderClrType) + { + valueExpression = Convert(valueExpression, converter.ProviderClrType); + } - valueExpression = ReplacingExpressionVisitor.Replace( - converter.ConvertFromProviderExpression.Parameters.Single(), - valueExpression, - converter.ConvertFromProviderExpression.Body); + valueExpression = ReplacingExpressionVisitor.Replace( + converter.ConvertFromProviderExpression.Parameters.Single(), + valueExpression, + converter.ConvertFromProviderExpression.Body); + } } if (valueExpression.Type != type) @@ -2546,10 +2767,45 @@ Expression valueExpression Expression replaceExpression; if (converter?.ConvertsNulls == true) { - replaceExpression = ReplacingExpressionVisitor.Replace( - converter.ConvertFromProviderExpression.Parameters.Single(), - Default(converter.ProviderClrType), - converter.ConvertFromProviderExpression.Body); + // we potentially have to repeat logic from above here. We can check if we computed converterExpression before + // if so, it means there are liftable constants in the ConvertFromProvider expression + // we can also reuse converter expression, just switch argument to the Invoke for default(provier type) or object + if (converterExpression != null) + { + var converterType = converter.GetType(); + var typedConverterType = converterType.GetGenericTypeImplementations(typeof(ValueConverter<,>)).FirstOrDefault(); + var invocationExpression = default(Expression); + if (typedConverterType != null) + { + if (converterExpression.Type != converter.GetType()) + { + converterExpression = Convert(converterExpression, converter.GetType()); + } + + invocationExpression = Invoke( + Property( + converterExpression, + nameof(ValueConverter.ConvertFromProviderTyped)), + Default(converter.ProviderClrType)); + } + else + { + invocationExpression = Invoke( + Property( + converterExpression, + nameof(ValueConverter.ConvertFromProvider)), + Default(typeof(object))); + } + + replaceExpression = invocationExpression; + } + else + { + replaceExpression = ReplacingExpressionVisitor.Replace( + converter.ConvertFromProviderExpression.Parameters.Single(), + Default(converter.ProviderClrType), + converter.ConvertFromProviderExpression.Body); + } if (replaceExpression.Type != type) { @@ -2579,23 +2835,104 @@ Expression valueExpression exceptionParameter, Call(dbDataReader, GetFieldValueMethod.MakeGenericMethod(typeof(object)), indexExpression), Constant(valueExpression.Type.MakeNullable(nullable), typeof(Type)), - Constant(property, typeof(IPropertyBase)))); + property == null + ? Default(typeof(IPropertyBase)) + : _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + property, + LiftableConstantExpressionHelpers.BuildMemberAccessLambdaForProperty(property), + property + "Property", + typeof(IPropertyBase)))); valueExpression = TryCatch(valueExpression, catchBlock); } return valueExpression; + + static Expression CreateTypeMappingExpression( + IPropertyBase? property, + Type type, + RelationalTypeMapping typeMapping, + ILiftableConstantFactory liftableConstantFactory) + { + Expression typeMappingExpression; + if (property != null) + { + typeMappingExpression = Call( + Convert( + liftableConstantFactory.CreateLiftableConstant( + property, + LiftableConstantExpressionHelpers.BuildMemberAccessLambdaForProperty(property), + property.Name + "Property", + typeof(IPropertyBase)), + typeof(IReadOnlyProperty)), + PropertyGetTypeMappingMethod); + } + else + { + // NOTE: this is unreliable way to get type mapping. Only doing this as last resort, hoping to "guess" the right one + Expression> resolverTemplate = + (c, _) => (RelationalTypeMapping)c.Dependencies.TypeMappingSource.FindMapping(_, c.Dependencies.Model, null)!; + + var body = ReplacingExpressionVisitor.Replace( + resolverTemplate.Parameters[1], + Constant(type), + resolverTemplate.Body); + + typeMappingExpression = liftableConstantFactory.CreateLiftableConstant( + typeMapping, + Lambda>(body, resolverTemplate.Parameters[0]), + "typeMapping", + typeof(RelationalTypeMapping)); + } + + return typeMappingExpression; + } + } + + private sealed class ConstantValidator : ExpressionVisitor + { + private bool _requiresLiftableConstant; + + public bool RequiresLiftableConstant(Expression expression) + { + _requiresLiftableConstant = false; + Visit(expression); + + return _requiresLiftableConstant; + } + + protected override Expression VisitConstant(ConstantExpression constantExpression) + { + if (!_requiresLiftableConstant && !LiftableConstantExpressionHelpers.IsLiteral(constantExpression.Value)) + { + _requiresLiftableConstant = true; + } + + return constantExpression; + } } private Expression CreateReadJsonPropertyValueExpression( ParameterExpression jsonReaderManagerParameter, IProperty property) { - var nullable = property.IsNullable; - var typeMapping = property.GetTypeMapping(); - - var jsonReaderWriterExpression = Constant( - property.GetJsonValueReaderWriter() ?? property.GetTypeMapping().JsonValueReaderWriter!); + var jsonReaderWriter = property.GetJsonValueReaderWriter() ?? property.GetTypeMapping().JsonValueReaderWriter!; + var prm = Parameter(typeof(MaterializerLiftableConstantContext), "c"); + var jsonReaderWriterExpression = _parentVisitor.Dependencies.LiftableConstantFactory.CreateLiftableConstant( + property.GetJsonValueReaderWriter() ?? property.GetTypeMapping().JsonValueReaderWriter!, + Lambda>( + Coalesce( + Call( + LiftableConstantExpressionHelpers.BuildMemberAccessForProperty(property, prm), + PropertyGetJsonValueReaderWriterMethod), + Property( + Call( + LiftableConstantExpressionHelpers.BuildMemberAccessForProperty(property, prm), + PropertyGetTypeMappingMethod), + nameof(CoreTypeMapping.JsonValueReaderWriter))), + prm), + property.Name + "PropertyName", + jsonReaderWriter.GetType()); var fromJsonMethod = jsonReaderWriterExpression.Type.GetMethod( nameof(JsonValueReaderWriter.FromJsonTyped), @@ -2603,9 +2940,9 @@ private Expression CreateReadJsonPropertyValueExpression( Expression resultExpression = Convert( Call(jsonReaderWriterExpression, fromJsonMethod, jsonReaderManagerParameter, Default(typeof(object))), - typeMapping.ClrType); + property.GetTypeMapping().ClrType); - if (nullable) + if (property.IsNullable) { // in case of null value we can't just use the JsonReader method, but rather check the current token type // if it's JsonTokenType.Null means value is null, only if it's not we are safe to read the value diff --git a/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitor.cs b/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitor.cs index ede5805ad30..19d4ecd9f39 100644 --- a/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitor.cs +++ b/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitor.cs @@ -70,17 +70,12 @@ protected virtual Expression VisitNonQuery(NonQueryExpression nonQueryExpression break; } - var relationalCommandCache = new RelationalCommandCache( - Dependencies.MemoryCache, - RelationalDependencies.QuerySqlGeneratorFactory, - RelationalDependencies.RelationalParameterBasedSqlProcessorFactory, - innerExpression, - _useRelationalNulls); + var relationalCommandCache = CreateRelationalCommandCacheExpression(innerExpression); return Call( QueryCompilationContext.IsAsync ? NonQueryAsyncMethodInfo : NonQueryMethodInfo, Convert(QueryCompilationContext.QueryContextParameter, typeof(RelationalQueryContext)), - Constant(relationalCommandCache), + relationalCommandCache, Constant(_contextType), Constant(nonQueryExpression.CommandSource), Constant(_threadSafetyChecksEnabled)); @@ -96,7 +91,14 @@ private static readonly MethodInfo NonQueryAsyncMethodInfo .GetDeclaredMethods(nameof(NonQueryResultAsync)) .Single(mi => mi.GetParameters().Length == 5); - private static int NonQueryResult( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static int NonQueryResult( RelationalQueryContext relationalQueryContext, RelationalCommandCache relationalCommandCache, Type contextType, @@ -167,7 +169,14 @@ private static int NonQueryResult( } } - private static Task NonQueryResultAsync( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static Task NonQueryResultAsync( RelationalQueryContext relationalQueryContext, RelationalCommandCache relationalCommandCache, Type contextType, @@ -268,32 +277,34 @@ protected override Expression VisitShapedQuery(ShapedQueryExpression shapedQuery QueryCompilationContext.Logger.MultipleCollectionIncludeWarning(); } + var readerColumnsExpression = CreateReaderColumnsExpression(readerColumns, Dependencies.LiftableConstantFactory); if (splitQuery) { - var relatedDataLoadersParameter = Constant( - QueryCompilationContext.IsAsync ? null : relatedDataLoaders?.Compile(), - typeof(Action)); + var relatedDataLoadersParameter = QueryCompilationContext.IsAsync || relatedDataLoaders == null + ? (Expression)Constant(null, typeof(Action)) + : relatedDataLoaders; - var relatedDataLoadersAsyncParameter = Constant( - QueryCompilationContext.IsAsync ? relatedDataLoaders?.Compile() : null, - typeof(Func)); + var relatedDataLoadersAsyncParameter = QueryCompilationContext.IsAsync && relatedDataLoaders != null + ? relatedDataLoaders! + : (Expression)Constant(null, typeof(Func)); return New( typeof(GroupBySplitQueryingEnumerable<,>).MakeGenericType( keySelector.ReturnType, elementSelector.ReturnType).GetConstructors()[0], Convert(QueryCompilationContext.QueryContextParameter, typeof(RelationalQueryContext)), - Constant(relationalCommandCache), - Constant(readerColumns, typeof(IReadOnlyList)), - Constant(keySelector.Compile()), - Constant(keyIdentifier.Compile()), - Constant(relationalGroupByResultExpression.KeyIdentifierValueComparers, typeof(IReadOnlyList)), - Constant(elementSelector.Compile()), + relationalCommandCache, + readerColumnsExpression, + keySelector, + keyIdentifier, + NewArrayInit( + typeof(Func), + relationalGroupByResultExpression.KeyIdentifierValueComparers.Select(vc => vc.ObjectEqualsExpression)), + elementSelector, relatedDataLoadersParameter, relatedDataLoadersAsyncParameter, Constant(_contextType), - Constant( - QueryCompilationContext.QueryTrackingBehavior == QueryTrackingBehavior.NoTrackingWithIdentityResolution), + Constant(QueryCompilationContext.QueryTrackingBehavior == QueryTrackingBehavior.NoTrackingWithIdentityResolution), Constant(_detailedErrorsEnabled), Constant(_threadSafetyChecksEnabled)); } @@ -303,15 +314,16 @@ protected override Expression VisitShapedQuery(ShapedQueryExpression shapedQuery keySelector.ReturnType, elementSelector.ReturnType).GetConstructors()[0], Convert(QueryCompilationContext.QueryContextParameter, typeof(RelationalQueryContext)), - Constant(relationalCommandCache), - Constant(readerColumns, typeof(IReadOnlyList)), - Constant(keySelector.Compile()), - Constant(keyIdentifier.Compile()), - Constant(relationalGroupByResultExpression.KeyIdentifierValueComparers, typeof(IReadOnlyList)), - Constant(elementSelector.Compile()), + relationalCommandCache, + readerColumnsExpression, + keySelector, + keyIdentifier, + NewArrayInit( + typeof(Func), + relationalGroupByResultExpression.KeyIdentifierValueComparers.Select(vc => vc.ObjectEqualsExpression)), + elementSelector, Constant(_contextType), - Constant( - QueryCompilationContext.QueryTrackingBehavior == QueryTrackingBehavior.NoTrackingWithIdentityResolution), + Constant(QueryCompilationContext.QueryTrackingBehavior == QueryTrackingBehavior.NoTrackingWithIdentityResolution), Constant(_detailedErrorsEnabled), Constant(_threadSafetyChecksEnabled)); } @@ -319,8 +331,8 @@ protected override Expression VisitShapedQuery(ShapedQueryExpression shapedQuery { var nonComposedFromSql = selectExpression.IsNonComposedFromSql(); var shaper = new ShaperProcessingExpressionVisitor(this, selectExpression, _tags, splitQuery, nonComposedFromSql).ProcessShaper( - shapedQueryExpression.ShaperExpression, - out var relationalCommandCache, out var readerColumns, out var relatedDataLoaders, ref collectionCount); + shapedQueryExpression.ShaperExpression, out var relationalCommandCache, out var readerColumns, + out var relatedDataLoaders, ref collectionCount); if (querySplittingBehavior == null && collectionCount > 1) @@ -328,60 +340,134 @@ protected override Expression VisitShapedQuery(ShapedQueryExpression shapedQuery QueryCompilationContext.Logger.MultipleCollectionIncludeWarning(); } + var readerColumnsExpression = CreateReaderColumnsExpression(readerColumns, Dependencies.LiftableConstantFactory); if (nonComposedFromSql) { - return New( - typeof(FromSqlQueryingEnumerable<>).MakeGenericType(shaper.ReturnType).GetConstructors()[0], + return Call( + typeof(FromSqlQueryingEnumerable).GetMethods() + .Single(m => m.Name == nameof(FromSqlQueryingEnumerable.Create)) + .MakeGenericMethod(shaper.ReturnType), Convert(QueryCompilationContext.QueryContextParameter, typeof(RelationalQueryContext)), - Constant(relationalCommandCache), - Constant(readerColumns, typeof(IReadOnlyList)), - Constant( - selectExpression.Projection.Select(pe => ((ColumnExpression)pe.Expression).Name).ToList(), - typeof(IReadOnlyList)), - Constant(shaper.Compile()), + relationalCommandCache, + readerColumnsExpression, + NewArrayInit( + typeof(string), + selectExpression.Projection.Select(pe => Constant(((ColumnExpression)pe.Expression).Name, typeof(string)))), + shaper, Constant(_contextType), - Constant( - QueryCompilationContext.QueryTrackingBehavior == QueryTrackingBehavior.NoTrackingWithIdentityResolution), + Constant(QueryCompilationContext.QueryTrackingBehavior == QueryTrackingBehavior.NoTrackingWithIdentityResolution), Constant(_detailedErrorsEnabled), Constant(_threadSafetyChecksEnabled)); } if (splitQuery) { - var relatedDataLoadersParameter = Constant( - QueryCompilationContext.IsAsync ? null : relatedDataLoaders?.Compile(), - typeof(Action)); - - var relatedDataLoadersAsyncParameter = Constant( - QueryCompilationContext.IsAsync ? relatedDataLoaders?.Compile() : null, - typeof(Func)); - - return New( - typeof(SplitQueryingEnumerable<>).MakeGenericType(shaper.ReturnType).GetConstructors().Single(), + var relatedDataLoadersParameter = + QueryCompilationContext.IsAsync || relatedDataLoaders is null + ? Constant(null, typeof(Action)) + : (Expression)relatedDataLoaders; + + var relatedDataLoadersAsyncParameter = + QueryCompilationContext.IsAsync && relatedDataLoaders is not null + ? (Expression)relatedDataLoaders + : Constant(null, typeof(Func)); + + return Call( + typeof(SplitQueryingEnumerable).GetMethods() + .Single(m => m.Name == nameof(FromSqlQueryingEnumerable.Create)) + .MakeGenericMethod(shaper.ReturnType), Convert(QueryCompilationContext.QueryContextParameter, typeof(RelationalQueryContext)), - Constant(relationalCommandCache), - Constant(readerColumns, typeof(IReadOnlyList)), - Constant(shaper.Compile()), + relationalCommandCache, + readerColumnsExpression, + shaper, relatedDataLoadersParameter, relatedDataLoadersAsyncParameter, Constant(_contextType), - Constant( - QueryCompilationContext.QueryTrackingBehavior == QueryTrackingBehavior.NoTrackingWithIdentityResolution), + Constant(QueryCompilationContext.QueryTrackingBehavior == QueryTrackingBehavior.NoTrackingWithIdentityResolution), Constant(_detailedErrorsEnabled), Constant(_threadSafetyChecksEnabled)); } - return New( - typeof(SingleQueryingEnumerable<>).MakeGenericType(shaper.ReturnType).GetConstructors()[0], + return Call( + typeof(SingleQueryingEnumerable).GetMethods() + .Single(m => m.Name == nameof(SingleQueryingEnumerable.Create)) + .MakeGenericMethod(shaper.ReturnType), Convert(QueryCompilationContext.QueryContextParameter, typeof(RelationalQueryContext)), - Constant(relationalCommandCache), - Constant(readerColumns, typeof(IReadOnlyList)), - Constant(shaper.Compile()), + relationalCommandCache, + readerColumnsExpression, + shaper, Constant(_contextType), - Constant( - QueryCompilationContext.QueryTrackingBehavior == QueryTrackingBehavior.NoTrackingWithIdentityResolution), + Constant(QueryCompilationContext.QueryTrackingBehavior == QueryTrackingBehavior.NoTrackingWithIdentityResolution), Constant(_detailedErrorsEnabled), Constant(_threadSafetyChecksEnabled)); } } + + private static Expression CreateReaderColumnsExpression( + IReadOnlyList? readerColumns, + ILiftableConstantFactory liftableConstantFactory) + { + if (readerColumns is null) + { + return Constant(readerColumns, typeof(ReaderColumn?[])); + } + + var materializerLiftableConstantContextParameter = Parameter(typeof(MaterializerLiftableConstantContext)); + var initializers = new List(); + + foreach (var readerColumn in readerColumns) + { + var currentReaderColumn = readerColumn; + if (currentReaderColumn is null) + { + initializers.Add(Constant(null, typeof(ReaderColumn))); + continue; + } + + var propertyExpression = LiftableConstantExpressionHelpers.BuildMemberAccessForProperty( + currentReaderColumn.Property, + materializerLiftableConstantContextParameter); + + initializers.Add( + New( + ReaderColumn.GetConstructor(currentReaderColumn.Type), + Constant(currentReaderColumn.IsNullable), + Constant(currentReaderColumn.Name, typeof(string)), + propertyExpression, + currentReaderColumn.GetFieldValueExpression)); + } + + var result = liftableConstantFactory.CreateLiftableConstant( + readerColumns, + Lambda>( + NewArrayInit( + typeof(ReaderColumn), + initializers), + materializerLiftableConstantContextParameter), + "readerColumns", + typeof(ReaderColumn[])); + + return result; + } + + private Expression CreateRelationalCommandCacheExpression(Expression queryExpression) + { + var relationalCommandCache = new RelationalCommandCache( + Dependencies.MemoryCache, + RelationalDependencies.QuerySqlGeneratorFactory, + RelationalDependencies.RelationalParameterBasedSqlProcessorFactory, + queryExpression, + _useRelationalNulls); + + return RelationalDependencies.RelationalLiftableConstantFactory.CreateLiftableConstant( + relationalCommandCache, + c => new RelationalCommandCache( + c.Dependencies.MemoryCache, + c.RelationalDependencies.QuerySqlGeneratorFactory, + c.RelationalDependencies.RelationalParameterBasedSqlProcessorFactory, + queryExpression, + _useRelationalNulls), + "relationalCommandCache", + typeof(RelationalCommandCache)); + } } diff --git a/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitorDependencies.cs b/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitorDependencies.cs index c32ba6d83b5..bca7ecafc93 100644 --- a/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitorDependencies.cs +++ b/src/EFCore.Relational/Query/RelationalShapedQueryCompilingExpressionVisitorDependencies.cs @@ -47,10 +47,12 @@ public sealed record RelationalShapedQueryCompilingExpressionVisitorDependencies [EntityFrameworkInternal] public RelationalShapedQueryCompilingExpressionVisitorDependencies( IQuerySqlGeneratorFactory querySqlGeneratorFactory, - IRelationalParameterBasedSqlProcessorFactory relationalParameterBasedSqlProcessorFactory) + IRelationalParameterBasedSqlProcessorFactory relationalParameterBasedSqlProcessorFactory, + IRelationalLiftableConstantFactory relationalLiftableConstantFactory) { QuerySqlGeneratorFactory = querySqlGeneratorFactory; RelationalParameterBasedSqlProcessorFactory = relationalParameterBasedSqlProcessorFactory; + RelationalLiftableConstantFactory = relationalLiftableConstantFactory; } /// @@ -62,4 +64,9 @@ public RelationalShapedQueryCompilingExpressionVisitorDependencies( /// The SQL processor based on parameter values. /// public IRelationalParameterBasedSqlProcessorFactory RelationalParameterBasedSqlProcessorFactory { get; init; } + + /// + /// The liftable constant factory. + /// + public IRelationalLiftableConstantFactory RelationalLiftableConstantFactory { get; init; } } diff --git a/src/EFCore.Relational/Query/RelationalSqlTranslatingExpressionVisitor.cs b/src/EFCore.Relational/Query/RelationalSqlTranslatingExpressionVisitor.cs index 85db91718ea..fb5439908b3 100644 --- a/src/EFCore.Relational/Query/RelationalSqlTranslatingExpressionVisitor.cs +++ b/src/EFCore.Relational/Query/RelationalSqlTranslatingExpressionVisitor.cs @@ -2107,7 +2107,14 @@ when memberInitExpression.Bindings.SingleOrDefault(mb => mb.Member.Name == compl _ => throw new UnreachableException() }; - private static T? ParameterValueExtractor( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static T? ParameterValueExtractor( QueryContext context, string baseParameterName, List? complexPropertyChain, @@ -2131,7 +2138,14 @@ when memberInitExpression.Bindings.SingleOrDefault(mb => mb.Member.Name == compl return baseValue == null ? (T?)(object?)null : (T?)property.GetGetter().GetClrValue(baseValue); } - private static List? ParameterListValueExtractor( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static List? ParameterListValueExtractor( QueryContext context, string baseParameterName, IProperty property) diff --git a/src/EFCore.Relational/Query/SqlExpressions/SelectExpression.cs b/src/EFCore.Relational/Query/SqlExpressions/SelectExpression.cs index e6156e3c736..2334d206162 100644 --- a/src/EFCore.Relational/Query/SqlExpressions/SelectExpression.cs +++ b/src/EFCore.Relational/Query/SqlExpressions/SelectExpression.cs @@ -2098,68 +2098,68 @@ StructuralTypeProjectionExpression ProcessStructuralType( var complexPropertyCache = new Dictionary(); var type = structuralProjection1.StructuralType; - foreach (var property in type.GetPropertiesInHierarchy()) + foreach (var property in type.GetPropertiesInHierarchy()) + { + var column1 = structuralProjection1.BindProperty(property); + var column2 = structuralProjection2.BindProperty(property); + var alias = GenerateUniqueColumnAlias(column1.Name); + var innerProjection = new ProjectionExpression(column1, alias); + select1._projection.Add(innerProjection); + select2._projection.Add(new ProjectionExpression(column2, alias)); + var outerColumn = CreateColumnExpression(innerProjection, setOperationAlias); + if (column1.IsNullable + || column2.IsNullable) { - var column1 = structuralProjection1.BindProperty(property); - var column2 = structuralProjection2.BindProperty(property); - var alias = GenerateUniqueColumnAlias(column1.Name); - var innerProjection = new ProjectionExpression(column1, alias); - select1._projection.Add(innerProjection); - select2._projection.Add(new ProjectionExpression(column2, alias)); - var outerColumn = CreateColumnExpression(innerProjection, setOperationAlias); - if (column1.IsNullable - || column2.IsNullable) - { - outerColumn = outerColumn.MakeNullable(); - } + outerColumn = outerColumn.MakeNullable(); + } - propertyExpressions[property] = outerColumn; + propertyExpressions[property] = outerColumn; - // Lift up any identifier columns to the set operation result (the outer). - // This is typically the entity primary key columns, but can also be all of a complex type's properties if Distinct - // was previously called. - if (outerIdentifiers.Length > 0) + // Lift up any identifier columns to the set operation result (the outer). + // This is typically the entity primary key columns, but can also be all of a complex type's properties if Distinct + // was previously called. + if (outerIdentifiers.Length > 0) + { + var index = select1._identifier.FindIndex(e => e.Column.Equals(column1)); + if (index != -1) { - var index = select1._identifier.FindIndex(e => e.Column.Equals(column1)); - if (index != -1) + if (select2._identifier[index].Column.Equals(column2)) { - if (select2._identifier[index].Column.Equals(column2)) - { - outerIdentifiers[index] = outerColumn; - } - else - { - // If select1 matched but select2 did not then we erase all identifiers - // TODO: We could make this little more robust by allow the indexes to be different. See issue#24475 - // i.e. Identifier ordering being different. - outerIdentifiers = []; - } + outerIdentifiers[index] = outerColumn; } - // If the top-level projection - not the current nested one - is a complex type and not an entity type, then add - // all its columns to the "otherExpressions" list (i.e. columns not part of a an entity primary key). This is - // the same as with a non-structural type projection. - else if (projection1.StructuralType is IComplexType) + else { - var outerTypeMapping = column1.TypeMapping ?? column1.TypeMapping; - if (outerTypeMapping == null) - { - throw new InvalidOperationException( - RelationalStrings.SetOperationsRequireAtLeastOneSideWithValidTypeMapping(setOperationType)); - } - - otherExpressions.Add((outerColumn, outerTypeMapping.KeyComparer)); + // If select1 matched but select2 did not then we erase all identifiers + // TODO: We could make this little more robust by allow the indexes to be different. See issue#24475 + // i.e. Identifier ordering being different. + outerIdentifiers = []; + } + } + // If the top-level projection - not the current nested one - is a complex type and not an entity type, then add + // all its columns to the "otherExpressions" list (i.e. columns not part of a an entity primary key). This is + // the same as with a non-structural type projection. + else if (projection1.StructuralType is IComplexType) + { + var outerTypeMapping = column1.TypeMapping ?? column1.TypeMapping; + if (outerTypeMapping == null) + { + throw new InvalidOperationException( + RelationalStrings.SetOperationsRequireAtLeastOneSideWithValidTypeMapping(setOperationType)); } + + otherExpressions.Add((outerColumn, outerTypeMapping.KeyComparer)); } } + } - foreach (var complexProperty in GetAllComplexPropertiesInHierarchy(type)) - { - var complexPropertyShaper1 = structuralProjection1.BindComplexProperty(complexProperty); + foreach (var complexProperty in GetAllComplexPropertiesInHierarchy(type)) + { + var complexPropertyShaper1 = structuralProjection1.BindComplexProperty(complexProperty); var complexPropertyShaper2 = structuralProjection2.BindComplexProperty(complexProperty); var resultComplexProjection = ProcessStructuralType( (StructuralTypeProjectionExpression)complexPropertyShaper1.ValueBufferExpression, - (StructuralTypeProjectionExpression)complexPropertyShaper2.ValueBufferExpression); + (StructuralTypeProjectionExpression)complexPropertyShaper2.ValueBufferExpression); var resultComplexShaper = new RelationalStructuralTypeShaperExpression( complexProperty.ComplexType, diff --git a/src/EFCore.Relational/Storage/ReaderColumn.cs b/src/EFCore.Relational/Storage/ReaderColumn.cs index e6aa7627825..138aa7cc8d4 100644 --- a/src/EFCore.Relational/Storage/ReaderColumn.cs +++ b/src/EFCore.Relational/Storage/ReaderColumn.cs @@ -29,12 +29,14 @@ public abstract class ReaderColumn /// A value indicating if the column is nullable. /// The name of the column. /// The property being read if any, null otherwise. - protected ReaderColumn(Type type, bool nullable, string? name, IPropertyBase? property) + /// A lambda expression to get field value for the column from the reader. + protected ReaderColumn(Type type, bool nullable, string? name, IPropertyBase? property, LambdaExpression getFieldValueExpression) { Type = type; IsNullable = nullable; Name = name; Property = property; + GetFieldValueExpression = getFieldValueExpression; } /// @@ -57,6 +59,11 @@ protected ReaderColumn(Type type, bool nullable, string? name, IPropertyBase? pr /// public virtual IPropertyBase? Property { get; } + /// + /// A lambda expression to get field value for the column from the reader. + /// + public virtual LambdaExpression GetFieldValueExpression { get; } + /// /// Creates an instance of . /// @@ -73,10 +80,17 @@ public static ReaderColumn Create( bool nullable, string? columnName, IPropertyBase? property, - object readFunc) + LambdaExpression readFunc) => (ReaderColumn)GetConstructor(type).Invoke([nullable, columnName, property, readFunc]); - private static ConstructorInfo GetConstructor(Type type) + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static ConstructorInfo GetConstructor(Type type) => Constructors.GetOrAdd( type, t => typeof(ReaderColumn<>).MakeGenericType(t).GetConstructors().First(ci => ci.GetParameters().Length == 4)); } diff --git a/src/EFCore.Relational/Storage/ReaderColumn`.cs b/src/EFCore.Relational/Storage/ReaderColumn`.cs index 9de7877e94f..b3ad9531ac2 100644 --- a/src/EFCore.Relational/Storage/ReaderColumn`.cs +++ b/src/EFCore.Relational/Storage/ReaderColumn`.cs @@ -24,15 +24,15 @@ public class ReaderColumn : ReaderColumn /// A value indicating if the column is nullable. /// The name of the column. /// The property being read if any, null otherwise. - /// A function to get field value for the column from the reader. + /// A lambda expression to get field value for the column from the reader. public ReaderColumn( bool nullable, string? name, IPropertyBase? property, - Func getFieldValue) - : base(typeof(T), nullable, name, property) + Expression> getFieldValueExpression) + : base(typeof(T), nullable, name, property, getFieldValueExpression) { - GetFieldValue = getFieldValue; + GetFieldValue = getFieldValueExpression.Compile(); } /// diff --git a/src/EFCore.SqlServer.HierarchyId/Storage/Json/SqlServerJsonHierarchyIdReaderWriter.cs b/src/EFCore.SqlServer.HierarchyId/Storage/Json/SqlServerJsonHierarchyIdReaderWriter.cs index 6e6e1bf0b0b..8330ab91f19 100644 --- a/src/EFCore.SqlServer.HierarchyId/Storage/Json/SqlServerJsonHierarchyIdReaderWriter.cs +++ b/src/EFCore.SqlServer.HierarchyId/Storage/Json/SqlServerJsonHierarchyIdReaderWriter.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.CodeAnalysis; using System.Text.Json; using Microsoft.EntityFrameworkCore.Storage.Json; @@ -27,4 +28,10 @@ public override HierarchyId FromJsonTyped(ref Utf8JsonReaderManager manager, obj /// public override void ToJsonTyped(Utf8JsonWriter writer, HierarchyId value) => writer.WriteStringValue(value.ToString()); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore.SqlServer.HierarchyId/Storage/Json/SqlServerJsonSqlHierarchyIdReaderWriter.cs b/src/EFCore.SqlServer.HierarchyId/Storage/Json/SqlServerJsonSqlHierarchyIdReaderWriter.cs index bedbb8bf860..90dd545dd93 100644 --- a/src/EFCore.SqlServer.HierarchyId/Storage/Json/SqlServerJsonSqlHierarchyIdReaderWriter.cs +++ b/src/EFCore.SqlServer.HierarchyId/Storage/Json/SqlServerJsonSqlHierarchyIdReaderWriter.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.CodeAnalysis; using System.Text.Json; using Microsoft.EntityFrameworkCore.Storage.Json; using Microsoft.SqlServer.Types; @@ -28,4 +29,10 @@ public override SqlHierarchyId FromJsonTyped(ref Utf8JsonReaderManager manager, /// public override void ToJsonTyped(Utf8JsonWriter writer, SqlHierarchyId value) => writer.WriteStringValue(value.ToString()); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore.SqlServer.NTS/Storage/Json/SqlServerJsonGeometryWktReaderWriter.cs b/src/EFCore.SqlServer.NTS/Storage/Json/SqlServerJsonGeometryWktReaderWriter.cs index a1ccc63910a..ebcc3f42e23 100644 --- a/src/EFCore.SqlServer.NTS/Storage/Json/SqlServerJsonGeometryWktReaderWriter.cs +++ b/src/EFCore.SqlServer.NTS/Storage/Json/SqlServerJsonGeometryWktReaderWriter.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.CodeAnalysis; using System.Text.Json; using Microsoft.EntityFrameworkCore.Storage.Json; using NetTopologySuite.Geometries; @@ -31,4 +32,10 @@ public override Geometry FromJsonTyped(ref Utf8JsonReaderManager manager, object /// public override void ToJsonTyped(Utf8JsonWriter writer, Geometry value) => writer.WriteStringValue(value.ToText()); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore.SqlServer/Query/Internal/SqlServerQueryCompilationContext.cs b/src/EFCore.SqlServer/Query/Internal/SqlServerQueryCompilationContext.cs index a5555d52ef7..7f1063149e0 100644 --- a/src/EFCore.SqlServer/Query/Internal/SqlServerQueryCompilationContext.cs +++ b/src/EFCore.SqlServer/Query/Internal/SqlServerQueryCompilationContext.cs @@ -51,4 +51,7 @@ public override bool IsBuffering /// doing so can result in application failures when updating to a new Entity Framework Core release. /// public virtual bool InAggregateFunction { get; set; } + + /// + public override bool SupportsPrecompiledQuery => true; } diff --git a/src/EFCore.SqlServer/Query/Internal/SqlServerSqlTranslatingExpressionVisitor.cs b/src/EFCore.SqlServer/Query/Internal/SqlServerSqlTranslatingExpressionVisitor.cs index c3cb7f851b2..438a00a3d86 100644 --- a/src/EFCore.SqlServer/Query/Internal/SqlServerSqlTranslatingExpressionVisitor.cs +++ b/src/EFCore.SqlServer/Query/Internal/SqlServerSqlTranslatingExpressionVisitor.cs @@ -355,7 +355,14 @@ StartsEndsWithContains.StartsWith or StartsEndsWithContains.EndsWith } } - private static string? ConstructLikePatternParameter( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static string? ConstructLikePatternParameter( QueryContext queryContext, string baseParameterName, StartsEndsWithContains methodType) @@ -378,10 +385,37 @@ StartsEndsWithContains.StartsWith or StartsEndsWithContains.EndsWith _ => throw new UnreachableException() }; - private enum StartsEndsWithContains + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public enum StartsEndsWithContains { + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// StartsWith, + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// EndsWith, + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// Contains } diff --git a/src/EFCore.Sqlite.Core/Extensions/SqliteServiceCollectionExtensions.cs b/src/EFCore.Sqlite.Core/Extensions/SqliteServiceCollectionExtensions.cs index 7145fe4c278..c109fcdded7 100644 --- a/src/EFCore.Sqlite.Core/Extensions/SqliteServiceCollectionExtensions.cs +++ b/src/EFCore.Sqlite.Core/Extensions/SqliteServiceCollectionExtensions.cs @@ -105,6 +105,7 @@ public static IServiceCollection AddEntityFrameworkSqlite(this IServiceCollectio .TryAdd() .TryAdd() .TryAdd() + .TryAdd() .TryAdd() .TryAdd() .TryAdd() diff --git a/src/EFCore.Sqlite.Core/Query/Internal/SqliteQueryCompilationContext.cs b/src/EFCore.Sqlite.Core/Query/Internal/SqliteQueryCompilationContext.cs new file mode 100644 index 00000000000..cf9f206c616 --- /dev/null +++ b/src/EFCore.Sqlite.Core/Query/Internal/SqliteQueryCompilationContext.cs @@ -0,0 +1,30 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +namespace Microsoft.EntityFrameworkCore.Sqlite.Query.Internal; + +/// +/// This is an internal API that supports the Entity Framework Core infrastructure and not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +public class SqliteQueryCompilationContext : RelationalQueryCompilationContext +{ + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public SqliteQueryCompilationContext( + QueryCompilationContextDependencies dependencies, + RelationalQueryCompilationContextDependencies relationalDependencies, + bool async) + : base(dependencies, relationalDependencies, async) + { + } + + /// + public override bool SupportsPrecompiledQuery => true; +} diff --git a/src/EFCore.Sqlite.Core/Query/Internal/SqliteQueryCompilationContextFactory.cs b/src/EFCore.Sqlite.Core/Query/Internal/SqliteQueryCompilationContextFactory.cs new file mode 100644 index 00000000000..0570b91fbc3 --- /dev/null +++ b/src/EFCore.Sqlite.Core/Query/Internal/SqliteQueryCompilationContextFactory.cs @@ -0,0 +1,46 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +namespace Microsoft.EntityFrameworkCore.Sqlite.Query.Internal; + +/// +/// This is an internal API that supports the Entity Framework Core infrastructure and not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +public class SqliteQueryCompilationContextFactory : IQueryCompilationContextFactory +{ + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public SqliteQueryCompilationContextFactory( + QueryCompilationContextDependencies dependencies, + RelationalQueryCompilationContextDependencies relationalDependencies) + { + Dependencies = dependencies; + RelationalDependencies = relationalDependencies; + } + + /// + /// Dependencies for this service. + /// + protected virtual QueryCompilationContextDependencies Dependencies { get; } + + /// + /// Relational provider-specific dependencies for this service. + /// + protected virtual RelationalQueryCompilationContextDependencies RelationalDependencies { get; } + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public virtual QueryCompilationContext Create(bool async) + => new SqliteQueryCompilationContext(Dependencies, RelationalDependencies, async); +} diff --git a/src/EFCore.Sqlite.Core/Query/Internal/SqliteQueryableMethodTranslatingExpressionVisitor.cs b/src/EFCore.Sqlite.Core/Query/Internal/SqliteQueryableMethodTranslatingExpressionVisitor.cs index dc9d782e20d..41265f09819 100644 --- a/src/EFCore.Sqlite.Core/Query/Internal/SqliteQueryableMethodTranslatingExpressionVisitor.cs +++ b/src/EFCore.Sqlite.Core/Query/Internal/SqliteQueryableMethodTranslatingExpressionVisitor.cs @@ -2,7 +2,6 @@ // The .NET Foundation licenses this file to you under the MIT license. using Microsoft.Data.Sqlite; -using Microsoft.EntityFrameworkCore.Internal; using Microsoft.EntityFrameworkCore.Query.SqlExpressions; using Microsoft.EntityFrameworkCore.Sqlite.Internal; using Microsoft.EntityFrameworkCore.Sqlite.Query.SqlExpressions.Internal; diff --git a/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonByteArrayReaderWriter.cs b/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonByteArrayReaderWriter.cs index 12376cc11af..ff3b4765253 100644 --- a/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonByteArrayReaderWriter.cs +++ b/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonByteArrayReaderWriter.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.CodeAnalysis; using System.Text.Json; using Microsoft.EntityFrameworkCore.Storage.Json; @@ -47,4 +48,10 @@ public override byte[] FromJsonTyped(ref Utf8JsonReaderManager manager, object? /// public override void ToJsonTyped(Utf8JsonWriter writer, byte[] value) => writer.WriteStringValue(Convert.ToHexString(value)); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonDateTimeOffsetReaderWriter.cs b/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonDateTimeOffsetReaderWriter.cs index d57cd9da2a1..f241a7e4cfb 100644 --- a/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonDateTimeOffsetReaderWriter.cs +++ b/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonDateTimeOffsetReaderWriter.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.CodeAnalysis; using System.Globalization; using System.Text.Encodings.Web; using System.Text.Json; @@ -56,4 +57,10 @@ public override void ToJsonTyped(Utf8JsonWriter writer, DateTimeOffset value) JsonEncodedText.Encode( string.Format(CultureInfo.InvariantCulture, DateTimeOffsetFormatConst, value), JavaScriptEncoder.UnsafeRelaxedJsonEscaping)); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonDateTimeReaderWriter.cs b/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonDateTimeReaderWriter.cs index 29e8025a919..5695e439131 100644 --- a/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonDateTimeReaderWriter.cs +++ b/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonDateTimeReaderWriter.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.CodeAnalysis; using System.Globalization; using System.Text.Json; using Microsoft.EntityFrameworkCore.Storage.Json; @@ -50,4 +51,10 @@ public override DateTime FromJsonTyped(ref Utf8JsonReaderManager manager, object /// public override void ToJsonTyped(Utf8JsonWriter writer, DateTime value) => writer.WriteStringValue(string.Format(CultureInfo.InvariantCulture, DateTimeFormatConst, value)); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonDecimalReaderWriter.cs b/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonDecimalReaderWriter.cs index d9cc869b838..48b2c12456f 100644 --- a/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonDecimalReaderWriter.cs +++ b/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonDecimalReaderWriter.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.CodeAnalysis; using System.Globalization; using System.Text.Json; using Microsoft.EntityFrameworkCore.Storage.Json; @@ -50,4 +51,10 @@ public override decimal FromJsonTyped(ref Utf8JsonReaderManager manager, object? /// public override void ToJsonTyped(Utf8JsonWriter writer, decimal value) => writer.WriteStringValue(string.Format(CultureInfo.InvariantCulture, DecimalFormatConst, value)); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonGuidReaderWriter.cs b/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonGuidReaderWriter.cs index 0b90ce03d1a..3267a1025a3 100644 --- a/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonGuidReaderWriter.cs +++ b/src/EFCore.Sqlite.Core/Storage/Json/Internal/SqliteJsonGuidReaderWriter.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.CodeAnalysis; using System.Text.Json; using Microsoft.EntityFrameworkCore.Storage.Json; @@ -47,4 +48,10 @@ public override Guid FromJsonTyped(ref Utf8JsonReaderManager manager, object? ex /// public override void ToJsonTyped(Utf8JsonWriter writer, Guid value) => writer.WriteStringValue(value.ToString().ToUpperInvariant()); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore.Sqlite.NTS/Storage/Json/SqliteJsonGeometryWktReaderWriter.cs b/src/EFCore.Sqlite.NTS/Storage/Json/SqliteJsonGeometryWktReaderWriter.cs index c6e784e402d..f04538c7bfe 100644 --- a/src/EFCore.Sqlite.NTS/Storage/Json/SqliteJsonGeometryWktReaderWriter.cs +++ b/src/EFCore.Sqlite.NTS/Storage/Json/SqliteJsonGeometryWktReaderWriter.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.CodeAnalysis; using System.Text.Json; using Microsoft.EntityFrameworkCore.Storage.Json; using NetTopologySuite.Geometries; @@ -31,4 +32,10 @@ public override Geometry FromJsonTyped(ref Utf8JsonReaderManager manager, object /// public override void ToJsonTyped(Utf8JsonWriter writer, Geometry value) => writer.WriteStringValue(value.ToText()); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/ChangeTracking/ListOfNullableValueTypesComparer.cs b/src/EFCore/ChangeTracking/ListOfNullableValueTypesComparer.cs index ba34d114ee6..6f57433853d 100644 --- a/src/EFCore/ChangeTracking/ListOfNullableValueTypesComparer.cs +++ b/src/EFCore/ChangeTracking/ListOfNullableValueTypesComparer.cs @@ -47,7 +47,14 @@ public ListOfNullableValueTypesComparer(ValueComparer elementComparer) /// public ValueComparer ElementComparer { get; } - private static bool Compare(IEnumerable? a, IEnumerable? b, ValueComparer elementComparer) + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static bool Compare(IEnumerable? a, IEnumerable? b, ValueComparer elementComparer) { if (ReferenceEquals(a, b)) { diff --git a/src/EFCore/ChangeTracking/ListOfValueTypesComparer.cs b/src/EFCore/ChangeTracking/ListOfValueTypesComparer.cs index 8ec353b7982..10e327beaa2 100644 --- a/src/EFCore/ChangeTracking/ListOfValueTypesComparer.cs +++ b/src/EFCore/ChangeTracking/ListOfValueTypesComparer.cs @@ -47,7 +47,14 @@ public ListOfValueTypesComparer(ValueComparer elementComparer) /// public ValueComparer ElementComparer { get; } - private static bool Compare(IEnumerable? a, IEnumerable? b, ValueComparer elementComparer) + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static bool Compare(IEnumerable? a, IEnumerable? b, ValueComparer elementComparer) { if (ReferenceEquals(a, b)) { diff --git a/src/EFCore/ChangeTracking/ValueComparer.cs b/src/EFCore/ChangeTracking/ValueComparer.cs index 2b5e80fd71f..1835113d36d 100644 --- a/src/EFCore/ChangeTracking/ValueComparer.cs +++ b/src/EFCore/ChangeTracking/ValueComparer.cs @@ -155,6 +155,11 @@ protected ValueComparer( /// public virtual LambdaExpression EqualsExpression { get; } + /// + /// The object comparison expression. + /// + public abstract LambdaExpression ObjectEqualsExpression { get; } + /// /// The hash code expression. /// diff --git a/src/EFCore/ChangeTracking/ValueComparer`.cs b/src/EFCore/ChangeTracking/ValueComparer`.cs index 84f7632f435..dea3d033566 100644 --- a/src/EFCore/ChangeTracking/ValueComparer`.cs +++ b/src/EFCore/ChangeTracking/ValueComparer`.cs @@ -5,6 +5,7 @@ using System.Diagnostics.CodeAnalysis; using Microsoft.EntityFrameworkCore.Internal; using ExpressionExtensions = Microsoft.EntityFrameworkCore.Infrastructure.ExpressionExtensions; +using static System.Linq.Expressions.Expression; namespace Microsoft.EntityFrameworkCore.ChangeTracking; @@ -38,6 +39,9 @@ public class ValueComparer private Func? _equals; private Func? _hashCode; private Func? _snapshot; + private LambdaExpression? _objectEqualsExpression; + private static readonly PropertyInfo StructuralComparisonsStructuralEqualityComparerProperty = + typeof(StructuralComparisons).GetProperty(nameof(StructuralComparisons.StructuralEqualityComparer))!; /// /// Creates a new with a default comparison @@ -96,19 +100,19 @@ public ValueComparer( protected static Expression> CreateDefaultEqualsExpression() { var type = typeof(T); - var param1 = Expression.Parameter(type, "v1"); - var param2 = Expression.Parameter(type, "v2"); + var param1 = Parameter(type, "v1"); + var param2 = Parameter(type, "v2"); // We exclude multi-dimensional arrays even though they're IStructuralEquatable because of // https://github.com/dotnet/runtime/issues/66472 if (typeof(IStructuralEquatable).IsAssignableFrom(type)) { - return Expression.Lambda>( - Expression.Call( - Expression.Constant(StructuralComparisons.StructuralEqualityComparer, typeof(IEqualityComparer)), + return Lambda>( + Call( + Property(null, StructuralComparisonsStructuralEqualityComparerProperty), EqualityComparerEqualsMethod, - Expression.Convert(param1, typeof(object)), - Expression.Convert(param2, typeof(object)) + Convert(param1, typeof(object)), + Convert(param2, typeof(object)) ), param1, param2); } @@ -121,8 +125,8 @@ public ValueComparer( || unwrappedType == typeof(decimal) || unwrappedType == typeof(object)) { - return Expression.Lambda>( - Expression.Equal(param1, param2), + return Lambda>( + Equal(param1, param2), param1, param2); } @@ -135,18 +139,18 @@ public ValueComparer( if (typedEquals != null) { - return Expression.Lambda>( + return Lambda>( type.IsNullableType() - ? Expression.OrElse( - Expression.AndAlso( - Expression.Equal(param1, Expression.Constant(null, type)), - Expression.Equal(param2, Expression.Constant(null, type))), - Expression.AndAlso( - Expression.AndAlso( - Expression.NotEqual(param1, Expression.Constant(null, type)), - Expression.NotEqual(param2, Expression.Constant(null, type))), - Expression.Call(param1, typedEquals, param2))) - : Expression.Call(param1, typedEquals, param2), + ? OrElse( + AndAlso( + Equal(param1, Constant(null, type)), + Equal(param2, Constant(null, type))), + AndAlso( + AndAlso( + NotEqual(param1, Constant(null, type)), + NotEqual(param2, Constant(null, type))), + Call(param1, typedEquals, param2))) + : Call(param1, typedEquals, param2), param1, param2); } @@ -165,10 +169,10 @@ public ValueComparer( type = type.BaseType; } - return Expression.Lambda>( + return Lambda>( typedEquals == null ? ExpressionExtensions.CreateEqualsExpression(param1, param2) - : Expression.Call(typedEquals, param1, param2), + : Call(typedEquals, param1, param2), param1, param2); } @@ -185,9 +189,9 @@ protected static Expression> CreateDefaultSnapshotExpression(bool fav return v => v; } - var sourceParameter = Expression.Parameter(typeof(T), "source"); - return Expression.Lambda>( - Expression.Call( + var sourceParameter = Parameter(typeof(T), "source"); + return Lambda>( + Call( EnumerableMethods.ToArray.MakeGenericMethod(typeof(T).GetElementType()!), sourceParameter), sourceParameter); @@ -204,16 +208,16 @@ protected static Expression> CreateDefaultHashCodeExpression(bool f { var type = typeof(T); var unwrappedType = type.UnwrapNullableType(); - var param = Expression.Parameter(type, "v"); + var param = Parameter(type, "v"); if (favorStructuralComparisons && typeof(IStructuralEquatable).IsAssignableFrom(type)) { - return Expression.Lambda>( - Expression.Call( - Expression.Constant(StructuralComparisons.StructuralEqualityComparer, typeof(IEqualityComparer)), + return Lambda>( + Call( + Property(null, StructuralComparisonsStructuralEqualityComparerProperty), EqualityComparerHashCodeMethod, - Expression.Convert(param, typeof(object)) + Convert(param, typeof(object)) ), param); } @@ -228,10 +232,10 @@ var expression || unwrappedType == typeof(ushort) || unwrappedType == typeof(sbyte) || unwrappedType == typeof(char) - ? (Expression)Expression.Convert(param, typeof(int)) - : Expression.Call(param, ObjectGetHashCodeMethod); + ? (Expression)Convert(param, typeof(int)) + : Call(param, ObjectGetHashCodeMethod); - return Expression.Lambda>(expression, param); + return Lambda>(expression, param); } /// @@ -248,6 +252,34 @@ public override bool Equals(object? left, object? right) return v1Null || v2Null ? v1Null && v2Null : Equals((T?)left, (T?)right); } + /// + public override LambdaExpression ObjectEqualsExpression + { + get + { + if (_objectEqualsExpression == null) + { + var left = Parameter(typeof(object), "left"); + var right = Parameter(typeof(object), "right"); + + _objectEqualsExpression = Lambda>( + Condition( + Equal(left, Constant(null)), + Equal(right, Constant(null)), + AndAlso( + NotEqual(right, Constant(null)), + Invoke( + EqualsExpression, + Convert(left, typeof(T)), + Convert(right, typeof(T))))), + left, + right); + } + + return _objectEqualsExpression; + } + } + /// /// Returns the hash code for the given instance. /// diff --git a/src/EFCore/EFCore.csproj b/src/EFCore/EFCore.csproj index 77b927ca73d..575c6c05276 100644 --- a/src/EFCore/EFCore.csproj +++ b/src/EFCore/EFCore.csproj @@ -13,6 +13,7 @@ Microsoft.EntityFrameworkCore.DbSet Microsoft.EntityFrameworkCore true true + $(NoWarn);EF9100 diff --git a/src/EFCore/Infrastructure/EntityFrameworkServicesBuilder.cs b/src/EFCore/Infrastructure/EntityFrameworkServicesBuilder.cs index 60b90236db4..c76d4aa9c84 100644 --- a/src/EFCore/Infrastructure/EntityFrameworkServicesBuilder.cs +++ b/src/EFCore/Infrastructure/EntityFrameworkServicesBuilder.cs @@ -83,6 +83,7 @@ public static readonly IDictionary CoreServices { typeof(IMemoryCache), new ServiceCharacteristics(ServiceLifetime.Singleton) }, { typeof(IEvaluatableExpressionFilter), new ServiceCharacteristics(ServiceLifetime.Singleton) }, { typeof(INavigationExpansionExtensibilityHelper), new ServiceCharacteristics(ServiceLifetime.Singleton) }, + { typeof(ILiftableConstantFactory), new ServiceCharacteristics(ServiceLifetime.Singleton) }, { typeof(IExceptionDetector), new ServiceCharacteristics(ServiceLifetime.Singleton) }, { typeof(IJsonValueReaderWriterSource), new ServiceCharacteristics(ServiceLifetime.Singleton) }, { typeof(IProviderConventionSetBuilder), new ServiceCharacteristics(ServiceLifetime.Scoped) }, @@ -125,6 +126,7 @@ public static readonly IDictionary CoreServices { typeof(IShapedQueryCompilingExpressionVisitorFactory), new ServiceCharacteristics(ServiceLifetime.Scoped) }, { typeof(IDbContextLogger), new ServiceCharacteristics(ServiceLifetime.Scoped) }, { typeof(IAdHocMapper), new ServiceCharacteristics(ServiceLifetime.Scoped) }, + { typeof(ILiftableConstantProcessor), new ServiceCharacteristics(ServiceLifetime.Scoped) }, { typeof(ILazyLoader), new ServiceCharacteristics(ServiceLifetime.Transient) }, { typeof(ILazyLoaderFactory), new ServiceCharacteristics(ServiceLifetime.Scoped) }, { typeof(IParameterBindingFactory), new ServiceCharacteristics(ServiceLifetime.Singleton, multipleRegistrations: true) }, @@ -309,6 +311,8 @@ public virtual EntityFrameworkServicesBuilder TryAddCoreServices() TryAdd(); TryAdd(); TryAdd(); + TryAdd(); + TryAdd(); TryAdd( p => p.GetService()?.FindExtension()?.DbContextLogger @@ -329,12 +333,12 @@ public virtual EntityFrameworkServicesBuilder TryAddCoreServices() .AddDependencySingleton() .AddDependencySingleton() .AddDependencySingleton() - .AddDependencySingleton() .AddDependencySingleton() .AddDependencySingleton() .AddDependencySingleton() .AddDependencySingleton() .AddDependencySingleton() + .AddDependencySingleton() .AddDependencyScoped() .AddDependencyScoped() .AddDependencyScoped() @@ -344,6 +348,7 @@ public virtual EntityFrameworkServicesBuilder TryAddCoreServices() .AddDependencyScoped() .AddDependencyScoped() .AddDependencyScoped() + .AddDependencyScoped() .AddDependencyScoped() .AddDependencyScoped() .AddDependencyScoped() diff --git a/src/EFCore/Query/ILiftableConstantFactory.cs b/src/EFCore/Query/ILiftableConstantFactory.cs new file mode 100644 index 00000000000..a342e05d572 --- /dev/null +++ b/src/EFCore/Query/ILiftableConstantFactory.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. + +using System.Diagnostics.CodeAnalysis; + +namespace Microsoft.EntityFrameworkCore.Query; + +/// +/// This is an experimental API used by the Entity Framework Core feature and it is not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +[Experimental(EFDiagnostics.PrecompiledQueryExperimental)] +public interface ILiftableConstantFactory +{ + /// + /// This is an experimental API used by the Entity Framework Core feature and it is not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + LiftableConstantExpression CreateLiftableConstant( + object? originalValue, + Expression> resolverExpression, + string variableName, + Type type); +} diff --git a/src/EFCore/Query/ILiftableConstantProcessor.cs b/src/EFCore/Query/ILiftableConstantProcessor.cs new file mode 100644 index 00000000000..13fee27c7a2 --- /dev/null +++ b/src/EFCore/Query/ILiftableConstantProcessor.cs @@ -0,0 +1,55 @@ +// 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.CodeAnalysis; + +namespace Microsoft.EntityFrameworkCore.Query; + +/// +/// This is an experimental API used by the Entity Framework Core feature and it is not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +[Experimental(EFDiagnostics.PrecompiledQueryExperimental)] +public interface ILiftableConstantProcessor +{ + /// + /// Exposes all constants that have been lifted during the last invocation of . + /// + IReadOnlyList<(ParameterExpression Parameter, Expression Expression)> LiftedConstants { get; } + + /// + /// Inlines all liftable constants as simple nodes in the tree, containing the result of + /// evaluating the liftable constants' resolvers. + /// + /// An expression containing nodes. + /// A value indicating whether precompiled queries are supported by the provider. + /// + /// An expression tree containing nodes instead of nodes. + /// + /// + /// Liftable constant inlining is performed in the regular, non-precompiled query pipeline flow. + /// + Expression InlineConstants(Expression expression, bool supportsPrecompiledQuery); + + /// + /// Lifts all nodes, embedding in their place and + /// exposing the parameter and resolver via . + /// + /// An expression containing nodes. + /// + /// The to be embedded in the liftable constant nodes' resolvers, instead of their lambda + /// parameter. + /// + /// + /// A set of variables already in use, for uniquification. Any generates variables will be added to this set. + /// + /// + /// An expression tree containing nodes instead of nodes. + /// + /// + /// Constant lifting is performed in the precompiled query pipeline flow. + /// + Expression LiftConstants(Expression expression, ParameterExpression contextParameter, HashSet variableNames); +} diff --git a/src/EFCore/Query/Internal/EntityMaterializerSource.cs b/src/EFCore/Query/Internal/EntityMaterializerSource.cs index 3cdeb4fab30..7c90bb0447f 100644 --- a/src/EFCore/Query/Internal/EntityMaterializerSource.cs +++ b/src/EFCore/Query/Internal/EntityMaterializerSource.cs @@ -18,6 +18,12 @@ public class EntityMaterializerSource : IEntityMaterializerSource private static readonly MethodInfo InjectableServiceInjectedMethod = typeof(IInjectableService).GetMethod(nameof(IInjectableService.Injected))!; + private static readonly ConstructorInfo EntityMaterializerSourceParametersCtor + = typeof(EntityMaterializerSourceParameters).GetConstructor([typeof(ITypeBase), typeof(string), typeof(QueryTrackingBehavior?)])!; + + private static readonly ConstructorInfo ParameterBindingInfoCtor + = typeof(ParameterBindingInfo).GetConstructor([typeof(EntityMaterializerSourceParameters), typeof(Expression)])!; + private ConcurrentDictionary>? _materializers; private ConcurrentDictionary>? _emptyMaterializers; private readonly List _bindingInterceptors; @@ -87,8 +93,6 @@ public Expression CreateMaterializeExpression( var constructorBinding = ModifyBindings(structuralType, structuralType.ConstructorBinding!); var bindingInfo = new ParameterBindingInfo(parameters, materializationContextExpression); - var blockExpressions = new List(); - var instanceVariable = Expression.Variable(constructorBinding.RuntimeType, entityInstanceName); bindingInfo.ServiceInstances.Add(instanceVariable); @@ -96,6 +100,7 @@ public Expression CreateMaterializeExpression( structuralType.GetProperties().Cast().Where(p => !p.IsShadowProperty()) .Concat(structuralType.GetComplexProperties().Where(p => !p.IsShadowProperty()))); + var blockExpressions = new List(); if (structuralType is IEntityType entityType) { var serviceProperties = entityType.GetServiceProperties().ToList(); diff --git a/src/EFCore/Query/Internal/NavigationExpandingExpressionVisitor.ExpressionVisitors.cs b/src/EFCore/Query/Internal/NavigationExpandingExpressionVisitor.ExpressionVisitors.cs index 26388a01c89..436179ceec8 100644 --- a/src/EFCore/Query/Internal/NavigationExpandingExpressionVisitor.ExpressionVisitors.cs +++ b/src/EFCore/Query/Internal/NavigationExpandingExpressionVisitor.ExpressionVisitors.cs @@ -501,7 +501,7 @@ outerKey is NewArrayExpression newArrayExpression private sealed class IncludeExpandingExpressionVisitor : ExpandingExpressionVisitor { private static readonly MethodInfo FetchJoinEntityMethodInfo = - typeof(IncludeExpandingExpressionVisitor).GetTypeInfo().GetDeclaredMethod(nameof(FetchJoinEntity))!; + typeof(NavigationExpandingExpressionVisitor).GetTypeInfo().GetDeclaredMethod(nameof(FetchJoinEntity))!; private readonly bool _queryStateManager; private readonly bool _ignoreAutoIncludes; @@ -892,11 +892,6 @@ private Expression ExpandIncludesHelper(Expression root, EntityReference entityR return result; } -#pragma warning disable IDE0060 // Remove unused parameter - private static TTarget FetchJoinEntity(TJoin joinEntity, TTarget targetEntity) - => targetEntity; -#pragma warning restore IDE0060 // Remove unused parameter - private static Expression RemapFilterExpressionForJoinEntity( ParameterExpression filterParameter, Expression filterExpressionBody, @@ -1383,4 +1378,14 @@ public override ExpressionType NodeType public IEntityType? EntityType { get; } } } + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static TTarget FetchJoinEntity(TJoin joinEntity, TTarget targetEntity) + => targetEntity; } diff --git a/src/EFCore/Query/LiftableConstantExpression.cs b/src/EFCore/Query/LiftableConstantExpression.cs new file mode 100644 index 00000000000..c67f1066763 --- /dev/null +++ b/src/EFCore/Query/LiftableConstantExpression.cs @@ -0,0 +1,104 @@ +// 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.CodeAnalysis; + +namespace Microsoft.EntityFrameworkCore.Query; + +/// +/// A node containing an expression expressing how to obtain a constant value, which may get lifted out of an expression tree. +/// +/// +/// +/// When the expression tree is compiled, the constant value can simply be evaluated beforehand, and a +/// expression can directly reference the result. +/// +/// +/// When the expression tree is translated to source code instead (in query pre-compilation), the expression can be rendered out +/// separately, to be assigned to a variable, and this node is replaced by a reference to that variable. +/// +/// +[DebuggerDisplay("{Microsoft.EntityFrameworkCore.Query.ExpressionPrinter.Print(this), nq}")] +[Experimental(EFDiagnostics.PrecompiledQueryExperimental)] +public class LiftableConstantExpression : Expression, IPrintableExpression +{ + /// + /// This is an experimental API used by the Entity Framework Core feature and it is not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public LiftableConstantExpression( + object? originalValue, + LambdaExpression resolverExpression, + string variableName, + Type type) + { + OriginalExpression = Constant(originalValue, type); + ResolverExpression = resolverExpression; + VariableName = char.ToLower(variableName[0]) + variableName[1..]; + Type = type; + } + + /// + /// This is an experimental API used by the Entity Framework Core feature and it is not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public virtual ConstantExpression OriginalExpression { get; } + + /// + /// This is an experimental API used by the Entity Framework Core feature and it is not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public virtual LambdaExpression ResolverExpression { get; } + + /// + /// This is an experimental API used by the Entity Framework Core feature and it is not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public virtual string VariableName { get; } + + /// + public override Type Type { get; } + + /// + public override ExpressionType NodeType + => ExpressionType.Extension; + + // TODO: Complete other expression stuff (equality, etc.) + + /// + protected override Expression VisitChildren(ExpressionVisitor visitor) + { + var resolverExpression = (LambdaExpression)visitor.Visit(ResolverExpression); + + return Update(resolverExpression); + } + + /// + /// Creates a new expression that is like this one, but using the supplied children. If all of the children are the same, it will + /// return this expression. + /// + /// The property of the result. + /// This expression if no children changed, or an expression with the updated children. + public virtual LiftableConstantExpression Update(LambdaExpression resolverExpression) + => resolverExpression != ResolverExpression + ? new LiftableConstantExpression(OriginalExpression, resolverExpression, VariableName, Type) + : this; + + /// + public void Print(ExpressionPrinter expressionPrinter) + { + expressionPrinter.Append("[LIFTABLE Constant: "); + expressionPrinter.Visit(OriginalExpression); + expressionPrinter.Append(" | Resolver: "); + expressionPrinter.Visit(ResolverExpression); + expressionPrinter.Append("]"); + } +} diff --git a/src/EFCore/Query/LiftableConstantExpressionDependencies.cs b/src/EFCore/Query/LiftableConstantExpressionDependencies.cs new file mode 100644 index 00000000000..d21d5f337f9 --- /dev/null +++ b/src/EFCore/Query/LiftableConstantExpressionDependencies.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.Diagnostics.CodeAnalysis; + +namespace Microsoft.EntityFrameworkCore.Query; + +/// +/// +/// Service dependencies parameter class for +/// +/// +/// This type is typically used by database providers (and other extensions). It is generally +/// not used in application code. +/// +/// +/// +/// +/// Do not construct instances of this class directly from either provider or application code as the +/// constructor signature may change as new dependencies are added. Instead, use this type in +/// your constructor so that an instance will be created and injected automatically by the +/// dependency injection container. To create an instance with some dependent services replaced, +/// first resolve the object from the dependency injection container, then replace selected +/// services using the C# 'with' operator. Do not call the constructor at any point in this process. +/// +/// +/// The service lifetime is . This means a single instance +/// is used by many instances. The implementation must be thread-safe. +/// This service cannot depend on services registered as . +/// +/// +[Experimental(EFDiagnostics.PrecompiledQueryExperimental)] +public sealed record LiftableConstantExpressionDependencies +{ +} diff --git a/src/EFCore/Query/LiftableConstantExpressionHelpers.cs b/src/EFCore/Query/LiftableConstantExpressionHelpers.cs new file mode 100644 index 00000000000..130a9a7f31d --- /dev/null +++ b/src/EFCore/Query/LiftableConstantExpressionHelpers.cs @@ -0,0 +1,292 @@ +// 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.Diagnostics.CodeAnalysis; +using System.Runtime.CompilerServices; +using Microsoft.EntityFrameworkCore.Internal; +using static System.Linq.Expressions.Expression; + +namespace Microsoft.EntityFrameworkCore.Query; + +/// +/// This is an internal API that supports the Entity Framework Core infrastructure and not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +[Experimental(EFDiagnostics.PrecompiledQueryExperimental)] +public class LiftableConstantExpressionHelpers +{ + private static readonly MethodInfo ModelFindEntiyTypeMethod = + typeof(IModel).GetRuntimeMethod(nameof(IModel.FindEntityType), [typeof(string)])!; + + private static readonly MethodInfo RuntimeModelFindAdHocEntiyTypeMethod = + typeof(RuntimeModel).GetRuntimeMethod(nameof(RuntimeModel.FindAdHocEntityType), [typeof(Type)])!; + + private static readonly MethodInfo TypeBaseFindComplexPropertyMethod = + typeof(ITypeBase).GetRuntimeMethod(nameof(ITypeBase.FindComplexProperty), [typeof(string)])!; + + private static readonly MethodInfo TypeBaseFindPropertyMethod = + typeof(ITypeBase).GetRuntimeMethod(nameof(ITypeBase.FindProperty), [typeof(string)])!; + + private static readonly MethodInfo TypeBaseFindServicePropertyMethod = + typeof(IEntityType).GetRuntimeMethod(nameof(IEntityType.FindServiceProperty), [typeof(string)])!; + + private static readonly MethodInfo EntityTypeFindNavigationMethod = + typeof(IEntityType).GetRuntimeMethod(nameof(IEntityType.FindNavigation), [typeof(string)])!; + + private static readonly MethodInfo EntityTypeFindSkipNavigationMethod = + typeof(IEntityType).GetRuntimeMethod(nameof(IEntityType.FindSkipNavigation), [typeof(string)])!; + + private static readonly MethodInfo NavigationBaseClrCollectionAccessorMethod = + typeof(INavigationBase).GetRuntimeMethod(nameof(INavigationBase.GetCollectionAccessor), [])!; + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public static bool IsLiteral(object? value) + { + return value switch + { + int or long or uint or ulong or short or sbyte or ushort or byte or double or float or decimal or string or char or bool => true, + null or Type or Enum => true, + TimeSpan or DateTime or DateTimeOffset or DateOnly or TimeOnly or Guid => true, + ITuple tuple + when tuple.GetType() is { IsGenericType: true } tupleType + && tupleType.Name.StartsWith("ValueTuple`", StringComparison.Ordinal) + && tupleType.Namespace == "System" + => IsTupleLiteral(tuple), + + Array array => IsCollectionOfLiterals(array), + IList list => IsCollectionOfLiterals(list), + + _ => false + }; + + bool IsTupleLiteral(ITuple tuple) + { + for (var i = 0; i < tuple.Length; i++) + { + if (!IsLiteral(tuple[i])) + { + return false; + } + } + + return true; + } + + bool IsCollectionOfLiterals(IEnumerable enumerable) + { + foreach (var enumerableElement in enumerable) + { + if (!IsLiteral(enumerableElement)) + { + return false; + } + } + + return true; + } + } + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public static Expression BuildMemberAccessForEntityOrComplexType(ITypeBase targetType, ParameterExpression liftableConstantContextParameter) + { + var (rootEntityType, complexTypes) = FindPathForEntityOrComplexType(targetType); + + Expression result; + + if (rootEntityType.IsAdHoc()) + { + result = Call( + Convert( + Property( + Property( + liftableConstantContextParameter, + nameof(MaterializerLiftableConstantContext.Dependencies)), + nameof(ShapedQueryCompilingExpressionVisitorDependencies.Model)), + typeof(RuntimeModel)), + RuntimeModelFindAdHocEntiyTypeMethod, + Constant(rootEntityType.ClrType)); + + } + else + { + result = Call( + Property( + Property( + liftableConstantContextParameter, + nameof(MaterializerLiftableConstantContext.Dependencies)), + nameof(ShapedQueryCompilingExpressionVisitorDependencies.Model)), + ModelFindEntiyTypeMethod, + Constant(rootEntityType.Name)); + } + + foreach (var complexType in complexTypes) + { + var complexPropertyName = complexType.ComplexProperty.Name; + result = Property( + Call(result, TypeBaseFindComplexPropertyMethod, Constant(complexPropertyName)), + nameof(IComplexProperty.ComplexType)); + } + + return result; + } + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public static Expression> BuildMemberAccessLambdaForEntityOrComplexType(ITypeBase type) + { + var prm = Parameter(typeof(MaterializerLiftableConstantContext)); + var body = BuildMemberAccessForEntityOrComplexType(type, prm); + + return Lambda>(body, prm); + } + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public static Expression BuildMemberAccessForProperty(IPropertyBase? property, ParameterExpression liftableConstantContextParameter) + { + if (property == null) + { + return Default(typeof(IPropertyBase)); + } + + var declaringType = property.DeclaringType; + var declaringTypeMemberAccessExpression = BuildMemberAccessForEntityOrComplexType(declaringType, liftableConstantContextParameter); + + return Call( + declaringTypeMemberAccessExpression, + property is IServiceProperty ? TypeBaseFindServicePropertyMethod : TypeBaseFindPropertyMethod, + Constant(property.Name)); + } + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public static Expression> BuildMemberAccessLambdaForProperty(IPropertyBase? property) + { + var prm = Parameter(typeof(MaterializerLiftableConstantContext)); + var body = BuildMemberAccessForProperty(property, prm); + + return Lambda>(body, prm); + } + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public static Expression BuildNavigationAccess(INavigationBase? navigation, ParameterExpression liftableConstantContextParameter) + { + if (navigation == null) + { + return Default(typeof(INavigationBase)); + } + + var declaringType = navigation.DeclaringType; + var declaringTypeExpression = BuildMemberAccessForEntityOrComplexType(declaringType, liftableConstantContextParameter); + + var result = Call( + declaringTypeExpression, + navigation is ISkipNavigation ? EntityTypeFindSkipNavigationMethod : EntityTypeFindNavigationMethod, + Constant(navigation.Name)); + + return result; + } + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public static Expression> BuildNavigationAccessLambda(INavigationBase? navigation) + { + var prm = Parameter(typeof(MaterializerLiftableConstantContext)); + var body = BuildNavigationAccess(navigation, prm); + + return Lambda>(body, prm); + } + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public static Expression BuildClrCollectionAccessor(INavigationBase? navigation, ParameterExpression liftableConstantContextParameter) + { + if (navigation == null) + { + return Default(typeof(IClrCollectionAccessor)); + } + + var navigationAccessExpression = BuildNavigationAccess(navigation, liftableConstantContextParameter); + var result = Call(navigationAccessExpression, NavigationBaseClrCollectionAccessorMethod); + + return result; + } + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public static Expression> BuildClrCollectionAccessorLambda(INavigationBase? navigation) + { + var prm = Parameter(typeof(MaterializerLiftableConstantContext)); + var body = BuildClrCollectionAccessor(navigation, prm); + + return Lambda>(body, prm); + } + + private static (IEntityType RootEntity, List ComplexTypes) FindPathForEntityOrComplexType(ITypeBase targetType) + { + if (targetType is IEntityType targetEntity) + { + return (targetEntity, []); + } + + var targetComplexType = (IComplexType)targetType; + var declaringType = targetComplexType.ComplexProperty.DeclaringType; + if (declaringType is IEntityType declaringEntityType) + { + return (declaringEntityType, [targetComplexType]); + } + + var complexTypes = new List(); + while (declaringType is IComplexType complexType) + { + complexTypes.Insert(0, complexType); + declaringType = complexType.ComplexProperty.DeclaringType; + } + + complexTypes.Add(targetComplexType); + + return ((IEntityType)declaringType, complexTypes); + } +} diff --git a/src/EFCore/Query/LiftableConstantFactory.cs b/src/EFCore/Query/LiftableConstantFactory.cs new file mode 100644 index 00000000000..3a379d9e0d4 --- /dev/null +++ b/src/EFCore/Query/LiftableConstantFactory.cs @@ -0,0 +1,37 @@ +// 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.CodeAnalysis; + +namespace Microsoft.EntityFrameworkCore.Query; + +/// +/// This is an experimental API used by the Entity Framework Core feature and it is not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +[Experimental(EFDiagnostics.PrecompiledQueryExperimental)] +public class LiftableConstantFactory(LiftableConstantExpressionDependencies dependencies) : ILiftableConstantFactory +{ + /// + /// This is an experimental API used by the Entity Framework Core feature and it is not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public virtual LiftableConstantExpressionDependencies Dependencies { get; } = dependencies; + + /// + /// This is an experimental API used by the Entity Framework Core feature and it is not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public virtual LiftableConstantExpression CreateLiftableConstant( + object? originalValue, + Expression> resolverExpression, + string variableName, + Type type) + => new(originalValue, resolverExpression, variableName, type); +} diff --git a/src/EFCore/Query/LiftableConstantProcessor.cs b/src/EFCore/Query/LiftableConstantProcessor.cs new file mode 100644 index 00000000000..2296037b66b --- /dev/null +++ b/src/EFCore/Query/LiftableConstantProcessor.cs @@ -0,0 +1,486 @@ +// 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.CodeAnalysis; + +namespace Microsoft.EntityFrameworkCore.Query; + +#pragma warning disable CS1591 + +/// +/// This is an experimental API used by the Entity Framework Core feature and it is not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +[Experimental(EFDiagnostics.PrecompiledQueryExperimental)] +public class LiftableConstantProcessor : ExpressionVisitor, ILiftableConstantProcessor +{ + private bool _inline; + private bool _precompiledQueriesSupported; + private readonly UnsupportedConstantChecker _unsupportedConstantChecker; + private readonly MaterializerLiftableConstantContext _materializerLiftableConstantContext; + + private sealed record LiftedConstant(ParameterExpression Parameter, Expression Expression, ParameterExpression? ReplacingParameter = null); + + private readonly List _liftedConstants = new(); + private readonly LiftedExpressionProcessor _liftedExpressionProcessor = new(); + private readonly LiftedConstantOptimizer _liftedConstantOptimizer = new(); + private ParameterExpression? _contextParameter; + + /// + /// Exposes all constants that have been lifted during the last invocation of . + /// + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + public virtual IReadOnlyList<(ParameterExpression Parameter, Expression Expression)> LiftedConstants { get; private set; } + = Array.Empty<(ParameterExpression Parameter, Expression Expression)>(); + + public LiftableConstantProcessor(ShapedQueryCompilingExpressionVisitorDependencies dependencies) + { + _materializerLiftableConstantContext = new(dependencies); + _unsupportedConstantChecker = new(this); + _liftedConstants.Clear(); + } + + /// + /// Inlines all liftable constants as simple nodes in the tree, containing the result of + /// evaluating the liftable constants' resolvers. + /// + /// An expression containing nodes. + /// A value indicating whether the provider supports precompiled queries. + /// + /// An expression tree containing nodes instead of nodes. + /// + /// + /// + /// Liftable constant inlining is performed in the regular, non-precompiled query pipeline flow. + /// + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + /// + public virtual Expression InlineConstants(Expression expression, bool supportsPrecompiledQuery) + { + _liftedConstants.Clear(); + _inline = true; + _precompiledQueriesSupported = supportsPrecompiledQuery; + + return Visit(expression); + } + + /// + /// Lifts all nodes, embedding in their place and + /// exposing the parameter and resolver via . + /// + /// An expression containing nodes. + /// + /// The to be embedded in the lifted constant nodes' resolvers, instead of their lambda + /// parameter. + /// + /// + /// A set of variables already in use, for uniquification. Any generates variables will be added to this set. + /// + /// + /// An expression tree containing nodes instead of nodes. + /// + public virtual Expression LiftConstants(Expression expression, ParameterExpression contextParameter, HashSet variableNames) + { + _liftedConstants.Clear(); + + _inline = false; + _contextParameter = contextParameter; + + var expressionAfterLifting = Visit(expression); + + // All liftable constant nodes have been lifted out. + // We'll now optimize them, looking for greatest common denominator tree fragments, in cases where e.g. two lifted constants look up + // the same entity type. + _liftedConstantOptimizer.Optimize(_liftedConstants); + + // Uniquify all variable names, taking into account possible remapping done in the optimization phase above + var replacedParameters = new Dictionary(); + // var (originalParameters, newParameters) = (new List(), new List()); + for (var i = 0; i < _liftedConstants.Count; i++) + { + var liftedConstant = _liftedConstants[i]; + + if (liftedConstant.ReplacingParameter is not null) + { + // This lifted constant is being removed, since it's a duplicate of another with the same expression. + // We still need to remap the parameter in the expression, but no uniquification etc. + replacedParameters.Add(liftedConstant.Parameter, + replacedParameters.TryGetValue(liftedConstant.ReplacingParameter, out var replacedReplacingParameter) + ? replacedReplacingParameter + : liftedConstant.ReplacingParameter); + _liftedConstants.RemoveAt(i--); + continue; + } + + var name = liftedConstant.Parameter.Name ?? "unknown"; + var baseName = name; + for (var j = 0; variableNames.Contains(name); j++) + { + name = baseName + j; + } + + variableNames.Add(name); + + if (name != liftedConstant.Parameter.Name) + { + var newParameter = Expression.Parameter(liftedConstant.Parameter.Type, name); + _liftedConstants[i] = liftedConstant with { Parameter = newParameter }; + replacedParameters.Add(liftedConstant.Parameter, newParameter); + } + } + + // Finally, apply all remapping (optimization, uniquification) to both the expression tree and to the lifted constant variable + // themselves. + + // var (originalParametersArray, newParametersArray) = (originalParameters.ToArray(), newParameters.ToArray()); + // var remappedExpression = ReplacingExpressionVisitor.Replace(originalParametersArray, newParametersArray, expressionAfterLifting); + var originalParameters = new Expression[replacedParameters.Count]; + var newParameters = new Expression[replacedParameters.Count]; + var index = 0; + foreach (var (originalParameter, newParameter) in replacedParameters) + { + originalParameters[index] = originalParameter; + newParameters[index] = newParameter; + index++; + } + var remappedExpression = ReplacingExpressionVisitor.Replace(originalParameters, newParameters, expressionAfterLifting); + + for (var i = 0; i < _liftedConstants.Count; i++) + { + var liftedConstant = _liftedConstants[i]; + var remappedLiftedConstantExpression = + ReplacingExpressionVisitor.Replace(originalParameters, newParameters, liftedConstant.Expression); + + if (remappedLiftedConstantExpression != liftedConstant.Expression) + { + _liftedConstants[i] = liftedConstant with { Expression = remappedLiftedConstantExpression }; + } + } + + LiftedConstants = _liftedConstants.Select(c => (c.Parameter, c.Expression)).ToArray(); + return remappedExpression; + } + + protected override Expression VisitExtension(Expression node) + { + if (node is LiftableConstantExpression liftedConstant) + { + return _inline + ? InlineConstant(liftedConstant) + : LiftConstant(liftedConstant); + } + + return base.VisitExtension(node); + } + + protected virtual ConstantExpression InlineConstant(LiftableConstantExpression liftableConstant) + { + if (liftableConstant.ResolverExpression is Expression> + resolverExpression) + { + // Make sure there aren't any problematic un-lifted constants within the resolver expression. + _unsupportedConstantChecker.Check(resolverExpression); + + var resolver = resolverExpression.Compile(preferInterpretation: true); + var value = resolver(_materializerLiftableConstantContext); + + return Expression.Constant(value, liftableConstant.Type); + } + + throw new InvalidOperationException( + $"Unknown resolved expression of type {liftableConstant.ResolverExpression.GetType().Name} found on liftable constant expression"); + } + + protected virtual ParameterExpression LiftConstant(LiftableConstantExpression liftableConstant) + { + var resolverLambda = liftableConstant.ResolverExpression; + var parameter = resolverLambda.Parameters[0]; + + // Extract the lambda body, replacing the lambda parameter with our lifted constant context parameter, and also inline any captured + // literals + var body = _liftedExpressionProcessor.Process(resolverLambda.Body, parameter, _contextParameter!); + + // If the lambda returns a value type, a Convert to object node gets needed that we need to unwrap + if (body is UnaryExpression { NodeType: ExpressionType.Convert } convertNode + && convertNode.Type == typeof(object)) + { + body = convertNode.Operand; + } + + // Register the lifted constant; note that the name will be uniquified later + var variableParameter = Expression.Parameter(liftableConstant.Type, liftableConstant.VariableName); + _liftedConstants.Add(new(variableParameter, body)); + + return variableParameter; + } + + protected override Expression VisitBinary(BinaryExpression binaryExpression) + { + var left = Visit(binaryExpression.Left); + var right = Visit(binaryExpression.Right); + var conversion = (LambdaExpression?)Visit(binaryExpression.Conversion); + + return binaryExpression.NodeType is ExpressionType.Assign + && left is MemberExpression { Member: FieldInfo { IsInitOnly: true } } initFieldMember + ? initFieldMember.Assign(right) + : binaryExpression.Update(left, conversion, right); + } + +#if DEBUG + + // TODO: issue #33482 - we should properly deal with NTS types rather than disabling them here + // especially using such a crude method + protected override Expression VisitMember(MemberExpression memberExpression) + => memberExpression is { Expression: ConstantExpression, Type.Name: "SqlServerBytesReader" or "GaiaGeoReader" } + ? memberExpression + : base.VisitMember(memberExpression); + + protected override Expression VisitConstant(ConstantExpression node) + { + _unsupportedConstantChecker.Check(node); + return node; + } +#endif + + private sealed class UnsupportedConstantChecker(LiftableConstantProcessor liftableConstantProcessor) : ExpressionVisitor + { + [Conditional("DEBUG")] + public void Check(Expression expression) + { + if (liftableConstantProcessor._precompiledQueriesSupported) + { + Visit(expression); + } + } + + protected override Expression VisitConstant(ConstantExpression node) + { + if (LiftableConstantExpressionHelpers.IsLiteral(node.Value) + // TODO: this part is temporary - we can't inline these constants but we need proper way to deal with them, + // without risk breaking existing scenarios + || node.Value is ParameterBindingInfo or RuntimeServiceProperty or IMaterializationInterceptor or IInstantiationBindingInterceptor + || node.Type.Name == "ProxyFactory") + { + return node; + } + else + { + throw new InvalidOperationException( + $"Shaper expression contains a non-literal constant of type '{node.Value!.GetType().Name}'. " + + $"Use a {nameof(LiftableConstantExpression)} to reference any non-literal constants."); + } + } + } + + private sealed class LiftedConstantOptimizer : ExpressionVisitor + { + private List _liftedConstants = null!; + + private sealed record ExpressionInfo(ExpressionStatus Status, ParameterExpression? Parameter = null, string? PreferredName = null); + private readonly Dictionary _indexedExpressions = new(ExpressionEqualityComparer.Instance); + private LiftedConstant _currentLiftedConstant = null!; + private bool _firstPass; + private int _index; + + public void Optimize(List liftedConstants) + { + _liftedConstants = liftedConstants; + _indexedExpressions.Clear(); + + _firstPass = true; + + // Phase 1: recursively seek out tree fragments which appear more than once across the lifted constants. These will be extracted + // out to separate variables. + foreach (var liftedConstant in liftedConstants) + { + _currentLiftedConstant = liftedConstant; + Visit(liftedConstant.Expression); + } + + // Filter out fragments which don't appear at least once + foreach (var (expression, expressionInfo) in _indexedExpressions) + { + if (expressionInfo.Status == ExpressionStatus.SeenOnce) + { + _indexedExpressions.Remove(expression); + continue; + } + + Check.DebugAssert(expressionInfo.Status == ExpressionStatus.SeenMultipleTimes, + "expressionInfo.Status == ExpressionStatus.SeenMultipleTimes"); + } + + // Second pass: extract common denominator tree fragments to separate variables + _firstPass = false; + for (_index = 0; _index < liftedConstants.Count; _index++) + { + _currentLiftedConstant = _liftedConstants[_index]; + if (_indexedExpressions.TryGetValue(_currentLiftedConstant.Expression, out var expressionInfo) + && expressionInfo.Status == ExpressionStatus.Extracted) + { + // This entire lifted constant has already been extracted before, so we no longer need it as a separate variable. + _liftedConstants[_index] = _currentLiftedConstant with { ReplacingParameter = expressionInfo.Parameter }; + + continue; + } + + var optimizedExpression = Visit(_currentLiftedConstant.Expression); + if (optimizedExpression != _currentLiftedConstant.Expression) + { + _liftedConstants[_index] = _currentLiftedConstant with { Expression = optimizedExpression }; + } + } + } + + [return: NotNullIfNotNull(nameof(node))] + public override Expression? Visit(Expression? node) + { + if (node is null) + { + return null; + } + + if (node is ParameterExpression or ConstantExpression || node.Type.IsAssignableTo(typeof(LambdaExpression))) + { + return node; + } + + if (_firstPass) + { + var preferredName = ReferenceEquals(node, _currentLiftedConstant.Expression) + ? _currentLiftedConstant.Parameter.Name + : null; + + if (!_indexedExpressions.TryGetValue(node, out var expressionInfo)) + { + // Unseen expression, add it to the dictionary with a null value, to indicate it's only a candidate at this point. + _indexedExpressions[node] = new(ExpressionStatus.SeenOnce, PreferredName: preferredName); + return base.Visit(node); + } + + // We've already seen this expression. + if (expressionInfo.Status == ExpressionStatus.SeenOnce + || expressionInfo.PreferredName is null && preferredName is not null) + { + // This is the 2nd time we're seeing the expression - mark it as a common denominator + _indexedExpressions[node] = _indexedExpressions[node] with + { + Status = ExpressionStatus.SeenMultipleTimes, + PreferredName = preferredName + }; + } + + // We've already seen and indexed this expression, no need to do it again + return node; + } + else + { + // 2nd pass + if (_indexedExpressions.TryGetValue(node, out var expressionInfo) && expressionInfo.Status != ExpressionStatus.SeenOnce) + { + // This fragment is common across multiple lifted constants. + if (expressionInfo.Status == ExpressionStatus.SeenMultipleTimes) + { + // This fragment hasn't yet been extracted out to its own variable in the 2nd pass. + + // If this happens to be a top-level node in the lifted constant, no need to extract an additional variable - just + // use that as the "extracted" parameter further down. + if (ReferenceEquals(node, _currentLiftedConstant.Expression)) + { + _indexedExpressions[node] = new(ExpressionStatus.Extracted, _currentLiftedConstant.Parameter); + return base.Visit(node); + } + + // Otherwise, we need to extract a new variable, integrating it just before this one. + var parameter = Expression.Parameter(node.Type, node switch + { + _ when expressionInfo.PreferredName is not null => expressionInfo.PreferredName, + MemberExpression me => char.ToLowerInvariant(me.Member.Name[0]) + me.Member.Name[1..], + MethodCallExpression mce => char.ToLowerInvariant(mce.Method.Name[0]) + mce.Method.Name[1..], + _ => "unknown" + }); + + var visitedNode = base.Visit(node); + _liftedConstants.Insert(_index++, new(parameter, visitedNode)); + + // Mark this node as having been extracted, to prevent it from getting extracted again + expressionInfo = _indexedExpressions[node] = new(ExpressionStatus.Extracted, parameter); + } + + Check.DebugAssert(expressionInfo.Parameter is not null, "expressionInfo.Parameter is not null"); + + return expressionInfo.Parameter; + } + + // This specific fragment only appears once across the lifted constants; keep going down. + return base.Visit(node); + } + } + + private enum ExpressionStatus + { + SeenOnce, + SeenMultipleTimes, + Extracted + } + } + + private sealed class LiftedExpressionProcessor : ExpressionVisitor + { + private ParameterExpression _originalParameter = null!; + private ParameterExpression _replacingParameter = null!; + + public Expression Process(Expression expression, ParameterExpression originalParameter, ParameterExpression replacingParameter) + { + _originalParameter = originalParameter; + _replacingParameter = replacingParameter; + + return Visit(expression); + } + + protected override Expression VisitMember(MemberExpression node) + { + // The expression to be lifted may contain a captured variable; for limited literal scenarios, inline that variable into the + // expression so we can render it out to C#. + + // TODO: For the general case, this needs to be a full blown "evaluatable" identifier (like ParameterExtractingEV), which can + // identify any fragments of the tree which don't depend on the lambda parameter, and evaluate them. + // But for now we're doing a reduced version. + + var visited = base.VisitMember(node); + + if (visited is MemberExpression + { + Expression: ConstantExpression { Value: { } constant }, + Member: var member + }) + { + return member switch + { + FieldInfo fi => Expression.Constant(fi.GetValue(constant), node.Type), + PropertyInfo pi => Expression.Constant(pi.GetValue(constant), node.Type), + _ => visited + }; + } + + return visited; + } + + protected override Expression VisitParameter(ParameterExpression node) + => ReferenceEquals(node, _originalParameter) + ? _replacingParameter + : base.VisitParameter(node); + } +} diff --git a/src/EFCore/Query/MaterializerLiftableConstantContext.cs b/src/EFCore/Query/MaterializerLiftableConstantContext.cs new file mode 100644 index 00000000000..a8c062cd902 --- /dev/null +++ b/src/EFCore/Query/MaterializerLiftableConstantContext.cs @@ -0,0 +1,15 @@ +// 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.CodeAnalysis; + +namespace Microsoft.EntityFrameworkCore.Query; + +/// +/// This is an experimental API used by the Entity Framework Core feature and it is not subject to +/// the same compatibility standards as public APIs. It may be changed or removed without notice in +/// any release. You should only use it directly in your code with extreme caution and knowing that +/// doing so can result in application failures when updating to a new Entity Framework Core release. +/// +[Experimental(EFDiagnostics.PrecompiledQueryExperimental)] +public record MaterializerLiftableConstantContext(ShapedQueryCompilingExpressionVisitorDependencies Dependencies); diff --git a/src/EFCore/Query/QueryCompilationContext.cs b/src/EFCore/Query/QueryCompilationContext.cs index def674ba898..8a86ed0d5ec 100644 --- a/src/EFCore/Query/QueryCompilationContext.cs +++ b/src/EFCore/Query/QueryCompilationContext.cs @@ -56,7 +56,8 @@ public class QueryCompilationContext private readonly IQueryTranslationPostprocessorFactory _queryTranslationPostprocessorFactory; private readonly IShapedQueryCompilingExpressionVisitorFactory _shapedQueryCompilingExpressionVisitorFactory; - private readonly ExpressionPrinter _expressionPrinter; + private readonly ExpressionPrinter _expressionPrinter = new(); + private readonly RuntimeParameterConstantLifter _runtimeParameterConstantLifter; private Dictionary? _runtimeParameters; @@ -82,8 +83,7 @@ public QueryCompilationContext( _queryableMethodTranslatingExpressionVisitorFactory = dependencies.QueryableMethodTranslatingExpressionVisitorFactory; _queryTranslationPostprocessorFactory = dependencies.QueryTranslationPostprocessorFactory; _shapedQueryCompilingExpressionVisitorFactory = dependencies.ShapedQueryCompilingExpressionVisitorFactory; - - _expressionPrinter = new ExpressionPrinter(); + _runtimeParameterConstantLifter = new(dependencies.LiftableConstantFactory); } /// @@ -148,6 +148,11 @@ public QueryCompilationContext( public virtual void AddTag(string tag) => Tags.Add(tag); + /// + /// A value indicating whether the provider supports precompiled query. Default value is . Providers that do support this feature should opt-in by setting this value to . + /// + public virtual bool SupportsPrecompiledQuery => false; + /// /// Creates the query executor func which gives results for this query. /// @@ -155,6 +160,33 @@ public virtual void AddTag(string tag) /// The query to generate executor for. /// Returns which can be invoked to get results of this query. public virtual Func CreateQueryExecutor(Expression query) + { + var queryExecutorExpression = CreateQueryExecutorExpression(query); + + // The materializer expression tree has liftable constant nodes, pointing to various constants that should be the same instances + // across invocations of the query. + // In normal mode, these nodes should simply be evaluated, and a ConstantExpression to those instances embedded directly in the + // tree (for precompiled queries we generate C# code for resolving those instances instead). + var queryExecutorAfterLiftingExpression = + (Expression>)Dependencies.LiftableConstantProcessor.InlineConstants(queryExecutorExpression, SupportsPrecompiledQuery); + + try + { + return queryExecutorAfterLiftingExpression.Compile(); + } + finally + { + Logger.QueryExecutionPlanned(Dependencies.Context, _expressionPrinter, queryExecutorExpression); + } + } + + /// + /// Creates the query executor func which gives results for this query. + /// + /// The result type of this query. + /// The query to generate executor for. + /// Returns which can be invoked to get results of this query. + public virtual Expression> CreateQueryExecutorExpression(Expression query) { var queryAndEventData = Logger.QueryCompilationStarting(Dependencies.Context, _expressionPrinter, query); query = queryAndEventData.Query; @@ -176,14 +208,7 @@ public virtual Func CreateQueryExecutor(Expressi query, QueryContextParameter); - try - { - return queryExecutorExpression.Compile(); - } - finally - { - Logger.QueryExecutionPlanned(Dependencies.Context, _expressionPrinter, queryExecutorExpression); - } + return queryExecutorExpression; } /// @@ -193,6 +218,14 @@ public virtual Func CreateQueryExecutor(Expressi /// public virtual ParameterExpression RegisterRuntimeParameter(string name, LambdaExpression valueExtractor) { + var valueExtractorBody = valueExtractor.Body; + if (SupportsPrecompiledQuery) + { + valueExtractorBody = _runtimeParameterConstantLifter.Visit(valueExtractorBody); + } + + valueExtractor = Expression.Lambda(valueExtractorBody, valueExtractor.Parameters); + if (valueExtractor.Parameters.Count != 1 || valueExtractor.Parameters[0] != QueryContextParameter) { @@ -230,4 +263,50 @@ public override Type Type public override ExpressionType NodeType => ExpressionType.Extension; } + + private sealed class RuntimeParameterConstantLifter(ILiftableConstantFactory liftableConstantFactory) : ExpressionVisitor + { + private readonly static MethodInfo ComplexPropertyListElementAddMethod = typeof(List).GetMethod(nameof(List.Add))!; + + protected override Expression VisitConstant(ConstantExpression constantExpression) + { + switch (constantExpression.Value) + { + case IProperty property: + { + return liftableConstantFactory.CreateLiftableConstant( + constantExpression.Value, + LiftableConstantExpressionHelpers.BuildMemberAccessLambdaForProperty(property), + property.Name + "Property", + typeof(IProperty)); + } + + case List complexPropertyChain: + { + var elementInitExpressions = new ElementInit[complexPropertyChain.Count]; + var prm = Expression.Parameter(typeof(MaterializerLiftableConstantContext)); + + for (var i = 0; i < complexPropertyChain.Count; i++) + { + var complexType = complexPropertyChain[i].ComplexType; + var complexTypeExpression = LiftableConstantExpressionHelpers.BuildMemberAccessForEntityOrComplexType(complexType, prm); + elementInitExpressions[i] = Expression.ElementInit( + ComplexPropertyListElementAddMethod, + Expression.Property(complexTypeExpression, nameof(IComplexType.ComplexProperty))); + } + + return liftableConstantFactory.CreateLiftableConstant( + constantExpression.Value, + Expression.Lambda>( + Expression.ListInit(Expression.New(typeof(List)), elementInitExpressions), + prm), + "ComplexPropertyChain", + constantExpression.Type); + } + + default: + return base.VisitConstant(constantExpression); + } + } + } } diff --git a/src/EFCore/Query/QueryCompilationContextDependencies.cs b/src/EFCore/Query/QueryCompilationContextDependencies.cs index 85ad3c009b9..3df7db3a297 100644 --- a/src/EFCore/Query/QueryCompilationContextDependencies.cs +++ b/src/EFCore/Query/QueryCompilationContextDependencies.cs @@ -53,6 +53,8 @@ public QueryCompilationContextDependencies( IQueryableMethodTranslatingExpressionVisitorFactory queryableMethodTranslatingExpressionVisitorFactory, IQueryTranslationPostprocessorFactory queryTranslationPostprocessorFactory, IShapedQueryCompilingExpressionVisitorFactory shapedQueryCompilingExpressionVisitorFactory, + ILiftableConstantFactory liftableConstantFactory, + ILiftableConstantProcessor liftableConstantProcessor, IExecutionStrategy executionStrategy, ICurrentDbContext currentContext, IDbContextOptions contextOptions, @@ -65,6 +67,8 @@ public QueryCompilationContextDependencies( QueryableMethodTranslatingExpressionVisitorFactory = queryableMethodTranslatingExpressionVisitorFactory; QueryTranslationPostprocessorFactory = queryTranslationPostprocessorFactory; ShapedQueryCompilingExpressionVisitorFactory = shapedQueryCompilingExpressionVisitorFactory; + LiftableConstantFactory = liftableConstantFactory; + LiftableConstantProcessor = liftableConstantProcessor; IsRetryingExecutionStrategy = executionStrategy.RetriesOnFailure; ContextOptions = contextOptions; Logger = logger; @@ -114,6 +118,16 @@ public QueryTrackingBehavior QueryTrackingBehavior /// public IShapedQueryCompilingExpressionVisitorFactory ShapedQueryCompilingExpressionVisitorFactory { get; init; } + /// + /// TODO + /// + public ILiftableConstantFactory LiftableConstantFactory { get; init; } + + /// + /// The liftable constant processor. + /// + public ILiftableConstantProcessor LiftableConstantProcessor { get; init; } + /// /// Whether the configured execution strategy can retry. /// diff --git a/src/EFCore/Query/ReplacingExpressionVisitor.cs b/src/EFCore/Query/ReplacingExpressionVisitor.cs index 666a63bb096..2d5989956d9 100644 --- a/src/EFCore/Query/ReplacingExpressionVisitor.cs +++ b/src/EFCore/Query/ReplacingExpressionVisitor.cs @@ -33,6 +33,16 @@ public class ReplacingExpressionVisitor : ExpressionVisitor public static Expression Replace(Expression original, Expression replacement, Expression tree) => new ReplacingExpressionVisitor(new[] { original }, new[] { replacement }).Visit(tree); + /// + /// Replaces one expression with another in given expression tree. + /// + /// A list of original expressions to replace. + /// A list of expressions to be used as replacements. + /// The expression tree in which replacement is going to be performed. + /// An expression tree with replacements made. + public static Expression Replace(Expression[] originals, Expression[] replacements, Expression tree) + => new ReplacingExpressionVisitor(originals, replacements).Visit(tree); + /// /// Creates a new instance of the class. /// @@ -48,7 +58,7 @@ public ReplacingExpressionVisitor(IReadOnlyList originals, IReadOnly [return: NotNullIfNotNull("expression")] public override Expression? Visit(Expression? expression) { - if (expression is null or ShapedQueryExpression or StructuralTypeShaperExpression or GroupByShaperExpression) + if (expression is null or ShapedQueryExpression or StructuralTypeShaperExpression or GroupByShaperExpression or LiftableConstantExpression) { return expression; } diff --git a/src/EFCore/Query/ShapedQueryCompilingExpressionVisitor.cs b/src/EFCore/Query/ShapedQueryCompilingExpressionVisitor.cs index 7e346199f23..3393e2bb592 100644 --- a/src/EFCore/Query/ShapedQueryCompilingExpressionVisitor.cs +++ b/src/EFCore/Query/ShapedQueryCompilingExpressionVisitor.cs @@ -1,8 +1,11 @@ // 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.CodeAnalysis; using JetBrains.Annotations; using Microsoft.EntityFrameworkCore.ChangeTracking.Internal; +using Microsoft.EntityFrameworkCore.Diagnostics.Internal; +using Microsoft.EntityFrameworkCore.Internal; using Microsoft.EntityFrameworkCore.Metadata.Internal; using static System.Linq.Expressions.Expression; @@ -39,6 +42,7 @@ private static readonly PropertyInfo CancellationTokenMemberInfo private readonly Expression _cancellationTokenParameter; private readonly EntityMaterializerInjectingExpressionVisitor _entityMaterializerInjectingExpressionVisitor; private readonly ConstantVerifyingExpressionVisitor _constantVerifyingExpressionVisitor; + private readonly MaterializationConditionConstantLifter _materializationConditionConstantLifter; /// /// Creates a new instance of the class. @@ -55,9 +59,12 @@ protected ShapedQueryCompilingExpressionVisitor( _entityMaterializerInjectingExpressionVisitor = new EntityMaterializerInjectingExpressionVisitor( dependencies.EntityMaterializerSource, - queryCompilationContext.QueryTrackingBehavior); + dependencies.LiftableConstantFactory, + queryCompilationContext.QueryTrackingBehavior, + queryCompilationContext.SupportsPrecompiledQuery); - _constantVerifyingExpressionVisitor = new ConstantVerifyingExpressionVisitor(dependencies.TypeMappingSource); + _constantVerifyingExpressionVisitor = new(dependencies.TypeMappingSource); + _materializationConditionConstantLifter = new(dependencies.LiftableConstantFactory); if (queryCompilationContext.IsAsync) { @@ -128,7 +135,14 @@ private static readonly MethodInfo SingleOrDefaultAsyncMethodInfo .GetDeclaredMethods(nameof(SingleOrDefaultAsync)) .Single(mi => mi.GetParameters().Length == 2); - private static async Task SingleAsync( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static async Task SingleAsync( IAsyncEnumerable asyncEnumerable, CancellationToken cancellationToken = default) { @@ -150,7 +164,14 @@ private static async Task SingleAsync( return result; } - private static async Task SingleOrDefaultAsync( + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [EntityFrameworkInternal] + public static async Task SingleOrDefaultAsync( IAsyncEnumerable asyncEnumerable, CancellationToken cancellationToken = default) { @@ -189,7 +210,55 @@ protected virtual Expression InjectEntityMaterializers(Expression expression) { VerifyNoClientConstant(expression); - return _entityMaterializerInjectingExpressionVisitor.Inject(expression); + var materializerExpression = _entityMaterializerInjectingExpressionVisitor.Inject(expression); + if (QueryCompilationContext.SupportsPrecompiledQuery) + { + materializerExpression = _materializationConditionConstantLifter.Visit(materializerExpression); + } + + return materializerExpression; + } + + private sealed class MaterializationConditionConstantLifter(ILiftableConstantFactory liftableConstantFactory) : ExpressionVisitor + { + private static readonly MethodInfo ServiceProviderGetService = + typeof(IServiceProvider).GetMethod(nameof(IServiceProvider.GetService), [typeof(Type)])!; + + protected override Expression VisitConstant(ConstantExpression constantExpression) + => constantExpression switch + { + { Value: IEntityType entityTypeValue } => liftableConstantFactory.CreateLiftableConstant( + constantExpression.Value, + LiftableConstantExpressionHelpers.BuildMemberAccessLambdaForEntityOrComplexType(entityTypeValue), + entityTypeValue.Name + "EntityType", + constantExpression.Type), + { Value: IComplexType complexTypeValue } => liftableConstantFactory.CreateLiftableConstant( + constantExpression.Value, + LiftableConstantExpressionHelpers.BuildMemberAccessLambdaForEntityOrComplexType(complexTypeValue), + complexTypeValue.Name + "ComplexType", + constantExpression.Type), + { Value: IProperty propertyValue } => liftableConstantFactory.CreateLiftableConstant( + constantExpression.Value, + LiftableConstantExpressionHelpers.BuildMemberAccessLambdaForProperty(propertyValue), + propertyValue.Name + "Property", + constantExpression.Type), + _ => base.VisitConstant(constantExpression) + }; + + protected override Expression VisitBinary(BinaryExpression binaryExpression) + { + var left = Visit(binaryExpression.Left); + var right = Visit(binaryExpression.Right); + var conversion = (LambdaExpression?)Visit(binaryExpression.Conversion); + + return binaryExpression.NodeType is ExpressionType.Assign + && left is MemberExpression { Member: FieldInfo { IsInitOnly: true } } initFieldMember + ? initFieldMember.Assign(right) + : binaryExpression.Update(left, conversion, right); + } + + protected override Expression VisitExtension(Expression node) + => node is LiftableConstantExpression ? node : base.VisitExtension(node); } /// @@ -199,6 +268,35 @@ protected virtual Expression InjectEntityMaterializers(Expression expression) protected virtual void VerifyNoClientConstant(Expression expression) => _constantVerifyingExpressionVisitor.Visit(expression); + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// + [UsedImplicitly] + [EntityFrameworkInternal] + public static Exception CreateNullKeyValueInNoTrackingQuery( + IEntityType entityType, + IReadOnlyList properties, + object?[] keyValues) + { + var index = -1; + for (var i = 0; i < keyValues.Length; i++) + { + if (keyValues[i] == null) + { + index = i; + break; + } + } + + var property = properties[index]; + + throw new InvalidOperationException( + CoreStrings.InvalidKeyValue(entityType.DisplayName(), property.Name)); + } + private sealed class ConstantVerifyingExpressionVisitor : ExpressionVisitor { private readonly ITypeMappingSource _typeMappingSource; @@ -289,23 +387,33 @@ private static readonly MethodInfo StartTrackingMethodInfo nameof(QueryContext.StartTracking), [typeof(IEntityType), typeof(object), typeof(ISnapshot).MakeByRefType()])!; private static readonly MethodInfo CreateNullKeyValueInNoTrackingQueryMethod - = typeof(EntityMaterializerInjectingExpressionVisitor) + = typeof(ShapedQueryCompilingExpressionVisitor) .GetTypeInfo().GetDeclaredMethod(nameof(CreateNullKeyValueInNoTrackingQuery))!; + private static readonly MethodInfo EntityTypeFindPrimaryKeyMethod = + typeof(IEntityType).GetMethod(nameof(IEntityType.FindPrimaryKey), [])!; + private readonly IEntityMaterializerSource _entityMaterializerSource; + private readonly ILiftableConstantFactory _liftableConstantFactory; private readonly QueryTrackingBehavior _queryTrackingBehavior; private readonly bool _queryStateManager; private readonly ISet _visitedEntityTypes = new HashSet(); + private readonly MaterializationConditionConstantLifter _materializationConditionConstantLifter; + private readonly bool _supportsPrecompiledQuery; private int _currentEntityIndex; public EntityMaterializerInjectingExpressionVisitor( IEntityMaterializerSource entityMaterializerSource, - QueryTrackingBehavior queryTrackingBehavior) + ILiftableConstantFactory liftableConstantFactory, + QueryTrackingBehavior queryTrackingBehavior, + bool supportsPrecompiledQuery) { _entityMaterializerSource = entityMaterializerSource; + _liftableConstantFactory = liftableConstantFactory; _queryTrackingBehavior = queryTrackingBehavior; - _queryStateManager = - queryTrackingBehavior is QueryTrackingBehavior.TrackAll or QueryTrackingBehavior.NoTrackingWithIdentityResolution; + _queryStateManager = queryTrackingBehavior is QueryTrackingBehavior.TrackAll or QueryTrackingBehavior.NoTrackingWithIdentityResolution; + _materializationConditionConstantLifter = new(liftableConstantFactory); + _supportsPrecompiledQuery = supportsPrecompiledQuery; } public Expression Inject(Expression expression) @@ -378,13 +486,24 @@ private Expression ProcessEntityShaper(StructuralTypeShaperExpression shaper) variables.Add(entryVariable); variables.Add(hasNullKeyVariable); + var resolverPrm = Parameter(typeof(MaterializerLiftableConstantContext), "c"); expressions.Add( Assign( entryVariable, Call( QueryCompilationContext.QueryContextParameter, TryGetEntryMethodInfo, - Constant(primaryKey), + _supportsPrecompiledQuery + ? _liftableConstantFactory.CreateLiftableConstant( + primaryKey, + Lambda>( + Call( + LiftableConstantExpressionHelpers.BuildMemberAccessForEntityOrComplexType(typeBase, resolverPrm), + EntityTypeFindPrimaryKeyMethod), + resolverPrm), + typeBase.Name + "Key", + typeof(IKey)) + : Constant(primaryKey), NewArrayInit( typeof(object), primaryKey.Properties @@ -432,6 +551,8 @@ private Expression ProcessEntityShaper(StructuralTypeShaperExpression shaper) else { var keyValuesVariable = Variable(typeof(object[]), "keyValues" + _currentEntityIndex); + var resolverPrm = Parameter(typeof(MaterializerLiftableConstantContext), "c"); + expressions.Add( IfThenElse( primaryKey.Properties.Select( @@ -454,8 +575,26 @@ private Expression ProcessEntityShaper(StructuralTypeShaperExpression shaper) typeof(object), p.GetIndex(), p)))), Call( CreateNullKeyValueInNoTrackingQueryMethod, - Constant(typeBase), - Constant(primaryKey.Properties), + _supportsPrecompiledQuery + ? _liftableConstantFactory.CreateLiftableConstant( + typeBase, + LiftableConstantExpressionHelpers.BuildMemberAccessLambdaForEntityOrComplexType(typeBase), + typeBase.Name + "EntityType", + typeof(IEntityType)) + : Constant(typeBase), + _supportsPrecompiledQuery + ? _liftableConstantFactory.CreateLiftableConstant( + primaryKey.Properties, + Lambda>( + Property( + Call( + LiftableConstantExpressionHelpers.BuildMemberAccessForEntityOrComplexType(typeBase, resolverPrm), + EntityTypeFindPrimaryKeyMethod), + nameof(IKey.Properties)), + resolverPrm), + typeBase.Name + "PrimaryKeyProperties", + typeof(IReadOnlyList)) + : Constant(primaryKey.Properties), keyValuesVariable)))); } } @@ -491,18 +630,24 @@ private Expression MaterializeEntity( expressions.Add( Assign( shadowValuesVariable, - Constant(Snapshot.Empty))); + _supportsPrecompiledQuery + ? _liftableConstantFactory.CreateLiftableConstant( + Snapshot.Empty, + static _ => Snapshot.Empty, + "emptySnapshot", + typeof(Snapshot)) + : Constant(Snapshot.Empty))); var returnType = typeBase.ClrType; var valueBufferExpression = Call(materializationContextVariable, MaterializationContext.GetValueBufferMethod); + + var materializationConditionBody = ReplacingExpressionVisitor.Replace( + shaper.MaterializationCondition.Parameters[0], + valueBufferExpression, + shaper.MaterializationCondition.Body); + var expressionContext = (returnType, materializationContextVariable, concreteEntityTypeVariable, shadowValuesVariable); - expressions.Add( - Assign( - concreteEntityTypeVariable, - ReplacingExpressionVisitor.Replace( - shaper.MaterializationCondition.Parameters[0], - valueBufferExpression, - shaper.MaterializationCondition.Body))); + expressions.Add(Assign(concreteEntityTypeVariable, materializationConditionBody)); var (primaryKey, concreteEntityTypes) = typeBase is IEntityType entityType ? (entityType.FindPrimaryKey(), entityType.GetConcreteDerivedTypesInclusive().Cast().ToArray()) @@ -511,9 +656,16 @@ private Expression MaterializeEntity( var switchCases = new SwitchCase[concreteEntityTypes.Length]; for (var i = 0; i < concreteEntityTypes.Length; i++) { + var concreteEntityType = concreteEntityTypes[i]; switchCases[i] = SwitchCase( CreateFullMaterializeExpression(concreteEntityTypes[i], expressionContext), - Constant(concreteEntityTypes[i], typeBase is IEntityType ? typeof(IEntityType) : typeof(IComplexType))); + _supportsPrecompiledQuery + ? _liftableConstantFactory.CreateLiftableConstant( + concreteEntityTypes[i], + LiftableConstantExpressionHelpers.BuildMemberAccessLambdaForEntityOrComplexType(concreteEntityType), + concreteEntityType.Name + (typeBase is IEntityType ? "EntityType" : "ComplexType"), + typeBase is IEntityType ? typeof(IEntityType) : typeof(IComplexType)) + : Constant(concreteEntityTypes[i], typeBase is IEntityType ? typeof(IEntityType) : typeof(IComplexType))); } var materializationExpression = Switch( @@ -605,27 +757,5 @@ private BlockExpression CreateFullMaterializeExpression( return Block(blockExpressions); } - - [UsedImplicitly] - private static Exception CreateNullKeyValueInNoTrackingQuery( - IEntityType entityType, - IReadOnlyList properties, - object?[] keyValues) - { - var index = -1; - for (var i = 0; i < keyValues.Length; i++) - { - if (keyValues[i] == null) - { - index = i; - break; - } - } - - var property = properties[index]; - - throw new InvalidOperationException( - CoreStrings.InvalidKeyValue(entityType.DisplayName(), property.Name)); - } } } diff --git a/src/EFCore/Query/ShapedQueryCompilingExpressionVisitorDependencies.cs b/src/EFCore/Query/ShapedQueryCompilingExpressionVisitorDependencies.cs index cc025f1e48d..a8eb8153921 100644 --- a/src/EFCore/Query/ShapedQueryCompilingExpressionVisitorDependencies.cs +++ b/src/EFCore/Query/ShapedQueryCompilingExpressionVisitorDependencies.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 Microsoft.EntityFrameworkCore.Internal; using Microsoft.Extensions.Caching.Memory; namespace Microsoft.EntityFrameworkCore.Query; @@ -51,12 +52,22 @@ public ShapedQueryCompilingExpressionVisitorDependencies( IEntityMaterializerSource entityMaterializerSource, ITypeMappingSource typeMappingSource, IMemoryCache memoryCache, - ICoreSingletonOptions coreSingletonOptions) + ICoreSingletonOptions coreSingletonOptions, + IModel model, + ILiftableConstantFactory liftableConstantFactory, + IDiagnosticsLogger queryLogger, + IEnumerable singletonInterceptors, + IDbContextServices contextServices) { EntityMaterializerSource = entityMaterializerSource; TypeMappingSource = typeMappingSource; MemoryCache = memoryCache; CoreSingletonOptions = coreSingletonOptions; + Model = model; + LiftableConstantFactory = liftableConstantFactory; + QueryLogger = queryLogger; + SingletonInterceptors = singletonInterceptors; + ContextServices = contextServices; } /// @@ -78,4 +89,29 @@ public ShapedQueryCompilingExpressionVisitorDependencies( /// Core singleton options. /// public ICoreSingletonOptions CoreSingletonOptions { get; init; } + + /// + /// The model. + /// + public IModel Model { get; init; } + + /// + /// The liftable constant factory. + /// + public ILiftableConstantFactory LiftableConstantFactory { get; init; } + + /// + /// The query logger. + /// + public IDiagnosticsLogger QueryLogger { get; init; } + + /// + /// Registered singleton interceptors. + /// + public IEnumerable SingletonInterceptors { get; init; } + + /// + /// TODO + /// + public IDbContextServices ContextServices { get; init; } } diff --git a/src/EFCore/Query/StructuralTypeShaperExpression.cs b/src/EFCore/Query/StructuralTypeShaperExpression.cs index 21800633f87..c063baf356c 100644 --- a/src/EFCore/Query/StructuralTypeShaperExpression.cs +++ b/src/EFCore/Query/StructuralTypeShaperExpression.cs @@ -23,8 +23,18 @@ public class StructuralTypeShaperExpression : Expression, IPrintableExpression private static readonly MethodInfo CreateUnableToDiscriminateExceptionMethod = typeof(StructuralTypeShaperExpression).GetTypeInfo().GetDeclaredMethod(nameof(CreateUnableToDiscriminateException))!; + private static readonly MethodInfo GetDiscriminatorValueMethod + = typeof(IReadOnlyEntityType).GetTypeInfo().GetDeclaredMethod(nameof(IReadOnlyEntityType.GetDiscriminatorValue))!; + + /// + /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to + /// the same compatibility standards as public APIs. It may be changed or removed without notice in + /// any release. You should only use it directly in your code with extreme caution and knowing that + /// doing so can result in application failures when updating to a new Entity Framework Core release. + /// [UsedImplicitly] - private static Exception CreateUnableToDiscriminateException(ITypeBase type, object discriminator) + [EntityFrameworkInternal] + public static Exception CreateUnableToDiscriminateException(ITypeBase type, object discriminator) => new InvalidOperationException(CoreStrings.UnableToDiscriminate(type.DisplayName(), discriminator.ToString())); /// @@ -80,7 +90,7 @@ protected StructuralTypeShaperExpression( /// The entity type for which materialization was requested. /// The expression containing value of discriminator. /// - /// An expression of representing materilization condition for the entity type. + /// An expression of representing materialization condition for the entity type. /// protected static Expression CreateUnableToDiscriminateExceptionExpression(ITypeBase type, Expression discriminatorValue) => Block( @@ -131,8 +141,12 @@ protected virtual LambdaExpression GenerateMaterializationCondition(ITypeBase ty var switchCases = new SwitchCase[concreteEntityTypes.Length]; for (var i = 0; i < concreteEntityTypes.Length; i++) { - var discriminatorValue = Constant(concreteEntityTypes[i].GetDiscriminatorValue(), discriminatorProperty.ClrType); - switchCases[i] = SwitchCase(Constant(concreteEntityTypes[i], typeof(IEntityType)), discriminatorValue); + var discriminatorValueObject = concreteEntityTypes[i].GetDiscriminatorValue(); + var discriminatorValueExpression = LiftableConstantExpressionHelpers.IsLiteral(discriminatorValueObject) + ? (Expression)Constant(discriminatorValueObject, discriminatorProperty.ClrType) + : Convert(Call(Constant(concreteEntityTypes[i]), GetDiscriminatorValueMethod), discriminatorProperty.ClrType); + + switchCases[i] = SwitchCase(Constant(concreteEntityTypes[i], typeof(IEntityType)), discriminatorValueExpression); } expressions.Add(Switch(discriminatorValueVariable, exception, switchCases)); @@ -142,11 +156,18 @@ protected virtual LambdaExpression GenerateMaterializationCondition(ITypeBase ty var conditions = exception; for (var i = concreteEntityTypes.Length - 1; i >= 0; i--) { + var discriminatorValueObject = concreteEntityTypes[i].GetDiscriminatorValue(); conditions = Condition( discriminatorComparer.ExtractEqualsBody( discriminatorValueVariable, - Constant( - concreteEntityTypes[i].GetDiscriminatorValue(), + LiftableConstantExpressionHelpers.IsLiteral(discriminatorValueObject) + ? Constant( + discriminatorValueObject, + discriminatorProperty.ClrType) + : Convert( + Call( + Constant(concreteEntityTypes[i], typeof(IEntityType)), + GetDiscriminatorValueMethod), discriminatorProperty.ClrType)), Constant(concreteEntityTypes[i], typeof(IEntityType)), conditions); diff --git a/src/EFCore/Storage/Json/JsonBoolReaderWriter.cs b/src/EFCore/Storage/Json/JsonBoolReaderWriter.cs index 5db25121a4b..d68eeb51f31 100644 --- a/src/EFCore/Storage/Json/JsonBoolReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonBoolReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -10,15 +11,15 @@ namespace Microsoft.EntityFrameworkCore.Storage.Json; /// public sealed class JsonBoolReaderWriter : JsonValueReaderWriter { + private JsonBoolReaderWriter() + { + } + /// /// The singleton instance of this stateless reader/writer. /// public static JsonBoolReaderWriter Instance { get; } = new(); - private JsonBoolReaderWriter() - { - } - /// public override bool FromJsonTyped(ref Utf8JsonReaderManager manager, object? existingObject = null) => manager.CurrentReader.GetBoolean(); @@ -26,4 +27,10 @@ public override bool FromJsonTyped(ref Utf8JsonReaderManager manager, object? ex /// public override void ToJsonTyped(Utf8JsonWriter writer, bool value) => writer.WriteBooleanValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonByteArrayReaderWriter.cs b/src/EFCore/Storage/Json/JsonByteArrayReaderWriter.cs index 5bbe3070fa1..54da73053d1 100644 --- a/src/EFCore/Storage/Json/JsonByteArrayReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonByteArrayReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override byte[] FromJsonTyped(ref Utf8JsonReaderManager manager, object? /// public override void ToJsonTyped(Utf8JsonWriter writer, byte[] value) => writer.WriteBase64StringValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonByteReaderWriter.cs b/src/EFCore/Storage/Json/JsonByteReaderWriter.cs index d9b22fa4e73..34ff7271f3b 100644 --- a/src/EFCore/Storage/Json/JsonByteReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonByteReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override byte FromJsonTyped(ref Utf8JsonReaderManager manager, object? ex /// public override void ToJsonTyped(Utf8JsonWriter writer, byte value) => writer.WriteNumberValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonCastValueReaderWriter.cs b/src/EFCore/Storage/Json/JsonCastValueReaderWriter.cs index 98fddf1eacf..f6aed536d73 100644 --- a/src/EFCore/Storage/Json/JsonCastValueReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonCastValueReaderWriter.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.CodeAnalysis; using System.Text.Json; using Microsoft.EntityFrameworkCore.Storage.Internal; @@ -34,4 +35,11 @@ public override void ToJsonTyped(Utf8JsonWriter writer, TConverted value) JsonValueReaderWriter ICompositeJsonValueReaderWriter.InnerReaderWriter => _providerReaderWriter; + + private readonly ConstructorInfo _constructorInfo = typeof(JsonCastValueReaderWriter).GetConstructor([typeof(JsonValueReaderWriter)])!; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => + Expression.New(_constructorInfo, ((ICompositeJsonValueReaderWriter)this).InnerReaderWriter.ConstructorExpression); } diff --git a/src/EFCore/Storage/Json/JsonCharReaderWriter.cs b/src/EFCore/Storage/Json/JsonCharReaderWriter.cs index 7eb278629cc..1d9b0f87dd1 100644 --- a/src/EFCore/Storage/Json/JsonCharReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonCharReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override char FromJsonTyped(ref Utf8JsonReaderManager manager, object? ex /// public override void ToJsonTyped(Utf8JsonWriter writer, char value) => writer.WriteStringValue(value.ToString()); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonCollectionOfNullableStructsReaderWriter.cs b/src/EFCore/Storage/Json/JsonCollectionOfNullableStructsReaderWriter.cs index cac740b68fc..dbffc3e1644 100644 --- a/src/EFCore/Storage/Json/JsonCollectionOfNullableStructsReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonCollectionOfNullableStructsReaderWriter.cs @@ -2,6 +2,7 @@ // The .NET Foundation licenses this file to you under the MIT license. using System.Collections.ObjectModel; +using System.Diagnostics.CodeAnalysis; using System.Text.Json; using Microsoft.EntityFrameworkCore.Storage.Internal; @@ -122,4 +123,12 @@ public override void ToJsonTyped(Utf8JsonWriter writer, IEnumerable v JsonValueReaderWriter ICompositeJsonValueReaderWriter.InnerReaderWriter => _elementReaderWriter; + + private readonly ConstructorInfo _constructorInfo = + typeof(JsonCollectionOfNullableStructsReaderWriter).GetConstructor([typeof(JsonValueReaderWriter)])!; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => + Expression.New(_constructorInfo, ((ICompositeJsonValueReaderWriter)this).InnerReaderWriter.ConstructorExpression); } diff --git a/src/EFCore/Storage/Json/JsonCollectionOfReferencesReaderWriter.cs b/src/EFCore/Storage/Json/JsonCollectionOfReferencesReaderWriter.cs index 6cb96b7b647..eb512a41ec2 100644 --- a/src/EFCore/Storage/Json/JsonCollectionOfReferencesReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonCollectionOfReferencesReaderWriter.cs @@ -2,6 +2,7 @@ // The .NET Foundation licenses this file to you under the MIT license. using System.Collections.ObjectModel; +using System.Diagnostics.CodeAnalysis; using System.Text.Json; using Microsoft.EntityFrameworkCore.Storage.Internal; @@ -120,4 +121,11 @@ public override void ToJsonTyped(Utf8JsonWriter writer, object? value) JsonValueReaderWriter ICompositeJsonValueReaderWriter.InnerReaderWriter => _elementReaderWriter; + + private readonly ConstructorInfo _constructorInfo = typeof(JsonCollectionOfReferencesReaderWriter).GetConstructor([typeof(JsonValueReaderWriter)])!; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => + Expression.New(_constructorInfo, ((ICompositeJsonValueReaderWriter)this).InnerReaderWriter.ConstructorExpression); } diff --git a/src/EFCore/Storage/Json/JsonCollectionOfStructsReaderWriter.cs b/src/EFCore/Storage/Json/JsonCollectionOfStructsReaderWriter.cs index 9f2c1722859..644bf4d5d56 100644 --- a/src/EFCore/Storage/Json/JsonCollectionOfStructsReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonCollectionOfStructsReaderWriter.cs @@ -2,6 +2,7 @@ // The .NET Foundation licenses this file to you under the MIT license. using System.Collections.ObjectModel; +using System.Diagnostics.CodeAnalysis; using System.Text.Json; using Microsoft.EntityFrameworkCore.Storage.Internal; @@ -111,4 +112,11 @@ public override void ToJsonTyped(Utf8JsonWriter writer, IEnumerable va JsonValueReaderWriter ICompositeJsonValueReaderWriter.InnerReaderWriter => _elementReaderWriter; + + private readonly ConstructorInfo _constructorInfo = typeof(JsonCollectionOfStructsReaderWriter).GetConstructor([typeof(JsonValueReaderWriter)])!; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => + Expression.New(_constructorInfo, ((ICompositeJsonValueReaderWriter)this).InnerReaderWriter.ConstructorExpression); } diff --git a/src/EFCore/Storage/Json/JsonConvertedValueReaderWriter.cs b/src/EFCore/Storage/Json/JsonConvertedValueReaderWriter.cs index c873191db76..33c9e8162c8 100644 --- a/src/EFCore/Storage/Json/JsonConvertedValueReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonConvertedValueReaderWriter.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.CodeAnalysis; using System.Text.Json; using Microsoft.EntityFrameworkCore.Storage.Internal; @@ -45,4 +46,14 @@ JsonValueReaderWriter ICompositeJsonValueReaderWriter.InnerReaderWriter ValueConverter IJsonConvertedValueReaderWriter.Converter => _converter; + + private readonly ConstructorInfo _constructorInfo = typeof(JsonConvertedValueReaderWriter).GetConstructor([typeof(JsonValueReaderWriter), typeof(ValueConverter)])!; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => + Expression.New( + _constructorInfo, + ((ICompositeJsonValueReaderWriter)this).InnerReaderWriter.ConstructorExpression, + ((IJsonConvertedValueReaderWriter)this).Converter.ConstructorExpression); } diff --git a/src/EFCore/Storage/Json/JsonDateOnlyReaderWriter.cs b/src/EFCore/Storage/Json/JsonDateOnlyReaderWriter.cs index 27ad1daa60b..50913e2f77c 100644 --- a/src/EFCore/Storage/Json/JsonDateOnlyReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonDateOnlyReaderWriter.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.CodeAnalysis; using System.Globalization; using System.Text.Json; @@ -27,4 +28,10 @@ public override DateOnly FromJsonTyped(ref Utf8JsonReaderManager manager, object /// public override void ToJsonTyped(Utf8JsonWriter writer, DateOnly value) => writer.WriteStringValue(value.ToString("o", CultureInfo.InvariantCulture)); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonDateTimeOffsetReaderWriter.cs b/src/EFCore/Storage/Json/JsonDateTimeOffsetReaderWriter.cs index 07fb536df59..da0cf7289d7 100644 --- a/src/EFCore/Storage/Json/JsonDateTimeOffsetReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonDateTimeOffsetReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override DateTimeOffset FromJsonTyped(ref Utf8JsonReaderManager manager, /// public override void ToJsonTyped(Utf8JsonWriter writer, DateTimeOffset value) => writer.WriteStringValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonDateTimeReaderWriter.cs b/src/EFCore/Storage/Json/JsonDateTimeReaderWriter.cs index bb9b26fd833..92ee127fa22 100644 --- a/src/EFCore/Storage/Json/JsonDateTimeReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonDateTimeReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override DateTime FromJsonTyped(ref Utf8JsonReaderManager manager, object /// public override void ToJsonTyped(Utf8JsonWriter writer, DateTime value) => writer.WriteStringValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonDecimalReaderWriter.cs b/src/EFCore/Storage/Json/JsonDecimalReaderWriter.cs index ba55bd51cd5..454a194dbf7 100644 --- a/src/EFCore/Storage/Json/JsonDecimalReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonDecimalReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override decimal FromJsonTyped(ref Utf8JsonReaderManager manager, object? /// public override void ToJsonTyped(Utf8JsonWriter writer, decimal value) => writer.WriteNumberValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonDoubleReaderWriter.cs b/src/EFCore/Storage/Json/JsonDoubleReaderWriter.cs index f2a5f16aa4b..d6b1f029b2c 100644 --- a/src/EFCore/Storage/Json/JsonDoubleReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonDoubleReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override double FromJsonTyped(ref Utf8JsonReaderManager manager, object? /// public override void ToJsonTyped(Utf8JsonWriter writer, double value) => writer.WriteNumberValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonFloatReaderWriter.cs b/src/EFCore/Storage/Json/JsonFloatReaderWriter.cs index e918acad473..5a2cd662a8f 100644 --- a/src/EFCore/Storage/Json/JsonFloatReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonFloatReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override float FromJsonTyped(ref Utf8JsonReaderManager manager, object? e /// public override void ToJsonTyped(Utf8JsonWriter writer, float value) => writer.WriteNumberValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonGuidReaderWriter.cs b/src/EFCore/Storage/Json/JsonGuidReaderWriter.cs index ff6074dafbe..64d074617dd 100644 --- a/src/EFCore/Storage/Json/JsonGuidReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonGuidReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override Guid FromJsonTyped(ref Utf8JsonReaderManager manager, object? ex /// public override void ToJsonTyped(Utf8JsonWriter writer, Guid value) => writer.WriteStringValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonInt16ReaderWriter.cs b/src/EFCore/Storage/Json/JsonInt16ReaderWriter.cs index f2ed6cba6b9..bd8e772039d 100644 --- a/src/EFCore/Storage/Json/JsonInt16ReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonInt16ReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override short FromJsonTyped(ref Utf8JsonReaderManager manager, object? e /// public override void ToJsonTyped(Utf8JsonWriter writer, short value) => writer.WriteNumberValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonInt32ReaderWriter.cs b/src/EFCore/Storage/Json/JsonInt32ReaderWriter.cs index 6eeadbaf117..af9d186bd34 100644 --- a/src/EFCore/Storage/Json/JsonInt32ReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonInt32ReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override int FromJsonTyped(ref Utf8JsonReaderManager manager, object? exi /// public override void ToJsonTyped(Utf8JsonWriter writer, int value) => writer.WriteNumberValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonInt64ReaderWriter.cs b/src/EFCore/Storage/Json/JsonInt64ReaderWriter.cs index ec13f7268b9..d198d40b920 100644 --- a/src/EFCore/Storage/Json/JsonInt64ReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonInt64ReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override long FromJsonTyped(ref Utf8JsonReaderManager manager, object? ex /// public override void ToJsonTyped(Utf8JsonWriter writer, long value) => writer.WriteNumberValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonNullReaderWriter.cs b/src/EFCore/Storage/Json/JsonNullReaderWriter.cs index a31d78e941f..f6e7b5d2d0b 100644 --- a/src/EFCore/Storage/Json/JsonNullReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonNullReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ private JsonNullReaderWriter() /// public override void ToJsonTyped(Utf8JsonWriter writer, object? value) => writer.WriteNullValue(); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonSByteReaderWriter.cs b/src/EFCore/Storage/Json/JsonSByteReaderWriter.cs index ab41a52bff0..1a9fc9a58b0 100644 --- a/src/EFCore/Storage/Json/JsonSByteReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonSByteReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override sbyte FromJsonTyped(ref Utf8JsonReaderManager manager, object? e /// public override void ToJsonTyped(Utf8JsonWriter writer, sbyte value) => writer.WriteNumberValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonSignedEnumReaderWriter.cs b/src/EFCore/Storage/Json/JsonSignedEnumReaderWriter.cs index 4870dc7ae2f..6e36d8f8bf5 100644 --- a/src/EFCore/Storage/Json/JsonSignedEnumReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonSignedEnumReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -27,4 +28,10 @@ public override TEnum FromJsonTyped(ref Utf8JsonReaderManager manager, object? e /// public override void ToJsonTyped(Utf8JsonWriter writer, TEnum value) => writer.WriteNumberValue((long)Convert.ChangeType(value, typeof(long))!); + + private readonly Expression>> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonStringReaderWriter.cs b/src/EFCore/Storage/Json/JsonStringReaderWriter.cs index b8e97006b37..2215bb0c5b7 100644 --- a/src/EFCore/Storage/Json/JsonStringReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonStringReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override string FromJsonTyped(ref Utf8JsonReaderManager manager, object? /// public override void ToJsonTyped(Utf8JsonWriter writer, string value) => writer.WriteStringValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonTimeOnlyReaderWriter.cs b/src/EFCore/Storage/Json/JsonTimeOnlyReaderWriter.cs index c209c777bc7..895b26a9235 100644 --- a/src/EFCore/Storage/Json/JsonTimeOnlyReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonTimeOnlyReaderWriter.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.CodeAnalysis; using System.Globalization; using System.Text.Json; @@ -27,4 +28,10 @@ public override TimeOnly FromJsonTyped(ref Utf8JsonReaderManager manager, object /// public override void ToJsonTyped(Utf8JsonWriter writer, TimeOnly value) => writer.WriteStringValue(value.ToString("o", CultureInfo.InvariantCulture)); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonTimeSpanReaderWriter.cs b/src/EFCore/Storage/Json/JsonTimeSpanReaderWriter.cs index f91b7fa9f95..ded85bc5292 100644 --- a/src/EFCore/Storage/Json/JsonTimeSpanReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonTimeSpanReaderWriter.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.CodeAnalysis; using System.Globalization; using System.Text.Json; @@ -27,4 +28,10 @@ public override TimeSpan FromJsonTyped(ref Utf8JsonReaderManager manager, object /// public override void ToJsonTyped(Utf8JsonWriter writer, TimeSpan value) => writer.WriteStringValue(value.ToString("g", CultureInfo.InvariantCulture)); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonUInt16ReaderWriter.cs b/src/EFCore/Storage/Json/JsonUInt16ReaderWriter.cs index 647f9be7fd6..b09cd1bb1f2 100644 --- a/src/EFCore/Storage/Json/JsonUInt16ReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonUInt16ReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override ushort FromJsonTyped(ref Utf8JsonReaderManager manager, object? /// public override void ToJsonTyped(Utf8JsonWriter writer, ushort value) => writer.WriteNumberValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonUInt32ReaderWriter.cs b/src/EFCore/Storage/Json/JsonUInt32ReaderWriter.cs index 876d288ff26..57d1d10f22f 100644 --- a/src/EFCore/Storage/Json/JsonUInt32ReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonUInt32ReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override uint FromJsonTyped(ref Utf8JsonReaderManager manager, object? ex /// public override void ToJsonTyped(Utf8JsonWriter writer, uint value) => writer.WriteNumberValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonUInt64ReaderWriter.cs b/src/EFCore/Storage/Json/JsonUInt64ReaderWriter.cs index e2f8ab7df04..a6f35f5c26d 100644 --- a/src/EFCore/Storage/Json/JsonUInt64ReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonUInt64ReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -26,4 +27,10 @@ public override ulong FromJsonTyped(ref Utf8JsonReaderManager manager, object? e /// public override void ToJsonTyped(Utf8JsonWriter writer, ulong value) => writer.WriteNumberValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonUnsignedEnumReaderWriter.cs b/src/EFCore/Storage/Json/JsonUnsignedEnumReaderWriter.cs index a2ee0b607bf..68a3a0fc683 100644 --- a/src/EFCore/Storage/Json/JsonUnsignedEnumReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonUnsignedEnumReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -27,4 +28,10 @@ public override TEnum FromJsonTyped(ref Utf8JsonReaderManager manager, object? e /// public override void ToJsonTyped(Utf8JsonWriter writer, TEnum value) => writer.WriteNumberValue((ulong)Convert.ChangeType(value, typeof(ulong))!); + + private readonly Expression>> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/Json/JsonValueReaderWriter.cs b/src/EFCore/Storage/Json/JsonValueReaderWriter.cs index 0536aa6ae43..4e07ab8325b 100644 --- a/src/EFCore/Storage/Json/JsonValueReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonValueReaderWriter.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.CodeAnalysis; using System.Text; using System.Text.Json; @@ -125,4 +126,10 @@ public string ToJsonString(object value) return null; } + + /// + /// The expression representing construction of this object. + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public abstract Expression ConstructorExpression { get; } } diff --git a/src/EFCore/Storage/Json/JsonWarningEnumReaderWriter.cs b/src/EFCore/Storage/Json/JsonWarningEnumReaderWriter.cs index 00eac1fffbf..115bfdefabe 100644 --- a/src/EFCore/Storage/Json/JsonWarningEnumReaderWriter.cs +++ b/src/EFCore/Storage/Json/JsonWarningEnumReaderWriter.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.CodeAnalysis; using System.Text.Json; namespace Microsoft.EntityFrameworkCore.Storage.Json; @@ -72,4 +73,10 @@ public override void ToJsonTyped(Utf8JsonWriter writer, TEnum value) writer.WriteNumberValue((ulong)Convert.ChangeType(value, typeof(ulong))); } } + + private readonly Expression>> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } diff --git a/src/EFCore/Storage/ValueConversion/BoolToStringConverter.cs b/src/EFCore/Storage/ValueConversion/BoolToStringConverter.cs index 9e2ff28a26e..ef3539e1f8f 100644 --- a/src/EFCore/Storage/ValueConversion/BoolToStringConverter.cs +++ b/src/EFCore/Storage/ValueConversion/BoolToStringConverter.cs @@ -1,6 +1,8 @@ // Licensed to the .NET Foundation under one or more agreements. // The .NET Foundation licenses this file to you under the MIT license. +using static System.Linq.Expressions.Expression; + namespace Microsoft.EntityFrameworkCore.Storage.ValueConversion; /// @@ -11,6 +13,10 @@ namespace Microsoft.EntityFrameworkCore.Storage.ValueConversion; /// public class BoolToStringConverter : BoolToTwoValuesConverter { + private static readonly MethodInfo StringToUpperInvariantMethod = typeof(string).GetMethod(nameof(string.ToUpperInvariant))!; + private static readonly MethodInfo StringCharsMethod = typeof(string).GetMethod("get_Chars", [typeof(int)])!; + private static readonly MethodInfo StringIsNullOrEmpty = typeof(string).GetMethod(nameof(string.IsNullOrEmpty), [typeof(string)])!; + /// /// Creates a new instance of this converter. A case-insensitive first character test is used /// when converting from the store. @@ -48,9 +54,33 @@ public BoolToStringConverter( private static Expression> FromProvider(string trueValue) { - var testChar = trueValue.ToUpperInvariant()[0]; + // v => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)trueValue.ToUpperInvariant()[0]; + var prm = Parameter(typeof(string), "v"); + var result = Lambda>( + AndAlso( + Not( + Call( + StringIsNullOrEmpty, + prm)), + Equal( + Convert( + Call( + Call( + prm, + StringToUpperInvariantMethod), + StringCharsMethod, + Constant(0)), + typeof(int)), + Convert( + Call( + Call( + Constant(trueValue), + StringToUpperInvariantMethod), + StringCharsMethod, + Constant(0)), + typeof(int)))), + prm); - return v => !string.IsNullOrEmpty(v) - && v.ToUpperInvariant()[0] == testChar; + return result; } } diff --git a/src/EFCore/Storage/ValueConversion/CollectionToJsonStringConverter.cs b/src/EFCore/Storage/ValueConversion/CollectionToJsonStringConverter.cs index 7742f1aa29e..a9af710ee78 100644 --- a/src/EFCore/Storage/ValueConversion/CollectionToJsonStringConverter.cs +++ b/src/EFCore/Storage/ValueConversion/CollectionToJsonStringConverter.cs @@ -2,6 +2,7 @@ // The .NET Foundation licenses this file to you under the MIT license. using Microsoft.EntityFrameworkCore.Storage.Json; +using static System.Linq.Expressions.Expression; namespace Microsoft.EntityFrameworkCore.Storage.ValueConversion; @@ -10,6 +11,12 @@ namespace Microsoft.EntityFrameworkCore.Storage.ValueConversion; /// public class CollectionToJsonStringConverter : ValueConverter, string> { + private static readonly MethodInfo ToJsonStringMethod + = typeof(JsonValueReaderWriter).GetMethod(nameof(JsonValueReaderWriter.ToJsonString), [typeof(object)])!; + + private static readonly MethodInfo FromJsonStringMethod + = typeof(JsonValueReaderWriter).GetMethod(nameof(JsonValueReaderWriter.FromJsonString), [typeof(string), typeof(object)])!; + /// /// Creates a new instance of this converter. /// @@ -19,12 +26,39 @@ public class CollectionToJsonStringConverter : ValueConverterThe reader/writer to use. public CollectionToJsonStringConverter(JsonValueReaderWriter collectionJsonReaderWriter) : base( - v => collectionJsonReaderWriter.ToJsonString(v), - v => (IEnumerable)collectionJsonReaderWriter.FromJsonString(v, null)) + ToJsonString(collectionJsonReaderWriter), + FromJsonString(collectionJsonReaderWriter)) { JsonReaderWriter = collectionJsonReaderWriter; } + private static Expression, string>> ToJsonString(JsonValueReaderWriter collectionJsonReaderWriter) + { + var prm = Parameter(typeof(IEnumerable), "v"); + + return Lambda, string>>( + Call( + collectionJsonReaderWriter.ConstructorExpression, + ToJsonStringMethod, + prm), + prm); + } + + private static Expression>> FromJsonString(JsonValueReaderWriter collectionJsonReaderWriter) + { + var prm = Parameter(typeof(string), "v"); + + return Lambda>>( + Convert( + Call( + collectionJsonReaderWriter.ConstructorExpression, + FromJsonStringMethod, + prm, + Constant(null, typeof(object))), + typeof(IEnumerable)), + prm); + } + /// /// The reader/writer to use. /// diff --git a/src/EFCore/Storage/ValueConversion/Internal/StringNumberConverter.cs b/src/EFCore/Storage/ValueConversion/Internal/StringNumberConverter.cs index 9502dae9666..30734b58bd3 100644 --- a/src/EFCore/Storage/ValueConversion/Internal/StringNumberConverter.cs +++ b/src/EFCore/Storage/ValueConversion/Internal/StringNumberConverter.cs @@ -13,6 +13,9 @@ namespace Microsoft.EntityFrameworkCore.Storage.ValueConversion.Internal; /// public class StringNumberConverter : ValueConverter { + private static readonly Expression> _cultureInfoInvariantCultureLambda = + () => CultureInfo.InvariantCulture; + /// /// This is an internal API that supports the Entity Framework Core infrastructure and not subject to /// the same compatibility standards as public APIs. It may be changed or removed without notice in @@ -63,7 +66,7 @@ protected static Expression> ToNumber() parseMethod, param, Expression.Constant(NumberStyles.Any), - Expression.Constant(CultureInfo.InvariantCulture, typeof(IFormatProvider))); + _cultureInfoInvariantCultureLambda.Body); if (typeof(TNumber).IsNullableType()) { @@ -101,7 +104,7 @@ protected static Expression> ToNumber() Expression expression = Expression.Call( formatMethod, - Expression.Constant(CultureInfo.InvariantCulture), + _cultureInfoInvariantCultureLambda.Body, Expression.Constant(type == typeof(float) || type == typeof(double) ? "{0:R}" : "{0}"), Expression.Convert(param, typeof(object))); diff --git a/src/EFCore/Storage/ValueConversion/StringToBytesConverter.cs b/src/EFCore/Storage/ValueConversion/StringToBytesConverter.cs index 4fedab8c122..63483ced254 100644 --- a/src/EFCore/Storage/ValueConversion/StringToBytesConverter.cs +++ b/src/EFCore/Storage/ValueConversion/StringToBytesConverter.cs @@ -13,6 +13,10 @@ namespace Microsoft.EntityFrameworkCore.Storage.ValueConversion; /// public class StringToBytesConverter : ValueConverter { + private static readonly MethodInfo EncodingGetBytesMethodInfo = typeof(Encoding).GetMethod(nameof(Encoding.GetBytes), [typeof(string)])!; + private static readonly MethodInfo EncodingGetStringMethodInfo = typeof(Encoding).GetMethod(nameof(Encoding.GetString), [typeof(byte[])])!; + private static readonly MethodInfo EncodingGetEncodingMethodInfo = typeof(Encoding).GetMethod(nameof(Encoding.GetEncoding), [typeof(int)])!; + /// /// Creates a new instance of this converter. /// @@ -28,8 +32,8 @@ public StringToBytesConverter( Encoding encoding, ConverterMappingHints? mappingHints = null) : base( - v => encoding.GetBytes(v!), - v => encoding.GetString(v!), + FromProvider(encoding), + ToProvider(encoding), mappingHints) { } @@ -39,4 +43,34 @@ public StringToBytesConverter( /// public static ValueConverterInfo DefaultInfo { get; } = new(typeof(string), typeof(byte[]), i => new StringToBytesConverter(Encoding.UTF8, i.MappingHints)); + + private static Expression> FromProvider(Encoding encoding) + { + // v => encoding.GetBytes(v!), + var prm = Expression.Parameter(typeof(string), "v"); + var result = Expression.Lambda>( + Expression.Call( + Expression.Call( + EncodingGetEncodingMethodInfo, + Expression.Constant(encoding.CodePage)), + EncodingGetBytesMethodInfo, prm), + prm); + + return result; + } + + private static Expression> ToProvider(Encoding encoding) + { + // v => encoding.GetString(v!) + var prm = Expression.Parameter(typeof(byte[]), "v"); + var result = Expression.Lambda>( + Expression.Call( + Expression.Call( + EncodingGetEncodingMethodInfo, + Expression.Constant(encoding.CodePage)), + EncodingGetStringMethodInfo, prm), + prm); + + return result; + } } diff --git a/src/EFCore/Storage/ValueConversion/ValueConverter.cs b/src/EFCore/Storage/ValueConversion/ValueConverter.cs index b7c0a00b058..bdef1e1a1d9 100644 --- a/src/EFCore/Storage/ValueConversion/ValueConverter.cs +++ b/src/EFCore/Storage/ValueConversion/ValueConverter.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.CodeAnalysis; using System.Runtime.CompilerServices; using Microsoft.EntityFrameworkCore.Storage.ValueConversion.Internal; @@ -251,4 +252,10 @@ var firstConverter ? firstConverter.MappingHints : secondConverter.MappingHints.With(firstConverter.MappingHints))!; } + + /// + /// The expression representing construction of this object. + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public abstract Expression ConstructorExpression { get; } } diff --git a/src/EFCore/Storage/ValueConversion/ValueConverter`.cs b/src/EFCore/Storage/ValueConversion/ValueConverter`.cs index 0bf8907b7e2..f202d63d929 100644 --- a/src/EFCore/Storage/ValueConversion/ValueConverter`.cs +++ b/src/EFCore/Storage/ValueConversion/ValueConverter`.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.CodeAnalysis; using Microsoft.EntityFrameworkCore.Internal; namespace Microsoft.EntityFrameworkCore.Storage.ValueConversion; @@ -18,6 +19,8 @@ public class ValueConverter : ValueConverter private Func? _convertFromProvider; private Func? _convertToProviderTyped; private Func? _convertFromProviderTyped; + private static readonly ConstructorInfo MappingHintsCtor + = typeof(ConverterMappingHints).GetConstructor([typeof(int?), typeof(int?), typeof(int?), typeof(bool?), typeof(Func)])!; /// /// Initializes a new instance of the class. @@ -172,4 +175,29 @@ public override Type ModelClrType /// public override Type ProviderClrType => typeof(TProvider); + + private readonly ConstructorInfo _constructorInfo = typeof(ValueConverter).GetConstructor( + [ + typeof(Expression>), + typeof(Expression>), + typeof(ConverterMappingHints) + ])!; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => + Expression.New( + _constructorInfo, + ConvertToProviderExpression, + ConvertFromProviderExpression, + MappingHints != null + ? Expression.New( + MappingHintsCtor, + Expression.Constant(MappingHints.Size, typeof(int?)), + Expression.Constant(MappingHints.Precision, typeof(int?)), + Expression.Constant(MappingHints.Scale, typeof(int?)), + Expression.Constant(MappingHints.IsUnicode, typeof(bool?)), + // valueGeneratorFactory is difficult to build using Expression trees and is obsolete + Expression.Default(typeof(Func))) + : Expression.Default(typeof(ConverterMappingHints))); } diff --git a/src/Shared/ExpressionExtensions.cs b/src/Shared/ExpressionExtensions.cs index 0835bd7ad95..e14bcf3099c 100644 --- a/src/Shared/ExpressionExtensions.cs +++ b/src/Shared/ExpressionExtensions.cs @@ -4,6 +4,7 @@ #nullable enable using System.Diagnostics.CodeAnalysis; +using Microsoft.EntityFrameworkCore.Query; // ReSharper disable once CheckNamespace namespace System.Linq.Expressions; @@ -45,7 +46,30 @@ private static Expression RemoveConvert(Expression expression) : expression; public static T GetConstantValue(this Expression expression) - => expression is ConstantExpression constantExpression - ? (T)constantExpression.Value! - : throw new InvalidOperationException(); + => expression switch + { + ConstantExpression constantExpression => (T)constantExpression.Value!, +#pragma warning disable EF9100 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. + LiftableConstantExpression liftableConstantExpression => (T)liftableConstantExpression.OriginalExpression.Value!, +#pragma warning restore EF9100 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. + _ => throw new InvalidOperationException() + }; + + public static bool TryGetNonNullConstantValue(this Expression expression, [NotNullWhen(true)][MaybeNullWhen(false)]out T value) + { + switch (expression) + { + case ConstantExpression constant when constant.Value is T typedValue: + value = typedValue; + return true; +#pragma warning disable EF9100 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. + case LiftableConstantExpression liftableConstant when liftableConstant.OriginalExpression.Value is T typedValue: +#pragma warning restore EF9100 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. + value = typedValue; + return true; + default: + value = default; + return false; + } + } } diff --git a/test/EFCore.Relational.Specification.Tests/Query/AdHocAdvancedMappingsQueryRelationalTestBase.cs b/test/EFCore.Relational.Specification.Tests/Query/AdHocAdvancedMappingsQueryRelationalTestBase.cs index 79a4d7cf64c..e33ea578f93 100644 --- a/test/EFCore.Relational.Specification.Tests/Query/AdHocAdvancedMappingsQueryRelationalTestBase.cs +++ b/test/EFCore.Relational.Specification.Tests/Query/AdHocAdvancedMappingsQueryRelationalTestBase.cs @@ -71,7 +71,8 @@ public virtual async Task Projecting_one_of_two_similar_complex_types_picks_the_ .Select( x => new { - x.B.A.Id, x.B.Info.Created, + x.B.A.Id, + x.B.Info.Created, }).ToList(); Assert.Equal(new DateTime(2000, 1, 1), query[0].Created); diff --git a/test/EFCore.Relational.Specification.Tests/Query/AdHocJsonQueryTestBase.cs b/test/EFCore.Relational.Specification.Tests/Query/AdHocJsonQueryTestBase.cs index b51ac0f8a32..d0467f17ecc 100644 --- a/test/EFCore.Relational.Specification.Tests/Query/AdHocJsonQueryTestBase.cs +++ b/test/EFCore.Relational.Specification.Tests/Query/AdHocJsonQueryTestBase.cs @@ -275,7 +275,8 @@ protected Task Seed30028(Context32939 ctx) { var entity = new Context32939.Entity32939 { - Empty = new Context32939.JsonEmpty32939(), FieldOnly = new Context32939.JsonFieldOnly32939() + Empty = new Context32939.JsonEmpty32939(), + FieldOnly = new Context32939.JsonFieldOnly32939() }; ctx.Entities.Add(entity); diff --git a/test/EFCore.Relational.Specification.Tests/Query/AdHocQuerySplittingQueryTestBase.cs b/test/EFCore.Relational.Specification.Tests/Query/AdHocQuerySplittingQueryTestBase.cs index 06c6fa948b5..db6a201e52f 100644 --- a/test/EFCore.Relational.Specification.Tests/Query/AdHocQuerySplittingQueryTestBase.cs +++ b/test/EFCore.Relational.Specification.Tests/Query/AdHocQuerySplittingQueryTestBase.cs @@ -249,7 +249,8 @@ protected virtual Task CreateTestStore25225() .Select( c => new Context25225.CollectionViewModel { - Id = c.Id, ParentId = c.ParentId, + Id = c.Id, + ParentId = c.ParentId, }) .ToArray() }); diff --git a/test/EFCore.Relational.Specification.Tests/Query/JsonQueryTestBase.cs b/test/EFCore.Relational.Specification.Tests/Query/JsonQueryTestBase.cs index 0015346d141..55c932ae72d 100644 --- a/test/EFCore.Relational.Specification.Tests/Query/JsonQueryTestBase.cs +++ b/test/EFCore.Relational.Specification.Tests/Query/JsonQueryTestBase.cs @@ -1311,7 +1311,7 @@ public virtual Task Json_collection_skip_take_in_projection_with_json_reference_ assertOrder: true, elementAsserter: (e, a) => AssertCollection(e, a, ordered: true)); - [ConditionalTheory] + [ConditionalTheory(Skip = "AOT: list comparer captures element comparer, see issue #33383")] [MemberData(nameof(IsAsyncData))] public virtual Task Json_collection_distinct_in_projection(bool async) => AssertQuery( @@ -1349,7 +1349,7 @@ public virtual Task Json_collection_leaf_filter_in_projection(bool async) assertOrder: true, elementAsserter: (e, a) => AssertCollection(e, a, ordered: true)); - [ConditionalTheory] + [ConditionalTheory(Skip = "AOT: list comparer captures element comparer, see issue #33383")] [MemberData(nameof(IsAsyncData))] public virtual Task Json_multiple_collection_projections(bool async) => AssertQuery( @@ -1376,7 +1376,7 @@ public virtual Task Json_multiple_collection_projections(bool async) AssertCollection(e.Fourth, a.Fourth); }); - [ConditionalTheory] + [ConditionalTheory(Skip = "AOT: list comparer captures element comparer, see issue #33383")] [MemberData(nameof(IsAsyncData))] public virtual Task Json_branch_collection_distinct_and_other_collection(bool async) => AssertQuery( diff --git a/test/EFCore.Relational.Specification.Tests/Query/OwnedEntityQueryRelationalTestBase.cs b/test/EFCore.Relational.Specification.Tests/Query/OwnedEntityQueryRelationalTestBase.cs index 55789c12376..12f1b63c13d 100644 --- a/test/EFCore.Relational.Specification.Tests/Query/OwnedEntityQueryRelationalTestBase.cs +++ b/test/EFCore.Relational.Specification.Tests/Query/OwnedEntityQueryRelationalTestBase.cs @@ -518,13 +518,15 @@ public Task SeedAsync() Add( new Monarch { - Name = "His August Majesty Guslav the Fifth", RulerOf = "The Union", + Name = "His August Majesty Guslav the Fifth", + RulerOf = "The Union", }); Add( new Monarch { - Name = "Emperor Uthman-ul-Dosht", RulerOf = "The Gurkish Empire", + Name = "Emperor Uthman-ul-Dosht", + RulerOf = "The Gurkish Empire", }); Add( diff --git a/test/EFCore.Relational.Specification.Tests/Query/UdfDbFunctionTestBase.cs b/test/EFCore.Relational.Specification.Tests/Query/UdfDbFunctionTestBase.cs index c43c07a2258..fdb44739e2d 100644 --- a/test/EFCore.Relational.Specification.Tests/Query/UdfDbFunctionTestBase.cs +++ b/test/EFCore.Relational.Specification.Tests/Query/UdfDbFunctionTestBase.cs @@ -1158,7 +1158,8 @@ public virtual void Scalar_Function_Anonymous_Type_Select_Nested_Instance() where c.Id == customerId select new { - c.LastName, OrderCount = context.StarValueInstance(starCount, context.CustomerOrderCountInstance(customerId)) + c.LastName, + OrderCount = context.StarValueInstance(starCount, context.CustomerOrderCountInstance(customerId)) }).Single(); Assert.Equal("Three", cust.LastName); @@ -1592,7 +1593,8 @@ public virtual void QF_Select_Direct_In_Anonymous() () => (from c in context.Customers select new { - c.Id, Prods = context.GetTopTwoSellingProducts().ToList(), + c.Id, + Prods = context.GetTopTwoSellingProducts().ToList(), }).ToList()).Message; Assert.Equal(RelationalStrings.InsufficientInformationToIdentifyElementOfCollectionJoin, message); @@ -1607,7 +1609,8 @@ public virtual void QF_Select_Direct_In_Anonymous_distinct() var query = (from c in context.Customers select new { - c.Id, Prods = context.GetTopTwoSellingProducts().Distinct().ToList(), + c.Id, + Prods = context.GetTopTwoSellingProducts().Distinct().ToList(), }).ToList(); } } @@ -1734,7 +1737,8 @@ public virtual void QF_Select_NonCorrelated_Subquery_In_Anonymous() () => (from c in context.Customers select new { - c.Id, Prods = context.GetTopTwoSellingProducts().Select(p => p.ProductId).ToList(), + c.Id, + Prods = context.GetTopTwoSellingProducts().Select(p => p.ProductId).ToList(), }).ToList()).Message; Assert.Equal(RelationalStrings.InsufficientInformationToIdentifyElementOfCollectionJoin, message); diff --git a/test/EFCore.Relational.Tests/Query/Internal/BufferedDataReaderTest.cs b/test/EFCore.Relational.Tests/Query/Internal/BufferedDataReaderTest.cs index 85cebb23e81..887f9899e97 100644 --- a/test/EFCore.Relational.Tests/Query/Internal/BufferedDataReaderTest.cs +++ b/test/EFCore.Relational.Tests/Query/Internal/BufferedDataReaderTest.cs @@ -177,9 +177,16 @@ private async Task Verify_method_result( columnType = typeof(object); } + var getFieldValueMethod = typeof(DbDataReader).GetMethod(nameof(DbDataReader.GetFieldValue)).MakeGenericMethod(columnType); + var prm = Expression.Parameter(typeof(DbDataReader), "r"); + var getFieldValueLambda = Expression.Lambda( + Expression.Call(prm, getFieldValueMethod, Expression.Constant(0)), + prm, + Expression.Parameter(typeof(int[]), "_")); + var columns = new[] { - ReaderColumn.Create(columnType, true, null, null, (Func)((r, _) => r.GetFieldValue(0))) + ReaderColumn.Create(columnType, true, null, null, getFieldValueLambda) }; var bufferedReader = new BufferedDataReader(reader, false); diff --git a/test/EFCore.Relational.Tests/RelationalApiConsistencyTest.cs b/test/EFCore.Relational.Tests/RelationalApiConsistencyTest.cs index b8373ba5df8..f54a1e559f2 100644 --- a/test/EFCore.Relational.Tests/RelationalApiConsistencyTest.cs +++ b/test/EFCore.Relational.Tests/RelationalApiConsistencyTest.cs @@ -3,6 +3,7 @@ using Microsoft.EntityFrameworkCore.Diagnostics.Internal; using Microsoft.EntityFrameworkCore.Migrations.Operations.Builders; +using static Microsoft.EntityFrameworkCore.Query.RelationalShapedQueryCompilingExpressionVisitor; namespace Microsoft.EntityFrameworkCore; @@ -552,7 +553,13 @@ public class RelationalApiConsistencyFixture : ApiConsistencyFixtureBase typeof(RelationalConnectionDiagnosticsLogger).GetMethod( nameof(IRelationalConnectionDiagnosticsLogger.ConnectionDisposingAsync)), typeof(RelationalConnectionDiagnosticsLogger).GetMethod( - nameof(IRelationalConnectionDiagnosticsLogger.ConnectionDisposedAsync)) + nameof(IRelationalConnectionDiagnosticsLogger.ConnectionDisposedAsync)), + + // internal methods made public for AOT + typeof(ShaperProcessingExpressionVisitor).GetMethod(nameof(ShaperProcessingExpressionVisitor.PopulateSplitIncludeCollectionAsync)), + typeof(ShaperProcessingExpressionVisitor).GetMethod(nameof(ShaperProcessingExpressionVisitor.PopulateSplitCollectionAsync)), + typeof(ShaperProcessingExpressionVisitor).GetMethod(nameof(ShaperProcessingExpressionVisitor.TaskAwaiter)), + typeof(RelationalShapedQueryCompilingExpressionVisitor).GetMethod(nameof(RelationalShapedQueryCompilingExpressionVisitor.NonQueryResultAsync)), ]; public override HashSet MetadataMethodExceptions { get; } = @@ -561,6 +568,17 @@ public class RelationalApiConsistencyFixture : ApiConsistencyFixtureBase typeof(IMutableStoredProcedure).GetMethod(nameof(IMutableStoredProcedure.AddResultColumn)) ]; + public override HashSet VirtualMethodExceptions { get; } = + [ + // non-sealed record +#pragma warning disable EF9100 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. + typeof(RelationalMaterializerLiftableConstantContext).GetMethod("get_RelationalDependencies"), + typeof(RelationalMaterializerLiftableConstantContext).GetMethod("set_RelationalDependencies"), + typeof(RelationalMaterializerLiftableConstantContext).GetMethod("Deconstruct", [typeof(ShapedQueryCompilingExpressionVisitorDependencies).MakeByRefType()]), + typeof(RelationalMaterializerLiftableConstantContext).GetMethod("Deconstruct", [typeof(ShapedQueryCompilingExpressionVisitorDependencies).MakeByRefType(), typeof(RelationalShapedQueryCompilingExpressionVisitorDependencies).MakeByRefType()]), +#pragma warning restore EF9100 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. + ]; + public List> RelationalMetadataMethods { get; } = []; protected override void Initialize() diff --git a/test/EFCore.Specification.Tests/ApiConsistencyTestBase.cs b/test/EFCore.Specification.Tests/ApiConsistencyTestBase.cs index 57d688e5c4b..3ad07fbeab3 100644 --- a/test/EFCore.Specification.Tests/ApiConsistencyTestBase.cs +++ b/test/EFCore.Specification.Tests/ApiConsistencyTestBase.cs @@ -1064,6 +1064,7 @@ where type.IsVisible from method in type.GetMethods(AnyInstance) where method.DeclaringType == type && !Fixture.NonVirtualMethods.Contains(method) + && !Fixture.VirtualMethodExceptions.Contains(method) && !method.IsVirtual && !method.Name.StartsWith("add_", StringComparison.Ordinal) && !method.Name.StartsWith("remove_", StringComparison.Ordinal) @@ -1275,6 +1276,15 @@ protected ApiConsistencyFixtureBase() public virtual Dictionary> UnmatchedMirrorMethods { get; } = new(); public virtual Dictionary MetadataMethodNameTransformers { get; } = new(); public virtual HashSet MetadataMethodExceptions { get; } = []; + public virtual HashSet VirtualMethodExceptions { get; } = + [ + // un-sealed record +#pragma warning disable EF9100 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. + typeof(MaterializerLiftableConstantContext).GetMethod("get_Dependencies"), + typeof(MaterializerLiftableConstantContext).GetMethod("set_Dependencies"), + typeof(MaterializerLiftableConstantContext).GetMethod("Deconstruct"), +#pragma warning restore EF9100 // Type is for evaluation purposes only and is subject to change or removal in future updates. Suppress this diagnostic to proceed. + ]; public virtual HashSet ComputedDependencyProperties { get; } = [ diff --git a/test/EFCore.Specification.Tests/JsonTypesTestBase.cs b/test/EFCore.Specification.Tests/JsonTypesTestBase.cs index 7b29f158c75..5a79e1a355e 100644 --- a/test/EFCore.Specification.Tests/JsonTypesTestBase.cs +++ b/test/EFCore.Specification.Tests/JsonTypesTestBase.cs @@ -3,6 +3,7 @@ using System.Collections; using System.Collections.ObjectModel; +using System.Diagnostics.CodeAnalysis; using System.Globalization; using System.Net; using System.Net.NetworkInformation; @@ -4242,6 +4243,12 @@ public override void ToJsonTyped(Utf8JsonWriter writer, Geometry value) serializer.Serialize(jsonWriter, value); writer.WriteRawValue(stringWriter.ToString()); } + + private readonly Expression> _instanceLambda = () => Instance; + + /// + [Experimental(EFDiagnostics.PrecompiledQueryExperimental)] + public override Expression ConstructorExpression => _instanceLambda.Body; } private readonly NullabilityInfoContext _nullabilityInfoContext = new(); diff --git a/test/EFCore.Specification.Tests/Query/AdHocMiscellaneousQueryTestBase.cs b/test/EFCore.Specification.Tests/Query/AdHocMiscellaneousQueryTestBase.cs index 22223eb470e..02e48fa62f5 100644 --- a/test/EFCore.Specification.Tests/Query/AdHocMiscellaneousQueryTestBase.cs +++ b/test/EFCore.Specification.Tests/Query/AdHocMiscellaneousQueryTestBase.cs @@ -238,15 +238,18 @@ public async Task SeedAsync() Contacts.AddRange( new ServiceOperatorContact { - UserName = "service.operator@esoterix.co.uk", ServiceOperator = ServiceOperators.OrderBy(o => o.Id).First() + UserName = "service.operator@esoterix.co.uk", + ServiceOperator = ServiceOperators.OrderBy(o => o.Id).First() }, new EmployerContact { - UserName = "uwe@esoterix.co.uk", Employer = Employers.OrderBy(e => e.Id).First(e => e.Name == "UWE") + UserName = "uwe@esoterix.co.uk", + Employer = Employers.OrderBy(e => e.Id).First(e => e.Name == "UWE") }, new EmployerContact { - UserName = "hp@esoterix.co.uk", Employer = Employers.OrderBy(e => e.Id).First(e => e.Name == "Hewlett Packard") + UserName = "hp@esoterix.co.uk", + Employer = Employers.OrderBy(e => e.Id).First(e => e.Name == "Hewlett Packard") }, new Contact { UserName = "noroles@esoterix.co.uk" }); @@ -1637,7 +1640,8 @@ public virtual async Task GroupBy_aggregate_on_right_side_of_join(bool async) o => o.OrderId, (o, g) => new { - Key = o, IsPending = g.Max(y => y.ShippingDate == null && y.CancellationDate == null ? o : (o - 10000000)) + Key = o, + IsPending = g.Max(y => y.ShippingDate == null && y.CancellationDate == null ? o : (o - 10000000)) }) .OrderBy(e => e.Key); @@ -1896,7 +1900,8 @@ group t.Id by t.Value into tg select new { - A = tg.Key, B = context.Tables.Where(t => t.Value == tg.Max() * 6).Max(t => (int?)t.Id), + A = tg.Key, + B = context.Tables.Where(t => t.Value == tg.Max() * 6).Max(t => (int?)t.Id), }; var orders = async @@ -2162,7 +2167,8 @@ public virtual async Task Filter_on_nested_DTO_with_interface_gets_simplified_co CountryId = m.Company.CountryId, Country = new Context31961.CountryDto() { - Id = m.Company.Country.Id, CountryName = m.Company.Country.CountryName, + Id = m.Company.Country.Id, + CountryName = m.Company.Country.CountryName, }, } : null, diff --git a/test/EFCore.Specification.Tests/Query/ComplexNavigationsQueryTestBase.cs b/test/EFCore.Specification.Tests/Query/ComplexNavigationsQueryTestBase.cs index c7deae9cbcc..5a23a69a79f 100644 --- a/test/EFCore.Specification.Tests/Query/ComplexNavigationsQueryTestBase.cs +++ b/test/EFCore.Specification.Tests/Query/ComplexNavigationsQueryTestBase.cs @@ -2242,7 +2242,8 @@ join l2 in ss.Set() } equals new { - A = EF.Property(l2, "Level1_Optional_Id"), B = EF.Property(l2, "OneToMany_Optional_Self_Inverse2Id") + A = EF.Property(l2, "Level1_Optional_Id"), + B = EF.Property(l2, "OneToMany_Optional_Self_Inverse2Id") } select l1); @@ -3300,7 +3301,8 @@ public virtual Task Member_over_null_check_ternary_and_nested_dto_type(bool asyn ? null : new Level2Dto { - Id = l1.OneToOne_Optional_FK1.Id, Name = l1.OneToOne_Optional_FK1.Name, + Id = l1.OneToOne_Optional_FK1.Id, + Name = l1.OneToOne_Optional_FK1.Name, } }) .OrderBy(e => e.Level2.Name) @@ -3458,7 +3460,8 @@ public virtual Task Composite_key_join_on_groupby_aggregate_projecting_only_grou o => new { o.Id, Condition = true }, i => new { - Id = i.Key, Condition = i.Sum > 10, + Id = i.Key, + Condition = i.Sum > 10, }, (o, i) => i.Key)); @@ -3763,7 +3766,8 @@ public virtual Task Project_shadow_properties8(bool async) ss => from x in ss.Set() select new { - x.Id, InheritanceLeaf2Id = EF.Property(x, "InheritanceLeaf2Id"), + x.Id, + InheritanceLeaf2Id = EF.Property(x, "InheritanceLeaf2Id"), }, elementSorter: e => e.Id); diff --git a/test/EFCore.Specification.Tests/Query/GearsOfWarQueryTestBase.cs b/test/EFCore.Specification.Tests/Query/GearsOfWarQueryTestBase.cs index fea5dc51208..493b6b4d3b2 100644 --- a/test/EFCore.Specification.Tests/Query/GearsOfWarQueryTestBase.cs +++ b/test/EFCore.Specification.Tests/Query/GearsOfWarQueryTestBase.cs @@ -653,7 +653,8 @@ public virtual Task Select_enum_has_flag(bool async) .Select( b => new { - hasFlagTrue = b.Rank.HasFlag(MilitaryRank.Corporal), hasFlagFalse = b.Rank.HasFlag(MilitaryRank.Sergeant) + hasFlagTrue = b.Rank.HasFlag(MilitaryRank.Corporal), + hasFlagFalse = b.Rank.HasFlag(MilitaryRank.Sergeant) })); [ConditionalTheory] @@ -1097,11 +1098,11 @@ public virtual Task Where_compare_anonymous_types(bool async) ss => from g in ss.Set() from o in ss.Set().OfType() where new - { - Name = g.LeaderNickname, - Squad = g.LeaderSquadId, - Five = 5 - } + { + Name = g.LeaderNickname, + Squad = g.LeaderSquadId, + Five = 5 + } == new { Name = o.Nickname, @@ -2807,7 +2808,7 @@ public virtual Task Comparing_two_collection_navigations_composite_key(bool asyn async, ss => from g1 in ss.Set() from g2 in ss.Set() - // ReSharper disable once PossibleUnintendedReferenceComparison + // ReSharper disable once PossibleUnintendedReferenceComparison where g1.Weapons == g2.Weapons orderby g1.Nickname select new { Nickname1 = g1.Nickname, Nickname2 = g2.Nickname }, @@ -6223,7 +6224,7 @@ public virtual Task Byte_array_filter_by_length_literal(bool async) [MemberData(nameof(IsAsyncData))] public virtual Task Byte_array_filter_by_length_parameter(bool async) { - var someByteArr = new[] { (byte)42, (byte)24}; + var someByteArr = new[] { (byte)42, (byte)24 }; return AssertQuery( async, ss => ss.Set().Where(w => w.Banner.Length == someByteArr.Length), diff --git a/test/EFCore.Specification.Tests/Query/NorthwindGroupByQueryTestBase.cs b/test/EFCore.Specification.Tests/Query/NorthwindGroupByQueryTestBase.cs index a19ff9c3cd8..345bdade48c 100644 --- a/test/EFCore.Specification.Tests/Query/NorthwindGroupByQueryTestBase.cs +++ b/test/EFCore.Specification.Tests/Query/NorthwindGroupByQueryTestBase.cs @@ -2548,7 +2548,8 @@ public virtual Task GroupBy_group_Distinct_Select_Distinct_aggregate(bool async) g => new { - g.Key, Max = g.Distinct().Select(e => e.OrderDate).Distinct().Max(), + g.Key, + Max = g.Distinct().Select(e => e.OrderDate).Distinct().Max(), }), elementSorter: e => e.Key); @@ -2563,7 +2564,8 @@ public virtual Task GroupBy_group_Where_Select_Distinct_aggregate(bool async) g => new { - g.Key, Max = g.Where(e => e.OrderDate.HasValue).Select(e => e.OrderDate).Distinct().Max(), + g.Key, + Max = g.Where(e => e.OrderDate.HasValue).Select(e => e.OrderDate).Distinct().Max(), }), elementSorter: e => e.Key); diff --git a/test/EFCore.Specification.Tests/Query/NorthwindMiscellaneousQueryTestBase.cs b/test/EFCore.Specification.Tests/Query/NorthwindMiscellaneousQueryTestBase.cs index a100eae5b3a..2afa772e4e9 100644 --- a/test/EFCore.Specification.Tests/Query/NorthwindMiscellaneousQueryTestBase.cs +++ b/test/EFCore.Specification.Tests/Query/NorthwindMiscellaneousQueryTestBase.cs @@ -723,7 +723,8 @@ public virtual Task Ternary_should_not_evaluate_both_sides_with_parameter(bool a o => new { // ReSharper disable SimplifyConditionalTernaryExpression - Data1 = param != null ? o.OrderDate == param.Value : true, Data2 = param == null ? true : o.OrderDate == param.Value + Data1 = param != null ? o.OrderDate == param.Value : true, + Data2 = param == null ? true : o.OrderDate == param.Value // ReSharper restore SimplifyConditionalTernaryExpression })); } diff --git a/test/EFCore.Specification.Tests/Query/OwnedEntityQueryTestBase.cs b/test/EFCore.Specification.Tests/Query/OwnedEntityQueryTestBase.cs index 189928999a3..4c91829a155 100644 --- a/test/EFCore.Specification.Tests/Query/OwnedEntityQueryTestBase.cs +++ b/test/EFCore.Specification.Tests/Query/OwnedEntityQueryTestBase.cs @@ -363,7 +363,8 @@ public virtual async Task Projecting_correlated_collection_property_for_owned_en var query = context.Warehouses.Select( x => new Context18582.WarehouseModel { - WarehouseCode = x.WarehouseCode, DestinationCountryCodes = x.DestinationCountries.Select(c => c.CountryCode).ToArray() + WarehouseCode = x.WarehouseCode, + DestinationCountryCodes = x.DestinationCountries.Select(c => c.CountryCode).ToArray() }).AsNoTracking(); var result = async @@ -750,7 +751,8 @@ public virtual async Task OwnsMany_correlated_projection(bool async) var results = await context.Contacts.Select( contact => new Context22089.ContactDto { - Id = contact.Id, Names = contact.Names.Select(name => new Context22089.NameDto()).ToArray() + Id = contact.Id, + Names = contact.Names.Select(name => new Context22089.NameDto()).ToArray() }) .ToListAsync(); } diff --git a/test/EFCore.Specification.Tests/Query/OwnedQueryTestBase.cs b/test/EFCore.Specification.Tests/Query/OwnedQueryTestBase.cs index 4f8953c0f73..fa171178571 100644 --- a/test/EFCore.Specification.Tests/Query/OwnedQueryTestBase.cs +++ b/test/EFCore.Specification.Tests/Query/OwnedQueryTestBase.cs @@ -25,7 +25,8 @@ public virtual async Task Can_query_owner_with_different_owned_types_having_same await context.AddAsync( new HeliumBalloon { - Id = Guid.NewGuid().ToString(), Gas = new Helium(), + Id = Guid.NewGuid().ToString(), + Gas = new Helium(), }); await context.AddAsync(new HydrogenBalloon { Id = Guid.NewGuid().ToString(), Gas = new Hydrogen() }); diff --git a/test/EFCore.Specification.Tests/Scaffolding/CompiledModelTestBase.cs b/test/EFCore.Specification.Tests/Scaffolding/CompiledModelTestBase.cs index 5ca13d44db1..6f8c9892551 100644 --- a/test/EFCore.Specification.Tests/Scaffolding/CompiledModelTestBase.cs +++ b/test/EFCore.Specification.Tests/Scaffolding/CompiledModelTestBase.cs @@ -834,6 +834,11 @@ public override Guid FromJsonTyped(ref Utf8JsonReaderManager manager, object? ex public override void ToJsonTyped(Utf8JsonWriter writer, Guid value) => writer.WriteStringValue(value); + + private readonly Expression> _ctorLambda = () => new(); + + /// + public override Expression ConstructorExpression => _ctorLambda.Body; } public class ManyTypes diff --git a/test/EFCore.Specification.Tests/SpatialTestBase.cs b/test/EFCore.Specification.Tests/SpatialTestBase.cs index 840519678b5..4b70d6efac5 100644 --- a/test/EFCore.Specification.Tests/SpatialTestBase.cs +++ b/test/EFCore.Specification.Tests/SpatialTestBase.cs @@ -42,7 +42,7 @@ public virtual void Values_arent_compared_by_reference() Assert.False(db.Entry(entity).Property(e => e.Point).IsModified); } - [ConditionalFact] + [ConditionalFact(Skip = "AOT: NTS is not supported")] public virtual async void Mutation_of_tracked_values_does_not_mutate_values_in_store() { Point CreatePoint(double y = 2.2) @@ -92,7 +92,7 @@ await ExecuteWithStrategyInTransactionAsync( }); } - [ConditionalFact] + [ConditionalFact(Skip = "AOT: NTS is not supported")] public virtual void Translators_handle_static_members() { using var db = Fixture.CreateContext(); @@ -108,7 +108,7 @@ orderby e.Id }).FirstOrDefault(); } - [ConditionalFact] + [ConditionalFact(Skip = "AOT: NTS is not supported")] public virtual void Can_roundtrip_Z_and_M() { using var db = Fixture.CreateContext(); diff --git a/test/EFCore.SqlServer.FunctionalTests/Query/AdHocMiscellaneousQuerySqlServerTest.cs b/test/EFCore.SqlServer.FunctionalTests/Query/AdHocMiscellaneousQuerySqlServerTest.cs index 7dbdedb982d..df9ec8389ff 100644 --- a/test/EFCore.SqlServer.FunctionalTests/Query/AdHocMiscellaneousQuerySqlServerTest.cs +++ b/test/EFCore.SqlServer.FunctionalTests/Query/AdHocMiscellaneousQuerySqlServerTest.cs @@ -1037,7 +1037,7 @@ public class List #region 23282 - [ConditionalFact] + [ConditionalFact(Skip = "AOT: NTS is not supported")] [SqlServerCondition(SqlServerCondition.SupportsSqlClr)] public virtual async Task Can_query_point_with_buffered_data_reader() { diff --git a/test/EFCore.SqlServer.FunctionalTests/Query/NonSharedPrimitiveCollectionsQuerySqlServerTest.cs b/test/EFCore.SqlServer.FunctionalTests/Query/NonSharedPrimitiveCollectionsQuerySqlServerTest.cs index fbaa5a4fd4d..825ab30fd1a 100644 --- a/test/EFCore.SqlServer.FunctionalTests/Query/NonSharedPrimitiveCollectionsQuerySqlServerTest.cs +++ b/test/EFCore.SqlServer.FunctionalTests/Query/NonSharedPrimitiveCollectionsQuerySqlServerTest.cs @@ -695,9 +695,19 @@ private static readonly MethodInfo SkipMethod #endregion - [ConditionalFact] - public override Task Column_with_custom_converter() - => base.Column_with_custom_converter(); + public override async Task Column_with_custom_converter() + { + await base.Column_with_custom_converter(); + + AssertSql( +""" +@__ints_0='1,2,3' (Size = 4000) + +SELECT TOP(2) [t].[Id], [t].[Ints] +FROM [TestEntity] AS [t] +WHERE [t].[Ints] = @__ints_0 +"""); + } public override async Task Parameter_with_inferred_value_converter() { diff --git a/test/EFCore.SqlServer.FunctionalTests/Query/QueryLoggingSqlServerTest.cs b/test/EFCore.SqlServer.FunctionalTests/Query/QueryLoggingSqlServerTest.cs index 509035f2fbc..8d0a62628d0 100644 --- a/test/EFCore.SqlServer.FunctionalTests/Query/QueryLoggingSqlServerTest.cs +++ b/test/EFCore.SqlServer.FunctionalTests/Query/QueryLoggingSqlServerTest.cs @@ -34,7 +34,7 @@ var customers "Compiling query expression: ", Fixture.TestSqlLoggerFactory.Log[0].Message); Assert.StartsWith( - "Generated query execution expression: " + Environment.NewLine + "'queryContext => new SingleQueryingEnumerable(", + "Generated query execution expression: " + Environment.NewLine + "'queryContext => SingleQueryingEnumerable.Create(", Fixture.TestSqlLoggerFactory.Log[1].Message); } @@ -48,7 +48,7 @@ var customers Assert.NotNull(customers); Assert.StartsWith( - "Generated query execution expression: " + Environment.NewLine + "'queryContext => new SplitQueryingEnumerable(", + "Generated query execution expression: " + Environment.NewLine + "'queryContext => SplitQueryingEnumerable.Create(", Fixture.TestSqlLoggerFactory.Log[1].Message); } diff --git a/test/EFCore.SqlServer.FunctionalTests/Query/RawSqlServerTest.cs b/test/EFCore.SqlServer.FunctionalTests/Query/RawSqlServerTest.cs index b6525c3fa55..0a06adc72f0 100644 --- a/test/EFCore.SqlServer.FunctionalTests/Query/RawSqlServerTest.cs +++ b/test/EFCore.SqlServer.FunctionalTests/Query/RawSqlServerTest.cs @@ -13,53 +13,50 @@ public class RawSqlServerTest : NonSharedModelTestBase [ConditionalFact] public virtual async Task ToQuery_can_use_FromSqlRaw() { - var contextFactory = await InitializeAsync(seed: c => c.SeedAsync()); + var contextFactory = await InitializeAsync(seed: c => c.SeedAsync()); + using var context = contextFactory.CreateContext(); + var query = context.Set().ToList(); - using (var context = contextFactory.CreateContext()) - { - var query = context.Set().ToList(); - - Assert.Equal(4, query.Count); + Assert.Equal(4, query.Count); - AssertSql( - """ + AssertSql( + """ SELECT o.Amount From Orders AS o -- RAW """); - } } - protected class MyContext13346(DbContextOptions options) : DbContext(options) + public class Context13346(DbContextOptions options) : DbContext(options) { - public virtual DbSet Orders { get; set; } + public virtual DbSet Orders { get; set; } protected override void OnModelCreating(ModelBuilder modelBuilder) { #pragma warning disable CS0618 // Type or member is obsolete - modelBuilder.Entity() + modelBuilder.Entity() .HasNoKey() - .ToQuery(() => Set().FromSqlRaw("SELECT o.Amount From Orders AS o -- RAW")); + .ToQuery(() => Set().FromSqlRaw("SELECT o.Amount From Orders AS o -- RAW")); #pragma warning restore CS0618 // Type or member is obsolete } public Task SeedAsync() { AddRange( - new Order13346 { Amount = 1 }, - new Order13346 { Amount = 2 }, - new Order13346 { Amount = 3 }, - new Order13346 { Amount = 4 } + new Order { Amount = 1 }, + new Order { Amount = 2 }, + new Order { Amount = 3 }, + new Order { Amount = 4 } ); return SaveChangesAsync(); } - public class Order13346 + public class Order { public int Id { get; set; } public int Amount { get; set; } } - public class OrderSummary13346 + public class OrderSummary { public int Amount { get; set; } } diff --git a/test/EFCore.SqlServer.FunctionalTests/Query/SpatialQuerySqlServerGeographyTest.cs b/test/EFCore.SqlServer.FunctionalTests/Query/SpatialQuerySqlServerGeographyTest.cs index 8b5dc90d599..e6b90c9d77c 100644 --- a/test/EFCore.SqlServer.FunctionalTests/Query/SpatialQuerySqlServerGeographyTest.cs +++ b/test/EFCore.SqlServer.FunctionalTests/Query/SpatialQuerySqlServerGeographyTest.cs @@ -218,7 +218,7 @@ public override Task Covers(bool async) public override Task Crosses(bool async) => Task.CompletedTask; - [ConditionalTheory] + [ConditionalTheory(Skip = "AOT: NTS is not supported")] [MemberData(nameof(IsAsyncData))] public virtual async Task CurveToLine(bool async) { diff --git a/test/EFCore.SqlServer.FunctionalTests/Query/SpatialQuerySqlServerGeometryTest.cs b/test/EFCore.SqlServer.FunctionalTests/Query/SpatialQuerySqlServerGeometryTest.cs index b7aaa29bd74..f0b4d8c986d 100644 --- a/test/EFCore.SqlServer.FunctionalTests/Query/SpatialQuerySqlServerGeometryTest.cs +++ b/test/EFCore.SqlServer.FunctionalTests/Query/SpatialQuerySqlServerGeometryTest.cs @@ -245,7 +245,7 @@ FROM [LineStringEntity] AS [l] """); } - [ConditionalTheory] + [ConditionalTheory(Skip = "AOT: NTS is not supported")] [MemberData(nameof(IsAsyncData))] public virtual async Task CurveToLine(bool async) { diff --git a/test/EFCore.SqlServer.FunctionalTests/Scaffolding/Baselines/BigModel/ManyTypesEntityType.cs b/test/EFCore.SqlServer.FunctionalTests/Scaffolding/Baselines/BigModel/ManyTypesEntityType.cs index 24f55453b93..3e6ffa7d0b7 100644 --- a/test/EFCore.SqlServer.FunctionalTests/Scaffolding/Baselines/BigModel/ManyTypesEntityType.cs +++ b/test/EFCore.SqlServer.FunctionalTests/Scaffolding/Baselines/BigModel/ManyTypesEntityType.cs @@ -333,12 +333,12 @@ public static RuntimeEntityType Create(RuntimeModel model, RuntimeEntityType bas dbType: System.Data.DbType.String), converter: new ValueConverter( (bool v) => (string)(v ? "B" : "A"), - (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)'B'), + (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)"B".ToUpperInvariant()[0]), jsonValueReaderWriter: new JsonConvertedValueReaderWriter( JsonStringReaderWriter.Instance, new ValueConverter( (bool v) => (string)(v ? "B" : "A"), - (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)'B'))); + (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)"B".ToUpperInvariant()[0]))); boolToStringConverterProperty.SetSentinelFromProviderValue("A"); boolToStringConverterProperty.AddAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.None); boolToStringConverterProperty.AddRuntimeAnnotation("UnsafeAccessors", new[] { ("ManyTypesEntityType.UnsafeAccessor_Microsoft_EntityFrameworkCore_Scaffolding_ManyTypes_BoolToStringConverterProperty", "TestNamespace") }); diff --git a/test/EFCore.SqlServer.FunctionalTests/Scaffolding/Baselines/BigModel_with_JSON_columns/ManyTypesEntityType.cs b/test/EFCore.SqlServer.FunctionalTests/Scaffolding/Baselines/BigModel_with_JSON_columns/ManyTypesEntityType.cs index 24f55453b93..3e6ffa7d0b7 100644 --- a/test/EFCore.SqlServer.FunctionalTests/Scaffolding/Baselines/BigModel_with_JSON_columns/ManyTypesEntityType.cs +++ b/test/EFCore.SqlServer.FunctionalTests/Scaffolding/Baselines/BigModel_with_JSON_columns/ManyTypesEntityType.cs @@ -333,12 +333,12 @@ public static RuntimeEntityType Create(RuntimeModel model, RuntimeEntityType bas dbType: System.Data.DbType.String), converter: new ValueConverter( (bool v) => (string)(v ? "B" : "A"), - (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)'B'), + (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)"B".ToUpperInvariant()[0]), jsonValueReaderWriter: new JsonConvertedValueReaderWriter( JsonStringReaderWriter.Instance, new ValueConverter( (bool v) => (string)(v ? "B" : "A"), - (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)'B'))); + (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)"B".ToUpperInvariant()[0]))); boolToStringConverterProperty.SetSentinelFromProviderValue("A"); boolToStringConverterProperty.AddAnnotation("SqlServer:ValueGenerationStrategy", SqlServerValueGenerationStrategy.None); boolToStringConverterProperty.AddRuntimeAnnotation("UnsafeAccessors", new[] { ("ManyTypesEntityType.UnsafeAccessor_Microsoft_EntityFrameworkCore_Scaffolding_ManyTypes_BoolToStringConverterProperty", "TestNamespace") }); diff --git a/test/EFCore.SqlServer.FunctionalTests/SqlServerValueGenerationScenariosTestBase.cs b/test/EFCore.SqlServer.FunctionalTests/SqlServerValueGenerationScenariosTestBase.cs index 77d2b88ed5f..2b5d2b0cdd4 100644 --- a/test/EFCore.SqlServer.FunctionalTests/SqlServerValueGenerationScenariosTestBase.cs +++ b/test/EFCore.SqlServer.FunctionalTests/SqlServerValueGenerationScenariosTestBase.cs @@ -699,7 +699,7 @@ public async Task Insert_with_non_key_default_value() } } - [ConditionalFact] + [ConditionalFact(Skip = "AOT: NTS is not supported")] [SqlServerCondition(SqlServerCondition.SupportsSqlClr)] public async Task Insert_with_non_key_default_spatial_value() { diff --git a/test/EFCore.Sqlite.FunctionalTests/Scaffolding/Baselines/BigModel/ManyTypesEntityType.cs b/test/EFCore.Sqlite.FunctionalTests/Scaffolding/Baselines/BigModel/ManyTypesEntityType.cs index f8de859a7fb..cf357d74fd2 100644 --- a/test/EFCore.Sqlite.FunctionalTests/Scaffolding/Baselines/BigModel/ManyTypesEntityType.cs +++ b/test/EFCore.Sqlite.FunctionalTests/Scaffolding/Baselines/BigModel/ManyTypesEntityType.cs @@ -242,12 +242,12 @@ public static RuntimeEntityType Create(RuntimeModel model, RuntimeEntityType bas size: 1), converter: new ValueConverter( (bool v) => (string)(v ? "B" : "A"), - (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)'B'), + (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)"B".ToUpperInvariant()[0]), jsonValueReaderWriter: new JsonConvertedValueReaderWriter( JsonStringReaderWriter.Instance, new ValueConverter( (bool v) => (string)(v ? "B" : "A"), - (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)'B'))); + (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)"B".ToUpperInvariant()[0]))); boolToStringConverterProperty.SetSentinelFromProviderValue("A"); boolToStringConverterProperty.AddRuntimeAnnotation("UnsafeAccessors", new[] { ("ManyTypesEntityType.UnsafeAccessor_Microsoft_EntityFrameworkCore_Scaffolding_ManyTypes_BoolToStringConverterProperty", "TestNamespace") }); diff --git a/test/EFCore.Sqlite.FunctionalTests/Scaffolding/Baselines/BigModel_with_JSON_columns/ManyTypesEntityType.cs b/test/EFCore.Sqlite.FunctionalTests/Scaffolding/Baselines/BigModel_with_JSON_columns/ManyTypesEntityType.cs index f8de859a7fb..cf357d74fd2 100644 --- a/test/EFCore.Sqlite.FunctionalTests/Scaffolding/Baselines/BigModel_with_JSON_columns/ManyTypesEntityType.cs +++ b/test/EFCore.Sqlite.FunctionalTests/Scaffolding/Baselines/BigModel_with_JSON_columns/ManyTypesEntityType.cs @@ -242,12 +242,12 @@ public static RuntimeEntityType Create(RuntimeModel model, RuntimeEntityType bas size: 1), converter: new ValueConverter( (bool v) => (string)(v ? "B" : "A"), - (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)'B'), + (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)"B".ToUpperInvariant()[0]), jsonValueReaderWriter: new JsonConvertedValueReaderWriter( JsonStringReaderWriter.Instance, new ValueConverter( (bool v) => (string)(v ? "B" : "A"), - (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)'B'))); + (string v) => !string.IsNullOrEmpty(v) && (int)v.ToUpperInvariant()[0] == (int)"B".ToUpperInvariant()[0]))); boolToStringConverterProperty.SetSentinelFromProviderValue("A"); boolToStringConverterProperty.AddRuntimeAnnotation("UnsafeAccessors", new[] { ("ManyTypesEntityType.UnsafeAccessor_Microsoft_EntityFrameworkCore_Scaffolding_ManyTypes_BoolToStringConverterProperty", "TestNamespace") }); diff --git a/test/EFCore.Tests/Metadata/Internal/PropertyTest.cs b/test/EFCore.Tests/Metadata/Internal/PropertyTest.cs index 2993dd43077..251e884d8d5 100644 --- a/test/EFCore.Tests/Metadata/Internal/PropertyTest.cs +++ b/test/EFCore.Tests/Metadata/Internal/PropertyTest.cs @@ -554,6 +554,10 @@ public override string FromJsonTyped(ref Utf8JsonReaderManager manager, object e public override void ToJsonTyped(Utf8JsonWriter writer, string value) => writer.WriteStringValue(value); + + private readonly Expression> _instanceLambda = () => new(); + + public override Expression ConstructorExpression => _instanceLambda.Body; } private class JasonValueReaderWriterWithPrivateInstance : JsonValueReaderWriter @@ -565,6 +569,10 @@ public override string FromJsonTyped(ref Utf8JsonReaderManager manager, object e public override void ToJsonTyped(Utf8JsonWriter writer, string value) => writer.WriteStringValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + public override Expression ConstructorExpression => _instanceLambda.Body; } private class JasonValueReaderWriterWithBadInstance : JsonValueReaderWriter @@ -576,6 +584,8 @@ public override string FromJsonTyped(ref Utf8JsonReaderManager manager, object e public override void ToJsonTyped(Utf8JsonWriter writer, string value) => writer.WriteStringValue(value); + + public override Expression ConstructorExpression => Expression.Default(typeof(JasonValueReaderWriterWithBadInstance)); } private class SimpleJasonValueReaderWriterWithInstance : JsonValueReaderWriter @@ -587,6 +597,10 @@ public override string FromJsonTyped(ref Utf8JsonReaderManager manager, object e public override void ToJsonTyped(Utf8JsonWriter writer, string value) => writer.WriteStringValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + public override Expression ConstructorExpression => _instanceLambda.Body; } private class SimpleJasonValueReaderWriterWithInstanceAndPrivateConstructor : JsonValueReaderWriter @@ -602,6 +616,10 @@ public override string FromJsonTyped(ref Utf8JsonReaderManager manager, object e public override void ToJsonTyped(Utf8JsonWriter writer, string value) => writer.WriteStringValue(value); + + private readonly Expression> _instanceLambda = () => Instance; + + public override Expression ConstructorExpression => _instanceLambda.Body; } private class NonDerivedJsonValueReaderWriter; @@ -616,6 +634,10 @@ public override void ToJson(Utf8JsonWriter writer, object value) public override Type ValueType => typeof(string); + + private readonly Expression> _instanceLambda = () => new(); + + public override Expression ConstructorExpression => _instanceLambda.Body; } private abstract class AbstractJasonValueReaderWriter : JsonValueReaderWriter; @@ -631,17 +653,24 @@ public override string FromJsonTyped(ref Utf8JsonReaderManager manager, object e public override void ToJsonTyped(Utf8JsonWriter writer, string value) => writer.WriteStringValue(value); + + private readonly Expression> _instanceLambda = () => new(); + + public override Expression ConstructorExpression => _instanceLambda.Body; } -#pragma warning disable CS9113 // Parameter '_' is unread private class NonParameterlessJsonValueReaderWriter(bool _) : JsonValueReaderWriter -#pragma warning restore CS9113 { public override string FromJsonTyped(ref Utf8JsonReaderManager manager, object existingObject = null) => manager.CurrentReader.GetString()!; public override void ToJsonTyped(Utf8JsonWriter writer, string value) => writer.WriteStringValue(value); + + private readonly ConstructorInfo _constructorInfo = typeof(NonParameterlessJsonValueReaderWriter).GetConstructor([typeof(bool)])!; + + public override Expression ConstructorExpression => + Expression.New(_constructorInfo, Expression.Constant(_)); } private static IMutableModel CreateModel() diff --git a/test/EFCore.Tests/Query/EntityMaterializerSourceTest.cs b/test/EFCore.Tests/Query/EntityMaterializerSourceTest.cs index 5214e55cd2f..de96391c436 100644 --- a/test/EFCore.Tests/Query/EntityMaterializerSourceTest.cs +++ b/test/EFCore.Tests/Query/EntityMaterializerSourceTest.cs @@ -3,6 +3,7 @@ using System.ComponentModel.DataAnnotations.Schema; using Microsoft.EntityFrameworkCore.Metadata.Internal; +using Microsoft.EntityFrameworkCore.Proxies.Internal; using Microsoft.EntityFrameworkCore.Query.Internal; // ReSharper disable UnusedMember.Local @@ -178,7 +179,7 @@ public void Can_create_materializer_for_entity_with_instance_factory_method(bool et.ConstructorBinding = new FactoryMethodBinding( TestProxyFactory.Instance, - typeof(TestProxyFactory).GetTypeInfo().GetDeclaredMethod(nameof(TestProxyFactory.Create))!, + typeof(TestProxyFactory).GetMethod(nameof(TestProxyFactory.Create), [typeof(IEntityType)])!, new List { new EntityTypeParameterBinding() }, et.ClrType); }); diff --git a/test/EFCore.Tests/Query/Internal/NavigationExpandingExpressionVisitorTests.cs b/test/EFCore.Tests/Query/Internal/NavigationExpandingExpressionVisitorTests.cs index 7d3492fdecc..aa8517389b9 100644 --- a/test/EFCore.Tests/Query/Internal/NavigationExpandingExpressionVisitorTests.cs +++ b/test/EFCore.Tests/Query/Internal/NavigationExpandingExpressionVisitorTests.cs @@ -27,6 +27,8 @@ public TestNavigationExpandingExpressionVisitor() null, null, null, + null, + null, new ExecutionStrategyTest.TestExecutionStrategy(new MyDemoContext()), new CurrentDbContext(new MyDemoContext()), null,