原文: https://howtodoinjava.com/spring-cloud/spring-cloud-api-gateway-zuul/

学习使用 Netflix Zuul 以及与 Spring Cloud 的牢固绑定来创建负载均衡器。 在这里,我们将主要关注 API 网关模式及其用法。 我们将构建一个 netflix zuul 示例,在此示例中,我们将创建一个微服务生态系统并测试其 Zuul API 网关在整个生态系统中的有效性和适用性。

这是一种非常常见的微服务模式,Zuul 的创建者 Netflix 大量智能地利用了这种模式,Netflix 声称所有 Netflix 流量首先进入 Zuul 集群,该集群主要负责基于不同的动态路由,监控,弹性和安全性。 基于 groovy 的自定义过滤器。

1. Zuul 在微服务生态系统中是什么位置?

构建微服务时,一个常见的问题是为系统的客户端应用程序提供唯一的网关。 您的服务被划分为小型微服务应用程序,否则用户不应该看到它们,否则可能会导致大量的开发/维护工作。 此外,在某些情况下,整个生态系统网络流量可能会通过单个点,这可能会影响群集的性能。

为解决此问题,Netflix(微服务的主要采用者)创建并开放了其 Zuul 代理服务器,并在之后将其开源,Pivotal 下面的 Spring 将其收纳在 Spring Cloud 技术栈中, 只需几个简单的步骤即可轻松有效地使用 zuul。

Zuul 是一种边缘服务,代理对多个支持服务的请求。 它为您的生态系统提供了统一的“前门”,允许任何浏览器,移动应用程序或其他用户界面使用来自多个主机的服务。 您可以将 Zuul 与其他 Netflix 堆栈组件(例如 Hystrix 进行容错)和 Eureka 进行服务发现进行集成,或将其用于管理系统中的路由规则,过滤器和负载平衡。 最重要的是,所有这些组件都可以通过 spring boot / cloud 方法由 spring 框架很好地适应。

Netflix Zuul 示例 – Zuul API 网关模式 – Spring Cloud 教程 - 图1

前端具有 Zuul 网关的微服务生态系统

2. Zuul 组件

Zuul 主要具有四种类型的过滤器,使我们能够拦截任何特定事务在请求处理的不同时间轴上的流量。 我们可以为特定的网址格式添加任意数量的过滤器。

  • 前置过滤器 – 在路由请求之前被调用。
  • 后置过滤器 – 在路由请求后调用。
  • 路由过滤器 – 用于路由请求。
  • 错误过滤器 – 在处理请求时发生错误时调用。

Netflix Zuul 示例 – Zuul API 网关模式 – Spring Cloud 教程 - 图2

Zuul 内部具有不同过滤器的请求处理流程

3. netflix zuul 示例概述

