>  기사  >  백엔드 개발  >  특정 컨트롤러 인스턴스에 두 번째 수준 도메인 이름을 바인딩하는 방법에 대한 자습서

특정 컨트롤러 인스턴스에 두 번째 수준 도메인 이름을 바인딩하는 방법에 대한 자습서

零下一度
零下一度원래의
2017-06-17 10:24:081472검색

이 글에서는 주로 Asp.net Core MVC의 특정 controller에 2차 도메인 이름을 바인딩하는 방법을 소개합니다. 필요한 친구는 참고하면 됩니다.

적용 시나리오: 엔터프라이즈 포털은 에 따라 다릅니다. Sina에는 스포츠, 엔터테인먼트 채널 등 다양한 섹션을 설정하세요. 어떤 경우에는 Sina Sports sports.sina.com.cn과 같이 섹션마다 다른 2차 도메인 이름을 설정해야 합니다.

 asp.net 코어 mvc에서 섹션의 효과를 얻으려면 섹션마다 다른 컨트롤러를 만들 수 있습니다(물론 다른 기술도 있으므로 구현 품질은 여기서 논의하지 않습니다). 예를 들어 스포츠 채널에 해당하는 컨트롤러는 스포츠 컨트롤러를 통해 시스템에 액세스할 때 호출됩니다.

 위의 시나리오를 마쳤으니 어떻게 구현하는지 살펴보겠습니다.

 asp.net 코어 mvc에 라우팅 규칙 구성이 있습니다. 구성은 Startup.Configure 메서드에 있습니다. 구체적인 코드는 다음과 같습니다.


app.UseMvc(routes =>
{
   routes.MapRoute(
      name: "default",
      template: "{controller=Home}/{action=Index}/{id?}",
      defaults: new { area="admin"});
});

 안타깝게도 도메인 이름에 대한 지원은 지원되지 않습니다. 현재 이해하고 있는 내용은 입니다. 질문이 있으면 정정해 주세요). RouteCollection을 통해 라우팅 규칙을 등록하고 RouteCollection에 추가합니다. 요청이 오면 RouterCollection은 일치하는 첫 번째 IRouter를 찾을 때까지 등록된 모든 IRouter 개체를 반복합니다. 프레임워크는 도메인 이름 구성 규칙을 지원하지 않지만 IRouter를 직접 구현하여 2차 도메인 이름 결정 논리를 구현할 수 있습니다. 임시로 이름을 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;
    }
}

위 내용은 특정 컨트롤러 인스턴스에 두 번째 수준 도메인 이름을 바인딩하는 방법에 대한 자습서의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.