Skip to content

Commit

Permalink
Identity resolution for loading detached entities (#29884)
Browse files Browse the repository at this point in the history
  • Loading branch information
ajcvickers authored Dec 21, 2022
1 parent bd5b44c commit 06a41bb
Show file tree
Hide file tree
Showing 31 changed files with 1,413 additions and 195 deletions.
5 changes: 1 addition & 4 deletions src/EFCore.Proxies/Proxies/Internal/ProxyFactory.cs
Original file line number Diff line number Diff line change
Expand Up @@ -79,10 +79,7 @@ public virtual object CreateLazyLoadingProxy(
throw new InvalidOperationException(ProxiesStrings.ProxyServicesMissing);
}

return CreateLazyLoadingProxy(
entityType,
context.GetService<ILazyLoader>(),
constructorArguments);
return CreateLazyLoadingProxy(entityType, loader, constructorArguments);
}

private object CreateLazyLoadingProxy(
Expand Down
Original file line number Diff line number Diff line change
@@ -1,10 +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 System.Collections;
using System.Runtime.CompilerServices;
using System.Text.Json;
using System.Xml.Linq;
using Microsoft.EntityFrameworkCore.Internal;
using Microsoft.EntityFrameworkCore.Query.Internal;

Expand Down Expand Up @@ -308,7 +306,8 @@ void ProcessCurrentElementRow()
if (!trackingQuery)
{
fixup(entity, relatedEntity);
if (inverseNavigation != null)
if (inverseNavigation != null
&& !inverseNavigation.IsCollection)
{
inverseNavigation.SetIsLoadedWhenNoTracking(relatedEntity);
}
Expand Down Expand Up @@ -477,7 +476,9 @@ private static async Task PopulateSplitIncludeCollectionAsync<TIncludingEntity,
// Execute and fetch data reader
var dataReader = await executionStrategy.ExecuteAsync(
(queryContext, relationalCommandCache, readerColumns, detailedErrorsEnabled),
((RelationalQueryContext, RelationalCommandCache, IReadOnlyList<ReaderColumn?>?, bool) tup, CancellationToken cancellationToken)
(
(RelationalQueryContext, RelationalCommandCache, IReadOnlyList<ReaderColumn?>?, bool) tup,
CancellationToken cancellationToken)
=> InitializeReaderAsync(tup.Item1, tup.Item2, tup.Item3, tup.Item4, cancellationToken),
verifySucceeded: null,
queryContext.CancellationToken)
Expand Down Expand Up @@ -800,7 +801,9 @@ private static async Task PopulateSplitCollectionAsync<TCollection, TElement, TR
// Execute and fetch data reader
var dataReader = await executionStrategy.ExecuteAsync(
(queryContext, relationalCommandCache, readerColumns, detailedErrorsEnabled),
((RelationalQueryContext, RelationalCommandCache, IReadOnlyList<ReaderColumn?>?, bool) tup, CancellationToken cancellationToken)
(
(RelationalQueryContext, RelationalCommandCache, IReadOnlyList<ReaderColumn?>?, bool) tup,
CancellationToken cancellationToken)
=> InitializeReaderAsync(tup.Item1, tup.Item2, tup.Item3, tup.Item4, cancellationToken),
verifySucceeded: null,
queryContext.CancellationToken)
Expand Down Expand Up @@ -929,7 +932,7 @@ private static void IncludeJsonEntityCollection<TIncludingEntity, TIncludedColle

if (nullable)
{
return default(TEntity);
return default;
}

throw new InvalidOperationException(
Expand Down Expand Up @@ -966,7 +969,7 @@ private static void IncludeJsonEntityCollection<TIncludingEntity, TIncludedColle
return result;
}

return default(TResult);
return default;
}

private static async Task TaskAwaiter(Func<Task>[] taskFactories)
Expand Down
92 changes: 86 additions & 6 deletions src/EFCore/ChangeTracking/CollectionEntry.cs
Original file line number Diff line number Diff line change
Expand Up @@ -207,11 +207,20 @@ public override bool IsModified

/// <summary>
/// Loads the entities referenced by this navigation property, unless <see cref="NavigationEntry.IsLoaded" />
/// is already set to true.
/// is already set to <see langword="true"/>.
/// </summary>
/// <remarks>
/// <para>
/// Note that entities that are already being tracked are not overwritten with new data from the database.
/// If the the entity represented by this entry is tracked, then entities with the same primary key value are not replaced
/// by new entities or overwritten with new data from the database. If the entity entity represented by this entry is not
/// tracked and the collection already contains entities, then calling this method will result in duplicate
/// instances in the collection or inverse collection for any entities with the same key value.
/// Use <see cref="LoadWithIdentityResolution" /> to avoid getting these duplicates.
/// </para>
/// <para>
/// For tracked entities, this method behaves in the same way and has the same performance as
/// <see cref="LoadWithIdentityResolution" />. For entities that are not tracked, this method can be faster than
/// <see cref="LoadWithIdentityResolution" />.
/// </para>
/// <para>
/// See <see href="https://aka.ms/efcore-docs-entity-entries">Accessing tracked entities in EF Core</see>
Expand All @@ -224,17 +233,54 @@ public override void Load()

if (!IsLoaded)
{
TargetLoader.Load(InternalEntry);
TargetLoader.Load(InternalEntry, forceIdentityResolution: false);
}
}

