[英]Nothing happened with custom IRouter in Asp.Net Core
我的路由器具有以下实现:
public class TenantUrlResolverRouter : IRouter
{
private readonly IRouter _defaultRouter;
public TenantUrlResolverRouter(IRouter defaultRouter)
{
_defaultRouter = defaultRouter;
}
public VirtualPathData GetVirtualPath(VirtualPathContext context)
{
return _defaultRouter.GetVirtualPath(context);
}
public async Task RouteAsync(RouteContext context)
{
var oldRouteData = context.RouteData;
var newRouteData = new RouteData(oldRouteData);
newRouteData.Values["library"] = "Default";
try
{
context.RouteData = newRouteData;
await _defaultRouter.RouteAsync(context);
}
finally
{
if (!context.IsHandled)
{
context.RouteData = oldRouteData;
}
}
}
}
然后在Startup.cs
定义它:
app.UseMvc(routes =>
{
routes.Routes.Add(
new TenantUrlResolverRouter(routes.DefaultHandler));
routes.MapRoute(
name: "default",
template: "{library=Unknown}/{controller=Home}/{action=Index}/{id?}");
});
但是什么也没发生, RouteContext
RouteData.Values
始终为空,我始终为Unknown ,而它需要为Default 。 这不是预定义模板的问题,因为它甚至在没有{library=Unknown}
情况下也可以工作,并且此{library}/{controller=Home}/{action=Index}/{id?}
也不起作用。
此自定义IRouter有什么问题?
您没有提供完整的路由值集。 为了路由到动作,您需要同时提供controller
和action
。
另外,您的路线中没有匹配条件。 匹配条件将确定传入的请求是否与当前路由匹配。 在内置路由中, url
和constraints
是匹配条件。 但是,在自定义路由中,您需要放置一个if
块,以确保任何不匹配的请求都将传递到下一条已注册的路由。
注意:这是自定义路由最强大的部分。 内置路由只能与URL匹配。 但是使用自定义路由,您可以匹配请求中的任何内容。 例如,您可以使路由仅对某个域或某个子域匹配。 您甚至可以使其与发布的表单值或会话状态之类的东西匹配。
public async Task RouteAsync(RouteContext context)
{
var requestPath = context.HttpContext.Request.Path.Value;
if (!string.IsNullOrEmpty(requestPath) && requestPath[0] == '/')
{
// Trim the leading slash
requestPath = requestPath.Substring(1);
}
var segments = requestPath.Split('/');
// Required: Match condition to determine if the incoming request
// matches this route. If not, you should allow the framework to
// match another route by doing nothing here.
if (segments.Length > 0 && segments[0].Equals("libraryname", StringComparison.OrdinalIgnoreCase))
{
var oldRouteData = context.RouteData;
var newRouteData = new RouteData(oldRouteData);
newRouteData.Values["library"] = segments[0];
newRouteData.Values["controller"] = segments[1];
newRouteData.Values["action"] = segments[2];
try
{
context.RouteData = newRouteData;
await _defaultRouter.RouteAsync(context);
}
finally
{
if (!context.IsHandled)
{
context.RouteData = oldRouteData;
}
}
}
}
见这个问题有关实现更多信息IRouter
。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.