单元测试自定义验证过滤器

MrB*_*liz 4 nunit unit-testing moq asp.net-web-api

我有以下属性:

public class ValidateModelAttribute : ActionFilterAttribute
    {
        public override void OnActionExecuting(HttpActionContext actionContext)
        {
            if (actionContext.ModelState.IsValid == false)
            {
                actionContext.Response = actionContext.Request.CreateErrorResponse(
                    HttpStatusCode.BadRequest, actionContext.ModelState);
            }
        }
    }  
Run Code Online (Sandbox Code Playgroud)

我有这个通用的扩展方法,用于确定属性是否应用于方法

public static bool ActionHasFilter(this ApiController controller, string action, Type filter)
    {
        var controllerType = controller.GetType();
        var method = controllerType.GetMethod(action);

        object[] filters = method.GetCustomAttributes(filter, true);

        return filters.Any(x=>x.GetType() == filter);

    }
Run Code Online (Sandbox Code Playgroud)

我的问题是如何在不测试控制器动作的情况下测试属性是否真实有效?

假设我有以下实体

public class UserViewModel
{
     [Required]
     public string Name {get; set;}
     [Required]
     [EmailAddress]
     public string Email {get;set;
}
Run Code Online (Sandbox Code Playgroud)

我将如何模拟上下文并检查模型是否有效?

我正在使用Nunit和Moq.

Pru*_*der 8

Spock的解决方案在正确的轨道上,但在代码上有点过于侵略,因为它使得ValidateModelAttribute类依赖于TestableHttpActionContext.我的实现使用一个公共属性,用于Request为单元测试"注入" 对象,而实现为属性继续使用以下Request对象ActionContext:

public class ValidateModelAttribute : ActionFilterAttribute
{
    public HttpRequestMessage TestRequestMessage { get; set; }

    public override void OnActionExecuting(HttpActionContext actionContext)
    {
        PerformValidation(actionContext, TestRequestMessage ?? actionContext.Request);
    }

    private void PerformValidation(HttpActionContext actionContext, HttpRequestMessage request)
    {
        if (actionContext.ModelState.IsValid == false)
        {
            actionContext.Response = request.CreateErrorResponse(HttpStatusCode.BadRequest, actionContext.ModelState);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

单元测试:

[Test]
public void OnActionExecuting_ValidModel_ResponseIsNotSet()
{
    var actionContext = new HttpActionContext();

    actionContext.ModelState.Clear();

    var attribute = new ValidateModelAttribute { TestRequestMessage = new HttpRequestMessage() };

    attribute.OnActionExecuting(actionContext);

    Assert.IsNull(actionContext.Response);
}

[Test]
public void OnActionExecuting_InvalidModel_ResponseIsSetToBadRequest()
{
    var actionContext = new HttpActionContext();

    actionContext.ModelState.AddModelError("key", "error");

    var attribute = new ValidateModelAttribute() { TestRequestMessage = new HttpRequestMessage() };

    attribute.OnActionExecuting(actionContext);

    Assert.AreEqual(HttpStatusCode.BadRequest, actionContext.Response.StatusCode);
}
Run Code Online (Sandbox Code Playgroud)

请注意,我没有使用实际模型来验证ModelState,因为这超出了单元测试的范围:我们想要测试ModelState结果,而不是ModelState自己.;-)