Zuul 使用教程 - 图1

zuul API网关,为微服务应用提供统一的对外访问接口
zuul 还提供过滤器,对所有微服务提供统一的请求校验

Zuul API 网关

1.新建项目

Zuul 使用教程 - 图2

Zuul 使用教程 - 图3

2.pom.xml还需要添加公共commons配置类

3.application.yml

  • zuul 路由配置可以省略,缺省以服务id作为访问路径
  1. spring:
  2. application:
  3. name: zuul
  4. server:
  5. port: 3001
  6. eureka:
  7. client:
  8. service-url:
  9. defaultZone: http://eureka1:2001/eureka, http://eureka2:2002/eureka
  10. zuul:
  11. routes:
  12. item-service: /item-service/**
  13. user-service: /user-service/**
  14. order-service: /order-service/**

4.主程序添加注解

添加 @EnableZuulProxy@EnableDIscoverClient 注解

  1. @EnableZuulProxy
  2. @EnableDiscoveryClient
  3. @SpringBootApplication
  4. public class ZuulApplication {
  5. public static void main(String[] args) {
  6. SpringApplication.run(Sp11ZuulApplication.class, args);
  7. }
  8. }

5.启动服务访问测试

zuul+ribbon负载均衡

  • zuul已经集成了ribbon, 默认已经实现了负载均衡

1.zuul+ribbon重试

pom.xml添加spring-retry依赖

  • 需要导入spring-retry依赖
  1. <dependency>
  2. <groupId>org.springframework.retry</groupId>
  3. <artifactId>spring-retry</artifactId>
  4. </dependency>

配置zuul开启重试,并配置 ribbon重试参数

  • 需要开启重试,默认不开启
  • zuul.retryable=true
  1. spring:
  2. application:
  3. name: zuul
  4. server:
  5. port: 3001
  6. eureka:
  7. client:
  8. service-url:
  9. defaultZone: http://eureka1:2001/eureka, http://eureka2:2002/eureka
  10. zuul:
  11. retryable: true
  12. # routes:
  13. # item-service: /item-service/**
  14. # user-service: /user-service/**
  15. # order-service: /order-service/**
  16. ribbon:
  17. ConnectTimeout: 1000
  18. ReadTimeout: 1000
  19. MaxAutoRetriesNextServer: 1
  20. MaxAutoRetries: 1

zuul+hystrix降级

0配置,已经启用了hystrix
降级代码:
实现FallbackProvider接口,实现降级代码
只需要添加 @Component 注解

  1. hystrix:
  2. command:
  3. default:
  4. execution:
  5. isolation:
  6. thread:
  7. timeoutInMilliseconds: 500

1.创建降级类

  • getRoute() 方法中指定应用此降级类的服务id,星号或null值可以通配所有服务
  1. @Component
  2. public class ItemFallback implements FallbackProvider {
  3. /**
  4. * 当前降级类,对那个服务有效
  5. * 如果返回 "item-service" , 那么当前降级类只对商品服务有效
  6. * 如果返回 * 或者null,那么对所有服务都有效
  7. * @return
  8. */
  9. @Override
  10. public String getRoute() {
  11. return "item-service";
  12. }
  13. @Override
  14. public ClientHttpResponse fallbackResponse(String route, Throwable cause) {
  15. return new ClientHttpResponse() {
  16. @Override
  17. public HttpStatus getStatusCode() throws IOException {
  18. return HttpStatus.OK;
  19. }
  20. @Override
  21. public int getRawStatusCode() throws IOException {
  22. return HttpStatus.OK.value();
  23. }
  24. @Override
  25. public String getStatusText() throws IOException {
  26. return HttpStatus.OK.getReasonPhrase();
  27. }
  28. @Override
  29. public void close() {
  30. }
  31. @Override
  32. public InputStream getBody() throws IOException {
  33. // JsonRest : {code:400,msg:"获取商品列表失败!" , data:null}
  34. String s = JsonResult.err().msg("获取商品列表失败!").toString();
  35. ByteArrayInputStream in = new ByteArrayInputStream(s.getBytes());
  36. return in;
  37. }
  38. @Override
  39. public HttpHeaders getHeaders() {
  40. // Content-Type: application-json
  41. HttpHeaders h = new HttpHeaders();
  42. h.setContentType(MediaType.APPLICATION_JSON);
  43. return h;
  44. }
  45. };
  46. }
  47. }

2.降低hystrix超时时间,以便测试降级

  1. spring:
  2. application:
  3. name: zuul
  4. server:
  5. port: 3001
  6. eureka:
  7. client:
  8. service-url:
  9. defaultZone: http://eureka1:2001/eureka, http://eureka2:2002/eureka
  10. zuul:
  11. retryable: true
  12. ribbon:
  13. ConnectTimeout: 1000
  14. ReadTimeout: 2000
  15. MaxAutoRetriesNextServer: 1
  16. MaxAutoRetries: 1
  17. hystrix:
  18. command:
  19. default:
  20. execution:
  21. isolation:
  22. thread:
  23. timeoutInMilliseconds: 500

3.启动服务,测试降级

http://localhost:3001/item-service/35
Zuul 使用教程 - 图4

zuul 请求过滤

Zuul 使用教程 - 图5
添加过滤器,继承ZuulFilter ,按照规则实现过滤方法. 不用做任何配置,只需要添加@Component 注解

1.定义过滤器,继承 ZuulFilter

  1. @Component
  2. public class AccessFilter extends ZuulFilter {
  3. /**
  4. * 过滤器的类型 : 前置 ,后置, 路由, 错误
  5. * @return
  6. */
  7. @Override
  8. public String filterType() {
  9. return FilterConstants.PRE_TYPE;
  10. }
  11. /**
  12. * 过滤器添加的位置
  13. * 返回6 ,加到第6个位置
  14. * @return
  15. */
  16. @Override
  17. public int filterOrder() {
  18. return 6;
  19. }
  20. /**
  21. * 针对当前请求进行判断,判断当前请求是否要执行这个过滤器的过滤代码
  22. * 如果访问 item-service 要检查权限
  23. * 如果访问其他服务则不检查权限,直接访问
  24. * @return
  25. */
  26. @Override
  27. public boolean shouldFilter() {
  28. RequestContext ctx= RequestContext.getCurrentContext();
  29. //获取当前请求的服务id
  30. String serviceId=(String)ctx.get(FilterConstants.SERVICE_ID_KEY);
  31. if(serviceId.equals("item-service")){
  32. return true;//要执行权限过滤
  33. }
  34. return false;//跳过过滤代码不执行
  35. }
  36. /**
  37. * 过滤代码,对用户权限进行检查
  38. */
  39. @Override
  40. public Object run() throws ZuulException {
  41. RequestContext ctx= RequestContext.getCurrentContext();
  42. HttpServletRequest request = ctx.getRequest();
  43. String token = request.getParameter("token");
  44. if(StringUtils.isEmpty(token)){
  45. //没有token,阻止继续调用
  46. ctx.setSendZuulResponse(false);
  47. //发送提示,提示没有用户登录
  48. ctx.setResponseStatusCode(JsonResult.NOT_LOGIN);
  49. ctx.setResponseBody(JsonResult.err().code(JsonResult.NOT_LOGIN).msg("not login!").toString());
  50. }
  51. return null; //zuul过度设计,返回值在现在的版本中没有使用
  52. }
  53. }

2.访问测试

  • 没有token参数不允许访问
  • 有token参数可以访问