在Ocelot中使用自定义的中间件(一)

2023-11-06 07:32

本文主要是介绍在Ocelot中使用自定义的中间件(一),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Ocelot是ASP.NET Core下的API网关的一种实现,在微服务架构领域发挥了非常重要的作用。本文不会从整个微服务架构的角度来介绍Ocelot,而是介绍一下最近在学习过程中遇到的一个问题,以及如何使用中间件(Middleware)来解决这样的问题。

问题描述

在上文中,我介绍了一种在Angular站点里基于Bootstrap切换主题的方法。之后,我将多个主题的boostrap.min.css文件放到一个ASP.NET Core Web API的站点上,并用静态文件的方式进行分发,在完成这部分工作之后,调用这个Web API,就可以从服务端获得主题信息以及所对应的样式文件。例如:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

// GET http://localhost:5010/api/themes

{

    "version": "1.0.0",

    "themes": [

        {

            "name": "蔚蓝 (Cerulean)",

            "description": "Cerulean",

            "category": "light",

            "cssMin": "http://localhost:5010/themes/cerulean/bootstrap.min.css",

            "navbarClass": "navbar-dark",

            "navbarBackgroundClass": "bg-primary",

            "footerTextClass": "text-light",

            "footerLinkClass": "text-light",

            "footerBackgroundClass": "bg-primary"

        },

        {

            "name": "机械 (Cyborg)",

            "description": "Cyborg",

            "category": "dark",

            "cssMin": "http://localhost:5010/themes/cyborg/bootstrap.min.css",

            "navbarClass": "navbar-dark",

            "navbarBackgroundClass": "bg-dark",

            "footerTextClass": "text-dark",

            "footerLinkClass": "text-dark",

            "footerBackgroundClass": "bg-light"

        }

    ]

}

当然,整个项目中不仅仅是有这个themes API,还有另外2-3个服务在后台运行,项目是基于微服务架构的。为了能够让前端有统一的API接口,我使用Ocelot作为服务端的API网关,以便为Angular站点提供API服务。于是,我定义了如下ReRoute规则:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

{

    "ReRoutes": [

        {

            "DownstreamPathTemplate": "/api/themes",

            "DownstreamScheme": "http",

            "DownstreamHostAndPorts": [

                {

                    "Host": "localhost",

                    "Port": 5010

                }

            ],

            "UpstreamPathTemplate": "/themes-api/themes",

            "UpstreamHttpMethod": [ "Get" ]

        }

    ]

}

假设API网关运行在http://localhost:9023,那么基于上面的ReRoute规则,通过访问http://localhost:9023/themes-api/themes,即可转发到后台的http://localhost:5010/api/themes,完成API的调用。运行一下,调用结果如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

// GET http://localhost:9023/themes-api/themes

{

    "version": "1.0.0",

    "themes": [

        {

            "name": "蔚蓝 (Cerulean)",

            "description": "Cerulean",

            "category": "light",

            "cssMin": "http://localhost:5010/themes/cerulean/bootstrap.min.css",

            "navbarClass": "navbar-dark",

            "navbarBackgroundClass": "bg-primary",

            "footerTextClass": "text-light",

            "footerLinkClass": "text-light",

            "footerBackgroundClass": "bg-primary"

        },

        {

            "name": "机械 (Cyborg)",

            "description": "Cyborg",

            "category": "dark",

            "cssMin": "http://localhost:5010/themes/cyborg/bootstrap.min.css",

            "navbarClass": "navbar-dark",

            "navbarBackgroundClass": "bg-dark",

            "footerTextClass": "text-dark",

            "footerLinkClass": "text-dark",

            "footerBackgroundClass": "bg-light"

        }

    ]

}

看上去一切正常,但是,每个主题设置的css文件地址仍然还是指向下游服务的URL地址,比如上面的cssMin中,还是使用的http://localhost:5010。从部署的角度,外部是无法访问除了API网关以外的其它服务的,于是,这就造成了css文件无法被访问的问题。

解决这个问题的思路很简单,就是API网关在返回response的时候,将cssMin的地址替换掉。如果在Ocelot的配置中加入以下ReRoute设置:

1

2

3

4

5

6

7

8

9

10

11

12

