Skip to content

Commit d606795

Browse files
authored
Map date/time to DateOnly/TimeOnly by default (npgsql#5948)
Closes npgsql#5328
1 parent 2f0bdd1 commit d606795

10 files changed

+92
-91
lines changed

src/Npgsql/Internal/Converters/Temporal/DateConverters.cs

+20-20
Original file line numberDiff line numberDiff line change
@@ -4,88 +4,88 @@
44
// ReSharper disable once CheckNamespace
55
namespace Npgsql.Internal.Converters;
66

7-
sealed class DateTimeDateConverter(bool dateTimeInfinityConversions) : PgBufferedConverter<DateTime>
7+
sealed class DateOnlyDateConverter(bool dateTimeInfinityConversions) : PgBufferedConverter<DateOnly>
88
{
9-
static readonly DateTime BaseValue = new(2000, 1, 1, 0, 0, 0);
9+
static readonly DateOnly BaseValue = new(2000, 1, 1);
1010

1111
public override bool CanConvert(DataFormat format, out BufferRequirements bufferRequirements)
1212
{
1313
bufferRequirements = BufferRequirements.CreateFixedSize(sizeof(int));
1414
return format is DataFormat.Binary;
1515
}
1616

17-
protected override DateTime ReadCore(PgReader reader)
17+
protected override DateOnly ReadCore(PgReader reader)
1818
=> reader.ReadInt32() switch
1919
{
2020
int.MaxValue => dateTimeInfinityConversions
21-
? DateTime.MaxValue
21+
? DateOnly.MaxValue
2222
: throw new InvalidCastException(NpgsqlStrings.CannotReadInfinityValue),
2323
int.MinValue => dateTimeInfinityConversions
24-
? DateTime.MinValue
24+
? DateOnly.MinValue
2525
: throw new InvalidCastException(NpgsqlStrings.CannotReadInfinityValue),
26-
var value => BaseValue + TimeSpan.FromDays(value)
26+
var value => BaseValue.AddDays(value)
2727
};
2828

29-
protected override void WriteCore(PgWriter writer, DateTime value)
29+
protected override void WriteCore(PgWriter writer, DateOnly value)
3030
{
3131
if (dateTimeInfinityConversions)
3232
{
33-
if (value == DateTime.MaxValue)
33+
if (value == DateOnly.MaxValue)
3434
{
3535
writer.WriteInt32(int.MaxValue);
3636
return;
3737
}
3838

39-
if (value == DateTime.MinValue)
39+
if (value == DateOnly.MinValue)
4040
{
4141
writer.WriteInt32(int.MinValue);
4242
return;
4343
}
4444
}
4545

46-
writer.WriteInt32((value.Date - BaseValue).Days);
46+
writer.WriteInt32(value.DayNumber - BaseValue.DayNumber);
4747
}
4848
}
4949

50-
sealed class DateOnlyDateConverter(bool dateTimeInfinityConversions) : PgBufferedConverter<DateOnly>
50+
sealed class DateTimeDateConverter(bool dateTimeInfinityConversions) : PgBufferedConverter<DateTime>
5151
{
52-
static readonly DateOnly BaseValue = new(2000, 1, 1);
52+
static readonly DateTime BaseValue = new(2000, 1, 1, 0, 0, 0);
5353

5454
public override bool CanConvert(DataFormat format, out BufferRequirements bufferRequirements)
5555
{
5656
bufferRequirements = BufferRequirements.CreateFixedSize(sizeof(int));
5757
return format is DataFormat.Binary;
5858
}
5959

60-
protected override DateOnly ReadCore(PgReader reader)
60+
protected override DateTime ReadCore(PgReader reader)
6161
=> reader.ReadInt32() switch
6262
{
6363
int.MaxValue => dateTimeInfinityConversions
64-
? DateOnly.MaxValue
64+
? DateTime.MaxValue
6565
: throw new InvalidCastException(NpgsqlStrings.CannotReadInfinityValue),
6666
int.MinValue => dateTimeInfinityConversions
67-
? DateOnly.MinValue
67+
? DateTime.MinValue
6868
: throw new InvalidCastException(NpgsqlStrings.CannotReadInfinityValue),
69-
var value => BaseValue.AddDays(value)
69+
var value => BaseValue + TimeSpan.FromDays(value)
7070
};
7171

72-
protected override void WriteCore(PgWriter writer, DateOnly value)
72+
protected override void WriteCore(PgWriter writer, DateTime value)
7373
{
7474
if (dateTimeInfinityConversions)
7575
{
76-
if (value == DateOnly.MaxValue)
76+
if (value == DateTime.MaxValue)
7777
{
7878
writer.WriteInt32(int.MaxValue);
7979
return;
8080
}
8181

82-
if (value == DateOnly.MinValue)
82+
if (value == DateTime.MinValue)
8383
{
8484
writer.WriteInt32(int.MinValue);
8585
return;
8686
}
8787
}
8888

89-
writer.WriteInt32(value.DayNumber - BaseValue.DayNumber);
89+
writer.WriteInt32((value.Date - BaseValue).Days);
9090
}
9191
}

src/Npgsql/Internal/Converters/Temporal/TimeConverters.cs

+6-6
Original file line numberDiff line numberDiff line change
@@ -3,26 +3,26 @@
33
// ReSharper disable once CheckNamespace
44
namespace Npgsql.Internal.Converters;
55

6-
sealed class TimeSpanTimeConverter : PgBufferedConverter<TimeSpan>
6+
sealed class TimeOnlyTimeConverter : PgBufferedConverter<TimeOnly>
77
{
88
public override bool CanConvert(DataFormat format, out BufferRequirements bufferRequirements)
99
{
1010
bufferRequirements = BufferRequirements.CreateFixedSize(sizeof(long));
1111
return format is DataFormat.Binary;
1212
}
13-
protected override TimeSpan ReadCore(PgReader reader) => new(reader.ReadInt64() * 10);
14-
protected override void WriteCore(PgWriter writer, TimeSpan value) => writer.WriteInt64(value.Ticks / 10);
13+
protected override TimeOnly ReadCore(PgReader reader) => new(reader.ReadInt64() * 10);
14+
protected override void WriteCore(PgWriter writer, TimeOnly value) => writer.WriteInt64(value.Ticks / 10);
1515
}
1616

17-
sealed class TimeOnlyTimeConverter : PgBufferedConverter<TimeOnly>
17+
sealed class TimeSpanTimeConverter : PgBufferedConverter<TimeSpan>
1818
{
1919
public override bool CanConvert(DataFormat format, out BufferRequirements bufferRequirements)
2020
{
2121
bufferRequirements = BufferRequirements.CreateFixedSize(sizeof(long));
2222
return format is DataFormat.Binary;
2323
}
24-
protected override TimeOnly ReadCore(PgReader reader) => new(reader.ReadInt64() * 10);
25-
protected override void WriteCore(PgWriter writer, TimeOnly value) => writer.WriteInt64(value.Ticks / 10);
24+
protected override TimeSpan ReadCore(PgReader reader) => new(reader.ReadInt64() * 10);
25+
protected override void WriteCore(PgWriter writer, TimeSpan value) => writer.WriteInt64(value.Ticks / 10);
2626
}
2727

2828
sealed class DateTimeOffsetTimeTzConverter : PgBufferedConverter<DateTimeOffset>

src/Npgsql/Internal/ResolverFactories/AdoTypeInfoResolverFactory.Multirange.cs

+8-9
Original file line numberDiff line numberDiff line change
@@ -159,24 +159,23 @@ static TypeInfoMappingCollection AddMappings(TypeInfoMappingCollection mappings)
159159
CreateListMultirangeConverter(CreateRangeConverter(new Int8Converter<long>(), options), options)));
160160

