Asp.Net Web API 2第十五课——Model Validation(模型验证)

前言html

阅读本文以前,您也能够到Asp.Net Web API 2 系列导航进行查看 http://www.cnblogs.com/aehyok/p/3446289.htmlweb

本文参考连接文章地址http://www.asp.net/web-api/overview/formats-and-model-binding/model-validation-in-aspnet-web-apijson

当客户端发送数据给你的Web API时,你一般但愿在作其它处理以前先对数据进行验证。api

Data Annotations——数据注解安全

 在ASP.NET Web API中,你可使用System.ComponentModel.DataAnnotations命名空间的注解属性来设置模型属性的验证规则。考虑如下模型:app

public class Product
{
    public int Id { get; set; }

    [Required]
    public string Name { get; set; }
    public decimal Price { get; set; }

    [Range(0,999)]
    public double Weight { get; set; }
}

若是你曾在ASP.NET MVC中使用过模型验证,这看上去是相似的。Required注解属性说明Name属性必须不为空。Range注解属性说明Weight必须在0-999之间。asp.net

假设客户端发送了一个带有下列JSON表示的POST请求:ide

{ "Id":4, "Price":2.99, "Weight":5 }

你能够看出,客户端并未包含被标记成required的Name属性。当Web API将该JSON转换成Product实例时,它会根据这些验证注解属性对Product进行验证。在控制器动做中,你能够检查该模型是否有效:post

public class ProductsController : ApiController
{
    public HttpResponseMessage Post(Product product)
    {
        if (ModelState.IsValid)
        {
            // Do something with the product (not shown).
            // 用product作一些事(未表示出来)

            return new HttpResponseMessage(HttpStatusCode.OK);
        }
        else
        {
            return new HttpResponseMessage(HttpStatusCode.BadRequest);
        }
    }
}

模型验证并不保证客户端数据是安全的。在应用程序的其它层面可能会须要附加验证(例如,数据层可能会强制外键约束)。ui

{"Id":4, "Name":"Gizmo"}

此处,客户端并未指定Price或Weight的值。JSON格式化器会将默认值(这里是零)赋给这些缺失的属性。

Under-Posting(递交不足)”:当客户端遗漏了某些属性时,便会发生“Under-posting”。例如,假设客户端发送以下:

此时模型的状态是有效的,由于零是这些属性的有效值。这是不是一个问题取决于你所处的场景。例如,在一个更新操做中,你可能但愿区分出“零”与“未设置”。为了强迫客户端要设置一个值,将该属性构形成nullable(可空的),并设置Required注解属性:

[Required]
public decimal? Price { get; set; }

Over-Posting(过份递交)”:客户端也可能发送比指望还多的数据。例如:

{"Id":4, "Name":"Gizmo", "Color":"Blue"}

此处,JSON包含了Product模型中存在的属性(“Color”)。在这种状况下,JSON格式化器会简单地忽略该值(XML格式化器却不一样)。若你的模型具备只读属性,Over-posting会产生问题。例如:

public class UserProfile
{
    public string Name { get; set; }
    public Uri Blog { get; set; }
    public bool IsAdmin { get; set; }  // uh-oh!(啊哦!)
}

若是你不想让用户对IsAdmin属性进行更新,并将其提高给管理员。最安全的策略是使用一个与容许客户端发送严格匹配的模型类:

public class UserProfileDTO
{
    public string Name { get; set; }
    public Uri Blog { get; set; }
    // Leave out "IsAdmin"
    // 略去了"IsAdmin"
}

Handling Validation Errors——处理验证错误

 当验证失败时,Web API并不会自动地将错误返回给客户端。这取决于控制器动做对模型状态及响应进行适当的检查。

你也能够建立一个动做过滤器,以便在控制器动做被调用以前,检查模型的状态。如下代码演示了一个例子:

using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Net.Http;
using System.Web.Http.Controllers;
using System.Web.Http.Filters;
using System.Web.Http.ModelBinding;

public class ModelValidationFilterAttribute : ActionFilterAttribute
{
    public override void OnActionExecuting(HttpActionContext actionContext)
    {
        if (actionContext.ModelState.IsValid == false)
        {
            // Return the validation errors in the response body.
            // 在响应体中返回验证错误
            var errors = new Dictionary<string, IEnumerable<string>>();
            foreach (KeyValuePair<string, ModelState> keyValue in actionContext.ModelState)
            {
                errors[keyValue.Key] = keyValue.Value.Errors.Select(e => e.ErrorMessage);
            }

            actionContext.Response = 
                actionContext.Request.CreateResponse(HttpStatusCode.BadRequest, errors);
        }
    }
}

若是模型验证失败,此过滤器会返回一个含有验证错误的HTTP响应。在此状况下,不会调用控制器动做。

HTTP/1.1 400 Bad Request
Server: ASP.NET Development Server/10.0.0.0
Date: Fri, 20 Jul 2012 21:42:18 GMT
Content-Type: application/json; charset=utf-8
Content-Length: 239
Connection: Close

{
    "product": [
        "Required property 'Name' not found in JSON. Line 1, position 18."
    ],

    "product.Name": [
        "The Name field is required."
    ],

    "product.Weight": [
        "The field Weight must be between 0 and 999."
    ]
}

若是你正在使用CodePlex上最新版的Web API,可使用HttpError类将验证错误返回给客户端。HttpError类在RC版(指Web API的预览版)中无效。

你能够将此过滤器全局性地运用于全部Web API控制器。在Application_Start方法中,将此过滤器添加到HttpConfiguration.Filters集合:

protected void Application_Start()
{
    // ...

    GlobalConfiguration.Configuration.Filters.Add(new ModelValidationFilterAttribute());
}

另外一种可选办法是,经过将此过滤器做为注解属性进行添加,你能够将它运用于个别控制器或控制器动做:

public class ProductsController : ApiController
{
    [ModelValidationFilter]
    public HttpResponseMessage Post(Product product)
    {
        // ...
    }
}
相关文章
相关标签/搜索