使用ServiceStack自动查询使用不实现IConvertible的值类型

时间:2016-05-23 17:12:11

标签: c# servicestack nodatime autoquery-servicestack

我尝试在查询参数上使用AutoQueryNodaTime.LocalDate,当我尝试使用该日期字段进行过滤时,我得到以下异常,特别是>MyDate=2020-01-01(排序不受影响):

[MyEndpoint: 5/23/2016 4:19:51 PM]: [REQUEST: {}] System.InvalidCastException: Invalid cast from 'System.String' to 'NodaTime.LocalDate'. at System.Convert.DefaultToType(IConvertible value, Type targetType, IFormatProvider provider) at ServiceStack.TypedQuery`2.AppendUntypedQueries(SqlExpression`1 q, Dictionary`2 dynamicParams, String defaultTerm, IAutoQueryOptions options, Dictionary`2 aliases) at ServiceStack.TypedQuery`2.CreateQuery(IDbConnection db, IQueryDb dto, Dictionary`2 dynamicParams, IAutoQueryOptions options) at ServiceStack.AutoQuery.CreateQuery[From](IQueryDb`1 dto, Dictionary`2 dynamicParams, IRequest req) at ServiceStack.AutoQueryServiceBase.Exec[From](IQueryDb`1 dto) at ServiceStack.Host.ServiceRunner`1.Execute(IRequest request, Object instance, TRequest requestDto)

我将其追踪到使用Convert.ChangeType(...)的{​​{3}},因为NodaTime.LocalDatestruct而不是enum

var value = strValue == null ? 
      null 
    : isMultiple ? 
      TypeSerializer.DeserializeFromString(strValue, Array.CreateInstance(fieldType, 0).GetType())
    : fieldType == typeof(string) ? 
      strValue
    : fieldType.IsValueType && !fieldType.IsEnum ? //This is true for NodaTime.LocalDate
      Convert.ChangeType(strValue, fieldType) :    //NodaTime.LocalDate does not implement IConvertible, so this throws
      TypeSerializer.DeserializeFromString(strValue, fieldType);

我正在使用我的this line of code序列化库,因此TypeSerializer.DeserializeFromString(strValue, fieldType)的行为是我在这种情况下实际需要的行为。

我看到的解决方法是:

  • 在查询字符串中使用MyDateDateBetween=2020-01-01,9999-12-31,因为该代码路径使用我指定的自定义序列化(繁琐)
  • 使用DateTime代替NodaTime.LocalDate(我想使用NodaTime.LocalDate
  • 不使用AutoQuery(我想)
  • NodaTime.LocalDate实施NodaTime ServiceStack(不太可能)

是否有其他方法可以让自动查询过滤器与不实现IConvertible的值类型一起使用?

1 个答案:

答案 0 :(得分:4)

我刚刚在新的ChangeTo()扩展程序中添加了这些行,并附加检查以检查实施IConvertible in this commit

public static object ChangeTo(this string strValue, Type type)
{
    if (type.IsValueType && !type.IsEnum
        && type.HasInterface(typeof(IConvertible)))
    {
        try
        {
            return Convert.ChangeType(strValue, type);
        }
        catch (Exception ex)
        {
            Tracer.Instance.WriteError(ex);
        }
    }
    return TypeSerializer.DeserializeFromString(strValue, type);
}

changed AutoQuery to use it所以NodaTime的LocalDate现在应该落入TypeSerializer。

此更改可从v4.0.57获得,现在为available on MyGet