161161
// datemultirange
162-
mappings.AddType<NpgsqlRange<DateTime>[]>(DataTypeNames.DateMultirange,
163-
static (options, mapping, _) =>
164-
mapping.CreateInfo(options, CreateArrayMultirangeConverter(
165-
CreateRangeConverter(new DateTimeDateConverter(options.EnableDateTimeInfinityConversions), options), options)),
166-
isDefault: true);
167-
mappings.AddType<List<NpgsqlRange<DateTime>>>(DataTypeNames.DateMultirange,
168-
static (options, mapping, _) =>
169-
mapping.CreateInfo(options, CreateListMultirangeConverter(
170-
CreateRangeConverter(new DateTimeDateConverter(options.EnableDateTimeInfinityConversions), options), options)));
171162
mappings.AddType<NpgsqlRange<DateOnly>[]>(DataTypeNames.DateMultirange,
172163
static (options, mapping, _) =>
173164
mapping.CreateInfo(options, CreateArrayMultirangeConverter(
174165
CreateRangeConverter(new DateOnlyDateConverter(options.EnableDateTimeInfinityConversions), options), options)),
175166
isDefault: true);
167+
mappings.AddType<NpgsqlRange<DateTime>[]>(DataTypeNames.DateMultirange,
168+
static (options, mapping, _) =>
169+
mapping.CreateInfo(options, CreateArrayMultirangeConverter(
170+
CreateRangeConverter(new DateTimeDateConverter(options.EnableDateTimeInfinityConversions), options), options)));
176171
mappings.AddType<List<NpgsqlRange<DateOnly>>>(DataTypeNames.DateMultirange,
177172
static (options, mapping, _) =>
178173
mapping.CreateInfo(options, CreateListMultirangeConverter(
179174
CreateRangeConverter(new DateOnlyDateConverter(options.EnableDateTimeInfinityConversions), options), options)));
175+
mappings.AddType<List<NpgsqlRange<DateTime>>>(DataTypeNames.DateMultirange,
176+
static (options, mapping, _) =>
177+
mapping.CreateInfo(options, CreateListMultirangeConverter(
178+
CreateRangeConverter(new DateTimeDateConverter(options.EnableDateTimeInfinityConversions), options), options)));
180179