/// <summary>
/// Loads the entities referenced by this navigation property, unless <see cref="NavigationEntry.IsLoaded" />
/// is already set to <see langword="true"/>.
/// </summary>
/// <remarks>
/// <para>
/// Entities with the same primary key value are not replaced by new entities or overwritten with new data from the database.
/// This navigation and its inverse will not contain duplicate entities.
/// </para>
/// <para>
/// For tracked entities, this method behaves in the same way and has the same performance as
/// <see cref="Load" />. For entities that are not tracked, this method can be slower than <see cref="Load" />.
/// </para>
/// <para>
/// See <see href="https://aka.ms/efcore-docs-entity-entries">Accessing tracked entities in EF Core</see>
/// and <see href="https://aka.ms/efcore-docs-load-related-data">Loading related entities</see> for more information and examples.
/// </para>
/// </remarks>
public override void LoadWithIdentityResolution()
{
EnsureInitialized();

if (!IsLoaded)
{
TargetLoader.Load(InternalEntry, forceIdentityResolution: true);
}
}

/// <summary>
/// Loads entities referenced by this navigation property, unless <see cref="NavigationEntry.IsLoaded" />
/// is already set to true.
/// is already set to <see langword="true"/>.
/// </summary>
/// <remarks>
/// <para>
/// Note that entities that are already being tracked are not overwritten with new data from the database.
/// If the the entity represented by this entry is tracked, then entities with the same primary key value are not replaced
/// by new entities or overwritten with new data from the database. If the entity entity represented by this entry is not
/// tracked and the collection already contains entities, then calling this method will result in duplicate
/// instances in the collection or inverse collection for any entities with the same key value.
/// Use <see cref="LoadWithIdentityResolutionAsync" /> to avoid getting these duplicates.
/// </para>
/// <para>
/// For tracked entities, this method behaves in the same way and has the same performance as
/// <see cref="LoadWithIdentityResolutionAsync" />. For entities that are not tracked, this method can be faster than
/// <see cref="LoadWithIdentityResolutionAsync" />.
/// </para>
/// <para>
/// Multiple active operations on the same context instance are not supported. Use <see langword="await" /> to ensure
Expand All @@ -254,7 +300,41 @@ public override Task LoadAsync(CancellationToken cancellationToken = default)

return IsLoaded
? Task.CompletedTask
: TargetLoader.LoadAsync(InternalEntry, cancellationToken);
: TargetLoader.LoadAsync(InternalEntry, forceIdentityResolution: false, cancellationToken);
}

/// <summary>
/// Loads entities referenced by this navigation property, unless <see cref="NavigationEntry.IsLoaded" />
/// is already set to <see langword="true"/>.
/// </summary>
/// <remarks>
/// <para>
/// Entities with the same primary key value are not replaced by new entities or overwritten with new data from the database.
/// This navigation and its inverse will not contain duplicate entities.
/// </para>
/// <para>
/// For tracked entities, this method behaves in the same way and has the same performance as
/// <see cref="LoadAsync" />. For entities that are not tracked, this method can be slower than <see cref="LoadAsync" />.
/// </para>
/// <para>
/// Multiple active operations on the same context instance are not supported. Use <see langword="await" /> to ensure
/// that any asynchronous operations have completed before calling another method on this context.
/// </para>
/// <para>
/// See <see href="https://aka.ms/efcore-docs-entity-entries">Accessing tracked entities in EF Core</see>
/// and <see href="https://aka.ms/efcore-docs-load-related-data">Loading related entities</see> for more information and examples.
/// </para>
/// </remarks>
/// <param name="cancellationToken">A <see cref="CancellationToken" /> to observe while waiting for the task to complete.</param>
/// <returns>A task that represents the asynchronous save operation.</returns>
/// <exception cref="OperationCanceledException">If the <see cref="CancellationToken" /> is canceled.</exception>
public override Task LoadWithIdentityResolutionAsync(CancellationToken cancellationToken = default)
{
EnsureInitialized();

return IsLoaded
? Task.CompletedTask
: TargetLoader.LoadAsync(InternalEntry, forceIdentityResolution: true, cancellationToken);
}

/// <summary>
Expand Down
8 changes: 8 additions & 0 deletions src/EFCore/ChangeTracking/Internal/IStateManager.cs
Original file line number Diff line number Diff line change
Expand Up @@ -126,6 +126,14 @@ InternalEntityEntry StartTrackingFromQuery(
/// </summary>
InternalEntityEntry? TryGetEntry(object entity, IEntityType type, bool throwOnTypeMismatch = true);

/// <summary>
/// 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.
/// </summary>
InternalEntityEntry? TryGetExistingEntry(object entity, IKey key);

/// <summary>
/// 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
Expand Down
31 changes: 31 additions & 0 deletions src/EFCore/ChangeTracking/Internal/StateManager.cs
Original file line number Diff line number Diff line change
Expand Up @@ -418,6 +418,37 @@ public virtual InternalEntityEntry StartTrackingFromQuery(
: entry
: null;

/// <summary>
/// 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.
/// </summary>
public virtual InternalEntityEntry? TryGetExistingEntry(object entity, IKey key)
{
var keyValues = GetKeyValues();
return keyValues == null ? null : TryGetEntry(key, keyValues);

object[]? GetKeyValues()
{
var entry = GetOrCreateEntry(entity);
var properties = key.Properties;
var propertyValues = new object[properties.Count];
for (var i = 0; i < propertyValues.Length; i++)
{
var propertyValue = entry[properties[i]];
if (propertyValue == null)
{
return null;
}

propertyValues[i] = propertyValue;
}

return propertyValues;
}
}

private IIdentityMap GetOrCreateIdentityMap(IKey key)
{
if (_identityMap0 == null)
Expand Down
Loading

0 comments on commit 06a41bb

Please sign in to comment.