Атрибут маршрутизации - показать вошедший в систему идентификатор пользователя в URL

Я использую пользовательскую аутентификацию MVC в своем проекте, и я не знаю, как мне украсить свой URL в соответствии с моим требованием. По сути, мне нужно, чтобы каждый раз, когда пользователь проходил проверку подлинности и получал доступ к моему приложению, он перенаправлялся на основной маршрут, отображаемый в конфигурации rote, который выглядит примерно так:

локальный:64843/ панель

Что мне нужно, так это предположим, что Джанет входит в систему и прошла проверку подлинности, тогда я хочу, чтобы URL целевой страницы выглядел так:

локальный:64843/Janet/ панель

Примечание. Поле имени пользователя (т. Е. Janet в этом контексте создается путем добавления настраиваемого поля в таблицу AspNetRoles).

А также мне нужна структура URL, чтобы она оставалась такой, когда я просматриваю другие страницы. Например, я нахожусь на странице создания представления продуктов, тогда URL должен быть таким:

локальный:64843/Janet/ Продукты / Создать

И эта структура должна быть для всех пользователей, которые входят в систему в соответствии с их именем пользователя.

Я новичок в атрибуции принципа маршрутизации, и любая помощь по этому вопросу будет принята с благодарностью.

Конфигурация маршрута

        public static void RegisterRoutes(RouteCollection routes)
        {
            routes.IgnoreRoute("{resource}.axd/{*pathInfo}");

            routes.MapRoute(
                name: "Username_Default",
                url: "{username}/{controller}/{action}/{id}",
                defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional },
                constraints: new { username = new OwinUsernameConstraint() }
            );


            routes.MapRoute(
                name: "Default",
                url: "{controller}/{action}/{id}",
                defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional }
            );
        }

Ограничение имени пользователя

    public class OwinUsernameConstraint : IRouteConstraint
    {
        private object synclock = new object();

        public bool Match(HttpContextBase httpContext, Route route, string parameterName, RouteValueDictionary values, RouteDirection routeDirection)
        {
            if (parameterName == null)
                throw new ArgumentNullException("parameterName");
            if (values == null)
                throw new ArgumentNullException("values");

            object value;
            if (values.TryGetValue(parameterName, out value) && value != null)
            {
                string valueString = Convert.ToString(value, CultureInfo.InvariantCulture);
                return this.GetUsernameList(httpContext).Contains(valueString);
            }
            return false;
        }

        private IEnumerable<string> GetUsernameList(HttpContextBase httpContext)
        {
            string key = "UsernameConstraint.GetUsernameList";
            var usernames = httpContext.Cache[key];
            if (usernames == null)
            {
                lock (synclock)
                {
                    usernames = httpContext.Cache[key];
                    if (usernames == null)
                    {
                        // Retrieve the list of usernames from the database
                        using (var db = ApplicationDbContext.Create())
                        {
                            usernames = (from users in db.Users
                                            select users.UserName).ToList();
                        }

                        httpContext.Cache.Insert(
                            key: key,
                            value: usernames,
                            dependencies: null,
                            absoluteExpiration: Cache.NoAbsoluteExpiration,
                            slidingExpiration: TimeSpan.FromSeconds(15),
                            priority: CacheItemPriority.NotRemovable,
                            onRemoveCallback: null);
                    }
                }
            }

            return (IEnumerable<string>)usernames;
        }
    }

Зарегистрировать контроллер

        public async Task<ActionResult> Register(RegisterViewModel model)
        {
            if (ModelState.IsValid)
            {
                var user = new ApplicationUser { UserName = model.UserName, Email = model.Email };
                var result = await UserManager.CreateAsync(user, model.Password);
                if (result.Succeeded)
                {
                    await SignInManager.SignInAsync(user, isPersistent:false, rememberBrowser:false);

                    // For more information on how to enable account confirmation and password reset please visit http://go.microsoft.com/fwlink/?LinkID=320771
                    // Send an email with this link
                    // string code = await UserManager.GenerateEmailConfirmationTokenAsync(user.Id);
                    // var callbackUrl = Url.Action("ConfirmEmail", "Account", new { userId = user.Id, code = code }, protocol: Request.Url.Scheme);
                    // await UserManager.SendEmailAsync(user.Id, "Confirm your account", "Please confirm your account by clicking <a href=\"" + callbackUrl + "\">here</a>");

                    return RedirectToAction("Index","Home");
                }
                AddErrors(result);
            }

            // If we got this far, something failed, redisplay form
            return View(model);
        }

Спасибо

1 ответ

Вы можете добавить собственный маршрут после успешного входа пользователя:

case SignInStatus.Success:
                using (RouteTable.Routes.GetWriteLock())
                {
                    RouteTable.Routes.Insert(0, new Route("{user}/{controller}/{action}/{id}", 
                        new RouteValueDictionary() { ["user"] = "Nick", ["controller"] = "Home", ["action"] = "Index", ["id"] = UrlParameter.Optional }, 
                        new MvcRouteHandler()));
                }
                return RedirectToLocal(returnUrl);

И удалите его, когда пользователь выйдет из системы:

AuthenticationManager.SignOut(DefaultAuthenticationTypes.ApplicationCookie);
        using (RouteTable.Routes.GetWriteLock())
        {
            RouteTable.Routes.RemoveAt(0);
        }
        return RedirectToAction("Index", "Home");
Другие вопросы по тегам