由於公司的工作安排,一直在研究其他技術,所以一直沒時間更新部落格,今天終於可以停下手頭的事情,寫一些新內容了。
應用情境:企業門戶網站會根據內容不同,設定不同的板塊,如新浪有體育,娛樂頻道,等等。有的情況下需要給不同的板塊設定不同的次層網域,如新浪體育sports.sina.com.cn。
在asp.net core mvc中,如果要實現板塊的效果,可能會給不同的板塊建立不同的控制器(當然也有其他的技術,這裡不討論實現方式的好壞),在這種情況下,如何給控制器綁定上專屬的次層網域,比如體育頻道對應的控制器叫SportController,通過sports.XXX.com網域名稱訪問系統的時候,直接進入SportController,並且通過這個次層網域無法訪問其他的控制器。
上面說完情境了,下面來看下如何?。
在asp.net core mvc中有路由規則配置,配置的地方在Startup.Configure方法中,具體代碼如下:
app.UseMvc(routes =>{ routes.MapRoute( name: "default", template: "{controller=Home}/{action=Index}/{id?}", defaults: new { area="admin"});});
遺憾的是不支援對網域名稱的支援(我目前瞭解的是,如果有問題,歡迎大家指正)。通過routes.MapRouter註冊路由規則,並加入到RouteCollection中,當某個請求過來後,RouterCollection迴圈所有註冊好的IRouter對象,找到第一個匹配的IRouter為止。雖然架構不支援網域名稱配置規則,但是我們可以自己去實現一個IRouter,在裡面實現次層網域判斷的邏輯,我這裡暫時起名為SubDomainRouter,具體實現代碼如下:
public class SubDomainRouter : RouteBase { private readonly IRouter _target; private readonly string _subDomain; public SubDomainRouter( IRouter target, string subDomain,//當前路由規則綁定的次層網域 string routeTemplate, RouteValueDictionary defaults, RouteValueDictionary constrains, IInlineConstraintResolver inlineConstraintResolver) : base(routeTemplate, subDomain, inlineConstraintResolver, defaults, constrains, new RouteValueDictionary(null)) { if (target == null) { throw new ArgumentNullException(nameof(target)); } if (subDomain == null) { throw new ArgumentNullException(nameof(subDomain)); } _subDomain = subDomain; _target = target; } public override Task RouteAsync(RouteContext context) { string domain = context.HttpContext.Request.Host.Host;//擷取當前請求網域名稱,然後跟_subDomain比較,如果不想等,直接忽略 if (string.IsNullOrEmpty(domain) || string.Compare(_subDomain, domain) != 0) { return Task.CompletedTask; } //如果網域名稱匹配,再去驗證訪問路徑是否匹配 return base.RouteAsync(context); } protected override Task OnRouteMatched(RouteContext context) { context.RouteData.Routers.Add(_target); return _target.RouteAsync(context); } protected override VirtualPathData OnVirtualPathGenerated(VirtualPathContext context) { return _target.GetVirtualPath(context); } }
從上面的代碼我們只看到了網域名稱檢測,但是如何把網域名稱定向到特定的控制器上,這就需要我們在註冊這個IRouter的時候做些文章,直接上代碼:
public static class RouteBuilderExtensions { public static IRouteBuilder MapDomainRoute( this IRouteBuilder routeBuilder,string domain,string area,string controller) { if(string.IsNullOrEmpty(area)||string.IsNullOrEmpty(controller)) { throw new ArgumentNullException("area or controller can not be null"); } var inlineConstraintResolver = routeBuilder .ServiceProvider .GetRequiredService<IInlineConstraintResolver>(); string template = ""; RouteValueDictionary defaults = new RouteValueDictionary(); RouteValueDictionary constrains = new RouteValueDictionary(); constrains.Add("area", area); defaults.Add("area", area); constrains.Add("controller", controller); defaults.Add("controller", string.IsNullOrEmpty(controller) ? "home" : controller); defaults.Add("action", "index"); template += "{action}/{id?}";//路徑規則中不再包含控制器資訊,但是上面通過constrains限定了尋找時所要求的控制器名稱 routeBuilder.Routes.Add(new SubDomainRouter(routeBuilder.DefaultHandler, domain, template, defaults, constrains, inlineConstraintResolver)); return routeBuilder; }}
最後我們就可以在Startup中註冊對應的規則,如下:
app.UseMvc( routes => { routes.MapDomainRoute("xxx.domain.com","areaname","controllername"); routes.MapRoute( name: "default", template: "{controller=Home}/{action=Index}/{id?}", defaults: new { area = "web" }); });
實現方法可能不是最好的,但是已經滿足了基本需求,如果大家有更好的方法,歡迎討論交流。