181180
return mappings;
182181
}

src/Npgsql/Internal/ResolverFactories/AdoTypeInfoResolverFactory.Range.cs

+6-5
Original file line numberDiff line numberDiff line change
@@ -87,15 +87,16 @@ static TypeInfoMappingCollection AddMappings(TypeInfoMappingCollection mappings)
8787
static (options, mapping, _) => mapping.CreateInfo(options, CreateRangeConverter(new Int8Converter<long>(), options)));
8888

8989
// daterange
90+
mappings.AddStructType<NpgsqlRange<DateOnly>>(DataTypeNames.DateRange,
91+
static (options, mapping, _) =>
92+
mapping.CreateInfo(options,
93+
CreateRangeConverter(new DateOnlyDateConverter(options.EnableDateTimeInfinityConversions), options)),
94+
isDefault: true);
9095
mappings.AddStructType<NpgsqlRange<DateTime>>(DataTypeNames.DateRange,
9196
static (options, mapping, _) => mapping.CreateInfo(options,
92-
CreateRangeConverter(new DateTimeDateConverter(options.EnableDateTimeInfinityConversions), options)),
93-
isDefault: true);
97+
CreateRangeConverter(new DateTimeDateConverter(options.EnableDateTimeInfinityConversions), options)));
9498
mappings.AddStructType<NpgsqlRange<int>>(DataTypeNames.DateRange,
9599
static (options, mapping, _) => mapping.CreateInfo(options, CreateRangeConverter(new Int4Converter<int>(), options)));
96-
mappings.AddStructType<NpgsqlRange<DateOnly>>(DataTypeNames.DateRange,
97-
static (options, mapping, _) =>
98-
mapping.CreateInfo(options, CreateRangeConverter(new DateOnlyDateConverter(options.EnableDateTimeInfinityConversions), options)));
99100

100101
return mappings;
101102
}

src/Npgsql/Internal/ResolverFactories/AdoTypeInfoResolverFactory.cs

+7-6
Original file line numberDiff line numberDiff line change
@@ -230,14 +230,15 @@ static TypeInfoMappingCollection AddMappings(TypeInfoMappingCollection mappings)
230230
static (options, mapping, _) => mapping.CreateInfo(options, new Int8Converter<long>()));
231231

232232
// Date
233+
mappings.AddStructType<DateOnly>(DataTypeNames.Date,
234+
static (options, mapping, _) =>
235+
mapping.CreateInfo(options, new DateOnlyDateConverter(options.EnableDateTimeInfinityConversions)), isDefault: true);
233236
mappings.AddStructType<DateTime>(DataTypeNames.Date,
234237
static (options, mapping, _) =>
235238
mapping.CreateInfo(options, new DateTimeDateConverter(options.EnableDateTimeInfinityConversions)),
236239
MatchRequirement.DataTypeName);
237240
mappings.AddStructType<int>(DataTypeNames.Date,
238241
static (options, mapping, _) => mapping.CreateInfo(options, new Int4Converter<int>()));
239-
mappings.AddStructType<DateOnly>(DataTypeNames.Date,
240-
static (options, mapping, _) => mapping.CreateInfo(options, new DateOnlyDateConverter(options.EnableDateTimeInfinityConversions)));
241242

