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

Bug: Int Enum with Flags can't read from database if more than one flag is set #814

Open
SpaceOgre opened this issue Apr 16, 2021 · 10 comments
Assignees
Labels
bug Something isn't working

Comments

@SpaceOgre
Copy link
Contributor

SpaceOgre commented Apr 16, 2021

Bug Description

I'm using int Enums with the Flags attribute and inserting with ExecuteNonQueryAsync and AnonymouseType is working but when I try and read a row that have more than one flag set I get the following crash.

It might be connected to #624.

This is how i do my setup:

RepoDb.SqlServerBootstrap.Initialize();
RepoDb.Converter.EnumDefaultDatabaseType = DbType.Int32;

And call to database:

await connection.ExecuteQueryAsync<Customer>("SELECT * FROM dbo.Customer");

Exception Message:

System.ArgumentNullException: Value cannot be null. (Parameter 'value')
   at System.Enum.TryParse(Type enumType, String value, Boolean ignoreCase, Boolean throwOnFailure, Object& result)
   at lambda_method89(Closure , DbDataReader )
   at RepoDb.Reflection.DataReader.ToEnumerableAsync[TResult](DbDataReader reader, IEnumerable`1 dbFields, IDbSetting dbSetting, CancellationToken cancellationToken)
   at RepoDb.DbConnectionExtension.ExecuteQueryAsyncInternalForType[TResult](IDbConnection connection, String commandText, Object param, Nullable`1 commandType, String cacheKey, Nullable`1 cacheItemExpiration, Nullable`1 commandTimeout, IDbTransaction transaction, ICache cache, CancellationToken cancellationToken, String tableName, Boolean skipCommandArrayParametersCheck)

Schema and Model:

Please share to us the schema of the table (not actual) that could help us replicate the issue if necessary.

CREATE TABLE dbo.Customer
(
  [CustomerNumber] int NOT NULL IDENTITY(1,1) PRIMARY KEY CLUSTERED
  , [ContractOwner] int NOT NULL
);

And also the model that corresponds the schema.

    public abstract record Customer
    {
        public int CustomerNumber { get; init; }
        public ContractOwner ContractOwner { get; init; }
    }

    [Flags]
    public enum ContractOwner
    {
        None = 0,
        Alfa = 1,
        Beta = 2,
        Other = 4
    }

Library Version:

Example: RepoDb v1.12.7 and RepoDb.SqlServer v1.1.3

@SpaceOgre SpaceOgre added the bug Something isn't working label Apr 16, 2021
@SpaceOgre
Copy link
Contributor Author

Just tested it with using nvarchar instead if int and that works, so it seems to be a problem with Flags and Enums as int.

@SpaceOgre
Copy link
Contributor Author

Something weird is going on I think, if I just change the database column from int to nvarchar then i works without doing anything else, RepoDb.Converter.EnumDefaultDatabaseType = DbType.Int32; is still set.
So the value in the database is still an int but the type is nvarchar.

@mikependon
Copy link
Owner

Interesting. We will investigate and get back to you in relation to this. Is this a blocker on your side? Or, can you utilize the NVARCHAR type for now?

@SpaceOgre
Copy link
Contributor Author

No blocker, the NVARCHAR works for now :) and nothing that is going to production anytime soon.

@SpaceOgre
Copy link
Contributor Author

@mikependon I might have to bump this to a bit more critical, we are now having problems with BulkInsertAsync and this.
When we do our normal insert we don't insert the flag column, that is handled later via an update like this:

        public async Task UpdateContractOwner(int customerNumber, ContractOwner contractOwner)
        {
            using var connection = new SqlConnection(ConnectionString);
            await connection.ExecuteNonQueryAsync(@"
                UPDATE dbo.Customer SET
                    ContractOwner = @contractOwner
                WHERE CustomerNumber = @customerNumber"
            , new { customerNumber, contractOwner });
        }

This inserts an int even though it is an NVARCHAR in the database, which is what we want. I guess this works is because there is no tabel mapping going on.

