Skip to content
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

Open
musictopia2 opened this issue Nov 21, 2021 · 29 comments
Open

Runtime error when using DateOnly #1728

musictopia2 opened this issue Nov 21, 2021 · 29 comments
Labels
blocked:external waiting on external factors outside of our control feature request

Comments

@musictopia2
Copy link

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

@musictopia2
Copy link
Author

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.
I would also need TimeOnly and TimeOnly? to be added for cases where only time is needed.

@mgravell
Copy link
Member

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?

@musictopia2
Copy link
Author

I am using the sql server one.

@mgravell
Copy link
Member

There are 2 official SQL Server drivers - System.Data.SqlClient, and Microsoft.Data.SqlClient; which are you using, and exactly what version?

@mgravell
Copy link
Member

mgravell commented Nov 21, 2021 via email

@musictopia2
Copy link
Author

The version I am using is this one.

@musictopia2
Copy link
Author

Microsoft.Data.SqlClient
version 1.1.1
Tried to add the xml but that was deleted.

@musictopia2
Copy link
Author

looks like they have that one up to 4.0.0.
I can use that one though if that would help as well.

@mgravell
Copy link
Member

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?

@mgravell
Copy link
Member

mgravell commented Nov 22, 2021

(to be clear: I mean "the underlying ADO.NET provider doesn't work with DateOnly/TimeOnly", not "Dapper doesn't ...")

@mgravell
Copy link
Member

Cross-referencing: dotnet/SqlClient#1009

@mgravell mgravell added feature request blocked:external waiting on external factors outside of our control labels Nov 22, 2021
@musictopia2
Copy link
Author

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.

@musictopia2
Copy link
Author

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.

@mgravell
Copy link
Member

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.

@musictopia2
Copy link
Author

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.

@roji
Copy link

roji commented Nov 26, 2021

What if no vendors ever fix them?

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.

@mgravell
Copy link
Member

@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

@roji
Copy link

roji commented Nov 26, 2021

Yep, exactly... Let me know if you run into any trouble!

@Rainmaker52
Copy link

Rainmaker52 commented Dec 9, 2021

I've just encountered this issue on SQLite as well.
Column in the database is TEXT, as SQLite does not have a date/time/datetime equivalent. Date is stored in ISO8601 format (YYYY-MM-dd), which DateOnly.Parse() accepts.

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:

internal record MyPOCO{
    internal string DateString { get; init; }
    internal DateOnly Date { 
        get
        {
            return DateOnly.Parse(this.DateString);
        }
    }
}

Then just change the SELECT statement to "SELECT Date AS DateString FROM MyTable"

The rest of my code can still use the Date property.

@mgravell
Copy link
Member

mgravell commented Dec 9, 2021

So, seeing as people are still bumping this, I thought I'd dust off the PR again

Updates:

  • SQLite v6 now works for parameters, but values come back as string, because: obviously why wouldn't it
  • Npgsql v6.0.1 now works for parameters, but values come back as DateTime/TimeSpan and there are some precision differences

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 time had microsecond precision?

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 added a commit that referenced this issue Dec 9, 2021
@roji
Copy link

roji commented Dec 9, 2021

@mgravell re return types, yeah - that's expected; I didn't change the default return type for PostgreSQL date and time to avoid breaking people (it would also mean the driver would have returned different types across different .NET TFMs). So these still return DateTime and TimeSpan, respectively. You can use GetFieldValue<DateOnly> to get what you want - hopefully that's something Dapper can do. I this this would also solve the SQLite-side issue (where there's no database-side types at all for these).

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:

Actual:   23:59:59.9999990
Expected: 23:59:59.9999999

The discrepancy is normal, since .NET has tick precision (100ns) whereas PostgreSQL has microsecond precision (1000ns). Are you seeing something different?

@Rainmaker52
Copy link

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

    [DapperSerialize(Convert.ToString)]
    [DapperDeserialize(DateOnly.Parse)]
    internal string DateString { get; init; }

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.

@mgravell
Copy link
Member

mgravell commented Dec 9, 2021

@roji ah, ta; I changed the rounding to millis and it still passes, so: great!

Re the GetFieldValue<>() - that's a bigger change; I'll need to think; maybe this is a good time to code a list of types that should use that approach. I also need to fix SQLite for this scenario, so... fun!

@roji
Copy link

roji commented Dec 9, 2021

Yeah, makes sense. Let me know if you need anything else.

@Anarios
Copy link

Anarios commented Mar 12, 2022

Any workarounds for this?

@musictopia2
Copy link
Author

Unfortunately, there is still none unfortunately.

@zanyar3
Copy link

zanyar3 commented Nov 30, 2022

There are 2 official SQL Server drivers - System.Data.SqlClient, and Microsoft.Data.SqlClient; which are you using, and exactly what version?

For both do not work

@celluj34
Copy link

celluj34 commented Mar 1, 2023

FYI I am using "Microsoft.Data.SqlClient" Version="5.1.0". I have introduced the type handlers from #1715 (comment) and they are working great - any chance we can get them in Dapper directly?

@buzz100
Copy link

buzz100 commented Apr 14, 2023

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."

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
blocked:external waiting on external factors outside of our control feature request
Projects
None yet
Development

No branches or pull requests

8 participants