C # MVC: контроллер сброса пароля пользователя: проблемы с адресами электронной почты в качестве имен пользователей - PullRequest
2 голосов
/ 10 января 2011

Я написал приведенный ниже код для сброса паролей пользователей (я использую api членства в aspnet) в приложении C # MVC и успешно протестирован на примере учебного приложения (MVC Music Store).Пропустите до конца, если вы хотите сначала прочитать описание проблемы.

Просмотр InactiveUsers (Частичное представление)

<%@ Control Language="C#" Inherits="System.Web.Mvc.ViewUserControl<System.Web.Security.MembershipUserCollection>" %>

<table class="normal" style="width: 100%; background-color: White;">
       <tr>
            <th>User Name</th>
            <th>Last Activity date</th>
            <th>Locked Out</th>
       </tr>
       <%foreach (MembershipUser user in Model){ %>


       <tr>
           <td><%: Html.RouteLink(user.UserName, "AdminPassword", new { username = user.UserName }) %></td>
           <td><%: user.LastActivityDate %></td>
           <td><%: user.IsLockedOut %></td>
       </tr>


       <% }%>
    </table>

Контроллер InactiveUsers

public ActionResult InactiveUsers()
    {
        var users = Membership.GetAllUsers();

        return View(users);

    }

changeUserPassword GET и POST-контроллеры

 public ActionResult changeUserPassword(string username)
        {
            ViewData["username"] = username;

            return View();
        }


        [HttpPost]
        public ActionResult changeUserPassword(ChangePasswordModel model, FormCollection values)
        {
            string username = values["username"];
            string password = values["password"];
            string confirmPassword = values["confirmPassword"];

            MembershipUser mu = Membership.GetUser(username);

            if (password == confirmPassword)
            {
                if (mu.ChangePassword(mu.ResetPassword(), password))
                {
                    return RedirectToAction("Index", "ControlPanel");
                }
                else
                {
                    ModelState.AddModelError("", "The current password does not meet requirements");
                }
            }

            return View();
        }

Я также изменил файл Global.asax.cs для обслуживания моего маршрута в части InactiveUsers:

        // Added in 10/01/11

        RouteTable.Routes.MapRoute(
            "AdminPassword", // routename
            "ControlPanel/changeUserPassword/{username}",
            new { controller = "ControlPanel", action = "changeUserPassword", username = UrlParameter.Optional }
            );

        // END

Теперь, когда я тестировал в музыкальном магазине MVC, все мои имена пользователей были просто словами, например, «Администратор», «Пользователь» и т. Д. Однако сейчас я применяю этот код в ситуации на моем рабочем месте, и он не работает.совсем как запланировано.Имена пользователей, используемые на моем рабочем месте, на самом деле являются адресами электронной почты, и я думаю именно это и является причиной проблемы.

Когда я нажимаю на RouteLink в частичном представлении InactiveUsers, он должен привести меня на страницу сброса пароля с URL-адресом, который выглядит следующим образом:

http://localhost:83/ControlPanel/changeUserPassword/example1@gmail.com, HOWEVER,

что происходит, когда я нажимаю на RouteLink, выдается сообщение об ошибке, указывающее, что представление changeUserPassword не может быть найдено, и URL выглядит следующим образом:

http://localhost:83/ControlPanel/changeUserPassword/example1%40gmail.com - Посмотрите, каксимвол '@' испортился?

Я также отлаживал код, и в моем GET changeUserPassword имя пользователя заполняется правильно: example1@gmail.com, так что я думаю, что это простоНеправильный URL-адрес?

Если я введу URL-адрес вручную, отобразится представление changeUserPassword, однако функция сброса пароля не работает.Исключение 'Ссылка на объект, не установленная для экземпляра объекта', выдается в строке if (mu.ChangePassword (mu.ResetPassword (), password)).

Думаю, смогу ли я решить первую проблему(Проблема с символом URL '@'), это может помочь мне вместе со второй проблемой.

Буду признателен за любую помощь:)

Трассировка стека - по запросу

Ошибка источника:

