gpt4 book ai didi

c# - 使用 RuleForEach 时如何访问正在验证的集合项?

转载 作者:IT王子 更新时间:2023-10-29 04:30:52 24 4
gpt4 key购买 nike

我正在使用 FluentValidation 来验证一个对象,因为这个对象有一个集合成员,所以我正在尝试使用 RuleForEach。例如,假设我们有 CustomerOrders,我们希望确保没有客户订单的总值(value)超过该客户允许的最大值:

this.RuleForEach(customer => customer.Orders)
.Must((customer, orders) => orders.Max(order => order.TotalValue) <= customer.MaxOrderValue)

到目前为止,还不错。但是,我还需要记录有关错误上下文的其他信息(例如,在数据文件中发现错误的位置)。我发现这很难通过 FluentValidation 实现,目前我最好的解决方案是使用 WithState 方法。例如,如果我发现客户的地址详细信息不正确,我可能会这样做:

this.RuleFor(customer => customer.Address)
.Must(...)
.WithState(customer => GetErrorContext(customer.Address))

(其中GetErrorContext是我提取相关细节的方法。

现在我遇到的问题是,在使用 RuleForEach 时,方法签名假定我将提供一个引用 Customer 的表达式,而不是导致验证失败的特定 Order。而且我好像也不知道是哪个订单出了问题。因此,我无法存储适当的上下文信息。

换句话说,我只能这样做:

this.RuleForEach(customer => customer.Orders)
.Must((customer, orders) => orders.Max(order => order.TotalValue) <= customer.MaxOrderValue)
.WithState(customer => ...)

...当我真的想这样做的时候:

this.RuleForEach(customer => customer.Orders)
.Must((customer, orders) => orders.Max(order => order.TotalValue) <= customer.MaxOrderValue)
.WithState(order => ...)

真的没有办法访问失败的集合项的详细信息(甚至索引)吗?

我想另一种看待它的方式是我希望 WithState 有一个等效的 WithStateForEach...

最佳答案

目前,FluentValidation 中没有允许以您想要的方式设置验证状态的功能。 RuleForEach 旨在防止为简单的集合项创建琐碎的验证器,它的实现并未涵盖所有可能的用例。

您可以为 Order 创建单独的验证器类,并使用 SetCollectionValidator 方法应用它。要在 Must 方法中访问 Customer.MaxOrderValue — 向 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's 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;
});
}

附言

不要忘记您的目标是实现验证,而不是到处使用 FluentValidation。有时我们将验证逻辑作为一个单独的方法来实现,它与 ViewModel 一起工作,并在 ASP.NET MVC 中填充 ModelState

如果您找不到符合您要求的解决方案,那么手动实现比使用库笨拙实现要好。

关于c# - 使用 RuleForEach 时如何访问正在验证的集合项?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27213058/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com