{

  "DownstreamPathTemplate": "/themes/{name}/bootstrap.min.css",

  "DownstreamScheme": "http",

  "DownstreamHostAndPorts": [

    {

      "Host": "localhost",

      "Port": 5010

    }

  ],

  "UpstreamPathTemplate": "/themes-api/theme-css/{name}",

  "UpstreamHttpMethod": [ "Get" ]

}

那么只需要将下游response中cssMin的值(比如http://localhost:5010/themes/cyborg/bootstrap.min.css)替换为Ocelot网关中设置的上游URL(比如http://localhost:9023/themes-api/theme-css/cyborg),然后将替换后的response返回给API调用方即可。这个过程,可以使用Ocelot中间件完成。

使用Ocelot中间件

Ocelot中间件是继承于OcelotMiddleware类的子类,并且可以在Startup.Configure方法中,通过app.UseOcelot方法将中间件注入到Ocelot管道中,然而,简单地调用IOcelotPipelineBuilder的UseMiddleware方法是不行的,它会导致整个Ocelot网关不可用。比如下面的方法是不行的:



这是因为没有将Ocelot的其它Middleware加入到管道中,Ocelot管道中只有ThemeCssMinUrlReplacer中间件。要解决这个问题,我目前的方法就是通过使用扩展方法,将所有Ocelot中间全部注册好,然后再注册自定义的中间件,比如:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

public static IOcelotPipelineBuilder BuildCustomOcelotPipeline(this IOcelotPipelineBuilder builder,

    OcelotPipelineConfiguration pipelineConfiguration)

{

    builder.UseExceptionHandlerMiddleware();

    builder.MapWhen(context => context.HttpContext.WebSockets.IsWebSocketRequest,

        app =>

        {

            app.UseDownstreamRouteFinderMiddleware();

            app.UseDownstreamRequestInitialiser();

            app.UseLoadBalancingMiddleware();

            app.UseDownstreamUrlCreatorMiddleware();

            app.UseWebSocketsProxyMiddleware();

        });

    builder.UseIfNotNull(pipelineConfiguration.PreErrorResponderMiddleware);

    builder.UseResponderMiddleware();

    builder.UseDownstreamRouteFinderMiddleware();

    builder.UseSecurityMiddleware();

    if (pipelineConfiguration.MapWhenOcelotPipeline != null)

    {

        foreach (var pipeline in pipelineConfiguration.MapWhenOcelotPipeline)

        {

            builder.MapWhen(pipeline);

        }

    }

    builder.UseHttpHeadersTransformationMiddleware();

    builder.UseDownstreamRequestInitialiser();

    builder.UseRateLimiting();

 

    builder.UseRequestIdMiddleware();

    builder.UseIfNotNull(pipelineConfiguration.PreAuthenticationMiddleware);

    if (pipelineConfiguration.AuthenticationMiddleware == null)

    {

        builder.UseAuthenticationMiddleware();

    }

    else

    {

        builder.Use(pipelineConfiguration.AuthenticationMiddleware);

    }

    builder.UseClaimsToClaimsMiddleware();

    builder.UseIfNotNull(pipelineConfiguration.PreAuthorisationMiddleware);

    if (pipelineConfiguration.AuthorisationMiddleware == null)

    {

        builder.UseAuthorisationMiddleware();

    }

    else

    {

        builder.Use(pipelineConfiguration.AuthorisationMiddleware);

    }

    builder.UseClaimsToHeadersMiddleware();

    builder.UseIfNotNull(pipelineConfiguration.PreQueryStringBuilderMiddleware);

    builder.UseClaimsToQueryStringMiddleware();

    builder.UseLoadBalancingMiddleware();

    builder.UseDownstreamUrlCreatorMiddleware();

    builder.UseOutputCacheMiddleware();

    builder.UseHttpRequesterMiddleware();

     

    return builder;

}

然后再调用app.UseOcelot即可:

1

2

3

4

5

6

app.UseOcelot((builder, config) =>

{

    builder.BuildCustomOcelotPipeline(config)

    .UseMiddleware<ThemeCssMinUrlReplacer>()

    .Build();

});

这种做法其实听起来不是特别的优雅,但是目前也没找到更合适的方式来解决Ocelot中间件注册的问题。

以下便是ThemeCssMinUrlReplacer中间件的代码,可以看到,我们使用正则表达式替换了cssMin的URL部分,使得css文件的地址可以正确被返回:



执行结果如下:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

// GET http://localhost:9023/themes-api/themes

{

  "version": "1.0.0",

  "themes": [

    {

      "name": "蔚蓝 (Cerulean)",

      "description": "Cerulean",

      "category": "light",

      "cssMin": "http://localhost:9023/themes-api/theme-css/cerulean",

      "navbarClass": "navbar-dark",

      "navbarBackgroundClass": "bg-primary",

      "footerTextClass": "text-light",

      "footerLinkClass": "text-light",

      "footerBackgroundClass": "bg-primary"

    },

    {

      "name": "机械 (Cyborg)",

      "description": "Cyborg",

      "category": "dark",

      "cssMin": "http://localhost:9023/themes-api/theme-css/cyborg",

      "navbarClass": "navbar-dark",

      "navbarBackgroundClass": "bg-dark",

      "footerTextClass": "text-dark",

      "footerLinkClass": "text-dark",

      "footerBackgroundClass": "bg-light"

    }

  ]

}

总结

本文介绍了使用Ocelot中间件实现下游服务response body的替换任务,在ThemeCssMinUrlReplacer的实现代码中,我们使用了context.DownstreamReRoute.DownstreamPathTemplate.Value来判断当前执行的URL是否需要由该中间件进行处理,以避免不必要的中间件逻辑执行。这个设计可以再优化一下,使用一个简单的框架让程序员可以通过Ocelot的配置文件来更为灵活地使用Ocelot中间件,下文介绍这部分内容。

这篇关于在Ocelot中使用自定义的中间件(一)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/355292

相关文章

Java使用ANTLR4对Lua脚本语法校验详解

《Java使用ANTLR4对Lua脚本语法校验详解》ANTLR是一个强大的解析器生成器,用于读取、处理、执行或翻译结构化文本或二进制文件,下面就跟随小编一起看看Java如何使用ANTLR4对Lua脚本... 目录什么是ANTLR?第一个例子ANTLR4 的工作流程Lua脚本语法校验准备一个Lua Gramm

Java Optional的使用技巧与最佳实践

《JavaOptional的使用技巧与最佳实践》在Java中,Optional是用于优雅处理null的容器类,其核心目标是显式提醒开发者处理空值场景,避免NullPointerExce... 目录一、Optional 的核心用途二、使用技巧与最佳实践三、常见误区与反模式四、替代方案与扩展五、总结在 Java

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析

Qt中QUndoView控件的具体使用

《Qt中QUndoView控件的具体使用》QUndoView是Qt框架中用于可视化显示QUndoStack内容的控件,本文主要介绍了Qt中QUndoView控件的具体使用,具有一定的参考价值,感兴趣的... 目录引言一、QUndoView 的用途二、工作原理三、 如何与 QUnDOStack 配合使用四、自

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

使用Python构建一个Hexo博客发布工具

《使用Python构建一个Hexo博客发布工具》虽然Hexo的命令行工具非常强大,但对于日常的博客撰写和发布过程,我总觉得缺少一个直观的图形界面来简化操作,下面我们就来看看如何使用Python构建一个... 目录引言Hexo博客系统简介设计需求技术选择代码实现主框架界面设计核心功能实现1. 发布文章2. 加

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的

使用Python开发一个带EPUB转换功能的Markdown编辑器

《使用Python开发一个带EPUB转换功能的Markdown编辑器》Markdown因其简单易用和强大的格式支持,成为了写作者、开发者及内容创作者的首选格式,本文将通过Python开发一个Markd... 目录应用概览代码结构与核心组件1. 初始化与布局 (__init__)2. 工具栏 (setup_t

Python虚拟环境终极(含PyCharm的使用教程)

《Python虚拟环境终极(含PyCharm的使用教程)》:本文主要介绍Python虚拟环境终极(含PyCharm的使用教程),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录一、为什么需要虚拟环境?二、虚拟环境创建方式对比三、命令行创建虚拟环境(venv)3.1 基础命令3

Python Transformer 库安装配置及使用方法

《PythonTransformer库安装配置及使用方法》HuggingFaceTransformers是自然语言处理(NLP)领域最流行的开源库之一,支持基于Transformer架构的预训练模... 目录python 中的 Transformer 库及使用方法一、库的概述二、安装与配置三、基础使用:Pi