Создано необработанное исключениево время выполнения текущего веб-запроса.Информацию о происхождении и местонахождении исключения можно определить с помощью приведенной ниже трассировки стека исключений.

Трассировка стека:

[InvalidOperationException: The view 'changeUserPassword' or its master was not found. The following locations were searched:
~/Views/ControlPanel/changeUserPassword.aspx
~/Views/ControlPanel/changeUserPassword.ascx
~/Views/Shared/changeUserPassword.aspx
~/Views/Shared/changeUserPassword.ascx]
   System.Web.Mvc.ViewResult.FindView(ControllerContext context) +495
   System.Web.Mvc.ViewResultBase.ExecuteResult(ControllerContext context) +208
   System.Web.Mvc.ControllerActionInvoker.InvokeActionResult(ControllerContext controllerContext, ActionResult actionResult) +39
   System.Web.Mvc.<>c__DisplayClass14.<InvokeActionResultWithFilters>b__11() +60
   System.Web.Mvc.ControllerActionInvoker.InvokeActionResultFilter(IResultFilter filter, ResultExecutingContext preContext, Func`1 continuation) +391
   System.Web.Mvc.<>c__DisplayClass16.<InvokeActionResultWithFilters>b__13() +61
   System.Web.Mvc.ControllerActionInvoker.InvokeActionResultWithFilters(ControllerContext controllerContext, IList`1 filters, ActionResult actionResult) +285
   System.Web.Mvc.ControllerActionInvoker.InvokeAction(ControllerContext controllerContext, String actionName) +830
   System.Web.Mvc.Controller.ExecuteCore() +136
   System.Web.Mvc.ControllerBase.Execute(RequestContext requestContext) +111
   System.Web.Mvc.ControllerBase.System.Web.Mvc.IController.Execute(RequestContext requestContext) +39
   System.Web.Mvc.<>c__DisplayClass8.<BeginProcessRequest>b__4() +65
   System.Web.Mvc.Async.<>c__DisplayClass1.<MakeVoidDelegate>b__0() +44
   System.Web.Mvc.Async.<>c__DisplayClass8`1.<BeginSynchronous>b__7(IAsyncResult _) +42
   System.Web.Mvc.Async.WrappedAsyncResult`1.End() +141
   System.Web.Mvc.Async.AsyncResultWrapper.End(IAsyncResult asyncResult, Object tag) +54
   System.Web.Mvc.Async.AsyncResultWrapper.End(IAsyncResult asyncResult, Object tag) +40
   System.Web.Mvc.MvcHandler.EndProcessRequest(IAsyncResult asyncResult) +52
   System.Web.Mvc.MvcHandler.System.Web.IHttpAsyncHandler.EndProcessRequest(IAsyncResult result) +38
   System.Web.CallHandlerExecutionStep.System.Web.HttpApplication.IExecutionStep.Execute() +8841105
   System.Web.HttpApplication.ExecuteStep(IExecutionStep step, Boolean& completedSynchronously) +184

Ответы [ 2 ]

0 голосов
/ 23 ноября 2011

Лучше использовать System.Web.HttpContext.Current.Server.UrlDecode(url1). независимо от того, закодирован ли URL-адрес или нет, он возвращает необработанный URL-адрес, например "http://localhost:83/ControlPanel/changeUserPassword/example1@gmail.com", и вы можете получить из него userName.

string userName = System.Web.HttpContext.Current.Server.UrlDecode(userName);
RouteTable.Routes.MapRoute(
            "AdminPassword", // routename 
            "ControlPanel/changeUserPassword/" + userName,
            new { controller = "ControlPanel", action = "changeUserPassword", username = UrlParameter.Optional }
            ); 
0 голосов
/ 10 января 2011

Вы понимаете, что% 40 - это версия @ -symbol в кодировке URL?Данные в вашем URL не перепутаны, они закодированы.Это необходимо, потому что ваши данные содержат символ, который имеет значение в URL, поэтому он должен быть закодирован таким образом, чтобы не нарушать URL.

Здесь вы можете попробовать кодировать / декодировать URL:1003 *http://meyerweb.com/eric/tools/dencoder/

...