forked from dotnet/efcore
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathConvertingValueComparer.cs
90 lines (76 loc) · 3.59 KB
/
ConvertingValueComparer.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
// 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.ChangeTracking.Internal;
using System.Diagnostics.CodeAnalysis;
using static Expression;
/// <summary>
/// A composable value comparer that accepts a value comparer, and exposes it as a value comparer for a base type.
/// Used when a collection comparer over e.g. object[] is needed over a specific element type (e.g. int)
/// </summary>
/// <remarks>
/// 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.
/// </remarks>
public class ConvertingValueComparer
<[DynamicallyAccessedMembers(
DynamicallyAccessedMemberTypes.PublicMethods
| DynamicallyAccessedMemberTypes.PublicProperties)]
TTo, TFrom> : ValueComparer<TTo>, IInfrastructure<ValueComparer>
{
private readonly ValueComparer<TFrom> _valueComparer;
/// <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 ConvertingValueComparer(ValueComparer<TFrom> valueComparer)
: base(
CreateEquals(valueComparer),
CreateHashCode(valueComparer),
CreateSnapshot(valueComparer))
=> _valueComparer = valueComparer;
private static Expression<Func<TTo?, TTo?, bool>> CreateEquals(ValueComparer<TFrom> valueComparer)
{
var p1 = Parameter(typeof(TTo), "v1");
var p2 = Parameter(typeof(TTo), "v2");
var body = typeof(TTo).IsAssignableFrom(typeof(TFrom))
? valueComparer.EqualsExpression.Body
: valueComparer.ExtractEqualsBody(
Convert(p1, typeof(TFrom)),
Convert(p2, typeof(TFrom)));
return Lambda<Func<TTo?, TTo?, bool>>(
body,
p1,
p2);
}
private static Expression<Func<TTo, int>> CreateHashCode(ValueComparer<TFrom> valueComparer)
{
var p = Parameter(typeof(TTo), "v");
var body = typeof(TTo).IsAssignableFrom(typeof(TFrom))
? valueComparer.HashCodeExpression.Body
: valueComparer.ExtractHashCodeBody(
Convert(p, typeof(TFrom)));
return Lambda<Func<TTo, int>>(
body,
p);
}
private static Expression<Func<TTo, TTo>> CreateSnapshot(ValueComparer<TFrom> valueComparer)
{
var p = Parameter(typeof(TTo), "v");
// types must match exactly as we have both covariance and contravariance case here
var body = typeof(TTo) == typeof(TFrom)
? valueComparer.SnapshotExpression.Body
: Convert(
valueComparer.ExtractSnapshotBody(
Convert(p, typeof(TFrom))),
typeof(TTo));
return Lambda<Func<TTo, TTo>>(
body,
p);
}
ValueComparer IInfrastructure<ValueComparer>.Instance
=> _valueComparer;
}