Атрибут маршрутизации - показать вошедший в систему идентификатор пользователя в 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");