-
Notifications
You must be signed in to change notification settings - Fork 292
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
LegacyRowVersionNullBehavior causes InvalidCastException when using SqlDataReader.GetFieldValue[T] #1175
Comments
I've added another test case that uses I don't think the |
I did add tests with the feature but if something has been missed can you take a look and see if you can help extend the test coverage with your failing scenarios? #998 I'll take a look when I can get time and see if there's a simple fix. |
Hi, I'm struggling to get the solution to build on my Linux development machine. I'll see if I can get it to build on Windows machine - I'll report back once I've done that. However, I've been looking at the change to SqlDataReader, assuming the problem lies in this class, I think the change that should've been made to maintain backwards compatibility is this:
This way, the code execution heads into the |
Building can be hard work to setup and isn't truly required if you can just work up tests to add alongside the ones already present trying to cover this feature. For example take [ConditionalFact(typeof(DataTestUtility), nameof(DataTestUtility.AreConnStringsSetup))]
public static void CheckNullRowVersionIsBDNull()
{
lock (s_rowVersionLock)
{
bool? originalValue = SetLegacyRowVersionNullBehaviour(false);
try
{
using (SqlConnection con = new SqlConnection(DataTestUtility.TCPConnectionString))
{
con.Open();
using (SqlCommand command = con.CreateCommand())
{
command.CommandText = "select cast(null as rowversion) rv";
using (SqlDataReader reader = command.ExecuteReader())
{
reader.Read();
Assert.True(reader.IsDBNull(0));
Assert.Equal(reader[0], DBNull.Value);
}
}
}
}
finally
{
SetLegacyRowVersionNullBehaviour(originalValue);
}
}
} As a template and change it to cover the things that currently aren't working, positive and negative cases for both legacy and new behaviour if possible. I can then drop those in and use them to explore and alter the behaviour. |
Hi, I've come up with these three tests: [ConditionalFact(typeof(DataTestUtility), nameof(DataTestUtility.AreConnStringsSetup))]
public static void CheckGetValuesNullRowVersionReturnsEmptyByteArray()
{
lock (s_rowVersionLock)
{
bool? originalValue = SetLegacyRowVersionNullBehaviour(true);
try
{
using (SqlConnection con = new SqlConnection(DataTestUtility.TCPConnectionString))
{
con.Open();
using (SqlCommand command = con.CreateCommand())
{
command.CommandText = "select cast(null as rowversion) rv";
using (SqlDataReader reader = command.ExecuteReader())
{
reader.Read();
var data = new object[1];
reader.GetValues(data);
Assert.True(data[0] is byte[] {Length: 0});
}
}
}
}
finally
{
SetLegacyRowVersionNullBehaviour(originalValue);
}
}
} [ConditionalFact(typeof(DataTestUtility), nameof(DataTestUtility.AreConnStringsSetup))]
public static void CheckGetValueNullRowVersionReturnsEmptyByteArray()
{
lock (s_rowVersionLock)
{
bool? originalValue = SetLegacyRowVersionNullBehaviour(true);
try
{
using (SqlConnection con = new SqlConnection(DataTestUtility.TCPConnectionString))
{
con.Open();
using (SqlCommand command = con.CreateCommand())
{
command.CommandText = "select cast(null as rowversion) rv";
using (SqlDataReader reader = command.ExecuteReader())
{
reader.Read();
var value = reader.GetValue(0);
Assert.True(value is byte[] {Length: 0});
}
}
}
}
finally
{
SetLegacyRowVersionNullBehaviour(originalValue);
}
}
} [ConditionalFact(typeof(DataTestUtility), nameof(DataTestUtility.AreConnStringsSetup))]
public static void CheckGetFieldValueNullRowVersionDoesNotThrow()
{
lock (s_rowVersionLock)
{
bool? originalValue = SetLegacyRowVersionNullBehaviour(true);
try
{
using (SqlConnection con = new SqlConnection(DataTestUtility.TCPConnectionString))
{
con.Open();
using (SqlCommand command = con.CreateCommand())
{
command.CommandText = "select cast(null as rowversion) rv";
using (SqlDataReader reader = command.ExecuteReader())
{
reader.Read();
var ex = Record.Exception(() => reader.GetFieldValue<byte[]>(0));
Assert.Null(ex);
}
}
}
}
finally
{
SetLegacyRowVersionNullBehaviour(originalValue);
}
}
} I've added the equivalent tests to my repo: andygjp/NullRowVersionIssue@f282a1f |
@andygjp thanks for the repro tests. I spent a short time yesterday on the failures and the behavior is as you described with or without the switch as far as I could see. We will get back to you shortly with an explanation if this is by design or a fix to address your concern. @Wraith2 is this something you like to work on or should we continue looking into this? |
I intend to look at it so you can assign it to me. |
@Wraith2 thanks for the continuous support. |
When I revert the change |
Ok, prospective fix is: switch (columnMetaData.type)
{
case SqlDbType.Timestamp: // rowversion has special behavior
if (isNull && !LocalAppContextSwitches.LegacyRowVersionNullBehavior)
{
_data[i].SetToNullOfType(SqlBuffer.StorageType.SqlBinary);
}
else
{
goto ReadDataValue;
}
break;
default:
if (isNull)
{
goto GetNullValue;
}
else
{
goto ReadDataValue;
}
ReadDataValue: // read the data value
if ((i > _sharedState._nextColumnDataToRead) || (!readHeaderOnly))
{
// If we're not in sequential access mode, we have to
// save the data we skip over so that the consumer
// can read it out of order
if (!_parser.TryReadSqlValue(_data[_sharedState._nextColumnDataToRead], columnMetaData, (int)dataLength, _stateObj,
_command != null ? _command.ColumnEncryptionSetting : SqlCommandColumnEncryptionSetting.UseConnectionSetting,
columnMetaData.column, _command))
{ // will read UDTs as VARBINARY.
return false;
}
_sharedState._nextColumnDataToRead++;
}
else
{
_sharedState._columnDataBytesRemaining = (long)dataLength;
}
break;
GetNullValue: // fetch a correctly typed null value
TdsParser.GetNullSqlValue(_data[_sharedState._nextColumnDataToRead],
columnMetaData,
_command != null ? _command.ColumnEncryptionSetting : SqlCommandColumnEncryptionSetting.UseConnectionSetting,
_parser.Connection);
if (!readHeaderOnly)
{
_sharedState._nextColumnDataToRead++;
}
break;
} but I'm not sure how other people feel about using gotos in switches. Done this way each value is only checked once so we don't waste time comparing things. It's a bit jumpy but cleaner than trying to express the same thing in pure if statements, you can clearly see the decisions being made and then named areas to implement them. What do you think @JRahnama ? |
Goto !? |
Maybe move GetNullValue up inline? |
Yes. Sometimes I find that it's the cleanest way to express things. |
Makes sense, and this code must be highly performant... |
I don't think a lot of changes are needed, issue was related to bug in if blocks. Edit: I ended up fixing it properly as cast between |
Describe the bug
I updated my project, which uses EF Core 5.0.8, to use the latest version SqlClient, 3.0.0, and got an InvalidCastException exception. Turning off
SqlServerDbContextOptionsBuilder.EnableRetryOnFailure()
prevents the error, but I need it on.Investigating further, the issue seems to come from
SqlDataReader.GetFieldValue[T](Int32 i)
.To reproduce
I've recreated the issue in a repo: https://github.com/andygjp/NullRowVersionIssue
Expected behavior
It should not throw an exception - the GetFieldValueFromSqlBufferInternal() function needs to have the same behaviour as TryReadColumnInternal(Int32, Bool), ie respect LocalAppContextSwitches.LegacyRowVersionNullBehavior
Further technical details
The text was updated successfully, but these errors were encountered: