Ответ 1
В настоящее время в FluentValidation нет функциональности, что позволяет установить состояние проверки так, как вы хотите. RuleForEach
был разработан для предотвращения создания тривиальных валидаторов для простых элементов коллекции, и реализация не охватывала все возможные варианты использования.
Вы можете создать отдельный класс проверки для Order
и применить его с помощью метода SetCollectionValidator
. Чтобы получить доступ к Customer.MaxOrderValue
в Must
- добавить свойство в Order
, ссылающееся на Customer
:
public class CustomerValidator
{
public CustomerValidator()
{
RuleFor(customer => customer.Orders).SetCollectionValidator(new OrderValidator());
}
}
public class OrderValidator
{
public OrderValidator()
{
RuleFor(order => order.TotalValue)
.Must((order, total) => total <= order.Customer.MaxOrderValue)
.WithState(order => GetErrorInfo(order)); // pass order info into state
}
}
Если вы все еще хотите использовать метод RuleForEach
, вы можете использовать сообщение об ошибке вместо пользовательского состояния, поскольку оно имеет доступ к объектам объектов родителя и дочернего элемента в одном из перегрузок:
public class CustomerValidator
{
public CustomerValidator()
{
RuleForEach(customer => customer.Orders)
.Must((customer, order) => order.TotalValue) <= customer.MaxOrderValue)
.WithMessage("order with Id = {0} have error. It total value exceeds {1}, that is maximum for {2}",
(customer, order) => order.Id,
(customer, order) => customer.MaxOrderValue,
(customer, order) => customer.Name);
}
}
Если вам нужно собрать все индексы (или идентификаторы) неудачных заказов - вы можете сделать это с помощью правила Custom
, как здесь:
public CustomerValidator()
{
Custom((customer, validationContext) =>
{
var isValid = true;
var failedOrders = new List<int>();
for (var i = 0; i < customer.Orders.Count; i++)
{
if (customer.Orders[i].TotalValue > customer.MaxOrderValue)
{
isValid = false;
failedOrders.Add(i);
}
}
if (!isValid){
var errorMessage = string.Format("Error: {0} orders TotalValue exceed maximum TotalValue allowed", string.Join(",", failedOrders));
return new ValidationFailure("", errorMessage) // return indexes of orders through error message
{
CustomState = GetOrdersErrorInfo(failedOrders) // set state object for parent model here
};
}
return null;
});
}
P.S.
Не забывайте, что ваша цель - выполнить проверку, а не использовать FluentValidation везде. Иногда мы реализуем логику проверки как отдельный метод, который работает с ViewModel
и заполняет ModelState
в ASP.NET MVC.
Если вы не можете найти решение, соответствующее вашим требованиям, тогда ручная реализация была бы лучше, чем корявая реализация с библиотекой.