概述:
是系统统一对外的入口,它是介于客户端和服务端之间的中间层,用来处理非业务功能,比如提供路由,监控,鉴权,缓存,限流等功能
作用:
- 负载均衡的作用:它的负载均衡依赖于ribbon,包括Fegin和RestTemplete都是依赖于ribbon实现的负载均衡
- 网关功能:其实是一系列过滤器,拥有前置过滤器、后置过滤器、路由、错误过滤器等
流程:
-
zuul是前台服务访问后台服务时使用的技术,当前台服务发送请求时可以对前台服务进行过滤,比如加一些验证策略(可以带一个token来访问zuul,根据token来保证前端服务发送的请求)
-
当验证通过后,会通过路由访问后台的接口,一般有一些路由机制,比如可以按照特定的名称进行访问后台接口,一般在数据库中配置路由地址
-
访问完接口后,后台相应数据到zuul,zuul可以对相应的数据进行加密处理
说明: Feign必须在配置中启动路由功能,若不启动,则无法实现路由功能,RestTemplete不需要配置,默认就有路由功能
使用:
- 主函数添加@EnableZuulProxy以及@EnableEurekaClient(zuul也是需要注册到注册中心的)
@EnableZuulProxy
@EnableEurekaClient
@SpringBootApplication
public class ZuulserverApplication {
public static void main(String[] args) {
SpringApplication.run(ZuulserverApplication.class, args);
}
}
- 访问地址:ip:网关端口号/服务注册名/路径/方法名 进行访问
自定义zuul过滤器实现登陆鉴权
- 新建一个文件继承ZuulFilter,重写里面的方法
- 添加@Component,让Spring扫描
@Component
public class testFilter extends ZuulFilter {
private Logger logger = LoggerFactory.getLogger(this.getClass());
@Autowired
private RedisFactory redisFactory;
/**
* 设置过滤器的类型
* pre 路由之前调用
* routing 路由请求时候调用
* post 路由之后调用
* error 发生错误时候调用
*/
@Override
public String filterType() {
return "pre";
}
//过滤器的顺序,数字越小表示顺序越高,越先执行
@Override
public int filterOrder() {
return 0;
}
/**
* 是否需要过滤器
*/
@Override
public boolean shouldFilter() {
return true;
}
/**
* 拦截时的具体操作
*/
@Override
public Object run() {
RequestContext ctx = RequestContext.getCurrentContext();
HttpServletResponse response = ctx.getResponse();//响应内容
HttpServletRequest request = ctx.getRequest();//请求内容
String url = request.getRequestURL().toString();//请求地址
RedisService redisService = redisFactory.getRedis();
String accessToken = request.getHeader("accessToken");//获取token
if (CommonUtil.isNotNullEmpty(accessToken)) {//判断是否为空
String userCode = redisService.getUserId(accessToken);
//登录超时
if (!CommonUtil.isNotNullEmpty(userCode)) {
ctx.setSendZuulResponse(false); //不进行路由
ctx.setResponseStatusCode(200);//设置返回状态码
try {
response.setContentType("text/json;charset=UTF-8");
response.getWriter().write("返回数据说明");
} catch (Exception e) {
e.printStackTrace();
}
return null;
} else {
//用户校验
redisService.validate(accessToken);
ctx.addZuulRequestHeader("accessToken", accessToken);
return null;
}
} else {
ctx.setSendZuulResponse(false); //不进行路由
ctx.setResponseStatusCode(200);//设置返回状态码
try {
response.setContentType("text/json;charset=UTF-8");
response.getWriter().write("返回数据说明");
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
return null;
}
}
应用场景
pre(前置):
限流、鉴权、参数校验等
post(后置):
统计、跨域等