2016-05-23 2 views
4

Я пытался использовать AutoQuery с NodaTime.LocalDate от параметра запроса и я получаю следующее исключение при попытке фильтровать с помощью этого поля даты, в частности >MyDate=2020-01-01 (упорядочение не влияет):Использование ServiceStack Autoquery С типами значений, которые не реализуют IConvertible

[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) 

Я отслеживал его до this line of code, который использует Convert.ChangeType(...), потому что NodaTime.LocalDate является struct и не 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); 

Я использую свою библиотеку сериализации NodaTime ServiceStack, поэтому поведение TypeSerializer.DeserializeFromString(strValue, fieldType) - это то, что я действительно хочу в этом случае.

обходных Я вижу, являются:

  • Использования MyDateDateBetween=2020-01-01,9999-12-31 в строке запроса, как этот код путь использует пользовательские сериализации Я указанная (громоздкие)
  • Использования DateTime вместо NodaTime.LocalDate (я хочу использовать NodaTime.LocalDate)
  • не использовать AutoQuery (я хочу)
  • NodaTime.LocalDate реализует IConvertible (вряд ли)

Есть ли другой способ получить фильтры автоматических запросов для работы со значениями типов, которые не реализуются IConvertible?

ответ

4

Я только добавил завернутый эти строки в новом методе ChangeTo() расширения с дополнительной проверкой для проверки реализации IConvertiblein 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 так LocalDate NodaTime должен теперь проваливается в TypeSerializer.

Это изменение доступно в версии v4.0.57, теперь это available on MyGet.

+0

Протестировано это, и это работает, спасибо! –

Смежные вопросы