c# - 在 ASP.NET Core 1.0 中使用 ModelState 和 RedirectToAction 进行异步

标签 c# async-await asp.net-core asp.net-core-mvc asp.net-core-1.0

我有一个方法来删除该对象。删除不拥有 View ,是“EditReport”中的“删除”按钮。成功删除“报告”上的重定向后。

[HttpPost]
[Route("{reportId:int}")]
[ValidateAntiForgeryToken]
public async Task<IActionResult> DeleteReport(int reportId)
{
    var success = await _reportService.DeleteReportControlAsync(reportId);
    if (success == false)
    {
        ModelState.AddModelError("Error", "Messages");
        return RedirectToAction("EditReport");
    }
    ModelState.AddModelError("OK", "Messages");
    return RedirectToAction("Report");
}

在 ASP.NET Core 1.0(完整的 .NET Framework)中,我使用以下属性在方法之间保存 ModelState。我从这里获取:https://stackoverflow.com/a/35987804/3878213

我最近转而对 Controller 、服务和存储库使用异步方法。 在我看来,action的属性也应该是异步的。 最初,我重写了代码属性,如下:

public class SetTempDataModelStateAttribute : Attribute, IAsyncActionFilter
    {
        public async Task OnActionExecutionAsync(ActionExecutingContext filterContext, ActionExecutionDelegate next)
        {    
            var controller = filterContext.Controller as Controller;
            var modelState = controller?.ViewData.ModelState;
            if (modelState != null)
            {
                var listError = modelState.ToDictionary(m => m.Key, m => m.Value.Errors
                    .Select(s => s.ErrorMessage)
                    .FirstOrDefault(s => s != null));
                var listErrorJson = await Task.Run(() => JsonConvert.SerializeObject(listError));
                controller.TempData["ModelState"] = listErrorJson;
            }
            await next();
        }
    }
public class RestoreModelStateFromTempDataAttribute : Attribute, IAsyncActionFilter
{
    public async Task OnActionExecutionAsync(ActionExecutingContext filterContext, ActionExecutionDelegate next)
    {
        var controller = filterContext.Controller as Controller;
        var tempData = controller?.TempData?.Keys;
        if (controller != null && tempData != null)
        {
            if (tempData.Contains("ModelState"))
            {
                var modelStateString = controller.TempData["ModelState"].ToString();
                var listError = await Task.Run(() => 
                    JsonConvert.DeserializeObject<Dictionary<string, string>>(modelStateString));
                var modelState = new ModelStateDictionary();
                foreach (var item in listError)
                {
                    modelState.AddModelError(item.Key, item.Value ?? "");
                }

                controller.ViewData.ModelState.Merge(modelState);
            }
        }
        await next();
    }
}

这对我来说似乎很合乎逻辑。但这段代码并没有按照我的预期工作。 但以下异步属性可以正常工作:

public class SetTempDataModelStateAttribute : ActionFilterAttribute
    {
        public override async Task OnActionExecutionAsync(ActionExecutingContext filterContext, ActionExecutionDelegate next)
        {
            await base.OnActionExecutionAsync(filterContext, next);

            var controller = filterContext.Controller as Controller;
            var modelState = controller?.ViewData.ModelState;
            if (modelState != null)
            {
                var listError = modelState.ToDictionary(m => m.Key, m => m.Value.Errors
                    .Select(s => s.ErrorMessage)
                    .FirstOrDefault(s => s != null));
                var listErrorJson = await Task.Run(() => JsonConvert.SerializeObject(listError));
                controller.TempData["ModelState"] = listErrorJson;
            }
            await next();
        }
    }
public class RestoreModelStateFromTempDataAttribute : ActionFilterAttribute
    {
        public override async Task OnActionExecutionAsync(ActionExecutingContext filterContext, ActionExecutionDelegate next)
        {
            await base.OnActionExecutionAsync(filterContext, next);

            var controller = filterContext.Controller as Controller;
            var tempData = controller?.TempData?.Keys;
            if (controller != null && tempData != null)
            {
                if (tempData.Contains("ModelState"))
                {
                    var modelStateString = controller.TempData["ModelState"].ToString();
                    var listError = await Task.Run(() => 
                        JsonConvert.DeserializeObject<Dictionary<string, string>>(modelStateString));
                    var modelState = new ModelStateDictionary();
                    foreach (var item in listError)
                    {
                        modelState.AddModelError(item.Key, item.Value ?? "");
                    }

                    controller.ViewData.ModelState.Merge(modelState);
                }
            }
            await next();
        }
    }

请告诉我这两个选项哪个更正确?为什么第一个选项不起作用?

最佳答案

Please tell me which of the two options more correct.

选项二更正确,因为它确实有效 - 正如您所注意到的。如果您不打算继承它,他们就会将其密封。在这种情况下使用 ActionFilterAttribute 似乎是合适的。事实上quick look @ the repo表明这实际上是一种非常常见的方法。

Why does the first option not work?

它没有调用base调用。关键是您的自定义实现中可能缺少base 调用中的代码。看看here .

关于c# - 在 ASP.NET Core 1.0 中使用 ModelState 和 RedirectToAction 进行异步,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39117361/

相关文章:

c# - 如何使这个异步? (异步、等待 - C#、MVC)

ASP.NET Core RC2配置自定义AppSettings

c# - EF Core 无法连接到服务器 - 与网络相关或特定于实例的错误

C# 审核覆盖 SaveChanges() 方法。如何从表中找出实体的外键及其值

c# - List.Cast<> 错误 "is a method which is not valid in the given context"

c# - 将 vb6 转换为 c#

c# - 将 JSON 数据写入 HttpWebRequest 时收到 "Request was cancelled"错误

c# - WebClient 与 HttpClient - 异步请求

c# - 需要帮助了解 Azure Functions 中异步/等待调用中的后台任务

c# - 使用 IStringLocalizer 对数据注释进行本地化