在Action外部访问ControlerBase(在CQRS处理程序中)

时间:2017-03-16 11:22:26

标签: asp.net-core cqrs mediatr

我将我的逻辑从Controller转移到CQRS处理程序,我不知道如何处理调用ControllerBase之类的代码:

[HttpPut("users/{id}")]
public IActionResult Put(UserForm userForm)
{
  var user = Users.Get(email);
  if (user == null) 
    return NotFound();
  ...
}

因此,当我将此逻辑移到行动之外时,显然我无法再访问NotFound了。我看到了许多解决方法:

  1. 改为抛出自定义的NotFoundException,然后在ExceptionMiddleware中捕获它并返回NotFoundResult。
  2. 将ControllerBase注入CQRS命令/查询。不喜欢这个选项,因为命令/查询将与ASP.NET核心基础结构相结合。
  3. 我快速谷歌搜索没有成功,所以可能是我错过了什么。处理这种情况的推荐方法是什么?

1 个答案:

答案 0 :(得分:1)

此逻辑/验证不属于Command / Query处理程序,因为它处理不同的事物(用户输入 - > BadRequest,资源状态(NotFound,Ok,Created(201)))用于某些场景(WebAPI,MVC) ,桌面,移动应用程序等)和响应高度依赖于基础设施(在REST中你将返回状态代码,MVC重定向,桌面消息等)

使用异常是不可接受的,因为这些是常见的情况,而不是例外情况(至少输入验证和“未找到”)。在处理程序中使用IActionResult也与基础结构紧密耦合。

在较大的项目中,我使用“结果”类,类似于ASP.NET核心标识中的IdentityResult

namespace MyProject.Common
{
    public class QueryResult
    {
        private static readonly QueryResult success = new QueryResult { Succeeded = true };
        private readonly List<QueryError> errors = new List<QueryError>();

        public static QueryResult Success { get { return success; } }

        public bool Succeeded { get; protected set; }
        public IEnumerable<QueryError> Errors { get { return errors; } }

        public static QueryResult Failed(params QueryError[] errors)
        {
            var result = new QueryResult { Succeeded = false };
            if (errors != null)
            {
                result.errors.AddRange(errors);
            }

            return result;
        }
    }

    public class QueryResult<T> : QueryResult where T : class
    {
        public T Result { get; protected set; }

        public static QueryResult<T> Suceeded(T result)
        {
            var queryResult = new QueryResult<T>
            {
                Succeeded = true,
                Result = result
            };

            return queryResult;
        }
    }
}

从命令(QueryResult)或查询(QueryResult<T>)返回这些内容。

然后,您可以编写返回正确IActionResult

的扩展方法
public static class QueryResultExtensions
{
    public IActionResult ToActionResult(this QueryResult result)
    {
        if(result.Success)
        {
            return new OkResult();
        }

        if(result.Errors != null)
        {
            // ToModelStateDictionary needs to be implemented by you ;)
            var errors = result.Errors.ToModelStateDictionary();
            return BadRequestResult(errors);
        }

        return BadRequestResult();
    }

    public IActionResult ToActionResult<T>(this QueryResult<T> result)
    {
        if(result.Success)
        {
            if(result.Result == null)
            {
                return new NotFoundResult();
            }

            return new OkObjectResult(result.Result);
        }

        if(result.Errors != null)
        {
            // ToModelStateDictionary needs to be implemented by you ;)
            var errors = result.Errors.ToModelStateDictionary();
            return BadRequestResult(errors);
        }

        return BadRequestResult();
    }
}

然后在控制器中调用它

[HttpPut("users/{id}")]
public IActionResult Put(CreateUserCommand command)
{
    var result = commandHandler.Handle(command);
    return result.ToActionResult();
}

您失去了控制返回代码的更细粒度的能力。您可以为命令/查询或不同的扩展方法创建不同的结果类型,以不同的方式处理它.ToCreateActionResult().ToQueryActionResult()等。