MVC InvalidOperationException 向用户添加角色

MVC InvalidOperationException adding role to user

我正在学习 this page 上的教程,了解如何在 C# MVC 中以编程方式向用户添加角色。但是,当代码执行时出现以下错误:

System.InvalidOperationException was unhandled by user code
  HResult=-2146233079
  Message=UserId not found.
  Source=Microsoft.AspNet.Identity.Core
  StackTrace:
       at Microsoft.AspNet.Identity.UserManager`2.<AddToRoleAsync>d__83.MoveNext()
    --- End of stack trace from previous location where exception was thrown ---
       at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task)
       at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task)
       at System.Runtime.CompilerServices.TaskAwaiter`1.GetResult()
       at Microsoft.AspNet.Identity.AsyncHelper.RunSync[TResult](Func`1 func)
       at Microsoft.AspNet.Identity.UserManagerExtensions.AddToRole[TUser,TKey](UserManager`2 manager, TKey userId, String role)
       at SchoolRegistration.Controllers.AccountController.RoleAddToUser(FormCollection collection) in c:\Users\Kyle\Documents\Visual Studio 2013\Projects\SchoolRegistration\SchoolRegistration\Controllers\AccountController.cs:line 98
       at lambda_method(Closure , ControllerBase , Object[] )
       at System.Web.Mvc.ActionMethodDispatcher.Execute(ControllerBase controller, Object[] parameters)
       at System.Web.Mvc.ReflectedActionDescriptor.Execute(ControllerContext controllerContext, IDictionary`2 parameters)
       at System.Web.Mvc.ControllerActionInvoker.InvokeActionMethod(ControllerContext controllerContext, ActionDescriptor actionDescriptor, IDictionary`2 parameters)
       at System.Web.Mvc.Async.AsyncControllerActionInvoker.ActionInvocation.InvokeSynchronousActionMethod()
       at System.Web.Mvc.Async.AsyncControllerActionInvoker.<BeginInvokeSynchronousActionMethod>b__39(IAsyncResult asyncResult, ActionInvocation innerInvokeState)
       at System.Web.Mvc.Async.AsyncResultWrapper.WrappedAsyncResult`2.CallEndDelegate(IAsyncResult asyncResult)
       at System.Web.Mvc.Async.AsyncResultWrapper.WrappedAsyncResultBase`1.End()
       at System.Web.Mvc.Async.AsyncResultWrapper.End[TResult](IAsyncResult asyncResult, Object tag)
       at System.Web.Mvc.Async.AsyncControllerActionInvoker.EndInvokeActionMethod(IAsyncResult asyncResult)
       at System.Web.Mvc.Async.AsyncControllerActionInvoker.AsyncInvocationWithFilters.<InvokeActionMethodFilterAsynchronouslyRecursive>b__3d()
       at System.Web.Mvc.Async.AsyncControllerActionInvoker.AsyncInvocationWithFilters.<>c__DisplayClass46.<InvokeActionMethodFilterAsynchronouslyRecursive>b__3f()
  InnerException: 

但我不知道是什么导致了这个错误。任何帮助将不胜感激。

代码:

在我的账户控制器中:

[HttpPost]
        [ValidateAntiForgeryToken]
        public ActionResult RoleAddToUser(FormCollection collection)
        {
            string UserName = collection["UserName"];
            string RoleName = collection["RoleName"];
            using (var context = new ApplicationDbContext()) {
            ApplicationUser user = context.Users.Where(u => u.UserName.Equals(UserName, StringComparison.CurrentCultureIgnoreCase)).FirstOrDefault();
            var user2 = new ApplicationUser() { UserName = UserName };
            var roleStore = new RoleStore<IdentityRole>(context);
            var roleManager = new RoleManager<IdentityRole>(roleStore);
            var userStore = new UserStore<ApplicationUser>(context);
            var userManager = new UserManager<ApplicationUser>(userStore);
            userManager.AddToRole(user2.Id, RoleName);
            var account = new AccountController();
            //account.UserManager.AddToRole(user.Id, RoleName);
            ViewBag.ResultMessage = "Role added successfully.";

            var list = context.Roles.OrderBy(r => r.Name).ToList().Select(rr => new SelectListItem { Value = rr.Name.ToString(), Text = rr.Name }).ToList();
            ViewBag.Roles = list;

        }
            return View("ManageUserRoles");
        }

更新:

好的,在采纳了大家的建议后,我重新编写了我的代码,如下所示:

[HttpPost]
        [ValidateAntiForgeryToken]
        public ActionResult RoleAddToUser(FormCollection collection)
        {
            string UserName = collection["UserName"];
            string RoleName = collection["RoleName"];
            using (var context = new ApplicationDbContext()) {
            ApplicationUser user = context.Users.Where(u => u.UserName.Equals(UserName, StringComparison.CurrentCultureIgnoreCase)).FirstOrDefault();
            var user2 = UserManager.FindByNameAsync(UserName);
            var roleStore = new RoleStore<IdentityRole>(context);
            var roleManager = new RoleManager<IdentityRole>(roleStore);
            var userStore = new UserStore<ApplicationUser>(context);
            var userManager = new UserManager<ApplicationUser>(userStore);
            userManager.AddToRole(user2.Id.ToString(),RoleName);
            var account = new AccountController();
            //account.UserManager.AddToRole(user.Id, RoleName);
            ViewBag.ResultMessage = "Role added successfully.";

            var list = context.Roles.OrderBy(r => r.Name).ToList().Select(rr => new SelectListItem { Value = rr.Name.ToString(), Text = rr.Name }).ToList();
            ViewBag.Roles = list;

        }
            return View("ManageUserRoles");
        }

奇怪的是,我遇到了完全相同的错误。但是,当我现在将鼠标悬停在我的参数上时,它实际上有一个有效的 id 而不是 null:

我确实注意到我在调试时收到的值与数据库中的值不同(这可能是由于哈希安全性?)

数据库值:0752a8d8-fb0e-44d9-b8b0-57ccbbff826b

更新 2:

对于 ID 不匹配的相同问题的任何人,请尝试使用 FindByName 方法而不是 FindByNameAsync。对我有用!

错误消息告诉您问题所在:

Message=UserId not found.

您从未设置 user2.Id,因此当您调用 AddToRole 时,它​​无法找到要将角色添加到的用户,因为 user2.Id 为空。

你在这里设置了user2: var user2 = new ApplicationUser() { UserName = UserName };

那么你在这里引用了user2.Id,但它还没有被设置。

userManager.AddToRole(user2.Id, RoleName);

先尝试通过 ID 查找用户。