FSharp.Data.SqlClient icon indicating copy to clipboard operation
FSharp.Data.SqlClient copied to clipboard

Conversion failed when using User Defined Table Types

Open daniellittledev opened this issue 3 years ago • 1 comments

Issue Summary

When using a User Defined Table Type it appears to write the value to the wrong column, resulting in a database conversion error. Using the SQL profiler I can see this error occurs before executing the SQL and only when the @Parameter has one or more items. If it is empty or removed the command executes successfully.

Error

System.Data.SqlClient.SqlException: 'Conversion failed when converting the nvarchar value 'TestName' to data type int.
The statement has been terminated.'

To Reproduce

Possibly share SQL Schema, snippets of data, and how you call it in F#.

Sample to reproduce the behavior:

Table and Table Type

Create Table [TestTable] (
	[C1] UniqueIdentifier Not Null,
	[C2] UniqueIdentifier Not Null,
	[C3] UniqueIdentifier Not Null,

	[C4] NVarChar(100) Not Null,
	[C5] NVarChar(100) Not Null,
	[C6] NVarChar(100) Null,
	[C7] NVarChar(1000) Not Null,

	[C8] Int Not Null,

	[C9] VarChar(10) Not Null,
	[C10] UniqueIdentifier Not Null,
	[C11] UniqueIdentifier Not Null,

	Constraint [PK_Id] Primary Key Clustered([C1], [C2], [C3])
);
Create Type [dbo].[TestTableItem] AS Table(
	[C3] UniqueIdentifier Not Null,
	[C5] NVarChar(100) Not Null,
	[C6] NVarChar(100) Null,
	[C7] NVarChar(1000) Not Null,
	[C8] Int Not Null,
	[C9] VarChar(10) Not Null,
	[C10] UniqueIdentifier Not Null
);

SqlCommandProvider

type AddSqlCommand = SqlCommandProvider<"
    Insert [TestTable]
    (
        [C1],
        [C2],
        [C11],
        [C4],
        [C2],
        [C5],
        [C6],
        [C7],
        [C8],
        [C9],
        [C10]
    )
    Select
        @C1,
        @C2,
        @C11,
        @C4,
        x.[C3],
        x.[C5],
        x.[C6],
        x.[C7],
        x.[C8],
        x.[C9],
        x.[C10]
    From @C12 x;
    " , staticConnectionString, TableVarMapping = "@C12=TestTableItem">

Usage

use cmd = new AddSqlCommand(connection, transaction = transaction)

let items =
    [
        AddSqlCommand.TestTableItem(
            C3 = Guid.NewGuid(),
            C5 = "TestName",
            C6 = "C6",
            C7 = "C7",
            C8 = 1,
            C9 = "C9",
            C10 = Guid.NewGuid()
        )
    ]

return!
    cmd.AsyncExecute(
        C1 = Guid.NewGuid(),
        C2 = Guid.NewGuid(),
        C4 = "C4 Name",
        C11 = Guid.NewGuid(),
        C12 = items
    )

Expected behavior

The query should successfully insert a new row.

What you can do

  • [x] I am willing to contribute a PR with a unit test showcasing the issue
  • [x] I am willing to test the bug fix before next release

daniellittledev avatar Nov 29 '22 00:11 daniellittledev

@daniellittledev I've tried to reproduce the issue in #425 but the added test doesn't throw a SqlException, do you mind trying it out and adjust the test accordingly to reproduce your issue?

smoothdeveloper avatar Dec 03 '22 11:12 smoothdeveloper