思路分析
-
用户进入网关开始登陆,网关过滤器进行判断,如果是登录,则路由到后台管理微服务进行登录
-
用户登录成功,后台管理微服务签发JWT TOKEN信息返回给用户
-
用户再次进入网关开始访问,网关过滤器接收用户携带的TOKEN
-
网关过滤器解析TOKEN ,判断是否有权限,如果有,则放行,如果没有则返回未认证错误
在网关微服务中新建全局过滤器
编写全局过滤器
package com.heima.admin.gateway.filter;
import com.heima.common.dtos.Payload;
import com.heima.common.util.JwtUtils;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;
import org.springframework.cloud.gateway.filter.GatewayFilterChain;
import org.springframework.cloud.gateway.filter.GlobalFilter;
import org.springframework.core.Ordered;
import org.springframework.http.HttpStatus;
import org.springframework.http.server.RequestPath;
import org.springframework.http.server.reactive.ServerHttpRequest;
import org.springframework.http.server.reactive.ServerHttpResponse;
import org.springframework.stereotype.Component;
import org.springframework.web.server.ServerWebExchange;
import reactor.core.publisher.Mono;
/**
* 网关全局过滤器
*/
@Slf4j
@Component
public class AuthFilter implements GlobalFilter, Ordered {
/**
* 网关全局过滤
* 如果是登录操作,路由到Admin服务进行登录
* 如果不是登录操作,就需要判断是否携带token
* 解析token,获取用户信息,判断是否有响应权限,如果有就放行,如果没有就返回认证错误信息
* @param exchange
* @param chain
* @return
*/
@Override
public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
// 获取当前请求
ServerHttpRequest request = exchange.getRequest();
ServerHttpResponse response = exchange.getResponse();
// 获取当前请求路径
String uriPath = request.getURI().getPath();
// 判断当前请求路径是否是登录
if(uriPath.contains("/login/in")){
// 直接放行到admin服务
return chain.filter(exchange);
}
// 不是登录请求,需要验证token
String token = request.getHeaders().getFirst("token");
if(StringUtils.isBlank(token)){
// 如果token为空,返回401未授权状态
response.setStatusCode(HttpStatus.UNAUTHORIZED);
return response.setComplete();
}
// 校验token
try{
Payload payload = JwtUtils.getInfoFromToken(token);
Integer userId = payload.getUserId();
// 把用户id 放入header中
ServerHttpRequest serverHttpRequest = request.mutate().headers(httpHeaders -> {
httpHeaders.add("userId", String.valueOf(userId));
}).build();
exchange.mutate().request(serverHttpRequest).build();
}catch (Exception e){
log.error("验证token出现错误!");
e.printStackTrace();
response.setStatusCode(HttpStatus.UNAUTHORIZED);
return response.setComplete();
}
return chain.filter(exchange);
}
/**
* 数字越小 优先级越高
* @return
*/
@Override
public int getOrder() {
return 0;
}
}