次層網域綁到特定的控制器執行個體教程

來源:互聯網
上載者:User
這篇文章主要介紹了Asp.net Core MVC中怎麼把次層網域綁定到特定的控制器上,需要的朋友可以參考下

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


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;    }}
相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在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.