|
|
|
|
@ -4,11 +4,11 @@ import cn.hutool.core.util.StrUtil;
|
|
|
|
|
import cn.iocoder.yudao.framework.common.pojo.CommonResult;
|
|
|
|
|
import cn.iocoder.yudao.framework.common.util.json.JsonUtils;
|
|
|
|
|
import cn.iocoder.yudao.gateway.util.SecurityFrameworkUtils;
|
|
|
|
|
import cn.iocoder.yudao.gateway.util.WebFrameworkUtils;
|
|
|
|
|
import cn.iocoder.yudao.module.system.api.oauth2.OAuth2TokenApi;
|
|
|
|
|
import cn.iocoder.yudao.module.system.api.oauth2.dto.OAuth2AccessTokenCheckRespDTO;
|
|
|
|
|
import cn.iocoder.yudao.module.system.api.oauth2.dto.OAuth2AccessTokenRespDTO;
|
|
|
|
|
import com.fasterxml.jackson.core.type.TypeReference;
|
|
|
|
|
import com.google.common.net.HttpHeaders;
|
|
|
|
|
import org.springframework.cloud.client.loadbalancer.reactive.ReactorLoadBalancerExchangeFilterFunction;
|
|
|
|
|
import org.springframework.cloud.gateway.filter.GatewayFilterChain;
|
|
|
|
|
import org.springframework.cloud.gateway.filter.GlobalFilter;
|
|
|
|
|
import org.springframework.core.Ordered;
|
|
|
|
|
@ -29,14 +29,21 @@ import java.util.function.Function;
|
|
|
|
|
*
|
|
|
|
|
* @author 芋道源码
|
|
|
|
|
*/
|
|
|
|
|
@Component // TODO 芋艿:要改成 configuration
|
|
|
|
|
@Component
|
|
|
|
|
public class TokenAuthenticationFilter implements GlobalFilter, Ordered {
|
|
|
|
|
|
|
|
|
|
// @Resource
|
|
|
|
|
// private OAuth2TokenApi oauth2TokenApi;
|
|
|
|
|
private static final TypeReference<CommonResult<OAuth2AccessTokenCheckRespDTO>> CHECK_RESULT_TYPE_REFERENCE
|
|
|
|
|
= new TypeReference<CommonResult<OAuth2AccessTokenCheckRespDTO>>() {};
|
|
|
|
|
|
|
|
|
|
@Resource
|
|
|
|
|
private WebClient webClient;
|
|
|
|
|
private final WebClient webClient;
|
|
|
|
|
|
|
|
|
|
public TokenAuthenticationFilter(ReactorLoadBalancerExchangeFilterFunction lbFunction) {
|
|
|
|
|
// Q:为什么不使用 OAuth2TokenApi 进行调用?
|
|
|
|
|
// A1:Spring Cloud OpenFeign 官方未内置 Reactive 的支持 https://docs.spring.io/spring-cloud-openfeign/docs/current/reference/html/#reactive-support
|
|
|
|
|
// A2:校验 Token 的 API 需要使用到 header[tenant-id] 传递租户编号,暂时不想编写 RequestInterceptor 实现
|
|
|
|
|
// 因此,这里采用 WebClient,通过 lbFunction 实现负载均衡
|
|
|
|
|
this.webClient = WebClient.builder().filter(lbFunction).build();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public Mono<Void> filter(final ServerWebExchange exchange, GatewayFilterChain chain) {
|
|
|
|
|
@ -46,46 +53,23 @@ public class TokenAuthenticationFilter implements GlobalFilter, Ordered {
|
|
|
|
|
return chain.filter(exchange);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// exchange = exchange.mutate().request(r -> r.headers(new Consumer<HttpHeaders>() {
|
|
|
|
|
// @Override
|
|
|
|
|
// public void accept(HttpHeaders headers) {
|
|
|
|
|
// headers.set("user-id", "1");
|
|
|
|
|
// }
|
|
|
|
|
// })).build();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// return Mono.fromCallable(new Callable<CommonResult<OAuth2AccessTokenCheckRespDTO>>() {
|
|
|
|
|
// @Override
|
|
|
|
|
// public CommonResult<OAuth2AccessTokenCheckRespDTO> call() throws Exception {
|
|
|
|
|
//// return oauth2TokenApi.checkAccessToken("1234");
|
|
|
|
|
// return CommonResult.success(new OAuth2AccessTokenCheckRespDTO().setUserId(1L));
|
|
|
|
|
// }
|
|
|
|
|
// }).subscribeOn(Schedulers.boundedElastic()).flatMap(new Function<CommonResult<OAuth2AccessTokenCheckRespDTO>, Mono<Void>>() {
|
|
|
|
|
// @Override
|
|
|
|
|
// public Mono<Void> apply(CommonResult<OAuth2AccessTokenCheckRespDTO> oAuth2AccessTokenCheckRespDTOCommonResult) {
|
|
|
|
|
// return chain.filter(exchange);
|
|
|
|
|
// }
|
|
|
|
|
// });
|
|
|
|
|
|
|
|
|
|
// 情况二,如果有 Token 令牌,则解析对应 userId、userType、tenantId 等字段,并通过 通过 Header 转发给服务
|
|
|
|
|
// TODO 芋艿:tenant-id
|
|
|
|
|
String tenantId = exchange.getRequest().getHeaders().getFirst("tenant-id");
|
|
|
|
|
return webClient.get()
|
|
|
|
|
.uri(OAuth2TokenApi.URL_CHECK, uriBuilder -> uriBuilder.queryParam("accessToken", token).build())
|
|
|
|
|
.header("tenant-id", tenantId)
|
|
|
|
|
.headers(httpHeaders -> WebFrameworkUtils.setTenantIdHeader(exchange, httpHeaders)) // 设置租户的 Header
|
|
|
|
|
.retrieve().bodyToMono(String.class) // 发起请求,设置 body 为 String 结果
|
|
|
|
|
// 处理请求的结果
|
|
|
|
|
.flatMap((Function<String, Mono<Void>>) body -> chain.filter(buildNewServerWebExchange(exchange, body)));
|
|
|
|
|
.flatMap((Function<String, Mono<Void>>) body -> chain.filter(buildNewServerWebExchange(exchange, body))); // 处理请求的结果
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private ServerWebExchange buildNewServerWebExchange(ServerWebExchange exchange, String body) {
|
|
|
|
|
// 校验 Token 令牌失败,则直接返回
|
|
|
|
|
CommonResult<?> result = JsonUtils.parseObject(body, CommonResult.class);
|
|
|
|
|
CommonResult<OAuth2AccessTokenCheckRespDTO> result = JsonUtils.parseObject(body, CHECK_RESULT_TYPE_REFERENCE);
|
|
|
|
|
if (result == null || result.isError()) {
|
|
|
|
|
return exchange;
|
|
|
|
|
}
|
|
|
|
|
// 创建新的 exchange 对象
|
|
|
|
|
return exchange.mutate().request(builder -> builder.header("login-user", result.getData().toString())).build();
|
|
|
|
|
|
|
|
|
|
// 将访问令牌封装成 LoginUser,并设置到 login-user 的请求头,使用 json 存储值
|
|
|
|
|
return exchange.mutate().request(builder -> SecurityFrameworkUtils.setLoginUserHeader(builder, result.getData())).build();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
|