Использование ServiceStack Autoquery с типами значений, которые не поддерживают 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
) - Не использовать AutoQuery (я хочу)
NodaTime.LocalDate
инвентарьIConvertible
(скорее всего, не)
Есть ли другой способ заставить фильтры автоматического запроса работать с типами значений, которые не реализуют IConvertible
?
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.