在ModelState.Errors中使用[JsonProperty(" name")]

时间:2016-11-16 19:11:06

标签: c# asp.net-web-api2

我们有几个模型通过JsonProperty覆盖名称,但是当我们通过ModelState获得验证错误时会出现问题。例如:

class MyModel
{
    [JsonProperty("id")]
    [Required]
    public string MyModelId {get;set;}
}

class MyModelController
{
    public IHttpActionResult Post([FromBody] MyModel model)
    {
        if (!ModelState.IsValid)
        {
            return HttpBadRequest(ModelState);
        }

        /* etc... */
    }
}

上述帖子会返回错误The MyModelId field is required.,但这并不准确。我们想这样说The id field is required.。我们尝试使用[DataMember(Name="id")],但得到的结果相同。

问题1:除了在每个[必需]属性上提供我们自己的错误消息之外,我们是否有办法让ModelState错误显示JSON属性名而不是C#属性名?

- 更新 -

我一直在玩这个,并找到了一个自己动手的"使用自定义属性名称重新创建错误消息的方法。我真的希望有一种内置的方式来做到这一点,但这似乎可以做到这一点...

https://gist.github.com/Blackbaud-JasonTremper/b64dc6ddb460afa1698daa6d075857e4

问题2 :可以假设ModelState.Key与<parameterName>.<reflectedProperty>语法匹配,或者是否存在可能不符合的情况?

问题3 :是否有更简单的方法来确定JSON参数名称应该是什么,而不是通过[DataMember][JsonProperty]属性上的反射进行搜索?

3 个答案:

答案 0 :(得分:2)

您是否尝试使用DisplayName属性?

displayname attribute vs display attribute

此外,您可以为[Required]属性分配错误消息。

[Required(ErrorMessage = "Name is required")]

答案 1 :(得分:2)

我也遇到了这个问题,我修改了链接中的一些代码以适应我的WebAPI。 modelState还将存储旧密钥,该密钥是模型的变量名称,加上Json属性名称。

  1. 首先,创建过滤器ValidateModelStateFilter
  2. 在控制器方法
  3. 之上添加[ValidateModelStateFilter]

    过滤器源代码:

    public class ValidateModelStateFilter : ActionFilterAttribute
    {
        public override void OnActionExecuting(HttpActionContext actionContext)
        {
            var descriptor = actionContext.ActionDescriptor;
            var modelState = actionContext.ModelState;
    
            if (descriptor != null)
            {
                var parameters = descriptor.GetParameters();
    
                var subParameterIssues = modelState.Keys
                                                   .Where(s => s.Contains("."))
                                                   .Where(s => modelState[s].Errors.Any())
                                                   .GroupBy(s => s.Substring(0, s.IndexOf('.')))
                                                   .ToDictionary(g => g.Key, g => g.ToArray());
    
                foreach (var parameter in parameters)
                {
                    var argument = actionContext.ActionArguments[parameter.ParameterName];
    
                    if (subParameterIssues.ContainsKey(parameter.ParameterName))
                    {
                        var subProperties = subParameterIssues[parameter.ParameterName];
                        foreach (var subProperty in subProperties)
                        {
                            var propName = subProperty.Substring(subProperty.IndexOf('.') + 1);
                            var property = parameter.ParameterType.GetProperty(propName);
                            var validationAttributes = property.GetCustomAttributes(typeof(ValidationAttribute), true);
    
                            var value = property.GetValue(argument);
    
                            modelState[subProperty].Errors.Clear();
                            foreach (var validationAttribute in validationAttributes)
                            {
                                var attr = (ValidationAttribute)validationAttribute;
                                if (!attr.IsValid(value))
                                {
                                    var parameterName = GetParameterName(property);
                                    // modelState.AddModelError(subProperty, attr.FormatErrorMessage(parameterName));
                                    modelState.AddModelError(parameterName, attr.FormatErrorMessage(parameterName));
                                }
                            }
                        }
                    }
    
    
                }
            }
    
        }
    
        private string GetParameterName(PropertyInfo property)
        {
            var dataMemberAttribute = property.GetCustomAttributes<DataMemberAttribute>().FirstOrDefault();
            if (dataMemberAttribute?.Name != null)
            {
                return dataMemberAttribute.Name;
            }
    
            var jsonProperty = property.GetCustomAttributes<JsonPropertyAttribute>().FirstOrDefault();
            if (jsonProperty?.PropertyName != null)
            {
                return jsonProperty.PropertyName;
            }
    
            return property.Name;
        }
    }
    

答案 2 :(得分:0)

您可以访问参数类型,获取json名称,然后用json名称替换属性名称。像这样:

var invalidParameters = (from m in actionContext.ModelState
                         where m.Value.Errors.Count > 0
                         select new InvalidParameter
                         {
                              ParameterName = m.Key,
                              ConstraintViolations = (from msg in m.Value.Errors select msg.ErrorMessage).ToArray()
                         }).AsEnumerable().ToArray();

if (actionContext.ActionDescriptor.Parameters.Count == 1)
{
    var nameMapper = new Dictionary<string, string>();

    foreach (var property in actionContext.ActionDescriptor.Parameters[0].ParameterType.GetProperties())
    {
        object[] attributes = property.GetCustomAttributes(typeof(JsonPropertyNameAttribute), false);

        if (attributes.Length != 1) continue;

        nameMapper.Add(property.Name, ((JsonPropertyNameAttribute) attributes[0]).Name);
    }

    var modifiedInvalidParameters = new List<InvalidParameter>();


    foreach (var invalidParameter in invalidParameters)
    {
        if(invalidParameter.ParameterName != null && nameMapper.TryGetValue(invalidParameter.ParameterName, out var mappedName))
        {
            var modifiedConstraintViolations = new List<string>();

            foreach (var constraintViolation in invalidParameter.ConstraintViolations ?? Enumerable.Empty<string>())
            {
                modifiedConstraintViolations.Add(constraintViolation.Replace(invalidParameter.ParameterName, mappedName));
            }

            modifiedInvalidParameters.Add(new InvalidParameter
            {
                ParameterName = mappedName,
                ConstraintViolations = modifiedConstraintViolations.ToArray()
            });
        }
        else
        {
            modifiedInvalidParameters.Add(invalidParameter);
        }
    }

    invalidParameters = modifiedInvalidParameters.ToArray();
}
public struct InvalidParameter
{
    [JsonPropertyName("parameter_name")]
    public string? ParameterName { get; set; }

    [JsonPropertyName("constraint_violations")]
    public string[]? ConstraintViolations { get; set; }
}