Использование автозапроса ServiceStack с типами значений, которые не реализуют IConvertible

Я пытался использовать AutoQuery с NodaTime.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(...), потому что 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)
  • Не использовать автозапрос (я хочу)
  • NodaTime.LocalDate реализует IConvertible (маловероятно )

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


person Anthony Carl    schedule 23.05.2016    source источник


Ответы (1)


Я только что добавил эти строки в новый метод расширения ChangeTo() с дополнительной проверкой реализации IConvertible в этом коммите:

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);
}

И изменил AutoQuery для его использования, поэтому LocalDate NodaTime теперь должен проходить через TypeSerializer.

Это изменение доступно в версии 4.0.57, которая теперь доступна на MyGet.

person mythz    schedule 23.05.2016