基于端点的路由 .net core 3.1 中缺少端点

Missing endpoints in endpoint based routing .net core 3.1

我正在使用由 WebApi 和 MVC 路由组成的多租户 .net 核心 Web 应用程序。对于正常行为,应用程序应该转到回退控制器,在该控制器中使用外部帮助将决定执行哪个 master/tenant 控制器 redirect.But 有时应用程序拒绝找到适当的回退操作。经过一些调查,我发现在这些时刻,并不是所有的端点都被注册了。经过两天的搜索,我发现这些问题大多影响Razor,但其中none适用于MVC/WebApi。我开始认为这是由于具有 .MapWhen() 的第二个分支引起的,但尚未确认。

因此,对于解决此问题的任何帮助,我将不胜感激。还要在下面附上我当前的路由配置:

配置服务方法:

services.AddControllersWithViews(options =>
    {
        options.Filters.Add(typeof(ReverseProxyFilter));
        options.Conventions.Add(new ApiExplorerGroupPerVersionConvention());
    })
    .AddRazorRuntimeCompilation()
    .AddApplicationPart(typeof(EmailNamespace).Assembly)
    .AddViewLocalization()
    .SetCompatibilityVersion(CompatibilityVersion.Latest)
    .AddNewtonsoftJson(options =>
        {
        options.SerializerSettings.ContractResolver = new CamelCasePropertyNamesContractResolver();
        options.SerializerSettings.DateParseHandling = DateParseHandling.DateTimeOffset;
        options.SerializerSettings.ReferenceLoopHandling = ReferenceLoopHandling.Ignore;
        })
    .AddControllersAsServices();

services.AddRazorPages();

services.Configure<ForwardedHeadersOptions>(options =>
    {
        options.ForwardedHeaders = ForwardedHeaders.XForwardedProto;
    });

services.Configure<RouteOptions>(routeOptions =>
    {
        routeOptions.ConstraintMap.Add("master", typeof(MasterRouteConstraint));
        routeOptions.ConstraintMap.Add("tenant", typeof(TenantRouteConstraint));
    });

配置方法:

app.UseMiddleware<TenantFilterMiddleware>();

app.UseHttpStatusCodeExceptionMiddleware();

app.UseResponseCaching();

app.UseRouting();
app.UseAuthentication();
app.UseAuthorization();

app.UseEndpoints(endpoints =>
{
    endpoints.MapHub<WebsocketsHub>("/hubs");
    endpoints.MapControllers();
    endpoints.MapRazorPages();
});

app.MapWhen(
    context => !context.Request.Path.Value.StartsWith("/api"),
    builder =>
        {
            builder.UseRouting();
            builder.UseEndpoints(endpoints =>
            {
                endpoints.MapFallbackToController("Index", "Fallback");
            });
    });

异常示例:

An unhandled exception occurred while processing the request.
InvalidOperationException: Cannot find the fallback endpoint specified by route values: 
{ 
    action: Index,
    controller: Fallback,
    area: 
}.
Microsoft.AspNetCore.Mvc.Routing.DynamicControllerEndpointMatcherPolicy.ApplyAsync(HttpContext HttpContext, CandidateSet candidates)

经过两天的尝试和研究,解决方案像往常一样非常简单。 解决这个问题的方法是稍微重写路由管道。据我所知,Microsoft 路由系统非常喜欢将分支放在默认分支之前而不是之后,在这种情况下,所有系统行为在 100% 的情况下都按预期开始工作。

所以不要这样:

app.UseRouting();
app.UseAuthentication();
app.UseAuthorization();

app.UseEndpoints(endpoints =>
{
    endpoints.MapHub<WebsocketsHub>("/hubs");
    endpoints.MapControllers();
    endpoints.MapRazorPages();
});

app.MapWhen(
    context => !context.Request.Path.Value.StartsWith("/api"),
    builder =>
        {
            builder.UseRouting();
            builder.UseEndpoints(endpoints =>
            {
                endpoints.MapFallbackToController("Index", "Fallback");
            });
    });

现在我有了这个:

app.MapWhen(context => !context.Request.Path.Value.StartsWith("/api"), ConfigureApiPipeline);

app.UseRouting();
app.UseAuthentication();
app.UseAuthorization();
app.UseSession();

app.UseEndpoints(MapBasicEndpoints);

其中 ConfigureApiPipeline() 函数如下所示:

app.UseRouting();
app.UseAuthentication();
app.UseAuthorization();
app.UseSession();
app.UseEndpoints(endpoints =>
    {
        MapBasicEndpoints(endpoints);
        endpoints.MapHealthChecks(SystemLivenessCheck.Path, new HealthCheckOptions()
                {
                    AllowCachingResponses = false,
                    ResultStatusCodes = new Dictionary<HealthStatus, int>
                    {
                        [HealthStatus.Healthy] = StatusCodes.Status200OK,
                        [HealthStatus.Degraded] = StatusCodes.Status503ServiceUnavailable,
                        [HealthStatus.Unhealthy] = StatusCodes.Status503ServiceUnavailable,
                    },
                });
        endpoints.MapFallbackToController("Index", "Fallback");
    });

MapBasicEndpoints是:

private static void MapBasicEndpoints(IEndpointRouteBuilder endpoints)
{
    endpoints.MapHub<WebsocketsHub>("/hubs");
    endpoints.MapControllers();
}