Чтобы захватить поддомен при сохранении стандартных функций маршрутизации MVC5 , используйте следующий класс SubdomainRoute
, полученный из Route
.
Кроме того, SubdomainRoute
позволяет субдомену необязательно указываться как параметр запроса , что делает sub.example.com/foo/bar
и example.com/foo/bar?subdomain=sub
эквивалентными. Это позволяет вам тестировать до того, как настроены субдомены DNS. Параметр запроса (при использовании) распространяется по новым ссылкам, сгенерированным Url.Action
и т. Д.
Параметр запроса также включает локальную отладку в Visual Studio 2013 без необходимости конфигурировать с помощью netsh или запускать от имени администратора . По умолчанию IIS Express привязывается только к localhost , если не повышен; он не будет привязан к синонимичным именам хостов, таким как sub.localtest.me .
class SubdomainRoute : Route
{
public SubdomainRoute(string url) : base(url, new MvcRouteHandler()) {}
public override RouteData GetRouteData(HttpContextBase httpContext)
{
var routeData = base.GetRouteData(httpContext);
if (routeData == null) return null; // Only look at the subdomain if this route matches in the first place.
string subdomain = httpContext.Request.Params["subdomain"]; // A subdomain specified as a query parameter takes precedence over the hostname.
if (subdomain == null) {
string host = httpContext.Request.Headers["Host"];
int index = host.IndexOf('.');
if (index >= 0)
subdomain = host.Substring(0, index);
}
if (subdomain != null)
routeData.Values["subdomain"] = subdomain;
return routeData;
}
public override VirtualPathData GetVirtualPath(RequestContext requestContext, RouteValueDictionary values)
{
object subdomainParam = requestContext.HttpContext.Request.Params["subdomain"];
if (subdomainParam != null)
values["subdomain"] = subdomainParam;
return base.GetVirtualPath(requestContext, values);
}
}
Для удобства, вызовите следующий метод MapSubdomainRoute
из вашего RegisterRoutes
метода, как если бы вы указали старый MapRoute
:
static void MapSubdomainRoute(this RouteCollection routes, string name, string url, object defaults = null, object constraints = null)
{
routes.Add(name, new SubdomainRoute(url) {
Defaults = new RouteValueDictionary(defaults),
Constraints = new RouteValueDictionary(constraints),
DataTokens = new RouteValueDictionary()
});
}
Наконец, для удобного доступа к субдомену (из истинного субдомена или параметра запроса) полезно создать базовый класс Controller с этим свойством Subdomain
:
protected string Subdomain
{
get { return (string)Request.RequestContext.RouteData.Values["subdomain"]; }
}