Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 405|回复: 0

解决微服务网关Ocelot使用AddStoreOcelotConfigurationInConsul后请求404问题

[复制链接]
  • TA的每日心情
    奋斗
    3 天前
  • 签到天数: 757 天

    [LV.10]以坛为家III

    2034

    主题

    2092

    帖子

    70万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    707886
    发表于 2021-7-21 15:39:43 | 显示全部楼层 |阅读模式

    一个小插曲,最近研究 netcore 微服务网关,在使用AddStoreOcelotConfigurationInConsul将配置存到consul后,任何经过网关的请求都出现404,并且没有任何有用的异常信息打印。这里先简单讲讲这个问题是如何发生的,及如何解决。

    之前在 ASP.NET Core 2 学习笔记(三)中间件 提到过大部分扩展的Middleware都会用一个静态方法包装,如:UseMvc()UseRewriter()等,而微服务网关Ocelot 包装了两个静态的异步方法 UseOcelot

    OcelotMiddlewareExtensions.cs

    public static async Task<IApplicationBuilder> UseOcelot(this IApplicationBuilder builder)
    {
        await builder.UseOcelot(new OcelotPipelineConfiguration());
    
        return builder;
    }
    
    public static async Task<IApplicationBuilder> UseOcelot(this IApplicationBuilder builder, OcelotPipelineConfiguration pipelineConfiguration)
    {
        var configuration = await CreateConfiguration(builder);
                
        CreateAdministrationArea(builder, configuration);
    
        if(UsingRafty(builder))
        {
            SetUpRafty(builder);
        }
    
        if (UsingEurekaServiceDiscoveryProvider(configuration))
        {
            builder.UseDiscoveryClient();
        }
    
        ConfigureDiagnosticListener(builder);
    
        var pipelineBuilder = new OcelotPipelineBuilder(builder.ApplicationServices);
    
        pipelineBuilder.BuildOcelotPipeline(pipelineConfiguration);
    
        var firstDelegate = pipelineBuilder.Build();
    
        /*
        inject first delegate into first piece of asp.net middleware..maybe not like this
        then because we are updating the http context in ocelot it comes out correct for
        rest of asp.net..
        */
    
        builder.Properties["analysis.NextMiddlewareName"] = "TransitionToOcelotMiddleware";
    
        builder.Use(async (context, task) =>
        {
            var downstreamContext = new DownstreamContext(context);
            await firstDelegate.Invoke(downstreamContext);
        });
    
        return builder;
    }  

    为什么会封装成异步的方法,可能是因为从底层一步一步写上来的。但是我个人认为 UseOcelot 是完全可以封装成同步方法,以避免误用。

    误用情形,如下:

    Startup.cs

    public async void Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
    {
        loggerFactory.AddConsole(Configuration.GetSection("Logging"));
        loggerFactory.AddDebug();
    
        if (env.IsDevelopment())
        {
            app.UseDeveloperExceptionPage();
        }
    
        await app.UseOcelot();
    }

    这里将 Startup.Configure() 方法写成了一个不规范的异步方法,其实是不被支持的。但是由于不规范,绕过了检测。

    规范写法,如下

    Startup.cs

    public async Task Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
    {
        loggerFactory.AddConsole(Configuration.GetSection("Logging"));
        loggerFactory.AddDebug();
    
        if (env.IsDevelopment())
        {
            app.UseDeveloperExceptionPage();
        }
    
        await app.UseOcelot();
    }

    检测到Configure返回类型非Void,直接异常:

    System.InvalidOperationException:“The 'Configure' method in the type 'OcelotConsul.ApiGateway.Core.Startup' must have a return type of 'Void'.”

    这里可以参考aspnet/Hosting关于Configure 异步问题的讨论:

    aspnet/Hosting#27

    aspnet/Hosting#29

    aspnet/Hosting#373

    aspnet/Hosting#1088

    由于用到了不规范的异步方法,使得线程并没有等待 UseOcelot 内部初始化完成就Host了起来,而造成了一些奇怪的异常,诸如:AddStoreOcelotConfigurationInConsul后请求404等问题。

    解决方案就是用 Task.Wait() 方法 阻塞线程,等待 UseOcelot 执行完成,再Host。如下:

    public void Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
    {
        loggerFactory.AddConsole(Configuration.GetSection("Logging"));
        loggerFactory.AddDebug();
    
        if (env.IsDevelopment())
        {
            app.UseDeveloperExceptionPage();
        }
    
        app.UseOcelot().Wait();
    }

     

    也许 UseOcelot 封装成同步方法会更好。已经给作者提了Issue

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-7-5 02:18 , Processed in 0.056875 second(s), 27 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表