-
-
Notifications
You must be signed in to change notification settings - Fork 3.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Runtime error when using DateOnly #1728
Comments
Here is what I think can fix the problem. Under static SqlMapper() in SQLMapper.cs file, if you add DateOnly and DateOnly?, that will fix the problem. |
There's also a branch (and possibly PR) that does this, but I was waiting on .NET 6 GA (which has now passed, so: yay). So: what RDBMS/provider are you seeing this with, so I can validate? |
I am using the sql server one. |
There are 2 official SQL Server drivers - System.Data.SqlClient, and Microsoft.Data.SqlClient; which are you using, and exactly what version? |
Sorry, I should have added: I don't mean to sound picky - my aim is to add
the relevant tests and deploy this tomorrow, so anything I can discover to
help shorten that: is good
…On Sun, 21 Nov 2021, 19:15 musictopia2, ***@***.***> wrote:
I am using the sql server one.
—
You are receiving this because you commented.
Reply to this email directly, view it on GitHub
<#1728 (comment)>,
or unsubscribe
<https://github.com/notifications/unsubscribe-auth/AAAEHMEANBB43LK6BFT3KB3UNFAMZANCNFSM5IPHCV7Q>
.
Triage notifications on the go with GitHub Mobile for iOS
<https://apps.apple.com/app/apple-store/id1477376905?ct=notification-email&mt=8&pt=524675>
or Android
<https://play.google.com/store/apps/details?id=com.github.android&referrer=utm_campaign%3Dnotification-email%26utm_medium%3Demail%26utm_source%3Dgithub>.
|
The version I am using is this one. |
Microsoft.Data.SqlClient |
looks like they have that one up to 4.0.0. |
So: I updated my test harness, and: it simply doesn't work for either SqlClient; both give an exception like: Message:
System.ArgumentException : No mapping exists from object type System.DateOnly to a known managed provider native type.
Stack Trace:
MetaType.GetMetaTypeFromValue(Type dataType, Object value, Boolean inferLen, Boolean streamAllowed)
MetaType.GetMetaTypeFromType(Type dataType)
SqlParameter.GetMetaTypeOnly()
SqlParameter.Validate(Int32 index, Boolean isCommandProc)
SqlCommand.BuildParamList(TdsParser parser, SqlParameterCollection parameters, Boolean includeReturnValue)
SqlCommand.BuildExecuteSql(CommandBehavior behavior, String commandText, SqlParameterCollection parameters, _SqlRPC& rpc)
SqlCommand.RunExecuteReaderTds(CommandBehavior cmdBehavior, RunBehavior runBehavior, Boolean returnStream, Boolean isAsync, Int32 timeout, Task& task, Boolean asyncWrite, Boolean inRetry, SqlDataReader ds, Boolean describeParameterEncryptionRequest)
SqlCommand.RunExecuteReader(CommandBehavior cmdBehavior, RunBehavior runBehavior, Boolean returnStream, TaskCompletionSource`1 completion, Int32 timeout, Task& task, Boolean& usedCache, Boolean asyncWrite, Boolean inRetry, String method)
SqlCommand.RunExecuteReader(CommandBehavior cmdBehavior, RunBehavior runBehavior, Boolean returnStream, String method)
SqlCommand.ExecuteReader(CommandBehavior behavior)
SqlCommand.ExecuteDbDataReader(CommandBehavior behavior) Do you have an example of it actually working with raw ADO.NET? |
(to be clear: I mean "the underlying ADO.NET provider doesn't work with |
Cross-referencing: dotnet/SqlClient#1009 |
Do you think there is a way for dapper to do some type of version so if dateonly was used, then it can somehow go to the external as datetime behind the scenes. Because not only its possible they may not ever fix it, but i found the newest version of microsofts version of the sql provider only works if a person uses ssl on a webpage which would not work if somebody had an intranet that is only for people on the local network with no connection to the internet. |
I see an issue that was posted here dotnet/efcore#24507 where they showed a converter a person can use as a temporary workaround in entity framework core. Can something like this be done to dapper? Would be disappointing if a person is forced to use entity framework core. If that is the case, then dapper would not work so well since without the dateonly support, businesses are blocked from even moving forward. |
I don't want to introduce lots of magic here. In particular, the "does it work or does it need a shim" question would be vendor-specific. I see no good sides to adding some hidden translation here - it seems.like that is something that can, and should, be done at the caller: so they know, with confidence, what they are sending. Not doing too much unpredictable voodoo is an unwritten design goal of Dapper. So no, I'd rather not do that. |
Any suggestion of a workaround while waiting for vendors to fix them. What if no vendors ever fix them? would be disappointing to have the new datatypes but the catch of not being able to use them in any database. |
FYI the new DateOnly and TimeOnly types are supported by Microsoft.Data.Sqlite, MySql and Npgsql (see this tracking issue). I agree with @mgravell that it's not Dapper's job to do hidden conversions etc. - it's the ADO.NET provider's job to support these types. |
@roji didn't realize it was in Npgsql already; I can use you for my smoke-test, then :) but: as discussed above, it'll be direct pass-thru, no magic |
Yep, exactly... Let me know if you run into any trouble! |
I've just encountered this issue on SQLite as well. I just wanted to share my quick work-around here, until the ADO provider for SQLite is fixed. Solution that worked for me was to change the POCO like this:
Then just change the SELECT statement to "SELECT Date AS DateString FROM MyTable" The rest of my code can still use the Date property. |
So, seeing as people are still bumping this, I thought I'd dust off the PR again Updates:
So for SQLite, this works: [Fact]
public void TestDateOnlyTimeOnly()
{
var now = DateTime.UtcNow;
var args = new { day = DateOnly.FromDateTime(now), time = TimeOnly.FromDateTime(now) };
var result = connection.QuerySingle("select @day as day, @time as time", args);
var day = DateOnly.Parse(Assert.IsType<string>(result.day));
var time = TimeOnly.Parse(Assert.IsType<string>(result.time));
Assert.Equal(args.day, day);
Assert.Equal(args.time, time);
} and for Postgresql, this needs rounding due to losing millisecond precision [Fact]
public void TestDateOnlyTimeOnly()
{
var now = DateTime.UtcNow;
var args = new { day = DateOnly.FromDateTime(now), time = TimeOnly.FromDateTime(now) };
var result = connection.QuerySingle("select @day::date as day, @time::time as time", args);
var day = DateOnly.FromDateTime(Assert.IsType<DateTime>(result.day));
var time = TimeOnly.FromTimeSpan(Assert.IsType<TimeSpan>(result.time));
Assert.Equal(args.day, day);
Assert.Equal(Round(args.time), Round(time));
static TimeOnly Round(TimeOnly value)
=> new TimeOnly(value.Hour, value.Minute, value.Second);
} @roji any comments on ^^^? am I doing it wrong? I thought Also: I need to put some more thoughts into how we handle return types here, i.e. should Dapper add translations between expected types (string and DateTime to DateOnly, string and TimeSpan to TimeOnly)? |
@mgravell re return types, yeah - that's expected; I didn't change the default return type for PostgreSQL Re the precision issue, can you provide more detail, ideally with an ADO.NET repro? Here's some ADO.NET code that shows what I think is correct behavior: await using var conn = new NpgsqlConnection("Host=localhost;Username=test;Password=test");
await conn.OpenAsync();
var expected = TimeOnly.MaxValue;
await using var command = new NpgsqlCommand("SELECT @p", conn)
{
Parameters = { new("p", expected) }
};
await using var reader = await command.ExecuteReaderAsync();
await reader.ReadAsync();
var actual = reader.GetFieldValue<TimeOnly>(0);
Console.WriteLine($"Actual: {actual:o}");
Console.WriteLine($"Expected: {expected:o}"); The results are:
The discrepancy is normal, since .NET has tick precision (100ns) whereas PostgreSQL has microsecond precision (1000ns). Are you seeing something different? |
As for more thoughts on how to handle this; I don't know if this would be considered "doing too much", but I'd kind of like the idea of member attributes. Where you'd have something like this
Where a static method needs to be passed in with exactly one argument. It's fairly trivial to write your own static method if you need something more complex. |
@roji ah, ta; I changed the rounding to millis and it still passes, so: great! Re the |
Yeah, makes sense. Let me know if you need anything else. |
Any workarounds for this? |
Unfortunately, there is still none unfortunately. |
For both do not work |
FYI I am using |
Probably not relevant but i've come across this https://learn.microsoft.com/en-us/ef/core/what-is-new/ef-core-8.0/whatsnew#dateonlytimeonly-supported-on-sql-server It sound like a "recent release of a [Microsoft.Data.SqlClient]" added features that made it possible for someone to add EF support for these types, maybe the change to Microsoft.Data.SqlClient may also allow dapper to support it with some work? "For SQL Server, the recent release of a Microsoft.Data.SqlClient package targeting .NET 6 has allowed ErikEJ to add support for these types at the ADO.NET level. This in turn paved the way for support in EF8 for DateOnly and TimeOnly as properties in entity types." |
I have a database that has a date property. However, the C# class is DateOnly since I only need the date, not the time. However, when I try to use a sql command and the c# class is DateOnly, the error is something like this
The member WhatDate1 of type System.DateOnly cannot be used as a parameter value
at Dapper.SqlMapper.LookupDbType(Type type, String name, Boolean demand, ITypeHandler& handler) in /_/Dapper/SqlMapper.cs:line 426
WhatDate1 is the field name
The text was updated successfully, but these errors were encountered: