声明,使用JDK8、springcloud2.01 、maven3.5.4、idea
Zuul 是什么,有什么作用:
核心作用3个:
过滤和路由、异常处理;
另外、Zuul 集成了负载均衡,在线面案例会有展示、
》
下面分别讲Zuul 的基础使用,以及这三个作用的案例演示、
<dependencies>
<dependency>
<groupId>org.springframework.cloudgroupId>
<artifactId>spring-cloud-starter-netflix-zuulartifactId>
dependency>
<dependency>
<groupId>org.springframework.cloudgroupId>
<artifactId>spring-cloud-starter-netflix-eureka-clientartifactId>
dependency>
dependencies>
由于存在Eureka 客户端,所以直接指定服务端的名字即可;
后面还有更简化的配置;
server:
port: 10010 #端口
spring:
application:
name: api-gateway # 网关名字
zuul:
routes:
user-service: # 路由的 id
path: /user-service/** #映射路径
# url: http://127.0.0.1:8081 #映射路径对应的实际的url地址
# 访问路径规则 添加 映射路径的规则
serviceId: user-service # 指定服务端的名称
简化配置:直接指定服务端的 ID 指向 映射路径 path
zuul.routes..path=/xxx/**: 来指定映射路径。是自定义的路由名
zuul.routes..serviceId=/user-service:来指定服务名。
简化配置:
zuul.routes.= 即简化成如下配置:
zuul:
routes:
user-service: /user-service/** # 这里是映射路径
默认配置: Zuul 默认配置规则就是服务path 对应就是服务ID ,意思就是不配置Zuul 也是OK的。
忽略配置:
想要禁用某一个路由规则 采用如下配置:
zuul:
ignored-services:
- user-service
- consumer
访问代理 http://localhost:10010/user-service/user/2
出现500报错,报错如下:
com.netflix.zuul.exception.ZuulException: Forwarding error
、、省略
Caused by: com.netflix.client.ClientException: Load balancer does not have available server for client: path
异常原因如下图:
明显路径映射不对,仔细查看发现yml 文件中path 、url 和user-service 在同一级路径了,都是坑;
修改如下图,恢复正常访问;
1、添加前缀: /api
使用 zuul.prefix : /api 的语法,yml具体配置如下:
访问:http://localhost:10010/api/user-service/user/2
zuul:
prefix: /api # 添加路由前缀
routes:
user-service: # 路由的 id
path: /user-service/** #映射路径
# url: http://127.0.0.1:8081 #映射路径对应的实际的url地址
# 访问路径规则 添加 映射路径的规则
serviceId: user-service # 指定服务端的名称
2018.12.5 更新
》
》
1、异常处理:
zuul 为我们提供了一个名叫 ZuulFallbackProvider 的接口、 在服务挂掉时候调用该接口返回信息,我们通过实现该接口,重写接口中fallbackResponse 方法,返回错误的处理规则、给出信息提示:
Component
public class ApiFallbackProvider implements ZuulFallbackProvider{
@Override
public String getRoute() {
return "eurekaclient";
}
@Override
public ClientHttpResponse fallbackResponse() {
return new ClientHttpResponse() {
@Override
public HttpStatus getStatusCode() throws IOException {
return HttpStatus.OK;
}
@Override
public int getRawStatusCode() throws IOException {
return 200;
}
@Override
public String getStatusText() throws IOException {
return "{code:0,message:\"服务器异常!\"}";
}
@Override
public void close() {
}
@Override
public InputStream getBody() throws IOException {
return new ByteArrayInputStream(getStatusText().getBytes());
}
@Override
public HttpHeaders getHeaders() {
HttpHeaders headers = new HttpHeaders();
headers.setContentType(MediaType.APPLICATION_JSON);
return headers;
}
};
}
2、动态路由:
Zuul 实现动态路由,是因为Zuul 集成的有负载均衡、有负载均衡的效果。
- 首先更改服务提供者 EurekaClient 端口、启动多个服务提供者,然后访问服务 http://localhost:8761/ 如下图,有2个服务提供者:
- 启动网关、 不断访问地址:http://localhost:8080/api/index 会看到如下,端口交替出现、意味着服务提供者交替提供服务;
就是默认的轮询的方式;
3、过滤、服务拦截:
1、服务网关还有个作用就是接口的安全性校验,这个时候我们就需要通过 zuul 进行统一拦截,zuul 通过继承过滤器 ZuulFilter 进行处理:
2、重写ZuulFilter 类中的 方法、
package com.syntactic.filter;
/**
* @auther SyntacticSugar
* @data 2018/12/5 0005下午 5:00
*/
@Component
public class ApiFilter extends ZuulFilter {
@Override
public String filterType() {
return "pre";
}
@Override
public int filterOrder() {
return 0;
}
@Override
public boolean shouldFilter() {
return true;
}
@Override
public Object run() {
// 逻辑代码
}
}
filterType 为过滤类型,可选值有 pre(路由之前)、routing(路由之时)、post(路由之后)、error(发生错误时调用)。
filterOrdery 为过滤的顺序,如果有多个过滤器,则数字越小越先执行
shouldFilter 表示是否过滤,这里可以做逻辑判断,true 为过滤,false 不过滤
run 为过滤器执行的具体逻辑;
以下逻辑做一个权限校验、
@Override
public Object run() {
// 逻辑代码
RequestContext currentContext = RequestContext.getCurrentContext();
HttpServletRequest request = currentContext.getRequest();
// 获取传递的参数、验证权限
String token = request.getParameter("token");
//
if (!"123456".equals(token)) {
currentContext.setSendZuulResponse(false);
currentContext.setResponseStatusCode(400);
// 把提示信息显示到 页面
try {
currentContext.getResponse().getWriter().write("token is invalid");
} catch (IOException e) {
e.printStackTrace();
}
}
return null;
}
访问:http://localhost:8080/api/index 非法
访问 http://localhost:8080/api/index?token=12345 有效