Skip to content

Commit f12d2cd

Browse files
committed
Temporal Tables API feedback - adding note to temporal methods that database stores time in UTC so local time may lead to unexpected results. Also indicating that in DateTime argument names for those methods.
1 parent 762e870 commit f12d2cd

File tree

1 file changed

+33
-23
lines changed

1 file changed

+33
-23
lines changed

src/EFCore.SqlServer/Extensions/SqlServerDbSetExtensions.cs

+33-23
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,6 @@
11
// Licensed to the .NET Foundation under one or more agreements.
22
// The .NET Foundation licenses this file to you under the MIT license.
33

4-
using System;
5-
using System.Linq;
64
using System.Linq.Expressions;
75
using Microsoft.EntityFrameworkCore.Query;
86
using Microsoft.EntityFrameworkCore.SqlServer.Query.Internal;
@@ -21,15 +19,18 @@ public static class SqlServerDbSetExtensions
2119
/// Applies temporal 'AsOf' operation on the given DbSet, which only returns elements that were present in the database at a given point in time.
2220
/// </para>
2321
/// <para>
22+
/// Temporal information is stored in UTC format on the database so any <see cref="DateTime" /> arguments in local time may lead to unexpected results.
23+
/// </para>
24+
/// <para>
2425
/// Temporal queries are always set as 'NoTracking'.
2526
/// </para>
2627
/// </summary>
2728
/// <param name="source">Source DbSet on which the temporal operation is applied.</param>
28-
/// <param name="pointInTime"><see cref="DateTime" /> representing a point in time for which the results should be returned.</param>
29+
/// <param name="utcPointInTime"><see cref="DateTime" /> representing a point in time for which the results should be returned.</param>
2930
/// <returns> An <see cref="IQueryable{T}" /> representing the entities at a given point in time.</returns>
3031
public static IQueryable<TEntity> TemporalAsOf<TEntity>(
3132
this DbSet<TEntity> source,
32-
DateTime pointInTime)
33+
DateTime utcPointInTime)
3334
where TEntity : class
3435
{
3536
Check.NotNull(source, nameof(source));
@@ -39,7 +40,7 @@ public static IQueryable<TEntity> TemporalAsOf<TEntity>(
3940
return queryableSource.Provider.CreateQuery<TEntity>(
4041
GenerateTemporalAsOfQueryRoot<TEntity>(
4142
queryableSource,
42-
pointInTime)).AsNoTracking();
43+
utcPointInTime)).AsNoTracking();
4344
}
4445

