Skip to content

Commit

Permalink
Make ownership handling during model building more consistent
Browse files Browse the repository at this point in the history
Store Owned configuration on EntityType
Add methods to IMutableModel and IConventionModel for creating owned entity types
Make sure ownerships have an associated principal to dependent navigation at all times
Store owned types in a field instead of an annotation

Fixes #25138
  • Loading branch information
AndriySvyryd authored Jul 9, 2021
1 parent d5c1cf1 commit d788e06
Show file tree
Hide file tree
Showing 78 changed files with 2,072 additions and 1,149 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -826,7 +826,7 @@ private bool TryRewriteEntityEquality(ExpressionType nodeType, Expression left,
var rightEntityType = rightEntityReference?.EntityType;
var entityType = leftEntityType ?? rightEntityType;

Debug.Assert(entityType != null, "At least either side should be entityReference so entityType should be non-null.");
Check.DebugAssert(entityType != null, "At least either side should be entityReference so entityType should be non-null.");

if (leftEntityType != null
&& rightEntityType != null
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -1408,7 +1408,7 @@ private bool TryRewriteEntityEquality(
var rightEntityType = rightEntityReference?.EntityType;
var entityType = leftEntityType ?? rightEntityType;

Debug.Assert(entityType != null, "At least either side should be entityReference so entityType should be non-null.");
Check.DebugAssert(entityType != null, "At least either side should be entityReference so entityType should be non-null.");

if (leftEntityType != null
&& rightEntityType != null
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,8 @@
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.

using System;
using System.Collections.Generic;
using System.Diagnostics.CodeAnalysis;
using System.Linq;
using Microsoft.EntityFrameworkCore.Diagnostics;
using Microsoft.EntityFrameworkCore.Metadata;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
using Microsoft.EntityFrameworkCore.Metadata.Internal;
Expand Down
24 changes: 5 additions & 19 deletions src/EFCore.Relational/Infrastructure/RelationalModelValidator.cs
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,6 @@

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using Microsoft.EntityFrameworkCore.Diagnostics;
using Microsoft.EntityFrameworkCore.Metadata;
Expand Down Expand Up @@ -733,22 +732,9 @@ protected virtual void ValidateSharedColumnsCompatibility(
{
foreach (var missingColumn in missingConcurrencyTokens)
{
var columnFound = false;
foreach (var property in entityType.GetAllBaseTypesAscending().SelectMany(t => t.GetDeclaredProperties()))
{
if (property.GetColumnName(storeObject) == missingColumn)
{
columnFound = true;
break;
}
}

if (!columnFound)
{
throw new InvalidOperationException(
RelationalStrings.MissingConcurrencyColumn(
entityType.DisplayName(), missingColumn, storeObject.DisplayName()));
}
throw new InvalidOperationException(
RelationalStrings.MissingConcurrencyColumn(
entityType.DisplayName(), missingColumn, storeObject.DisplayName()));
}
}
}
Expand Down Expand Up @@ -1477,8 +1463,8 @@ protected virtual void ValidateIndexProperties(
}
else if (overlappingTables.Count == 0)
{
Debug.Assert(firstPropertyTables != null, nameof(firstPropertyTables));
Debug.Assert(lastPropertyTables != null, nameof(lastPropertyTables));
Check.DebugAssert(firstPropertyTables != null, nameof(firstPropertyTables));
Check.DebugAssert(lastPropertyTables != null, nameof(lastPropertyTables));

logger.IndexPropertiesMappedToNonOverlappingTables(
entityType,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -193,21 +193,22 @@ public static bool IsConcurrencyTokenMissing(
IReadOnlyEntityType entityType,
IReadOnlyList<IReadOnlyEntityType> mappedTypes)
{
if (entityType.FindPrimaryKey() == null)
if (entityType.FindPrimaryKey() == null
|| propertiesMappedToConcurrencyColumn.Count == 0)
{
return false;
}

var propertyMissing = false;
var propertyMissing = true;
foreach (var mappedProperty in propertiesMappedToConcurrencyColumn)
{
var declaringEntityType = mappedProperty.DeclaringEntityType;
if (declaringEntityType.IsAssignableFrom(entityType)
|| entityType.IsAssignableFrom(declaringEntityType)
|| declaringEntityType.IsInOwnershipPath(entityType)
|| entityType.IsInOwnershipPath(declaringEntityType))
{
// The concurrency token is in the same hierarchy or in the same aggregate
// The concurrency token is on the base type or in the same aggregate
propertyMissing = false;
continue;
}

Expand All @@ -222,11 +223,8 @@ public static bool IsConcurrencyTokenMissing(
|| entityType.IsAssignableFrom(fk.PrincipalEntityType)))
{
// The concurrency token is on a type that shares the row with a base or derived type
continue;
propertyMissing = false;
}

propertyMissing = true;
break;
}

return propertyMissing;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -70,26 +70,12 @@ private void ProcessDbFunctionAdded(
var entityType = model.FindEntityType(elementType);
if (entityType?.IsOwned() == true
|| model.IsOwned(elementType)
|| (entityType == null && model.GetEntityTypes().Any(e => e.ClrType == elementType)))
|| (entityType == null && model.FindEntityTypes(elementType).Any()))
{
return;
}

IConventionEntityTypeBuilder? entityTypeBuilder;
if (entityType != null)
{
entityTypeBuilder = entityType.Builder;
}
else
{
entityTypeBuilder = dbFunctionBuilder.ModelBuilder.Entity(elementType);
if (entityTypeBuilder == null)
{
return;
}

entityType = entityTypeBuilder.Metadata;
}
dbFunctionBuilder.ModelBuilder.Entity(elementType);
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -1358,7 +1358,7 @@ private bool TryRewriteEntityEquality(
var rightEntityType = rightEntityReference?.EntityType;
var entityType = leftEntityType ?? rightEntityType;

Debug.Assert(entityType != null, "At least one side should be entityReference so entityType should be non-null.");
Check.DebugAssert(entityType != null, "At least one side should be entityReference so entityType should be non-null.");

if (leftEntityType != null
&& rightEntityType != null
Expand Down
3 changes: 2 additions & 1 deletion src/EFCore.Relational/Storage/RelationalConnection.cs
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,8 @@ public abstract class RelationalConnection : IRelationalConnection, ITransaction
private bool _connectionOwned;
private int _openedCount;
private bool _openedInternally;
private int? _commandTimeout, _defaultCommandTimeout;
private int? _commandTimeout;
private readonly int? _defaultCommandTimeout;
private readonly ConcurrentStack<Transaction> _ambientTransactions = new();
private DbConnection? _connection;
private readonly IRelationalCommandBuilder _relationalCommandBuilder;
Expand Down

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Original file line number Diff line number Diff line change
Expand Up @@ -138,7 +138,7 @@ public static IndexBuilder<TEntity> IncludeProperties<TEntity>(
/// <returns> A builder to further configure the index. </returns>
public static IndexBuilder<TEntity> IncludeProperties<TEntity>(
this IndexBuilder<TEntity> indexBuilder,
Expression<Func<TEntity, object>> includeExpression)
Expression<Func<TEntity, object?>> includeExpression)
{
Check.NotNull(indexBuilder, nameof(indexBuilder));
Check.NotNull(includeExpression, nameof(includeExpression));
Expand Down
7 changes: 4 additions & 3 deletions src/EFCore/ChangeTracking/EntityEntry.cs
Original file line number Diff line number Diff line change
Expand Up @@ -325,7 +325,8 @@ public virtual PropertyValues CurrentValues
/// </para>
/// <para>
/// Note that whenever real original property values are not available (e.g. entity was not yet
/// persisted to the database) this will default to the current property values of this entity.
/// persisted to the database or was retrieved in a non-tracking query) this will default to the
/// current property values of this entity.
/// </para>
/// </summary>
/// <value> The original values. </value>
Expand All @@ -338,14 +339,14 @@ public virtual PropertyValues OriginalValues
/// <summary>
/// <para>
/// Queries the database for copies of the values of the tracked entity as they currently
/// exist in the database. If the entity is not found in the database, then null is returned.
/// exist in the database. If the entity is not found in the database, then <see langword="null"/> is returned.
/// </para>
/// <para>
/// Note that changing the values in the returned dictionary will not update the values
/// in the database.
/// </para>
/// </summary>
/// <returns> The store values, or null if the entity does not exist in the database. </returns>
/// <returns> The store values, or <see langword="null"/> if the entity does not exist in the database. </returns>
public virtual PropertyValues? GetDatabaseValues()
{
var values = Finder.GetDatabaseValues(InternalEntry);
Expand Down
1 change: 0 additions & 1 deletion src/EFCore/Design/CSharpRuntimeAnnotationCodeGenerator.cs
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,6 @@ public virtual void Generate(IModel model, CSharpRuntimeAnnotationCodeGeneratorP
}
else
{
parameters.Annotations.Remove(CoreAnnotationNames.OwnedTypes);
parameters.Annotations.Remove(CoreAnnotationNames.PropertyAccessMode);
}

Expand Down
2 changes: 1 addition & 1 deletion src/EFCore/Diagnostics/CoreEventId.cs
Original file line number Diff line number Diff line change
Expand Up @@ -146,7 +146,7 @@ private static EventId MakeUpdateId(Id id)
public static readonly EventId SaveChangesFailed = MakeUpdateId(Id.SaveChangesFailed);

/// <summary>
/// The same entity is being tracked as a different weak entity type.
/// The same entity is being tracked as a different shared entity entity type.
/// This event is in the <see cref="DbLoggerCategory.Update" /> category.
/// </summary>
public static readonly EventId DuplicateDependentEntityTypeInstanceWarning =
Expand Down
3 changes: 2 additions & 1 deletion src/EFCore/Infrastructure/IModelCacheKeyFactory.cs
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,8 @@ public interface IModelCacheKeyFactory
/// </param>
/// <returns> The created key. </returns>
[Obsolete("Use the overload with most parameters")]
object Create(DbContext context);
object Create(DbContext context)
=> Create(context, true);

/// <summary>
/// Gets the model cache key for a given context.
Expand Down
51 changes: 19 additions & 32 deletions src/EFCore/Infrastructure/ModelValidator.cs
Original file line number Diff line number Diff line change
Expand Up @@ -141,7 +141,7 @@ protected virtual void ValidatePropertyMapping(
{
Check.NotNull(model, nameof(model));

if (!(model is IConventionModel conventionModel))
if (model is not IConventionModel conventionModel)
{
return;
}
Expand All @@ -167,10 +167,8 @@ protected virtual void ValidatePropertyMapping(
continue;
}

var clrProperties = new HashSet<string>(StringComparer.Ordinal);

var runtimeProperties = entityType.GetRuntimeProperties();

var clrProperties = new HashSet<string>(StringComparer.Ordinal);
clrProperties.UnionWith(
runtimeProperties.Values
.Where(pi => pi.IsCandidateProperty(needsWrite: false))
Expand All @@ -181,6 +179,7 @@ protected virtual void ValidatePropertyMapping(
.Concat(entityType.GetNavigations())
.Concat(entityType.GetSkipNavigations())
.Concat(entityType.GetServiceProperties()).Select(p => p.Name));

if (entityType.IsPropertyBag)
{
clrProperties.ExceptWith(_dictionaryProperties);
Expand All @@ -191,7 +190,6 @@ protected virtual void ValidatePropertyMapping(
continue;
}

var configuration = ((Model)entityType.Model).Configuration;
foreach (var clrPropertyName in clrProperties)
{
if (entityType.FindIgnoredConfigurationSource(clrPropertyName) != null)
Expand All @@ -212,53 +210,40 @@ protected virtual void ValidatePropertyMapping(
continue;
}

Dependencies.MemberClassifier.GetNavigationCandidates(entityType).TryGetValue(clrProperty, out var targetType);
var targetType = Dependencies.MemberClassifier.FindCandidateNavigationPropertyType(
clrProperty, conventionModel, out var targetOwned);
if (targetType == null
|| targetSequenceType == null)
&& clrProperty.FindSetterProperty() == null)
{
if (clrProperty.FindSetterProperty() == null)
{
continue;
}

var sharedType = clrProperty.GetMemberType();
if (conventionModel.IsShared(sharedType))
{
targetType = sharedType;
}
continue;
}

var isTargetSharedOrOwned = targetType != null
&& (conventionModel.IsShared(targetType)
|| conventionModel.IsOwned(targetType));

if (targetType?.IsValidEntityType() == true
&& (isTargetSharedOrOwned
|| conventionModel.FindEntityType(targetType) != null
|| targetType.GetRuntimeProperties().Any(p => p.IsCandidateProperty())))
if (targetType != null)
{
var targetShared = conventionModel.IsShared(targetType);
targetOwned ??= conventionModel.IsOwned(targetType);
// ReSharper disable CheckForReferenceEqualityInstead.1
// ReSharper disable CheckForReferenceEqualityInstead.3
if ((!entityType.IsKeyless
|| targetSequenceType == null)
&& entityType.GetDerivedTypes().All(
dt => dt.GetDeclaredNavigations().FirstOrDefault(n => n.Name == clrProperty.GetSimpleMemberName())
== null)
&& (!isTargetSharedOrOwned
&& (!(targetShared || targetOwned.Value)
|| (!targetType.Equals(entityType.ClrType)
&& (!entityType.IsInOwnershipPath(targetType)
|| (entityType.FindOwnership()!.PrincipalEntityType.ClrType.Equals(targetType)
&& targetSequenceType == null)))))
{
if (conventionModel.IsOwned(entityType.ClrType)
&& conventionModel.IsOwned(targetType))
if (entityType.IsOwned()
&& targetOwned.Value)
{
throw new InvalidOperationException(
CoreStrings.AmbiguousOwnedNavigation(
entityType.DisplayName() + "." + clrProperty.Name, targetType.ShortDisplayName()));
}

if (model.IsShared(targetType))
if (targetShared)
{
throw new InvalidOperationException(
CoreStrings.NonConfiguredNavigationToSharedType(clrProperty.Name, entityType.DisplayName()));
Expand Down Expand Up @@ -705,9 +690,10 @@ protected virtual void ValidateOwnership(

if (ownerships.Count == 1)
{
Check.DebugAssert(entityType.IsOwned(), $"Expected the entity type {entityType.DisplayName()} to be marked as owned");

var ownership = ownerships[0];
if (entityType.BaseType != null
&& ownership.DeclaringEntityType == entityType)
if (entityType.BaseType != null)
{
throw new InvalidOperationException(CoreStrings.OwnedDerivedType(entityType.DisplayName()));
}
Expand Down Expand Up @@ -742,7 +728,8 @@ protected virtual void ValidateOwnership(
ownership.PrincipalEntityType.DisplayName()));
}
}
else if (((IMutableModel)model).IsOwned(entityType.ClrType))
else if (((IConventionModel)model).IsOwned(entityType.ClrType)
|| entityType.IsOwned())
{
throw new InvalidOperationException(CoreStrings.OwnerlessOwnedType(entityType.DisplayName()));
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -138,7 +138,7 @@ public virtual EntityTypeBuilder UsingEntity(

if (newJoinEntityType == null)
{
newJoinEntityType = ModelBuilder.Entity(joinEntityType, ConfigurationSource.Explicit)!.Metadata;
newJoinEntityType = ModelBuilder.Entity(joinEntityType, ConfigurationSource.Explicit, shouldBeOwned: false)!.Metadata;
}

var entityTypeBuilder = new EntityTypeBuilder(newJoinEntityType);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -94,7 +94,7 @@ public virtual EntityTypeBuilder<TJoinEntity> UsingEntity<TJoinEntity>(

if (joinEntityType == null)
{
joinEntityType = ModelBuilder.Entity(typeof(TJoinEntity), ConfigurationSource.Explicit)!.Metadata;
joinEntityType = ModelBuilder.Entity(typeof(TJoinEntity), ConfigurationSource.Explicit, shouldBeOwned: false)!.Metadata;
}

var entityTypeBuilder = new EntityTypeBuilder<TJoinEntity>(joinEntityType);
Expand Down
4 changes: 2 additions & 2 deletions src/EFCore/Metadata/Builders/DiscriminatorBuilder.cs
Original file line number Diff line number Diff line change
Expand Up @@ -92,7 +92,7 @@ public virtual DiscriminatorBuilder HasValue<TEntity>(object? value)
public virtual DiscriminatorBuilder HasValue(Type entityType, object? value)
{
var entityTypeBuilder = EntityTypeBuilder.ModelBuilder.Entity(
entityType, ConfigurationSource.Explicit, shouldBeOwned: null);
entityType, ConfigurationSource.Explicit);

return HasValue(entityTypeBuilder, value, ConfigurationSource.Explicit)!;
}
Expand All @@ -106,7 +106,7 @@ public virtual DiscriminatorBuilder HasValue(Type entityType, object? value)
public virtual DiscriminatorBuilder HasValue(string entityTypeName, object? value)
{
var entityTypeBuilder = EntityTypeBuilder.ModelBuilder.Entity(
entityTypeName, ConfigurationSource.Explicit, shouldBeOwned: null);
entityTypeName, ConfigurationSource.Explicit);

return HasValue(entityTypeBuilder, value, ConfigurationSource.Explicit)!;
}
Expand Down
Loading

0 comments on commit d788e06

Please sign in to comment.