Я оцениваю FluentValidation в ServiceStack для обработки автоматической проверки запросов DTO:
Plugins.Add(new ValidationFeature());
container.RegisterValidators(typeof(MyValidator).Assembly);
Ошибки возвращаются клиенту путем сериализации ErrorResponse
DTO и могут выглядеть так:
{
"ErrorCode": "GreaterThan",
"Message": "'Age' must be greater than '0'.",
"Errors": [
{
"ErrorCode": "GreaterThan",
"FieldName": "Age",
"Message": "'Age' must be greater than '0'."
},
{
"ErrorCode": "NotEmpty",
"FieldName": "Company",
"Message": "'Company' should not be empty."
}
]
}
Я хотел бы знать, можно ли возвращать ошибки, используя другой ответ DTO. Например:
{
"code": "123",
"error": "'Age' must be greater than '0'."
}
Я знаю, что можно явно использовать валидатор в службе:
public MyService : Service
{
private readonly IValidator<MyRequestDto> validator;
public MyService(IValidator<MyRequestDto> validator)
{
this.validator = validator;
}
public object Get(MyRequestDto request)
{
var result = this.validator.Validate(request);
if (!result.IsValid)
{
throw new SomeCustomException(result);
}
... at this stage request DTO validation has passed
}
}
Но вопрос в том, можно ли, чтобы эта ошибка проверки была перехвачена неявно где-то, чтобы я мог заменить ответ DTO и иметь более чистый сервис:
public MyService : Service
{
public object Get(MyRequestDto request)
{
... at this stage request DTO validation has passed
}
}
UPDATE:
После дальнейшего копания в исходный код, похоже, что это сжигается в ValidationFeature
и, более конкретно, в фильтр запроса, который он регистрирует:
public class ValidationFilters
{
public void RequestFilter(IHttpRequest req, IHttpResponse res, object requestDto)
{
var validator = ValidatorCache.GetValidator(req, requestDto.GetType());
if (validator == null) return;
var validatorWithHttpRequest = validator as IRequiresHttpRequest;
if (validatorWithHttpRequest != null)
validatorWithHttpRequest.HttpRequest = req;
var ruleSet = req.HttpMethod;
var validationResult = validator.Validate(
new ValidationContext(requestDto, null, new MultiRuleSetValidatorSelector(ruleSet)));
if (validationResult.IsValid) return;
var errorResponse = DtoUtils.CreateErrorResponse(
requestDto, validationResult.ToErrorResult());
res.WriteToResponse(req, errorResponse);
}
}
Написав специальную функцию проверки, я смог добиться желаемого эффекта. Но может быть, там более элегантный способ?