4546
/// <summary>
@@ -53,17 +54,20 @@ public static IQueryable<TEntity> TemporalAsOf<TEntity>(
5354
/// All versions of entities in that were present within the time range are returned, so it is possible to return multiple entities with the same key.
5455
/// </para>
5556
/// <para>
57+
/// Temporal information is stored in UTC format on the database so any <see cref="DateTime" /> arguments in local time may lead to unexpected results.
58+
/// </para>
59+
/// <para>
5660
/// Temporal queries are always set as 'NoTracking'.
5761
/// </para>
5862
/// </summary>
5963
/// <param name="source">Source DbSet on which the temporal operation is applied.</param>
60-
/// <param name="from">Point in time representing the start of the period for which results should be returned.</param>
61-
/// <param name="to">Point in time representing the end of the period for which results should be returned.</param>
64+
/// <param name="utcFrom">Point in time representing the start of the period for which results should be returned.</param>
65+
/// <param name="utcTo">Point in time representing the end of the period for which results should be returned.</param>
6266
/// <returns> An <see cref="IQueryable{T}" /> representing the entities present in a given time range.</returns>
6367
public static IQueryable<TEntity> TemporalFromTo<TEntity>(
6468
this DbSet<TEntity> source,
65-
DateTime from,
66-
DateTime to)
69+
DateTime utcFrom,
70+
DateTime utcTo)
6771
where TEntity : class
6872
{
6973
Check.NotNull(source, nameof(source));
@@ -73,8 +77,8 @@ public static IQueryable<TEntity> TemporalFromTo<TEntity>(
7377
return queryableSource.Provider.CreateQuery<TEntity>(
7478
GenerateRangeTemporalQueryRoot<TEntity>(
7579
queryableSource,
76-
from,
77-
to,
80+
utcFrom,
81+
utcTo,
7882
TemporalOperationType.FromTo)).AsNoTracking();
7983
}
8084

@@ -89,17 +93,20 @@ public static IQueryable<TEntity> TemporalFromTo<TEntity>(
8993
/// All versions of entities in that were present within the time range are returned, so it is possible to return multiple entities with the same key.
9094
/// </para>
9195
/// <para>
96+
/// Temporal information is stored in UTC format on the database so any <see cref="DateTime" /> arguments in local time may lead to unexpected results.
97+
/// </para>
98+
/// <para>
9299
/// Temporal queries are always set as 'NoTracking'.
93100
/// </para>
94101
/// </summary>
95102
/// <param name="source">Source DbSet on which the temporal operation is applied.</param>
96-
/// <param name="from">Point in time representing the start of the period for which results should be returned.</param>
97-
/// <param name="to">Point in time representing the end of the period for which results should be returned.</param>
103+
/// <param name="utcFrom">Point in time representing the start of the period for which results should be returned.</param>
104+
/// <param name="utcTo">Point in time representing the end of the period for which results should be returned.</param>
98105
/// <returns> An <see cref="IQueryable{T}" /> representing the entities present in a given time range.</returns>
99106
public static IQueryable<TEntity> TemporalBetween<TEntity>(
100107
this DbSet<TEntity> source,
101-
DateTime from,
102-
DateTime to)
108+
DateTime utcFrom,
109+
DateTime utcTo)
103110
where TEntity : class
104111
{
105112
Check.NotNull(source, nameof(source));
@@ -109,8 +116,8 @@ public static IQueryable<TEntity> TemporalBetween<TEntity>(
109116
return queryableSource.Provider.CreateQuery<TEntity>(
110117
GenerateRangeTemporalQueryRoot<TEntity>(
111118
queryableSource,
112-
from,
113-
to,
119+
utcFrom,
120+
utcTo,
114121
TemporalOperationType.Between)).AsNoTracking();
115122
}
116123

@@ -125,17 +132,20 @@ public static IQueryable<TEntity> TemporalBetween<TEntity>(
125132
/// All versions of entities in that were present within the time range are returned, so it is possible to return multiple entities with the same key.
126133
/// </para>
127134
/// <para>
135+
/// Temporal information is stored in UTC format on the database so any <see cref="DateTime" /> arguments in local time may lead to unexpected results.
136+
/// </para>
137+
/// <para>
128138
/// Temporal queries are always set as 'NoTracking'.
129139
/// </para>
130140
/// </summary>
131141
/// <param name="source">Source DbSet on which the temporal operation is applied.</param>
132-
/// <param name="from">Point in time representing the start of the period for which results should be returned.</param>
133-
/// <param name="to">Point in time representing the end of the period for which results should be returned.</param>
142+
/// <param name="utcFrom">Point in time representing the start of the period for which results should be returned.</param>
143+
/// <param name="utcTo">Point in time representing the end of the period for which results should be returned.</param>
134144
/// <returns> An <see cref="IQueryable{T}" /> representing the entities present in a given time range.</returns>
135145
public static IQueryable<TEntity> TemporalContainedIn<TEntity>(
136146
this DbSet<TEntity> source,
137-
DateTime from,
138-
DateTime to)
147+
DateTime utcFrom,
148+
DateTime utcTo)
139149
where TEntity : class
140150
{
141151
Check.NotNull(source, nameof(source));
@@ -145,8 +155,8 @@ public static IQueryable<TEntity> TemporalContainedIn<TEntity>(
145155
return queryableSource.Provider.CreateQuery<TEntity>(
146156
GenerateRangeTemporalQueryRoot<TEntity>(
147157
queryableSource,
148-
from,
149-
to,
158+
utcFrom,
159+
utcTo,
150160
TemporalOperationType.ContainedIn)).AsNoTracking();
151161
}
152162

0 commit comments

Comments
 (0)