现在,让我们使用 Zuul 代理创建一个简单而有意义的生态系统,以免脏污。 我们将创建以下工件来演示整个过程:

  • 学生微服务 – 基于 Spring Boot 的微服务,它将仅公开一个 URL 以启用某些搜索功能。 为简单起见,我们只返回硬编码的值,但在现实世界中,我们可以连接到此服务的任何位置以获取数据。

  • Zuul 网关服务代理 – 将再次基于 Spring Boot,它将基本上拦截所有学生服务的流量并应用一系列请求过滤器,然后路由到基础服务,并在响应服务时再次路由, 它将应用一些响应过滤。 由于它是网关,因此我们可以有效地使用过滤器来采取许多有趣且有用的操作。
    网关服务的一些常见责任是:

    • 在网关层应用微服务身份验证和安全性以保护实际服务
    • 我们可以通过启用一些日志记录以获取边缘的有意义的数据和统计信息,从而对进入生态系统的所有流量进行微服务洞察并监视,以便为我们提供准确的生产视图。
    • 动态路由可以根据需要将请求路由到不同的后端群集。
    • 我们可以通过逐渐增加到新集群的流量来进行运行时压力测试,以便在许多情况下(例如, 群集具有新的硬件和网络设置,或者已部署了新版本的生产代码。
    • 我们可以进行动态减载,即为每种类型的请求分配容量,并丢弃超出限制的请求。
    • 我们可以应用静态响应处理,即直接在边缘构建一些响应,而不是将其转发到内部集群进行处理。

技术栈和运行时

  • Java 1.8 和 Eclipse IDE 作为开发环境
  • Spring Cloud Zuul 作为网关代理供应器
  • Spring Boot 作为应用程序框架
  • Spring Rest 用于将微服务公开为 REST
  • Maven 作为构建工具

4. 创建学生微服务

请按照以下步骤开发学生微服务,该服务将公开几个 REST 终结点,这些终结点以后可通过 zuul 代理进行访问。 稍后我们将研究 zuul 部分,让我们首先创建学生服务。

4.1. 创建 Spring Boot 项目

spring 初始化器页面创建一个具有相关性(即WebRest Repositories)的 Spring Boot 项目。 给出其他 Maven GAV 坐标并下载项目。

Netflix Zuul 示例 – Zuul API 网关模式 – Spring Cloud 教程 - 图3

学生服务 Maven 项目生成

将项目解压缩并将其作为existing maven project导入 Eclipse。 在此步骤中,使用命令mvn clean install重新构建 Maven,以便正确下载所有 Maven 依赖项。

4.2. 添加一些 REST 端点

现在,我们将仅向该服务添加一些 REST 端点,以便稍后测试代理。 为此,我们需要通过添加注解@RestController来添加一个 REST 控制器。 为简单起见,我们将添加一个模型类Student

完成所有更改后,该类将如下所示。

  1. package com.example.springboostudentservice;
  2. import java.util.Date;
  3. import org.springframework.boot.SpringApplication;
  4. import org.springframework.boot.autoconfigure.SpringBootApplication;
  5. import org.springframework.web.bind.annotation.PathVariable;
  6. import org.springframework.web.bind.annotation.RequestMapping;
  7. import org.springframework.web.bind.annotation.RestController;
  8. @RestController
  9. @SpringBootApplication
  10. public class SpringBootStudentServiceApplication
  11. {
  12. @RequestMapping(value = "/echoStudentName/{name}")
  13. public String echoStudentName(@PathVariable(name = "name") String name)
  14. {
  15. return "hello <strong style=\"color: red;\">" + name + " </strong> Responsed on : " + new Date();
  16. }
  17. @RequestMapping(value = "/getStudentDetails/{name}")
  18. public Student getStudentDetails(@PathVariable(name = "name") String name)
  19. {
  20. return new Student(name, "Pune", "MCA");
  21. }
  22. public static void main(String[] args)
  23. {
  24. SpringApplication.run(SpringBootStudentServiceApplication.class, args);
  25. }
  26. }
  27. class Student
  28. {
  29. String name;
  30. String address;
  31. String cls;
  32. public Student(String name, String address, String cls) {
  33. super();
  34. this.name = name;
  35. this.address = address;
  36. this.cls = cls;
  37. }
  38. public String getName() {
  39. return name;
  40. }
  41. public String getAddress() {
  42. return address;
  43. }
  44. public String getCls() {
  45. return cls;
  46. }
  47. }

4.3. 应用配置

现在打开application.properties文件并添加这些条目。

  1. spring.application.name=student
  2. server.port=8090

在这里,我们通过属性spring.application.name=student为该服务命名,并且通过server.port=8090定义默认端口。 我们需要覆盖默认端口,因为我们将在本地主机中运行不同微服务的多个实例。

4.4. 验证学生服务

最后,使用命令mvn clean install进行 maven 构建,并通过运行命令java -jar target\spring-boot-zuulgatwayproxy-student-service-0.0.1-SNAPSHOT.jar作为 Spring Boot 应用程序启动该项目。 现在,一旦服务器启动,请转到浏览器并测试端点是否正常运行。

http://localhost:8090/echoStudentName/Sajal

Netflix Zuul 示例 – Zuul API 网关模式 – Spring Cloud 教程 - 图4

浏览器输出

http://localhost:8090/getStudentDetails/Sajal

Netflix Zuul 示例 – Zuul API 网关模式 – Spring Cloud 教程 - 图5

浏览器输出

现在,我们将使用 Zuul 创建实际的代理服务。

5. 创建 Zuul 网关服务代理

这将再次是基于 Spring Boot 的微服务,但它具有特殊功能。 它将使用 zuul 创建一个 API 网关代理,该代理将代理学生服务。 稍后,我们可以添加任何数量的微服务,例如学生服务,并能够创建强大的微服务生态系统。

5.1. 创建 Spring Boot 项目

从具有Zuul依赖关系的 spring 初始化器页面创建一个 Spring Boot 项目。 给出其他 Maven GAV 坐标并下载项目。

Netflix Zuul 示例 – Zuul API 网关模式 – Spring Cloud 教程 - 图6

Zuul 代理服务 Maven 项目生成

将项目解压缩并将其作为现有的 maven 项目导入 Eclipse。 在此步骤中,使用命令mvn clean install重新构建 Maven,以便正确下载所有 Maven 依赖项。

5.2. 启用 Zuul 服务代理

现在,在src文件夹中的 Spring 运行应用程序类上添加@EnableZuulProxy注解。 使用此注解,此工件将像 Zuul 服务代理一样工作,并将启用 API 网关层的所有功能,如前所述。 然后,我们将添加一些过滤器和路由配置。

  1. package com.example.springbootzuulgatwayproxy;
  2. import org.springframework.boot.SpringApplication;
  3. import org.springframework.boot.autoconfigure.SpringBootApplication;
  4. import org.springframework.cloud.netflix.zuul.EnableZuulProxy;
  5. import org.springframework.context.annotation.Bean;
  6. import com.example.springbootzuulgatwayproxy.filters.ErrorFilter;
  7. import com.example.springbootzuulgatwayproxy.filters.PostFilter;
  8. import com.example.springbootzuulgatwayproxy.filters.PreFilter;
  9. import com.example.springbootzuulgatwayproxy.filters.RouteFilter;
  10. @SpringBootApplication
  11. @EnableZuulProxy
  12. public class SpringBootZuulgatwayproxyApplication {
  13. public static void main(String[] args) {
  14. SpringApplication.run(SpringBootZuulgatwayproxyApplication.class, args);
  15. }
  16. @Bean
  17. public PreFilter preFilter() {
  18. return new PreFilter();
  19. }
  20. @Bean
  21. public PostFilter postFilter() {
  22. return new PostFilter();
  23. }
  24. @Bean
  25. public ErrorFilter errorFilter() {
  26. return new ErrorFilter();
  27. }
  28. @Bean
  29. public RouteFilter routeFilter() {
  30. return new RouteFilter();
  31. }
  32. }

5.3. Zuul 路由配置

打开application.properties并添加以下条目:

  1. #Zuul routes. Here for /student path, we are routing to localhost:8090 with extra path after that.
  2. zuul.routes.student.url=http://localhost:8090
  3. #Ribbon is auto integrated with Zuul and for this exercise we are not using that.
  4. ribbon.eureka.enabled=false
  5. #Will start the gateway server @8080
  6. server.port=8080

在这里,zuul.routes.student.url将路由所有请求/student的流量到实际的学生服务服务器。 这样,我们也可以将路由添加到其他服务。
ribbon.eureka.enabled与 Zuul 自动集成。
server.port – 需要覆盖默认端口,因为我们将在本地主机中运行不同微服务的多个实例。

5.4. 添加 Zuul 过滤器

如我们已经描述的,我们现在将添加一些过滤器,Zuul 支持 4 种类型的过滤器,即prepostrouteerror。 在这里,我们将创建每种类型的过滤器。

要编写过滤器,我们基本上需要执行以下步骤:

  • 需要扩展com.netflix.zuul.ZuulFilter
  • 需要覆盖filterTypefilterOrdershouldFilterrun方法。 这里filterType方法只能返回四个字符串 – pre/post/route/error中的任何一个。 依赖此值,过滤器将像特定过滤器一样工作。
  • run方法是根据我们的要求放置过滤器逻辑的地方。
  • 同样,我们可以根据需要添加任意数量的任何特定过滤器,在这种类型的过滤器执行阶段,这种情况filterOrder将用于确定该过滤器的顺序。

前置过滤器代码 – 我们将添加以下前置过滤器。 目前,出于测试目的,过滤器除了println以外无所作为。 但是实际上,这些功能足以执行前面提到的许多重要切面。

  1. package com.example.springbootzuulgatwayproxy.filters;
  2. import javax.servlet.http.HttpServletRequest;
  3. import com.netflix.zuul.ZuulFilter;
  4. import com.netflix.zuul.context.RequestContext;
  5. public class PreFilter extends ZuulFilter {
  6. @Override
  7. public String filterType() {
  8. return "pre";
  9. }
  10. @Override
  11. public int filterOrder() {
  12. return 1;
  13. }
  14. @Override
  15. public boolean shouldFilter() {
  16. return true;
  17. }
  18. @Override
  19. public Object run() {
  20. RequestContext ctx = RequestContext.getCurrentContext();
  21. HttpServletRequest request = ctx.getRequest();
  22. System.out.println("Request Method : " + request.getMethod() + " Request URL : " + request.getRequestURL().toString());
  23. return null;
  24. }
  25. }

后置过滤器

  1. package com.example.springbootzuulgatwayproxy.filters;
  2. import com.netflix.zuul.ZuulFilter;
  3. public class PostFilter extends ZuulFilter {
  4. @Override
  5. public String filterType() {
  6. return "post";
  7. }
  8. @Override
  9. public int filterOrder() {
  10. return 1;
  11. }
  12. @Override
  13. public boolean shouldFilter() {
  14. return true;
  15. }
  16. @Override
  17. public Object run() {
  18. System.out.println("Inside Response Filter");
  19. return null;
  20. }
  21. }

路由过滤器

  1. package com.example.springbootzuulgatwayproxy.filters;
  2. import com.netflix.zuul.ZuulFilter;
  3. public class RouteFilter extends ZuulFilter {
  4. @Override
  5. public String filterType() {
  6. return "route";
  7. }
  8. @Override
  9. public int filterOrder() {
  10. return 1;
  11. }
  12. @Override
  13. public boolean shouldFilter() {
  14. return true;
  15. }
  16. @Override
  17. public Object run() {
  18. System.out.println("Inside Route Filter");
  19. return null;
  20. }
  21. }

错误过滤器

  1. package com.example.springbootzuulgatwayproxy.filters;
  2. import com.netflix.zuul.ZuulFilter;
  3. public class ErrorFilter extends ZuulFilter {
  4. @Override
  5. public String filterType() {
  6. return "error";
  7. }
  8. @Override
  9. public int filterOrder() {
  10. return 1;
  11. }
  12. @Override
  13. public boolean shouldFilter() {
  14. return true;
  15. }
  16. @Override
  17. public Object run() {
  18. System.out.println("Inside Route Filter");
  19. return null;
  20. }
  21. }

5.5. 注册 zuul 过滤器

创建这些过滤器的 Bean 定义以进行自动注册和启用。

  1. @Bean
  2. public PreFilter preFilter() {
  3. return new PreFilter();
  4. }
  5. @Bean
  6. public PostFilter postFilter() {
  7. return new PostFilter();
  8. }
  9. @Bean
  10. public ErrorFilter errorFilter() {
  11. return new ErrorFilter();
  12. }
  13. @Bean
  14. public RouteFilter routeFilter() {
  15. return new RouteFilter();
  16. }

6. Netflix Zuul 示例演示

因此,我们启用了 Zuul,添加了所需的配置并开发了过滤器。 因此,我们可以进行基本测试以了解整个过程。

使用命令mvn clean install进行 Maven 构建,并通过运行命令java -jar target\spring-boot-zuulgatwayproxy-0.0.1-SNAPSHOT.jar作为 Spring Boot 应用程序启动该项目。 现在,服务器启动后,进入浏览器并通过访问学生服务(名称为/student)来测试端点是否正常运行。

http://localhost:8080/student/getStudentDetails/Sajal

Netflix Zuul 示例 – Zuul API 网关模式 – Spring Cloud 教程 - 图7

代理服务输出

http://localhost:8080/student/echoStudentName/Sajal

Netflix Zuul 示例 – Zuul API 网关模式 – Spring Cloud 教程 - 图8

代理服务输出

7. 总结

关于 netflix zuul 过滤器示例的全部内容。 我建议您自己做,添加更多基础服务,并通过代理路由请求,应用不同类型的过滤器,并在过滤器中添加真实的逻辑。

下载源码

将我的问题放在评论部分。

学习愉快!