But we also have a migration stage that is using BulkInsertAsync with Entity mapping, this is now inserting the names of the flags rather than the int value and we are getting a crash since we only did NVARCHAR(2). We can ofc increase the NVARCHAR to a value that can fit all combinations but we really want to use int.

Is there some way we can force the BulkInsert to use Int event though the database is using NVARCHAR? The [TypeMap(DbType.Int32)] attribute does not help.

@mikependon
Copy link
Owner

By default, the library is using the string types when pushing the data into the database, specifically if it uses the Execute<Methods> and if your column type in the data database is NVARCHAR. If you uses the Execute<Methods> with INT column type in the database, it should throw an exception (unless you explicitly set the default enum database type to DbType.Int32 as explained here).

Note: You do not need to define the database type if you are using the model-based methods.

In your case, we cannot do any further for BulkInsert as it is the default behavior of the underlying mechanism which is the ADO .NET. I would strongly suggest that you use either of the approaches below.

  • Make a model that has an INT property and use it for bulk insert (you can also use anonymous type)
  • Or, consider using the string type enumeration when saving to database (there should be no more action item here, as per your case)

But since you mentioned that when you call the ExecuteNonQueryAsync it saves an INT type, it sounds to me that you are using the older version of the library, I really believe that this should not be the behavior. The Execute<Methods> should pass the string representation of the ENUM into the database as explained above. Or, ensure to use the latest beta version.

@mikependon
Copy link
Owner

Oops, wait, I can replicate your problem if the enum is marked as Flags. We will look into this.

@mikependon
Copy link
Owner

I think, the ideal fix for this should save the string representation of the bit-wise enum (i.e: "Apple | Banana | Orange") if the underlying column is of NVARCHAR type, otherwise, it will save the INT type. This is now true to the normal enumeration, which is not a bit-wise.

@SpaceOgre
Copy link
Contributor Author

Yes that is how I would like for it to work as well, but I think it needs to respect the DbType set by Fluent, Attribut, Default etc as well.

@mikependon mikependon pinned this issue Jul 30, 2021
@mikependon mikependon unpinned this issue Sep 15, 2021
@w5l
Copy link

w5l commented Mar 2, 2022

Just ran into this as well.

Given the following enum, and an int database column with value 5 (so A+C).

[Flags]
enum ExampleFlagEnum
{
    A = 1 << 0,
    B = 1 << 1,
    C = 1 << 2
}

Throws an error:

   at System.Enum.EnumResult.SetFailure(ParseFailureKind failure, String failureParameter)
   at System.Enum.TryParseEnum(Type enumType, String value, Boolean ignoreCase, EnumResult& parseResult)
   at System.Enum.Parse(Type enumType, String value, Boolean ignoreCase)
   at RepoDb.Reflection.DataReader.<ToEnumerableAsync>d__1`1.MoveNext()

I've narrowed that down to the following code:

internal static Expression ConvertExpressionToEnumExpressionForNonString(Expression expression,
Type toEnumType)
{
var method = StaticType.Enum.GetMethod("GetName", new[] { StaticType.Type, StaticType.Object });
if (method == null)
{
throw new InvalidOperationException($"There is no enum 'GetName' method found between '{toEnumType.FullName}' and '{StaticType.String.FullName}'.");
}
var parameters = new Expression[]
{
Expression.Constant(toEnumType),
ConvertExpressionToTypeExpression(expression, StaticType.Object)
};
return ConvertExpressionToEnumExpressionForString(Expression.Call(method, parameters), toEnumType);
}

Which generates the following expression: System.Enum.GetName(typeof(ExampleFlagEnum), 5). But, value 5 is not an enum value (because it's a combination), so GetName returns null. This gets passed to ConvertExpressionToEnumExpressionForString(...) which calls System.Enum.Parse(...) on that null value and crashes.

What's the reason for going from a database int value via Enum.GetName to string, and then to Parse to get the enum value, instead of just casting directly? A simple call to Enum.ToObject(...) would suffice for conversion from numeric values regardless of actual underlying type of the enum.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working
Projects
None yet
Development

No branches or pull requests

3 participants