242243
// Interval
243244
mappings.AddStructType<TimeSpan>(DataTypeNames.Interval,
@@ -246,12 +247,12 @@ static TypeInfoMappingCollection AddMappings(TypeInfoMappingCollection mappings)
246247
static (options, mapping, _) => mapping.CreateInfo(options, new NpgsqlIntervalConverter()));
247248

248249
// Time
250+
mappings.AddStructType<TimeOnly>(DataTypeNames.Time,
251+
static (options, mapping, _) => mapping.CreateInfo(options, new TimeOnlyTimeConverter()), isDefault: true);
249252
mappings.AddStructType<TimeSpan>(DataTypeNames.Time,
250-
static (options, mapping, _) => mapping.CreateInfo(options, new TimeSpanTimeConverter()), isDefault: true);
253+
static (options, mapping, _) => mapping.CreateInfo(options, new TimeSpanTimeConverter()));
251254
mappings.AddStructType<long>(DataTypeNames.Time,
252255
static (options, mapping, _) => mapping.CreateInfo(options, new Int8Converter<long>()));
253-
mappings.AddStructType<TimeOnly>(DataTypeNames.Time,
254-
static (options, mapping, _) => mapping.CreateInfo(options, new TimeOnlyTimeConverter()));
255256

256257
// TimeTz
257258
mappings.AddStructType<DateTimeOffset>(DataTypeNames.TimeTz,
@@ -446,9 +447,9 @@ static TypeInfoMappingCollection AddMappings(TypeInfoMappingCollection mappings)
446447
mappings.AddStructArrayType<long>(DataTypeNames.TimestampTz);
447448

448449
// Date
450+
mappings.AddStructArrayType<DateOnly>(DataTypeNames.Date);
449451
mappings.AddStructArrayType<DateTime>(DataTypeNames.Date);
450452
mappings.AddStructArrayType<int>(DataTypeNames.Date);
451-
mappings.AddStructArrayType<DateOnly>(DataTypeNames.Date);
452453

453454
// Interval
454455
mappings.AddStructArrayType<TimeSpan>(DataTypeNames.Interval);

test/Npgsql.Tests/CommandBuilderTests.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -341,7 +341,7 @@ PRIMARY KEY (Cod)
341341

342342
Assert.That(row[0], Is.EqualTo("key1"));
343343
Assert.That(row[1], Is.EqualTo("description"));
344-
Assert.That(row[2], Is.EqualTo(new DateTime(2018, 7, 3)));
344+
Assert.That(row[2], Is.EqualTo(new DateOnly(2018, 7, 3)));
345345
Assert.That(row[3], Is.EqualTo(new DateTime(2018, 7, 3, 7, 2, 0)));
346346
Assert.That(row[4], Is.EqualTo(123));
347347
Assert.That(row[5], Is.EqualTo(123.4));

test/Npgsql.Tests/ExceptionTests.cs

+3
Original file line numberDiff line numberDiff line change
@@ -250,6 +250,8 @@ PostgresException CreateWithSqlState(string sqlState)
250250

251251
#pragma warning disable SYSLIB0011
252252
#pragma warning disable SYSLIB0050
253+
254+
#if !NET9_0_OR_GREATER // BinaryFormatter serialization and deserialization have been removed. See https://aka.ms/binaryformatter for more information.
253255
[Test]
254256
public void Serialization()
255257
{
@@ -283,6 +285,7 @@ public void Serialization()
283285
Assert.That(expected.Line, Is.EqualTo(actual.Line));
284286
Assert.That(expected.Routine, Is.EqualTo(actual.Routine));
285287
}
288+
#endif
286289

287290
SerializationInfo CreateSerializationInfo() => new(typeof(PostgresException), new FormatterConverter());
288291
#pragma warning restore SYSLIB0011

test/Npgsql.Tests/ReaderTests.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -475,7 +475,7 @@ public async Task GetValues()
475475
dr.Read();
476476
var values = new object[4];
477477
Assert.That(dr.GetValues(values), Is.EqualTo(3));
478-
Assert.That(values, Is.EqualTo(new object?[] { "hello", 1, new DateTime(2014, 1, 1), null }));
478+
Assert.That(values, Is.EqualTo(new object?[] { "hello", 1, new DateOnly(2014, 1, 1), null }));
479479
}
480480
using (var dr = await command.ExecuteReaderAsync(Behavior))
481481
{

0 commit comments

Comments
 (0)