asp.net core mvc中如何把次層網域綁定到特定的控制器上

來源:互聯網
上載者:User
  由於公司的工作安排,一直在研究其他技術,所以一直沒時間更新部落格,今天終於可以停下手頭的事情,寫一些新內容了。

  應用情境:企業門戶網站會根據內容不同,設定不同的板塊,如新浪有體育,娛樂頻道,等等。有的情況下需要給不同的板塊設定不同的次層網域,如新浪體育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" });        });

  實現方法可能不是最好的,但是已經滿足了基本需求,如果大家有更好的方法,歡迎討論交流。

  

  

  

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.