Aby przechwycić poddomenę, zachowując standardowe funkcje routingu MVC5 , użyj następującej SubdomainRoute
klasy pochodnej Route
.
Dodatkowo SubdomainRoute
umożliwia opcjonalne określenie subdomeny jako parametru zapytania , tworzenia sub.example.com/foo/bar
i example.com/foo/bar?subdomain=sub
równoważności. Umożliwia to testowanie przed skonfigurowaniem poddomen DNS. Parametr zapytania (gdy jest używany) jest propagowany przez nowe łącza generowane przez Url.Action
itp.
Parametr zapytania umożliwia także lokalne debugowanie w Visual Studio 2013 bez konieczności konfigurowania za pomocą netsh lub uruchamiania jako Administrator . Domyślnie IIS Express łączy się z localhost tylko wtedy, gdy nie jest podniesiony; nie będzie wiązać się z synonimicznymi nazwami hostów, takimi jak 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);
}
}
Dla wygody wywołaj następującą MapSubdomainRoute
metodę ze swojej RegisterRoutes
metody, tak jak zwykły stary 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()
});
}
Wreszcie, aby wygodnie uzyskać dostęp do subdomeny (z prawdziwej subdomeny lub parametru zapytania), pomocne jest utworzenie podstawowej klasy kontrolera z tą Subdomain
właściwością:
protected string Subdomain
{
get { return (string)Request.RequestContext.RouteData.Values["subdomain"]; }
}